午夜剧场伦理_日本一道高清_国产又黄又硬_91黄色网战_女同久久另类69精品国产_妹妹的朋友在线

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Java中List集合對(duì)象去重及按屬性去重的8種方法

瀏覽:15日期:2022-08-25 13:21:03

Java中List集合對(duì)象去重及按屬性去重的8種方法

最近在寫(xiě)一些關(guān)于java基礎(chǔ)的文章,但是我又不想按照教科書(shū)的方式去寫(xiě)知識(shí)點(diǎn)的文章,因?yàn)橐饬x不大。基礎(chǔ)知識(shí)太多了,如何將這些知識(shí)歸納總結(jié),總結(jié)出優(yōu)缺點(diǎn)或者是使用場(chǎng)景才是對(duì)知識(shí)的升華。所以我更想把java相關(guān)的基礎(chǔ)知識(shí)進(jìn)行穿針引線,進(jìn)行整體上的總結(jié)。

總結(jié)java中創(chuàng)建并寫(xiě)文件的5種方式 總結(jié)java從文件中讀取數(shù)據(jù)的6種方法 總結(jié)java創(chuàng)建文件夾的4種方法及其優(yōu)缺點(diǎn) 總結(jié)java中刪除文件或文件夾的7種方法 總結(jié)java中文件拷貝剪切的5種方式

比如之前我已經(jīng)寫(xiě)了上面的這些內(nèi)容,如果對(duì)java基礎(chǔ)知識(shí)總結(jié)系列感興趣的同學(xué)可以關(guān)注我的博客(文末給出我的博客地址)。

一、本文梗概

這一篇文章我想寫(xiě)一下List集合元素去重的8種方法,實(shí)際上通過(guò)靈活的運(yùn)用、排列組合不一定是8種,可能有18種方法。

對(duì)象元素整體去重的4種方法 按照對(duì)象屬性去重的4種方法

為了在下文中進(jìn)行測(cè)試內(nèi)容講解,我們先做一些初始化數(shù)據(jù)

public class ListRmDuplicate { private List<String> list; private List<Player> playerList; @BeforeEach public void setup() { list = new ArrayList<>(); list.add('kobe'); list.add('james'); list.add('curry'); list.add('zimug'); list.add('zimug'); playerList= new ArrayList<>(); playerList.add(new Player('kobe','10000')); //科比萬(wàn)歲 playerList.add(new Player('james','32')); playerList.add(new Player('curry','30')); playerList.add(new Player('zimug','27')); // 注意這里名字重復(fù) playerList.add(new Player('zimug','18')); //注意這里名字和年齡重復(fù) playerList.add(new Player('zimug','18')); //注意這里名字和年齡重復(fù) }}

Player對(duì)象就是一個(gè)普通的java對(duì)象,有兩個(gè)成員變量name與age,實(shí)現(xiàn)了帶參數(shù)構(gòu)造函數(shù)、toString、equals和hashCode方法、以及GET/SET方法。

二、集合元素整體去重

下文中四種方法對(duì)List中的String類(lèi)型以集合元素對(duì)象為單位整體去重。如果你的List放入的是Object對(duì)象,需要你去實(shí)現(xiàn)對(duì)象的equals和hashCode方法,去重的代碼實(shí)現(xiàn)方法和List<String>去重是一樣的。

第一種方法

是大家最容易想到的,先把List數(shù)據(jù)放入Set,因?yàn)镾et數(shù)據(jù)結(jié)構(gòu)本身具有去重的功能,所以再將SET轉(zhuǎn)為L(zhǎng)ist之后就是去重之后的結(jié)果。這種方法在去重之后會(huì)改變?cè)械腖ist元素順序,因?yàn)镠ashSet本身是無(wú)序的,而TreeSet排序也不是List種元素的原有順序。

