当前位置: 首页 > news >正文

Java学习----Set接口

今日学习内容总结如下:

HashSet类

HashSet类直接实现了Set接口, 其底层其实是包装了一个HashMap去实现的
以需要存储的数据作为map的key值,以常量PRESENT作为value值

private transient HashMap<E, Object> map;
private static final Object PRESENT=new Object();

 HashSet采用HashCode算法来存取集合中的元素,因此具有比较好的读取和查找性能

Set<Person> set=new HashSet<>();
Person p1=new Person();
Person p2=new Person();
set.add(p1);
set.add(p2);
System.out.println(set.size());
//这里显示值为2,因为 Person类中的hashCode方法和equals方法都来自于Object类
Set<Person> set=new HashSet<>();
Person p1=new Person(1L,"yan");
Person p2=new Person(1L,"yan");
set.add(p1);
set.add(p2);
System.out.println(p1.equals(p2));
System.out.println(set.size()); //返回值仍旧为2

在Person类中添加方法

public boolean equals(Object obj){
    if(obj!=null && obj instanceof Person){
        Person p=(Person)obj;
        //具体的比较内容取决于业务规则,这里不进行判空了(偷懒)
        return this.id==p.id && this.name.equals(p.name);
    }
    return false;
}
//再次运行程序输出值还是2

问题在于hashcode值

public int hashCode(){
    return this.id.hashCode();
}

原因在于:向HashSet中添加元素时首先执行的是对象的hashcode值比较,如果两个
对象的hashcode值相等时才会继续调用equals方法;如果两个对象的hashcode值不
相等则不会调用equals方法

潜规则:不是Java的语法强制要求
    要求当两个对象的equals为true时,hashCode值必须相等

向set中添加元素到底比较是采用==还是equals?

Set<String> set=new HashSet<>();
String s1="abc";
String s2=new String("abc");
System.out.println(s1==s2);
set.add(s1);
set.add(s2);
System.out.println(set.size());  //返回为1

HashSet实际上是通过使用HashMap的key实现的,所有key对应的value都是一个常量

散列算法

散列法Hashing是一种将字符组成的字符串转换为固定长度(一般是更短长度)的数值或索引值的方法,称为散列法,也叫哈希法。

由于通过更短的哈希值比用原始值进行数据库搜索更快,这种方法一般用来在数据库中建立索引并进行搜索,同时还用在各种解密算法中

当然在存储时需要解决哈希碰撞问题

通常处理碰撞的方法有开放寻址Open Addressing法和链地址法。

String类型中的hashCode方法的实现:

public int hashCode() {
        int h = hash;
        if (h == 0 && value.length > 0) {
            hash = h = isLatin1() ? StringLatin1.hashCode(value)
                                  : StringUTF16.hashCode(value);
        }
        return h;
    }

由于自定义类都会直接或者间接的继承于java.lang.Object,所以所有的类中都有hashCode方法

public native int hashCode();

HashSet的特征

无序:不仅不能保证元素插入的顺序(如果需要顺序则可以使用LinkedHashSet)
    ,而且在元素在以后的顺序中也可能变化(这是由HashSet按HashCode存储对象
    (元素)决定的,对象变化则可能导致HashCode变化)

    如果需要访问的顺序和插入的顺序一致,可以使用HashSet的子类LinkedHashSet

不允许重复 [equals和hashcode]

HashSet是线程非安全的,方法上没有同步约束

如何判断两个对象相等

实现Set接口的HashSet,依靠HashMap来实现的。

    我们应该为要存放到散列表的各个对象定义hashCode()和equals()

HashSet的equals和hashCode

前面说过,Set集合是不允许重复元素的,否则将会引发各种奇怪的问题。
    
    那么HashSet如何判断元素重复呢?
    HashSet需要同时通过equals和HashCode来判断两个元素是否相等,具体规则
    是,如果两个元素通过equals为true,并且两个元素的hashCode相等,则这两
    个元素相等(即重复)。
    所以如果要重写保存在HashSet中的对象的equals方法,也要重写hashCode方
    法,重写前后hashCode返回的结果相等(即保证保存在同一个位置)。所有参
    与计算 hashCode() 返回值的关键属性,都应该用于作为 equals() 比较的
    标准。

