数组转集合、集合转数组、字符串数组与int型、long型数组等的转换以及排序问题

==================类型转换==================

  在项目中常常会遇到数组转集合、集合转数组、数组之间类型转换等操做

 

1.数组转集合

为了实现把一个数组转换成一个ArrayList,不少Java程序员会使用以下的代码:html

        String str[] = {"1","2","3"};
        List<String> strings = Arrays.asList(str);

  Arrays.asList确实会返回一个ArrayList对象,可是该类是Arrays类 中一个私有静态内部类,而不是常见的java.util.ArrayList类。这个java.util.Arrays.ArrayList类具备 set(),get(),contains()等方法,可是不具备任何添加或移除元素的任何方法。由于该类的大小(size)是固定的。若是添加元素是会报错的(可是若是转换后的集合只是用来进行查询不进行增长元素也能够这样转换):java

        String str[] = {"1","2","3"};
        List<String> strings = Arrays.asList(str);
        strings.add("eee");

 

报错以下:程序员

Exception in thread "main" java.lang.UnsupportedOperationExceptionapache

at java.util.AbstractList.add(AbstractList.java:148)数组

at java.util.AbstractList.add(AbstractList.java:108)ide

at Test.test1(Test.java:31)spa

at Test.main(Test.java:24)3d

 

为了建立出一个真正的ArrayList,代码应该以下所示:(这种方法建立的集合能够进行集合的增长)code

        String str[] = {"1","2","3"};
        List<String> strings = new ArrayList<String>(Arrays.asList(str));
        strings.add("4");
        System.out.println(strings);

 

更加高效的代码以下:htm

        String str[] = {"1","2","3"};
        List<String> strings = new ArrayList<String>(str.length);
        Collections.addAll(strings,str);
        strings.add("4");
        System.out.println(strings);

 

2.集合转数组

(1)错误演示

不少人习惯下面用法:

        List<String> strings = new ArrayList<String>();
        String[] objects = (String[]) strings.toArray();

 

编译经过,运行报错以下:

