2021-04-22Set集合

Set集合概述和特點(diǎn)
Set集合特點(diǎn)
  • 不包含重復(fù)元素的集合
  • 沒有帶索引的方法侥钳,所以不能使用普通for循環(huán)遍歷
    Set 集合練習(xí)
  • 存儲(chǔ)字符串并遍歷
/**
 * HashSet:對(duì)集合的迭代順序不作任何保證
 */
import java.util.HashSet;
import java.util.Set;
public class SetDemo {
    public static void main(String[] args) {
        // 創(chuàng)建集合對(duì)象 Set是接口,不能直接實(shí)例化
        Set<String> set = new HashSet<String>();
        // 添加元素
        set.add("hello");
        set.add("world");
        set.add("java");
        set.add("hello");
        //遍歷
        for (String s : set) {
            System.out.println(s);
        }
    }
}
代碼運(yùn)行截圖.png
哈希值

哈希值:是JDK根據(jù)對(duì)象的地址或者字符串或者數(shù)字算出來的int類型的數(shù)值

Object類中有一個(gè)方法可以獲取對(duì)象的哈希值

  • public int hashCode(): 返回對(duì)象的哈希碼值

對(duì)象的哈希值特點(diǎn):

  • 同一個(gè)對(duì)象多次調(diào)用hashCode() 方法返回的哈希值是相同的
  • 默認(rèn)情況下讲仰,不同對(duì)象的哈希值是不同的慕趴。而重寫hashCode()方法,可以實(shí)現(xiàn)讓不同對(duì)象的哈希值相同
HashSet集合概述和特點(diǎn)

HashSet集合特點(diǎn)

  • 底層數(shù)據(jù)結(jié)構(gòu)是哈希表
  • 對(duì)集合的迭代順序不作任何保證鄙陡,也就是說不保證存儲(chǔ)和取出的元素順序一致
  • 沒有帶索引的方法冕房,所以不能使用普通for循環(huán)遍歷
  • 由于是Set集合,所以是不包含重復(fù)元素的集合
HashSet集合保證元素唯一性源碼分析

HashSet集合添加到一個(gè)元素的過程:


添加元素過程圖.png

HashSet集合存儲(chǔ)元素:

  • 要保證元素唯一性趁矾,需要重寫hashCode()equals()
常見數(shù)據(jù)結(jié)構(gòu)之哈希表

哈希表

  • JDK8之前耙册,底層采用數(shù)組+鏈表實(shí)現(xiàn),可以說是一個(gè)元素為鏈表的數(shù)組
  • JDK8之后毫捣,在長(zhǎng)度比較長(zhǎng)的時(shí)候详拙,底層實(shí)現(xiàn)了優(yōu)化
HashSet集合存儲(chǔ)學(xué)生對(duì)象并遍歷

需求:創(chuàng)建一個(gè)存儲(chǔ)學(xué)生對(duì)象的集合帝际,存儲(chǔ)多個(gè)學(xué)生對(duì)象,使用程序?qū)崿F(xiàn)在控制臺(tái)遍歷該集合
要求:學(xué)生對(duì)象的成員變量值相同饶辙,我們認(rèn)為是同一個(gè)對(duì)象
思路:

  1. 定義學(xué)生類
  2. 創(chuàng)建HashSet對(duì)象
  3. 創(chuàng)建學(xué)生對(duì)象
  4. 把學(xué)生添加到集合
  5. 循環(huán)遍歷(增強(qiáng)for循環(huán))
  6. 在學(xué)生類中重寫兩個(gè)方法(此步最為關(guān)鍵蹲诀,如果缺少此步操作,將會(huì)錄入成員變量值相同的成員對(duì)象弃揽。重寫方法按下Alt+Insert快捷鍵即可脯爪,找到hashCode()和equals()如下自動(dòng)生成即可)
    截圖.png