结论:要求当两个同类型对象equals为true时,必须hashCode值一致。事实上
    equals方法和hashCode方法没有任何必然联系,所以说前面的规则是个潜规则

LinkedHashSet

public class LinkedHashSet<E> 
		extends HashSet<E> 
		implements Set<E>, Cloneable, Serializable

LinkedHashSet是HashSet的一个子类,LinkedHashSet也根据HashCode的值来决
    定元素的存储位置,但同时它还用一个链表来维护元素的插入顺序,插入的时候即
    要计算hashCode又要维护链表,而遍历的时候只需要按链表来访问元素

采用双向链表记录添加元素的顺序

LinkedHashMap类中的节点定义

static class Entry<K,V> extends HashMap.Node<K,V> {
        		Entry<K,V> before, after;
		        Entry(int hash, K key, V value, Node<K,V> next) {
		            super(hash, key, value, next);
		        }
    		}
Set<String> set=new LinkedHashSet<>();
for(int i=0;i<10;i++)
	set.add("str-"+i);
Iterator<String> it=set.iterator();
while(it.hasNext()){
    String tmp=it.next();
    System.out.println(tmp);
}

因为底层采用链表和哈希表的算法。链表保证元素的添加顺序,哈希表保证元素的唯一性

TreeSet类

TreeSet实现了SortedSet接口,顾名思义这是一种排序的Set集合

public class TreeSet<E> 
 			extends AbstractSet<E> 
 			implements NavigableSet<E>, Cloneable, Serializable

数据存储采用的是

private transient NavigableMap<E,Object> m;

public interface NavigableMap<K,V> extends SortedMap<K,V>

查看jdk源码发现底层是用TreeMap实现的,本质上是一个红黑树原理。正因为它是排序了的,所以相对HashSet来说,TreeSet提供了一些额外的按排序位置访问元素的方法,例如first(), last(), lower(), higher(), subSet(),headSet(), tailSet()

Random r=new Random();
Set<Integer> lhs=new LinkedHashSet<>(); //用于记录生成的对技术的顺序
Set<Integer> set=new TreeSet<>();
while(set.size()<10){
    int k=r.nextInt(100);
    lhs.add(k);
    set.add(k);
}
System.out.println("生成的数据为:")
System.out.println(lhs);
System.out.println("插入TreeSet后的数据为:");
System.out.println(set);

TreeSet的排序分两种类型,一种是自然排序,另一种是定制排序。

对象大小的比较

Java中为了实现对象的比较引入了compare算法

这里用于实现了两个日期的比较大小,如果返回为0则表示两个对象相等,如果返回为1则表示d1大于参数;如果返回-1表示d1小于参数

实现Comparable接口,在类定义中添加比较规则

public class Person implements Comparable<Person>{
    private Long id;
    private String name;
    public int compareTo(Person person){
        if(name!=null && name.trim().length()>0)
            return this.name.compareTo(person.name);
        return 0;
    }
}

 

Set<Person> set=new TreeSet<>();
for(int i=0;i<10;i++)
    set.add(new Person(1L+i, (9-i)+"name"));
set.forEach(System.out::println);

自然排序(在元素中写排序规则)

TreeSet 会调用compareTo方法比较元素大小,然后按升序排序(从小到达)。所以自然排序中的元素对象,都必须实现了Comparable接口,否则会跑出异常。对于TreeSet判断元素是否重复的标准,也是调用元素从Comparable接口继承而来compareTo方法,如果返回0则是重复元素。Java的常见类都已经实现了Comparable接口

给Person类上添加针对Comparable接口的实现:考虑具体的业务规则,按照类中什么属性进行排序比较

定制排序(在集合中写排序规则)

TreeSet还有一种排序就是定制排序,定制排序时候,需要关联一个 Comparator对象,由Comparator提供排序逻辑