@Testvoid testRemove1() { /*Set<String> set = new HashSet<>(list); List<String> newList = new ArrayList<>(set);*/ //去重并排序的方法(如果是字符串,按字母表排序。如果是對(duì)象,按Comparable接口實(shí)現(xiàn)排序) //List<String> newList = new ArrayList<>(new TreeSet<>(list)); //簡(jiǎn)寫(xiě)的方法 List<String> newList = new ArrayList<>(new HashSet<>(list)); System.out.println( '去重后的集合: ' + newList);}

控制臺(tái)打印結(jié)果如下:

去重后的集合: [kobe, james, zimug, curry]

第二種方法

使用就比較簡(jiǎn)單,先用stream方法將集合轉(zhuǎn)換成流,然后distinct去重,最后在將Stream流collect收集為L(zhǎng)ist。

@Testvoid testRemove2() { List<String> newList = list.stream().distinct().collect(Collectors.toList()); System.out.println( '去重后的集合: ' + newList);}

控制臺(tái)打印結(jié)果如下:

去重后的集合: [kobe, james, curry, zimug]

第三種方法

這種方法利用了set.add(T),如果T元素已經(jīng)存在集合中,就返回false。利用這個(gè)方法進(jìn)行是否重復(fù)的數(shù)據(jù)判斷,如果不重復(fù)就放入一個(gè)新的newList中,這個(gè)newList就是最終的去重結(jié)果

//三個(gè)集合類(lèi)list、newList、set,能夠保證順序@Testvoid testRemove3() { Set<String> set = new HashSet<>(); List<String> newList = new ArrayList<>(); for (String str :list) { if(set.add(str)){ //重復(fù)的話返回false newList.add(str); } } System.out.println( '去重后的集合: ' + newList);}

控制臺(tái)打印結(jié)果和第二種方法一致。

第四種方法

這種方法已經(jīng)脫離了使用Set集合進(jìn)行去重的思維,而是使用newList.contains(T)方法,在向新的List添加數(shù)據(jù)的時(shí)候判斷這個(gè)數(shù)據(jù)是否已經(jīng)存在,如果存在就不添加,從而達(dá)到去重的效果。

//優(yōu)化 List、newList、set,能夠保證順序@Testvoid testRemove4() { List<String> newList = new ArrayList<>(); for (String cd:list) { if(!newList.contains(cd)){ //主動(dòng)判斷是否包含重復(fù)元素 newList.add(cd); } } System.out.println( '去重后的集合: ' + newList);}

控制臺(tái)打印結(jié)果和第二種方法一致。

三、按照集合元素對(duì)象屬性去重

其實(shí)在實(shí)際的工作中,按照集合元素對(duì)象整體去重的應(yīng)用的還比較少,更多的是要求我們按照元素對(duì)象的某些屬性進(jìn)行去重。看到這里請(qǐng)大家回頭去看一下上文中,構(gòu)造的初始化數(shù)據(jù)playerList,特別注意其中的一些重復(fù)元素,以及成員變量重復(fù)。

第一種方法

為T(mén)reeSet實(shí)現(xiàn)Comparator接口,如果我們希望按照Player的name屬性進(jìn)行去重,就去在Comparator接口中比較name。下文中寫(xiě)了兩種實(shí)現(xiàn)Comparator接口方法:

lambda表達(dá)式:(o1, o2) -> o1.getName().compareTo(o2.getName()) 方法引用:Comparator.comparing(Player::getName)

@Testvoid testRemove5() { //Set<Player> playerSet = new TreeSet<>((o1, o2) -> o1.getName().compareTo(o2.getName())); Set<Player> playerSet = new TreeSet<>(Comparator.comparing(Player::getName)); playerSet.addAll(playerList); /*new ArrayList<>(playerSet).forEach(player->{ System.out.println(player.toString()); });*/ //將去重之后的結(jié)果打印出來(lái) new ArrayList<>(playerSet).forEach(System.out::println);}

輸出結(jié)果如下:三個(gè)zimug因?yàn)閚ame重復(fù),另外兩個(gè)被去重。但是因?yàn)槭褂玫搅薚reeSet,list中元素被重新排序。

Player{name=’curry’, age=’30’}Player{name=’james’, age=’32’}Player{name=’kobe’, age=’10000’}Player{name=’zimug’, age=’27’}

第二種方法

這種方法是網(wǎng)上很多的文章中用來(lái)顯示自己很牛的方法,但是在筆者看來(lái)有點(diǎn)脫了褲子放屁,多此一舉。既然大家都說(shuō)有這種方法,我不寫(xiě)好像我不牛一樣。我為什么說(shuō)這種方法是“脫了褲子放屁”?

首先用stream()把list集合轉(zhuǎn)換成流 然后用collect及toCollection把流轉(zhuǎn)換成集合 然后剩下的就和第一種方法一樣了

前兩步不是脫了褲子放屁么?看看就得了,實(shí)際應(yīng)用意義不大,但是如果是為了學(xué)習(xí)Stream流的使用方法,搞出這么一個(gè)例子還是有可取之處的。

@Testvoid testRemove6() { List<Player> newList = playerList.stream().collect(Collectors .collectingAndThen( Collectors.toCollection(() -> new TreeSet<>(Comparator.comparing(Player::getName))), ArrayList::new)); newList.forEach(System.out::println);}

控制臺(tái)打印輸出和第一種方法一樣。

第三種方法

這種方法也是筆者建議大家使用的一種方法,咋一看好像代碼量更大了,但實(shí)際上這種方法是應(yīng)用比較簡(jiǎn)單的方法。