LinkedHashSet集合概述和特點(diǎn)
LinkedHashSet集合特點(diǎn)
  • 哈希表和鏈表實(shí)現(xiàn)的Set接口,具有可預(yù)測(cè)的迭代次序
  • 由鏈表保證元素有序矿微,也就是說元素的存儲(chǔ)和取出順序是一致的(非元素大小順序)
  • 由哈希表表示元素唯一痕慢,也就是說沒有重復(fù)的元素
    LinkedHashSet集合練習(xí)
public class LinkedHashSetDemo {
    public static void main(String[] args) {
        // 創(chuàng)建對(duì)象
        LinkedHashSet<String> lhs = new LinkedHashSet<String>();
        // 添加對(duì)象
        lhs.add("hello");
        lhs.add("world");
        lhs.add("java");
        lhs.add("java");
        for (String s : lhs) {
            System.out.println(s);
        }
        System.out.println(lhs);
    }
}
代碼運(yùn)行截圖.png
TreeSet集合概述和特點(diǎn)

TreeSet集合特點(diǎn)

  • 元素有序,這里的順序不是指存儲(chǔ)和取出的順序涌矢,而是按照一定的規(guī)則進(jìn)行排序掖举,具體的排序方式取決于構(gòu)造方法
    TreeSet():根據(jù)其元素的自然排序進(jìn)行排序
    TreeSet(Comparator comparator):根據(jù)指定的比較器進(jìn)行排序
  • 沒有帶索引的方法,所以不能使用普通for循環(huán)遍歷[可以使用迭代器和增強(qiáng)for進(jìn)行循環(huán)遍歷]
  • 由于是Set集合娜庇,所以不包含重復(fù)元素的集合
    TreeSet集合練習(xí)
public class TreeDemo {
    public static void main(String[] args) {
        TreeSet<Integer> ts = new TreeSet<Integer>();  // 基本類型進(jìn)行存儲(chǔ)的時(shí)候塔次,
        // 里面用的是對(duì)應(yīng)的包裝類類型
        ts.add(7);
        ts.add(39);
        ts.add(28);
        ts.add(20);
        ts.add(40);
        ts.add(20);  // 不包含重復(fù)元素

        for (Integer i : ts) {
            System.out.println(i);
        }
    }
}
代碼運(yùn)行結(jié)果.png
自然排序Comparable的使用
  • 存儲(chǔ)學(xué)生對(duì)象并遍歷,創(chuàng)建TreeSet集合使用無參構(gòu)造方法
  • 要求:按照年齡從小到大排序思灌,年齡相同時(shí)俺叭,按照姓名的字母順序排序
    為滿足條件:要讓學(xué)生類實(shí)現(xiàn)Comparable<Student>接口
    而且需要進(jìn)行重寫操作
   @Override
    public int compareTo(Student s) {
//        return 0;  // 不存儲(chǔ)的元素
//        return 1;   // 升序。按添加順序存儲(chǔ)
//        return -1;   // 降序
//        int num = this.age - s.age;  // 按年齡大小升序:this是s2,s是s1.
        int num = s.age - this.age;   // 按年齡大小升序降序
        int num2 = num == 0?this.name.compareTo(s.name):num;
        return num2;           // 使?jié)M足年齡相同姓名不同的學(xué)生也可以添加成功
    }
public class TreeSetDemo02 {
    public static void main(String[] args) {
        // 創(chuàng)建集合對(duì)象
        TreeSet<Student> ts = new TreeSet<Student>();
        Student s1 = new Student("xishi",29);
        Student s2 = new Student("wangzhojun",33);
        Student s3 = new Student("diaochao",30);
        Student s4 = new Student("yangyuhuan",28);
        Student s5 = new Student("linqingxie",28);
        Student s6 = new Student("linqingxie",28);
        // 把學(xué)生添加到集合
        ts.add(s1);
        ts.add(s2);
        ts.add(s3);
        ts.add(s4);
        ts.add(s5);
        ts.add(s6);
        // 遍歷集合
        for(Student s:ts) {
            System.out.println(s.getName()+","+s.getAge());
        }
    }
}
代碼運(yùn)行截圖.png

