今天的菜品是這樣的:
List十籍、Set蛆封、數(shù)組、Map這些常用的數(shù)據(jù)結(jié)構(gòu)之間是怎么轉(zhuǎn)換的勾栗?
太長(zhǎng)不看版
- List和Set互轉(zhuǎn)用構(gòu)造器
- List和Set轉(zhuǎn)數(shù)組用toArray()方法
- 數(shù)組轉(zhuǎn)List和Set用構(gòu)造器嵌套Arrays.asList()方法
- List轉(zhuǎn)Map惨篱,先將List轉(zhuǎn)為Stream,再利用Stream的collect(Collectors.toMap())方法轉(zhuǎn)為Map
- List/Set轉(zhuǎn)Stream用stream()方法
- 數(shù)組轉(zhuǎn)Stream用Arrays.stream()方法
- 可以用Comparator.comparing()方法構(gòu)造比較器围俘,可以用thenComparing()方法構(gòu)造多字段排序的構(gòu)造器砸讳,可以用reversed()方法翻轉(zhuǎn)比較器的排序規(guī)則。
下面是正文:
List界牡、Set和Array之間的互轉(zhuǎn)
List和Set互轉(zhuǎn)
List和Set之間的相互轉(zhuǎn)換主要靠List和Set的有參構(gòu)造器簿寂。這種方法同樣適用于List或Set不同實(shí)現(xiàn)類之間的互相轉(zhuǎn)換。
List/Set和Array互轉(zhuǎn)
數(shù)組轉(zhuǎn)List/Set時(shí)宿亡,首先可以利用Arrays.asList方法轉(zhuǎn)換為L(zhǎng)ist常遂,這時(shí)轉(zhuǎn)成的List對(duì)象是一個(gè)名為ArrayList的內(nèi)部類的對(duì)象,它是原數(shù)組的視圖挽荠,有很多操作不支持克胳。因此我們需要再利用上一節(jié)中的方法用List/Set的構(gòu)造器再包一層。List/Set轉(zhuǎn)為數(shù)組時(shí)圈匆,可以直接用數(shù)據(jù)的toArray方法漠另,由于Java中泛型的限制,toArray方法需要傳一個(gè)數(shù)組作為參數(shù)跃赚,這個(gè)數(shù)組通常傳一個(gè)新建的相應(yīng)類型的0元素?cái)?shù)組即可笆搓。
Integer[] array = {1, 2, 3, 4, 5};
// array -> list
List<Integer> listFromArray = new ArrayList<>(Arrays.asList(array));
System.out.println(listFromArray);
// list -> array
Integer[] arrayFromList = listFromArray.toArray(new Integer[0]);
System.out.println(Arrays.toString(arrayFromList));
// list -> set
Set<Integer> setFromList = new HashSet<>(listFromArray);
System.out.println(setFromList);
// set -> list
List<Integer> listFromSet = new ArrayList<>(setFromList);
System.out.println(listFromSet);
// set -> array
Integer[] arrayFromSet = setFromList.toArray(new Integer[0]);
System.out.println(Arrays.toString(arrayFromSet));
// array -> set
Set<Integer> setFromArray = new HashSet<>(Arrays.asList(array));
System.out.println(setFromArray);
List、Set、Array和流(Stream)的互轉(zhuǎn)
Stream(流)是Jdk8開始引入的一個(gè)抽象概念砚作,它并不是實(shí)體的數(shù)據(jù)結(jié)構(gòu)窘奏,不保存對(duì)象嘹锁,但是它可以在讓要處理的元素在管道中傳輸葫录,在管道的各個(gè)節(jié)點(diǎn)上進(jìn)行篩選、排序领猾、聚合等處理米同。可以把List等數(shù)據(jù)結(jié)構(gòu)想象成水桶摔竿,Stream就是水管面粮,我們可以把水桶中的水抽到水管中進(jìn)行處理,但是處理過后一定要再用一個(gè)容器去容納這些水继低。
List/Set和Stream互轉(zhuǎn)
List/Set轉(zhuǎn)為Stream熬苍,可以利用這些類本身的stream()方法。Stream轉(zhuǎn)為L(zhǎng)ist/Set時(shí)袁翁,可以利用Stream的collect方法柴底,將Collectors.toList()或Collectors.toSet()作為參數(shù)。
Array和Stream互轉(zhuǎn)
Array轉(zhuǎn)為Stream粱胜,可以使用Arrays.Stream方法或者Stream.of方法柄驻,這些方法的參數(shù)都是泛型不定長(zhǎng)參數(shù),可以將數(shù)組作為參數(shù)傳入焙压。但是需注意鸿脓,如果要轉(zhuǎn)換的數(shù)組是基本數(shù)據(jù)類型(int、long和double)涯曲,那么必須使用Arrays.Stream方法野哭,因?yàn)樵摲椒榛緮?shù)據(jù)類型設(shè)定了重載方法。如果使用Stream.of方法幻件,整個(gè)數(shù)組會(huì)被當(dāng)成一個(gè)元素拨黔,從而將數(shù)組轉(zhuǎn)換為了包含一個(gè)元素的數(shù)組的流。另外注意傲武,除了int蓉驹、long和double外,其他基本數(shù)據(jù)類型沒有對(duì)應(yīng)的基本流揪利。
// array -> stream
Integer[] array = {1, 2, 3, 4, 5};
int[] intArray = {1, 2, 3, 4, 5};
// 1. Stream.of沒有為int[]單獨(dú)設(shè)置重載方法态兴,因此整個(gè)int[]對(duì)象會(huì)被當(dāng)做一個(gè)參數(shù)
Stream<Integer> stream1 = Stream.of(array);
Stream<int[]> stream2 = Stream.of(intArray);
// 2. Arrays.stream為int[]單獨(dú)設(shè)置了重載方法,因此會(huì)被轉(zhuǎn)換成基本數(shù)據(jù)流IntStream(流里存的元素是int類型)
Stream<Integer> stream3 = Arrays.stream(array);
IntStream intStream = Arrays.stream(intArray);
// stream -> array
Integer[] arrayFromStream = stream1.toArray(Integer[]::new);
int[] intArrayFromStream = intStream.toArray();
// list -> stream
List<Integer> listFromArray = new ArrayList<>(Arrays.asList(array));
Stream<Integer> streamFromList = listFromArray.stream();
// stream -> list
List<Integer> listFromStream = streamFromList.collect(Collectors.toList());
// set -> stream
Set<Integer> setFromArray = new HashSet<>(Arrays.asList(array));
Stream<Integer> streamFromSet = setFromArray.stream();
// stream -> set
Set<Integer> setFromStream = streamFromSet.collect(Collectors.toSet());
List轉(zhuǎn)為Map
想要將List轉(zhuǎn)為Map疟位,我們可以先把集合轉(zhuǎn)成流瞻润,再調(diào)用流的歸并操作collect。collect方法的數(shù)據(jù)通過調(diào)用Collectors.toMap方法獲取,該方法的兩個(gè)參數(shù)分別傳根據(jù)List中元素生成Map的key和value的策略绍撞。
List<Student> students = new ArrayList<>();
students.add(student);
// 將list轉(zhuǎn)為學(xué)生ID和學(xué)生對(duì)象的映射表
Map<Integer, Student> studentMap = students.stream()
.collect(Collectors.toMap(Student::getId, s->s));
數(shù)據(jù)分組及后續(xù)處理
想要對(duì)數(shù)據(jù)進(jìn)行分組正勒,我們可以先把集合轉(zhuǎn)成流,再調(diào)用流的歸并操作collect傻铣。collect方法的數(shù)據(jù)通過調(diào)用Collectors.groupingBy方法獲取章贞,groupingBy方法可以傳一到兩個(gè)參數(shù),第一個(gè)參數(shù)為分組的依據(jù)非洲,如果傳第二個(gè)參數(shù)鸭限,則它代表分組后的后續(xù)(downstream)操作。以求學(xué)生平均分為例:
List<Student> students = new ArrayList<>();
// 求男/女學(xué)生的平均分
Map<Integer, Double> averageScoreMap = students.stream()
.collect(Collectors.groupingBy(Student::getGender, Collectors.averagingDouble(Student::getScore)));
數(shù)據(jù)的篩選两踏、按多個(gè)條件排序
filter
如果我們只需要獲得男學(xué)生的平均分败京,就不需要進(jìn)行分組操作了,可以直接使用filter方法進(jìn)行過濾操作梦染,如下:
final Double average = students.stream()
.filter(s -> s.getGender() == 1)
.collect(Collectors.averagingDouble(Student::getScore));
過濾操作傳入的參數(shù)是一個(gè)Predicate類型的對(duì)象赡麦,它代表元素保留下來(lái)的條件。
sort
如果需要對(duì)集合進(jìn)行排序帕识,我們可以使用List的sort方法泛粹,也可以使用Stream的sorted方法。這些方法要求我們傳一個(gè)Comparator對(duì)象渡冻,作為排序的依據(jù)戚扳。如:
students.sort((s1, s2) -> {
double diff = s1.getScore() - s2.getScore();
return diff > 0 ? 1 : diff == 0 ? 0 : -1;
});
Comparator.comparing
如果我們是根據(jù)類中的某個(gè)字段對(duì)某個(gè)類的對(duì)象進(jìn)行排序的話,我們可以使用Comparator.comparing這個(gè)靜態(tài)方法構(gòu)造Comparator對(duì)象族吻。因此帽借,上述代碼可以簡(jiǎn)化為:
students.sort(Comparator.comparing(Student::getScore));
Comparator的thenComparing方法
有時(shí)我們需要對(duì)多個(gè)字段進(jìn)行排序,這時(shí)我們可以利用Comparator對(duì)象的thenComparing方法構(gòu)造一個(gè)用于多字段排序的比較器超歌,方法中需要傳一個(gè)新的比較器或比較策略砍艾。如對(duì)學(xué)生先按班級(jí)再按分?jǐn)?shù)進(jìn)行排序:
students.sort(Comparator.comparing(Student::getClassNumber)
.thenComparing(Student::getScore));
Comparator的reversed方法
Comparator.comparing默認(rèn)是按自然排序,即由小到大排序巍举。有的時(shí)候我們需要按從大到小的順序進(jìn)行排序脆荷,此時(shí)我們可以調(diào)用Comparator的reversed方法饮笛,如對(duì)學(xué)生分?jǐn)?shù)按由高到低排序:
students.sort(Comparator
.comparing(Student::getScore)
.reversed());
本文介紹了常用數(shù)據(jù)結(jié)構(gòu)相互轉(zhuǎn)換的方法以及一些數(shù)據(jù)的常見處理方式截珍。今天的文章就到這里,有什么想要聊的話題偷拔,歡迎留言炭分。喜歡文章的同學(xué)也可以鼓勵(lì)我一下哦~