7月24号_set集合

Set接口

  • set一个最大的特点就是无序,无重复

今天主要内容

  • HashSet

    • HashSet的工作原理存储自定义对象保证元素唯一性的原理
  • LinkedHashSet

  • 两个练习

    • 使用Scanner类从键盘读取一输入,去掉其中重复字符,打印出不同字符

    • 将集合中的重复元素去掉

  • TreeSet

    • TreeSet的应用举例

HashSet

  • HashSet存储自定义对象时保证元素唯一性

    • 重写hashCode()方法和equals()方法

    • 此时理解hashCode实际上相当于身份证,在内存中每一个对象都有一个属于自己的hashCode

    • 单纯修改equals方法不生效,只有在重写了hashCode方法才可以

      • 因为只有在hashCode一致时,才去调用equals方法
  • 程序

  • Employee类

      public class Employee{
    
          //data field
          private String name;
          private double salary;
          private LocalDate hireDay;
          
          //constructor
          public Employee(String name,double salary,int year,int month,int day){
              this.name = name;
              this.salary = salary;
              this.hireDay = LocalDate.of(year,month,day);
          }
          
          //method
          public String getName(){
              return name;    
          }
          
          public double getSalary(){
              return salary;  
          }
          
          public LocalDate gethireDay(){
              return hireDay;
          }
          
          public void raiseSalary(double byPercent){
              salary += salary * byPercent/100;
          }
      
          public String toString() {
              return "name : " + name + " salary : " + salary + " hireDay : " + hireDay;      
          }
          
          @Override
          public int hashCode() { 
              /*
              hashCode这样编码就是为了能够少调用equals方法
              减少hashCode相同的可能,因为hashCode相同时就会调用equal方法
              为什么选择31?
                  1. 31是一个质数,只能被1和自己本身整除
                  2. 31记不大又不小
                  3. 31是2的5次方,2左移5位
              */
              
              final int prime = 31;
              int result = 1;
              result = prime * result + ((hireDay == null) ? 0 : hireDay.hashCode());
              result = prime * result + ((name == null) ? 0 : name.hashCode());
              long temp;
              temp = Double.doubleToLongBits(salary);
              result = prime * result + (int) (temp ^ (temp >>> 32));
              return result;
          }
      
          @Override
          public boolean equals(Object obj) {
              if (this == obj)
                  return true;
              if (obj == null)
                  return false;
              if (getClass() != obj.getClass())
                  return false;
              Employee other = (Employee) obj;
              /*
              if (hireDay == null) {
                  if (other.hireDay != null)
                      return false;
              } else if (!hireDay.equals(other.hireDay))
                  return false;
              if (name == null) {
                  if (other.name != null)
                      return false;
              } else if (!name.equals(other.name))
                  return false;
              if (Double.doubleToLongBits(salary) != Double.doubleToLongBits(other.salary))
                  return false;
              
              return true;
              */
              
              return (this.name.equals(other.name) && (this.salary == other.salary && this.hireDay.equals(other.hireDay)));
          }   
      }
    

HashSet的工作原理存储自定义对象保证元素唯一性的原理

  1. HashSet原理

    • 我们使用Set集合都是需要去掉重复元素的,在存储时逐个调用equals/方法进行比较效率极低,因此,利用利用哈希算法提高去重复的效率,降低使用equals的次数

    • 当HashSet调用add方法存储对象时,先调用对象的hashCode方法得到哈希值,然后在集合中查找是否有哈希值一样的对象

      • 若没有,则直接存入集合
      • 若有,则和哈希值相同的对象逐个进行equals比较,比较结果为false时存入,true时不存
  2. 将自定义的类对象存入HashSet中去重复复

    • 类必须重写equals方法和hashCode方法
    • hashCode():属性相同得对象返回值必须相同,属性不同的返回值尽量不同,提高效率
    • equals方法,属性相同则返回true,属性不同返回flase,返回flase时存储

LinkedHashSet的概述

  • 底层用链表实现,是set集合中唯一一个能保证怎么存就怎么取的集合对象

  • HashSet的子类,因此也保证元素的唯一性,与HashSet原理一致

  • 编写程序,获取10个1-20的随机数,要求随机数不可重复,并把最终结果输出

    • 分析:
        1. Random类创建随机数对象
        1. 需要存储10个不重复随机数,选择HashSet
        1. 如果HashSet的size是小于10就可以不断存储,大于10就停止存储
      • 通过Random类中nextInt(n)方法获取1-20之间的随机数,并将随机数存储到 HashSet中
      • 遍历HashSet
  • Code

      public static void main(String[] args) {
          //1. Random类创建随机数对象
          Random random = new Random();
          
          //定义HashSet
          HashSet<Integer> hSet = new HashSet<>();
          
          //存储随机数
          while(hSet.size() < 10) {
              //创建随机数
              hSet.add(random.nextInt(20) + 1);        //产生1-20之间的随机数
          }
          
          System.out.println(hSet);           
      }
    
      /*
       * 在JDK1.8中输出结果为:
       * -----------------------------------
       * [1, 2, 3, 19, 8, 9, 11, 12, 13, 15]
       * -----------------------------------
       * */
    