結(jié)論

  • 用TreeSet集合存儲(chǔ)自定義對(duì)象泰偿,無參構(gòu)造方法使用的是自然排序對(duì)元素進(jìn)行排序的
  • 自然排序熄守,就是讓元素所屬的類實(shí)現(xiàn)Comparable接口,重寫compareTo(To)方法
  • 重寫方法時(shí)耗跛,一定要注意排序規(guī)則必須按照要求的主要條件和次要條件來寫
比較器排序Comparator的使用
  • 存儲(chǔ)學(xué)生對(duì)象并遍歷裕照。創(chuàng)建 TreeSet集合使用帶參構(gòu)造
  • 要求:按照年齡從小到大排序,年齡相同時(shí)调塌,按照姓名的字母順序排序
    結(jié)論:
  • 用TreeSet集合存儲(chǔ)自定義對(duì)象晋南,帶參構(gòu)造方法接收的是比較器排序對(duì)元素進(jìn)行排序的
  • 比較器排序,就是讓集合構(gòu)造方法接收Comparator的實(shí)現(xiàn)類對(duì)象羔砾,重寫compare(To1,To2)方法
  • 重寫方法時(shí)负间,一定要注意排序規(guī)則必須按照要求的主要條件和次要條件來寫
public class TreeSetDemo02 {
    public static void main(String[] args) {
        // 創(chuàng)建集合對(duì)象
        TreeSet<Student> ts = new TreeSet<Student>(new Comparator<Student>() {
            @Override
            public int compare(Student s1, Student s2) {
                // this.age -  s.age
                // s1,s2分別指的時(shí)this,s
                int num = s1.getAge() - s2.getAge();
                int num2 = num == 0 ? s1.getName().compareTo(s2.getName()) : num;
                return num2;
            }
        });
        Student s1 = new Student("xishi", 29);
        Student s2 = new Student("yangyuhuan", 28);
        Student s3 = new Student("linqingxie", 27);
        Student s4 = new Student("linqingxie", 28);
        // 把學(xué)生添加到集合
        ts.add(s1);
        ts.add(s2);
        ts.add(s3);
        ts.add(s4);
        // 遍歷集合
        for (Student s : ts) {
            System.out.println(s.getName() + "," + s.getAge());
        }
    }
}

案例:

public class TreeSetDemo01 {
    public static void main(String[] args) {
        TreeSet<Student> ts = new TreeSet<Student>(new Comparator<Student>() {
            @Override
            public int compare(Student s1, Student s2) {
                int i = s1.getSum() - s2.getSum();
                int i2 = i == 0 ? s1.getChinese() - s2.getChinese() : i;
                int i3 = i2 == 0 ? s1.getName().compareTo(s2.getName()) : i2;
                return i3;
            }
        });
        Student s1 = new Student("風(fēng)清揚(yáng)", 99, 87);
        Student s2 = new Student("令狐沖", 94, 95);
        Student s3 = new Student("張曼玉", 92, 91);
        Student s4 = new Student("劉巖", 90, 93);
        Student s5 = new Student("劉一備", 90, 93);

        ts.add(s1);
        ts.add(s2);
        ts.add(s3);
        ts.add(s4);
        ts.add(s5);

        for (Student s : ts) {
            System.out.println(s.getName() + "," + s.getChinese() + "," + s.getMath() + " " + s.getSum());
        }
    }
}
代碼運(yùn)行截圖.png

案例2:不重復(fù)的隨機(jī)數(shù)
需求:編寫一個(gè)程序姜凄,獲取10個(gè)1-20之間的隨機(jī)數(shù)政溃,要求隨機(jī)數(shù)不能重復(fù),并在控制臺(tái)輸出
思路:

  1. 創(chuàng)建Set集合對(duì)象
  2. 創(chuàng)建隨機(jī)數(shù)對(duì)象
  3. 判斷集合的長(zhǎng)度是不是小于10
    是:產(chǎn)生一個(gè)隨機(jī)數(shù)态秧,添加到集合
    4.遍歷集合