Predicate(有人管這個(gè)叫斷言,從英文的角度作為名詞可以翻譯為謂詞,作為動(dòng)詞可以翻譯為斷言)。謂詞就是用來(lái)修飾主語(yǔ)的,比如:喜歡唱歌的小鳥(niǎo),喜歡唱歌就是謂詞,用來(lái)限定主語(yǔ)的范圍。所以我們這里是用來(lái)filter過(guò)濾的,也是用來(lái)限制主語(yǔ)范圍的,所以我認(rèn)為翻譯為謂詞更合適。隨便吧,看你怎么覺(jué)得怎么理解合理、好記,你就怎么來(lái)。

首先我們定義一個(gè)謂詞Predicate用來(lái)過(guò)濾,過(guò)濾的條件是distinctByKey。謂詞返回ture元素保留,返回false元素被過(guò)濾掉。 當(dāng)然我們的需求是過(guò)濾掉重復(fù)元素。我們?nèi)ブ剡壿嬍峭ㄟ^(guò)map的putIfAbsent實(shí)現(xiàn)的。putIfAbsent方法添加鍵值對(duì),如果map集合中沒(méi)有該key對(duì)應(yīng)的值,則直接添加,并返回null,如果已經(jīng)存在對(duì)應(yīng)的值,則依舊為原來(lái)的值。 如果putIfAbsent返回null表示添加數(shù)據(jù)成功(不重復(fù)),如果putIfAbsent返回value(value==null :false),則滿(mǎn)足了distinctByKey謂詞的條件元素被過(guò)濾掉。

這種方法雖然看上去代碼量增大了,但是distinctByKey謂詞方法只需要被定義一次,就可以無(wú)限復(fù)用。

@Testvoid testRemove7() { List<Player> newList = new ArrayList<>(); playerList.stream().filter(distinctByKey(p -> p.getName())) //filter保留true的值 .forEach(newList::add); newList.forEach(System.out::println);}static <T> Predicate<T> distinctByKey(Function<? super T, ?> keyExtractor) { Map<Object,Boolean> seen = new ConcurrentHashMap<>(); //putIfAbsent方法添加鍵值對(duì),如果map集合中沒(méi)有該key對(duì)應(yīng)的值,則直接添加,并返回null,如果已經(jīng)存在對(duì)應(yīng)的值,則依舊為原來(lái)的值。 //如果返回null表示添加數(shù)據(jù)成功(不重復(fù)),不重復(fù)(null==null :TRUE) return t -> seen.putIfAbsent(keyExtractor.apply(t), Boolean.TRUE) == null;}

輸出結(jié)果如下:三個(gè)zimug因?yàn)閚ame重復(fù),另外兩個(gè)被去重。并且沒(méi)有打亂List的原始順序

Player{name=’kobe’, age=’10000’}Player{name=’james’, age=’32’}Player{name=’curry’, age=’30’}Player{name=’zimug’, age=’27’}

第四種方法

第四種方法實(shí)際上不是新方法,上面的例子都是按某一個(gè)對(duì)象屬性進(jìn)行去重,如果我們想按照某幾個(gè)元素進(jìn)行去重,就需要對(duì)上面的三種方法進(jìn)行改造。我只改造其中一個(gè),另外幾個(gè)改造的原理是一樣的,就是把多個(gè)比較屬性加起來(lái),作為一個(gè)String屬性進(jìn)行比較。

@Testvoid testRemove8() { Set<Player> playerSet = new TreeSet<>(Comparator.comparing(o -> (o.getName() + '' + o.getAge()))); playerSet.addAll(playerList); new ArrayList<>(playerSet).forEach(System.out::println);}

總結(jié)

到此這篇關(guān)于Java中List集合對(duì)象去重及按屬性去重的8種方法的文章就介紹到這了,更多相關(guān)java list集合去重內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 一级 黄 色 片69 | 爱爱视频网址 | 国产免费一区二区三区免费视频 | 国产超碰在线观看 | 都市激情视频 | 性久久久久久 | 97超碰超碰 | 激情文学综合网 | 色先锋影音资源 | 久久午夜免费视频 | 午夜资源| a级一级黄色片 | 中文字幕日韩一区二区三区 | 极品颜值美女露脸啪啪 | 久久婷婷六月 | 在线精品亚洲欧美日韩国产 | 成人免费精品动漫网站 | 九月色婷婷| 一级黄色淫片 | 美女久久久久久久 | 亚瑟av在线 | 亚洲成人第一网站 | 日韩精品在线观看视频 | 天天爆操 | 日韩精品伦理 | a√天堂网 | 欧洲a级片| 国产一区视频在线 | 91网站免费在线观看 | 久久精品在线观看视频 | a一级黄色片 | 日韩精品在线视频观看 | 欧美日韩a | 久久官网| 成人动漫视频在线观看 | 日韩成人一区二区 | 婷婷av网| 午夜视频精品 | 日本五十路女优 | 亚洲天堂精品在线 | 欧美激情16p|