Exception in thread "main" java.lang.ClassCastException: [Ljava.lang.Object; cannot be cast to [Ljava.lang.String;

at Test.test1(Test.java:32)

at Test.main(Test.java:26)

学过JVM的应该知道上面的意思是Object数组不能转变为String数组。[表明觉得数组,L表明数组的元素是引用类型,后面是具体的元素类型

  对于这个现象咱们能够这么解释:Java中容许向上和向下转型,可是这个转型是否成功是根据Java虚拟机中这个对象的类型来实现的。Java虚拟机中保存 了每一个对象的类型。而数组也是一个对象。数组的类型是[Ljava.lang.Object。把[Ljava.lang.Object转换成 [Ljava.lang.String是显然不可能的事情,由于这里是一个向下转型,而虚拟机只保存了这是一个Object的数组,不能保证数组中的元素 是String的,因此这个转型不能成功。数组里面的元素只是元素的引用,不是存储的具体元素,因此数组中元素的类型仍是保存在Java虚拟机中的。

根据上面的解释,咱们能够把这个问题概括到下面这个模型:

Object objs[]=new Object[10];
String strs[]=(String[])objs;

 

这样子和刚才上面编译错误是同样的。若是咱们修改一下这个代码,以下:

String strs[]=new String[10];
Object objs[]=strs;

 

这样子就能够编译经过了。因此这个问题咱们能够归结为一个Java转型规则的问题。

 

(2)正确的作法:(延伸一点直接打印数组打印的是数组的)

最菜鸡的作法是:

        List<String> list = new ArrayList<String>();
        list.add("1");
        String strings[]=new String[list.size()];
        for(int i=0,j=list.size();i<j;i++){
            strings[i]=list.get(i);
        }

        System.out.println(strings);
        System.out.println(strings.getClass());
        System.out.println(Arrays.toString(strings));

结果:

[Ljava.lang.String;@20724356

class [Ljava.lang.String;

[1]

 

比较简便的作法:

        List<String> list = new ArrayList<String>();
        list.add("1");
        String[] strings = new String[list.size()];
        list.toArray(strings);

        System.out.println(strings);
        System.out.println(strings.getClass());
        System.out.println(Arrays.toString(strings));

结果同上。

 

3.数组转数组--代码用到了commons-beanutils包

最多见的就是字符串数组类型转int、long数组,或者字符串类型转Integer、Long、Integer型转int(也就是包装类型转原始类型)。

最原始的for循环转换赋值在这里就不试了。

 

数组类型的转换:

import org.apache.commons.beanutils.ConvertUtils;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;

public class Test {
    public static void main(String[] args) {
        String str[] = { "1", "2", "3" };

       // 字符串数组转long数组
        long[] str2lon = (long[]) ConvertUtils.convert(str, long.class);
        System.out.println(str2lon);
        // 字符串数组转Long数组
        Long[] str2Lon = (Long[]) ConvertUtils.convert(str, Long.class);
        System.out.println(str2Lon);


        // 字符串数组转int数组
        int[] str2int = (int[]) ConvertUtils.convert(str, int.class);
        System.out.println(str2int);
        // 字符串数组转Integer数组
        Integer[] str2Int = (Integer[]) ConvertUtils.convert(str, Integer.class);
        System.out.println(str2Int);

        // int型数组转为String数组
        String int2Str[] = (String[]) ConvertUtils.convert(str2int, String[].class);
        System.out.println(int2Str);
        // Integer型数组转为String数组
        String Int2Str[] = (String[]) ConvertUtils.convert(str2Int, String[].class);
        System.out.println(Int2Str);

        // long型数组转为String数组
        String lon2str[] = (String[]) ConvertUtils.convert(str2lon, String[].class);
        System.out.println(lon2str);
        String Lon2str[] = (String[]) ConvertUtils.convert(str2Lon, String[].class);
        System.out.println(Lon2str);
    }
}

[J@15a6d5e1

[Ljava.lang.Long;@7c23b1e1

[I@b736a73

[Ljava.lang.Integer;@4651a9e4

[Ljava.lang.String;@1b68dbcd

[Ljava.lang.String;@1367dca

[Ljava.lang.String;@207c5965

[Ljava.lang.String;@43d1068c

 

关于包装类型转原始类,能够用commons-lang包的ArrayUtils操做,参考:http://www.javashuo.com/article/p-thahcpjv-cx.html

 

 补充:补充一点JVM相关知识

在Java中,任何类型都有class,包括基本数据类型与void。在Java中 [   表明数组,  [[  表明二维数组。依次类推。

其余的描述标识符以下:

B---基本数据类型byte

C---基本数据类型char

D---基本数据类型double

F---基本数据类型float

I---基本数据类型int

J---基本数据类型long

S---基本数据类型short

Z---基本数据类型boolean

V---特殊类型void

L---对象类型(也就是引用类型)。例如 Ljava/lang/Object.

 

须要注意的是八种基本数据类型也有  calss,并且其对应包装类型有一个TYPE成员变量就是其基本数据类型。特殊类型void也有class。基本类型的数组与引用类型数组也都有class

public static final Class<Integer>  TYPE = (Class<Integer>) Class.getPrimitiveClass("int");

 

例如:

        System.out.println(int.class);
        System.out.println(Integer.class);
        System.out.println(Integer.TYPE);

        System.out.println(Integer[].class);
        System.out.println(int[].class);
        System.out.println(void.class);

结果:

int
class java.lang.Integer
int
class [Ljava.lang.Integer;
class [I
void

 

注意int不是对象,因此没有getClass方法。只有int.class

 

 

==================数组与集合排序==================

  JDK自带的排序方法能够知足大部分要求。咱们知道要在集合中排序,须要使用能够排序的集合或者本身手动排序。使用可排序的集合如TreeMap,TreeSet。若是手动排序就是用Collections.sort传入一个比较器便可。

 

 1.数组排序

  在使用Arrays.sort()对int、double、long等基本类型的数组进行排序时,只有正序排序的方法。要实现倒序排序,只有使用Integer、Double、Long等代替。

  包装类型比较的时候能够传入比较器,以下:

import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;

public class PlainTest {

    public static void main(String[] args) {
        // 基本数据类型Arrays.sort正序
        char[] chars = { 'a', 'b', 'd', 'e', 'c', 'e', 'd', 'a' };
        Arrays.sort(chars);
        System.out.println(chars);

        // 原生数组正序
        Character[] characters = { 'a', 'b', 'd', 'e', 'c', 'e', 'd', 'a' };
        Arrays.sort(characters);
        System.out.println(Arrays.toString(characters));

        // 原生数组逆序
        Arrays.sort(characters, Collections.reverseOrder());
        System.out.println(Arrays.toString(characters));

        // 本身实现逆序号排序
        Arrays.sort(characters, new Comparator<Character>() {
            @Override
            public int compare(Character o1, Character o2) {
                if (o1 > o2) {
                    return -1;
                } else if (o1 < o2) {
                    return 1;
                }
                return 0;
            }
        });
        System.out.println(Arrays.toString(characters));
    }

}

结果:

aabcddee
[a, a, b, c, d, d, e, e]
[e, e, d, d, c, b, a, a]
[e, e, d, d, c, b, a, a]

 

查看源码:(逆序是经过反转compare来实现的)

    public static <T> Comparator<T> reverseOrder() {
        return (Comparator<T>) ReverseComparator.REVERSE_ORDER;
    }

    private static class ReverseComparator
        implements Comparator<Comparable<Object>>, Serializable {

        private static final long serialVersionUID = 7207038068494060240L;

        static final ReverseComparator REVERSE_ORDER
            = new ReverseComparator();

        public int compare(Comparable<Object> c1, Comparable<Object> c2) {
            return c2.compareTo(c1);
        }

        private Object readResolve() { return reverseOrder(); }
    }

 

2.集合排序

  集合排序与数组相似。

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;

public class PlainTest {

    public static void main(String[] args) {
        Character[] characters = { 'a', 'b', 'd', 'e', 'c', 'e', 'd', 'a' };
        List<Character> list = new ArrayList<Character>(Arrays.asList(characters));
        System.out.println(list);

        // 集合正序
        Collections.sort(list);
        System.out.println(list);

        // 集合逆序
        Collections.sort(list, Collections.reverseOrder());
        System.out.println(list);

        // 本身实现逆序号排序
        Collections.sort(list, new Comparator<Character>() {
            @Override
            public int compare(Character o1, Character o2) {
                if (o1 > o2) {
                    return -1;
                } else if (o1 < o2) {
                    return 1;
                }
                return 0;
            }
        });
        System.out.println(list);
    }

}

结果:

[a, b, d, e, c, e, d, a][a, a, b, c, d, d, e, e][e, e, d, d, c, b, a, a][e, e, d, d, c, b, a, a]

相关文章
相关标签/搜索