public class SetDemo {
    public static void main(String[] args) {
        // 創(chuàng)建集合對(duì)象 Set是接口董虱,不能直接實(shí)例化
//        Set<Integer> set = new HashSet<Integer>(); //  無序排列
        TreeSet<Integer> set = new TreeSet<Integer>();   // 元素大小有序排列輸出
//        LinkedHashSet<Integer> set = new LinkedHashSet<Integer>(); // 元素存儲(chǔ)和取出有序
        // 創(chuàng)建隨機(jī)數(shù)對(duì)象
        Random r = new Random();
        // 判斷集合的長(zhǎng)度是不是小于10
        while (set.size() < 10) {  // 從0 開始比較的
            // 產(chǎn)生一個(gè)隨機(jī)數(shù),添加到集合
            int number = r.nextInt(20) + 1;
            set.add(number);
        }
        // 遍歷集合
        for (Integer i : set) {
            System.out.println(i);
        }
    }
}
代碼運(yùn)行截圖.png
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市愤诱,隨后出現(xiàn)的幾起案子云头,更是在濱河造成了極大的恐慌,老刑警劉巖淫半,帶你破解...
    沈念sama閱讀 212,816評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件溃槐,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡撮慨,警方通過查閱死者的電腦和手機(jī)竿痰,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,729評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來砌溺,“玉大人,你說我怎么就攤上這事变隔」娣ィ” “怎么了?”我有些...
    開封第一講書人閱讀 158,300評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵匣缘,是天一觀的道長(zhǎng)猖闪。 經(jīng)常有香客問我,道長(zhǎng)肌厨,這世上最難降的妖魔是什么培慌? 我笑而不...
    開封第一講書人閱讀 56,780評(píng)論 1 285
  • 正文 為了忘掉前任,我火速辦了婚禮柑爸,結(jié)果婚禮上吵护,老公的妹妹穿的比我還像新娘。我一直安慰自己表鳍,他們只是感情好馅而,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,890評(píng)論 6 385
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著譬圣,像睡著了一般瓮恭。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上厘熟,一...
    開封第一講書人閱讀 50,084評(píng)論 1 291
  • 那天屯蹦,我揣著相機(jī)與錄音,去河邊找鬼绳姨。 笑死登澜,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的就缆。 我是一名探鬼主播帖渠,決...
    沈念sama閱讀 39,151評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼竭宰!你這毒婦竟也來了空郊?” 一聲冷哼從身側(cè)響起份招,我...
    開封第一講書人閱讀 37,912評(píng)論 0 268
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎狞甚,沒想到半個(gè)月后锁摔,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,355評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡哼审,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,666評(píng)論 2 327
  • 正文 我和宋清朗相戀三年谐腰,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片涩盾。...
    茶點(diǎn)故事閱讀 38,809評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡十气,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出春霍,到底是詐尸還是另有隱情砸西,我是刑警寧澤,帶...
    沈念sama閱讀 34,504評(píng)論 4 334
  • 正文 年R本政府宣布址儒,位于F島的核電站芹枷,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏莲趣。R本人自食惡果不足惜鸳慈,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,150評(píng)論 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望喧伞。 院中可真熱鬧走芋,春花似錦、人聲如沸絮识。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,882評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽次舌。三九已至熄攘,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間彼念,已是汗流浹背挪圾。 一陣腳步聲響...
    開封第一講書人閱讀 32,121評(píng)論 1 267
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留逐沙,地道東北人哲思。 一個(gè)月前我還...
    沈念sama閱讀 46,628評(píng)論 2 362
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像吩案,于是被迫代替她去往敵國(guó)和親棚赔。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,724評(píng)論 2 351

推薦閱讀更多精彩內(nèi)容