集合框架练习一

  • 需求: 使用Scanner类从键盘读取一输入,去掉其中重复字符,打印出不同字符

  • 程序

      public class TestSet_1 {
          public static void main(String[] args) {
              //从键盘获取一行
              Scanner input = new Scanner(System.in);
              
              System.out.print("请输入一行:");
              String s = input.nextLine();
              
              //创建一个HashSet存放字符
              HashSet<Character> hSet = new HashSet<>();
              
              /*
              int i = 0;
              while(i < s.length()) {
                  hSet.add(s.charAt(i));
                  i++;
              }*/
              
              char[] arr = s.toCharArray();
              
              for(char c : arr) {
                  hSet.add(c);
              }
              
              System.out.println(hSet);       
          }
      }
      /*
       *  在JDK1.8中输出结果为:
      -------------------------
       *   请输入一行:aabbccddeeff
        [a, b, c, d, e, f]
       * */
    

集合框架练习二

  • 需求:将集合中的重复元素去掉

  • 程序

      public class TestSet_2 {
          public static void main(String[] args) {
              List<String> list = new ArrayList<String>();
              
              list.add("a");
              list.add("b");
              list.add("b");
              list.add("a");
              list.add("a");
              list.add("b");
              list.add("c");
              
              System.out.println("list中元素: " + list);
              
              fun(list);
              
              System.out.println("去重后list中元素: " + list);
          }
      
          private static void fun(List<String> list) {
              
              //创建一个Set集合
              HashSet<String> hSet = new HashSet<>();
              
              //将list中元素存入hSet中
              hSet.addAll(list);
              
              //清除list中的数据
              list.clear();
              
              //将hSet重新赋值回list
              list.addAll(hSet);      
          }
      }
      /*
       *   在JDK1.8中输出结果为:
       *   ----------------------------------
       *  list中元素: [a, b, b, a, a, b, c]
           去重后list中元素: [a, b, c]
         ------------------------------------
       * */
    

TreeSet

1. TreeSet特点

  • TreeSet集合是用来对元素进行排序的,同样也可以保证元素的唯一性

  • TreeSet可指定一个顺序,对象存入后会按照指定顺序排列

2. TreeSet使用方式

A. 自然排序(Comparable)

* TreeSet类中add()方法把存入的对象提升为Comparable类型

* 调用对象的compareTo方法和集合中元素进行比较

* 根据compareTo方法返回的结果进行存储

B. 比较器顺序(Comparator)

* 创建TreeSet的时候可创建一个Comparator

* 如果传入了Comparator的子类对象,那么TreeSet就会按照比较器中顺序进行排序

* add()方法内部自动调用Comparator接口中compare方法排序

C. 两种方式区别

* TreeSet调用无参构造器时,默认按照类中Comparable的顺序

* TreeSet如果传入Comparator,优先按照Comparator