构造器

 public TreeSet(Comparator<? super E> comparator) {
        this(new TreeMap<>(comparator));
    }
Set<Person> set=new TreeSet<>((obj1, obj2)->{
    Person p1=(Person)obj1;
    Person p2=(Person)obj2;
    return p2.getId().compareTo(p1.getId());
});
for(int i=0;i<10;i++)
	set.add(new Person(1L+i, (9-i)+"name"));
set.forEach(System.out::println);

相关文章:

  • To enable Secure Boots and Flash Encryption using the ESP Flash download tool
  • FastFlow(2)---任务调度Task Schedule
  • 根据当前日期获取前一天日期-小工具
  • 金仓数据库KingbaseES客户端应用参考手册--12. sys_dumpall
  • 最详细说明spring cloud和Spring Cloud Alibaba的联系和区别
  • 【0基础学算法】二分查找 (超详细讲解+私人笔记+源码)
  • 计算机网络作业(存储单位k、KB、MB、GB、TB、PB;手机运行内存和内存的区别)
  • 正则表达式 (Regex) 2022教程
  • 第五章Redis 的发布和订阅
  • Dueling Network Architectures for Deep Reinforcement Learning(Dueling-DQN)
  • Vue 3.2 + TypeScript + Pinia + Vite2 + Element-Plus 管理系统(开源啦 )
  • vue工程化vue-cli创建项目以及图形创建vue项目
  • 浏览器http提交protobuf二进制数据正常,微信小程序失败解决方案
  • 实现 QQuickImageProvider 的若干问题的思路
  • 算法——查找
  • JavaScript 如何正确处理 Unicode 编码问题!
  • 2017前端实习生面试总结
  • 78. Subsets
  • JS笔记四:作用域、变量(函数)提升
  • leetcode讲解--894. All Possible Full Binary Trees
  • Linux后台研发超实用命令总结
  • PAT A1050
  • PyCharm搭建GO开发环境(GO语言学习第1课)
  • Service Worker
  • tweak 支持第三方库
  • Vue全家桶实现一个Web App
  • Vue实战(四)登录/注册页的实现
  • 干货 | 以太坊Mist负责人教你建立无服务器应用
  • 技术:超级实用的电脑小技巧
  • 猫头鹰的深夜翻译:JDK9 NotNullOrElse方法
  • 浅谈Golang中select的用法
  • 视频flv转mp4最快的几种方法(就是不用格式工厂)
  • 说说动画卡顿的解决方案
  • ​业务双活的数据切换思路设计(下)
  • !! 2.对十份论文和报告中的关于OpenCV和Android NDK开发的总结
  • ![CDATA[ ]] 是什么东东
  • # Swust 12th acm 邀请赛# [ K ] 三角形判定 [题解]
  • #define、const、typedef的差别
  • #我与Java虚拟机的故事#连载13:有这本书就够了
  • (2020)Java后端开发----(面试题和笔试题)
  • (M)unity2D敌人的创建、人物属性设置,遇敌掉血
  • (附源码)SSM环卫人员管理平台 计算机毕设36412
  • (附源码)小程序儿童艺术培训机构教育管理小程序 毕业设计 201740
  • (紀錄)[ASP.NET MVC][jQuery]-2 純手工打造屬於自己的 jQuery GridView (含完整程式碼下載)...
  • (一)Java算法:二分查找
  • (转)Sublime Text3配置Lua运行环境
  • (转)母版页和相对路径
  • .MSSQLSERVER 导入导出 命令集--堪称经典,值得借鉴!
  • .net 4.0发布后不能正常显示图片问题
  • .net core开源商城系统源码,支持可视化布局小程序
  • .NET Micro Framework 4.2 beta 源码探析
  • .NET Windows:删除文件夹后立即判断,有可能依然存在
  • .NET/ASP.NETMVC 深入剖析 Model元数据、HtmlHelper、自定义模板、模板的装饰者模式(二)...
  • .NET中统一的存储过程调用方法(收藏)
  • .Net组件程序设计之线程、并发管理(一)