3. TreeSet存储自定义对象

  • TreeSet中的对象元素必须实现Comparable接口

    • 也就是重写compareTo方法必须被重写

      • 当compareTo方法返回0——集合中元素均相等,则集合中只有一个元素

      • 当compareTo方法返回正数——集合怎么存就怎么取

      • 当CompareTo方法返回负数——集合会倒序存储

  • 原理图

    image
  • 程序

  • Employee类——实现Comparable接口——重写compareTo()方法

      public class Employee implements Comparable<Employee>{
    
          //data field
          private String name;
          private int salary;
          private LocalDate hireDay;
          
          //constructor
          public Employee(String name,int salary,int year,int month,int day){
              this.name = name;
              this.salary = salary;
              this.hireDay = LocalDate.of(year,month,day);
          }
          
          //method
          public String getName(){
              return name;    
          }
          
          public double getSalary(){
              return salary;  
          }
          
          public LocalDate gethireDay(){
              return hireDay;
          }
          
          public void raiseSalary(double byPercent){
              salary += salary * byPercent/100;
          }
      
          public String toString() {
              return "name : " + name + " salary : " + salary + " hireDay : " + hireDay;      
          }
          
          @Override
          public int hashCode() { 
              /*
              hashCode这样编码就是为了能够少调用equals方法
              减少hashCode相同的可能,因为hashCode相同时就会调用equal方法
              为什么选择31?
                  1. 31是一个质数,只能被1和自己本身整除
                  2. 31记不大又不小
                  3. 31是2的5次方,2左移5位
              */
              
              final int prime = 31;
              int result = 1;
              result = prime * result + ((hireDay == null) ? 0 : hireDay.hashCode());
              result = prime * result + ((name == null) ? 0 : name.hashCode());
              long temp;
              temp = Double.doubleToLongBits(salary);
              result = prime * result + (int) (temp ^ (temp >>> 32));
              return result;
          }
      
          @Override
          public boolean equals(Object obj) {
              if (this == obj)
                  return true;
              if (obj == null)
                  return false;
              if (getClass() != obj.getClass())
                  return false;
              Employee other = (Employee) obj;
              /*
              if (hireDay == null) {
                  if (other.hireDay != null)
                      return false;
              } else if (!hireDay.equals(other.hireDay))
                  return false;
              if (name == null) {
                  if (other.name != null)
                      return false;
              } else if (!name.equals(other.name))
                  return false;
              if (Double.doubleToLongBits(salary) != Double.doubleToLongBits(other.salary))
                  return false;
              
              return true;
              */
              
              return (this.name.equals(other.name) && (this.salary == other.salary && this.hireDay.equals(other.hireDay)));
          }
      
          @Override
          public int compareTo(Employee o) {
              int num = this.salary - o.salary;
              
              return num == 0 ? this.name.compareTo(o.name) : num;
              //先按照薪水排序,再按照名字排序               
          }   
      }
    
  • TestTreeSet类

      public static void main(String[] args) {
          TreeSet<Employee> tSet = new TreeSet<>();
          
          tSet.add(new Employee("haha", 200, 2018, 7, 24));
          tSet.add(new Employee("haha", -1, 2018, 7, 24));
          tSet.add(new Employee("haha", 310, 2018, 7, 24));
          tSet.add(new Employee("haha", 10, 2018, 7, 24));
          
          for(Employee employee : tSet) {
              System.out.println(employee);
          }       
      }
    
      /*
       *   在JDK1.8中输出结果为:
       *   ----------------------------------------------
       *  name : haha salary : -1.0 hireDay : 2018-07-24
          name : haha salary : 10.0 hireDay : 2018-07-24
          name : haha salary : 200.0 hireDay : 2018-07-24
          name : haha salary : 310.0 hireDay : 2018-07-24
          ------------------------------------------------
       * */
    

4. TreeSet应用举例——将字符串按照长度排序

原理
  • 使用TreeSet构造器

    • TreeSet(Comparator<? super E> comparator)
  • 创建一个类实现Comparator接口

    • 重写compare方法
  • 将重写了compare方法的类作为参数传给TreeSet

程序
  • CompareByLength实现Comparator接口

      class CompareByLength implements Comparator<String>{
          @Override
          public int compare(String o1, String o2) {
              
              int num = o1.length() - o2.length();            //先比较长度
              
              return num == 0 ? o1.compareTo(o2) : num;       //内容为次要条件
          }
      }
    
  • 将字符串按照长度排序

      public static void main(String[] args) {
          TreeSet<String> tSet = new TreeSet<>(new CompareByLength());
          
          tSet.add("hahaha");
          tSet.add("haha");
          tSet.add("ha");
          tSet.add("h");
          
          System.out.println(tSet);
      }
      /*
       *  在JDK1.8中输出结果为:
       * ----------------------
       * [h, ha, haha, hahaha]
       * ----------------------
       * */
    
排序原理图

[图片上传失败...(image-cee7b4-1532526925688)]

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,608评论 18 399
  • List的学习 今天的主要内容 集合框架(去除ArrayList中重复字符串元素方式)(掌握) 集合框架(去除Ar...
    须臾之北阅读 210评论 0 0
  • 一. Java基础部分.................................................
    wy_sure阅读 3,809评论 0 11
  • 首先,先了解一下iOS版本号的规则 CFBundleShortVersionString(Version) Spe...
    SySean阅读 12,414评论 3 19
  • “北冥有鱼,其名为鲲。鲲之大,不知其几千里也” 古有神木者,其名大椿。椿之久也,以八千岁为春,八千岁为秋。 ...
    小鱼是不老少女阅读 275评论 0 0