java中的集合主要分为三种类型
#java中的集合主要分为三种类型:
##数组:⼏乎所有集合实现的底层都有数据的⾝影存在,因此我们⾸先需要了解⼀下数组。以下这段话摘⾃《Thinking In Algorithm》,感觉很不错现在拿出来跟⼤家分享。
##《Thinking In Algorithm》之数组
##集合:接下来是集合,同样我们摘⾃⽹络上的⼀段解释,很不错也通俗易懂,与⼤家分享:
##集合与数组
##总结⼀下上⾯两段话:
##数组的⼤⼩是固定不变的,并且同⼀个数组只能存储相同类型的数据,该数据类型可以是基本类型也可以是引⽤类型。Java中集合可以存储操作不同类型和⼤⼩不固定的数据,但是Java中集合只能存储引⽤类型,不能存储基本类型。
##1、Set(集)
Set:注重独⼀⽆⼆的性质,该体系集合可以知道某物是否已近存在于集合中,不会存储重复的元素
⽤于存储⽆序(存⼊和取出的顺序不⼀定相同)元素,值不能重复。
对象的相等性
引⽤到堆上同⼀个对象的两个引⽤是相等的。如果对两个引⽤调⽤hashCode⽅法,会得到相同的结果,如果对象所属的类没有覆盖Object 的hashCode⽅法的话,hashCode会返回每个对象特有的序号(java是依据对象的内存地址计算出的此序号),所以两个不同的对象的hashCode值是不可能相等的。
如果想要让两个不同的Person对象视为相等的,就必须覆盖Object继下来的hashCode⽅法和equals⽅法,因为Object  hashCode⽅法返回的是该对象的内存地址,所以必须重写hashCode⽅法,才能保证两个不同的对象具有相同的hashCode,同时也需要两个不同对象⽐较equals⽅法会返回true
该集合中没有特有的⽅法,直接继承⾃Collection。
—| Itreable 接⼝ 实现该接⼝可以使⽤增强for循环
—| Collection 描述所有集合共性的接⼝
—| List接⼝ 可以有重复元素的集合
—| ArrayList
—| LinkedList
—| Set接⼝ 不可以有重复元素的集合
案例:set集合添加元素并使⽤迭代器迭代元素。
public class Demo4 {
public static void main(String[] args) {
//Set 集合存和取的顺序不⼀致。
Set hs = new HashSet();
hs.add("世界军事");
hs.add("兵器知识");
hs.add("舰船知识");
hs.add("汉和防务");
System.out.println(hs);
// [舰船知识, 世界军事, 兵器知识, 汉和防务]
Iterator it = hs.iterator();
while (it.hasNext()) {
System.out.());
}
}
}
###HashSet
| Itreable 接⼝ 实现该接⼝可以使⽤增强for循环
—| Collection 描述所有集合共性的接⼝
—| List接⼝ 可以有重复元素的集合
—| ArrayList
—| LinkedList
—| Set接⼝ 不可以有重复元素的集合
—| HashSet 线程不安全,存取速度快。底层是以哈希表实现的。
###HashSet
哈希表边存放的是哈希值。HashSet存储元素的顺序并不是按照存⼊时的顺序(和List显然不同) 是按照哈希值来存的所以取数据也是按照哈希值取得。
HashSet不存⼊重复元素的规则.使⽤hashcode和equals
由于Set集合是不能存⼊重复元素的集合。那么HashSet也是具备这⼀特性的。HashSet如何检查重复?HashSet会通过元素的
hashcode()和equals⽅法进⾏判断元素师否重复。
当你试图把对象加⼊HashSet时,HashSet会使⽤对象的hashCode来判断对象加⼊的位置。同时也会与其他已经加⼊的对象的hashCode 进⾏⽐较,如果没有相等的hashCode,HashSet就会假设对象没有重复出现。
简单⼀句话,如果对象的hashCode值是不同的,那么HashSet会认为对象是不可能相等的。
因此我们⾃定义类的时候需要重写hashCode,来确保对象具有相同的hashCode值。
如果元素(对象)的hashCode值相同,是不是就⽆法存⼊HashSet中了? 当然不是,会继续使⽤equals 进⾏⽐较.如果 equals为true 那么HashSet认为新加⼊的对象重复了,所以加⼊失败。如果equals 为false那么HashSet 认为新加⼊的对象没有重复.新元素可以存⼊.
###总结:
###元素的哈希值是通过元素的hashcode⽅法 来获取的, HashSet⾸先判断两个元素的哈希值,如果哈
希值⼀样,接着会⽐较equals⽅法如果 equls结果为true ,HashSet就视为同⼀个元素。如果equals 为false就不是同⼀个元素。
###哈希值相同equals为false的元素是怎么存储呢,就是在同样的哈希值下顺延(可以认为哈希值相同的元素放在⼀个哈希桶中)。也就是哈希⼀样的存⼀列。
HashSet:通过hashCode值来确定元素在内存中的位置。⼀个hashCode位置上可以存放多个元素。
当hashcode() 值相同equals() 返回为true 时,hashset 集合认为这两个元素是相同的元素.只存储⼀个(重复元素⽆法放⼊)。调⽤原理:先判断hashcode ⽅法的值,如果相同才会去判断equals 如果不相同,是不会调⽤equals⽅法的。
HashSet到底是如何判断两个元素重复。
通过hashCode⽅法和equals⽅法来保证元素的唯⼀性,add()返回的是boolean类型
判断两个元素是否相同,先要判断元素的hashCode值是否⼀致,只有在该值⼀致的情况下,才会判断equals⽅法,如果存储在HashSet 中的两个对象hashCode⽅法的值相同equals⽅法返回的结果是true,那么HashSet认为这两个元素是相同元素,只存储⼀个(重复元素⽆法存⼊)。
注意:HashSet集合在判断元素是否相同先判断hashCode⽅法,如果相同才会判断equals。如果不相同,是不会调⽤equals⽅法的。
HashSet 和ArrayList集合都有判断元素是否相同的⽅法,
boolean contains(Object o)
HashSet使⽤hashCode和equals⽅法,ArrayList使⽤了equals⽅法
public static void main(String[] args) {
// Set 集合存和取的顺序不⼀致。
Set hs = new HashSet();
hs.add("世界军事");
hs.add("兵器知识");
hs.add("舰船知识");
hs.add("汉和防务");
// 返回此 set 中的元素的数量
System.out.println(hs.size()); // 4
// 如果此 set 尚未包含指定元素,则返回 true
boolean add = hs.add("世界军事"); // false
System.out.println(add);
// 返回此 set 中的元素的数量
System.out.println(hs.size());// 4
Iterator it = hs.iterator();
while (it.hasNext()) {
System.out.());
}
}
}
使⽤HashSet存储⾃定义对象,并尝试添加重复对象(对象的重复的判定)
public class Demo4 {
public static void main(String[] args) {
HashSet hs = new HashSet();
hs.add(new Person(“jack”, 20));
hs.add(new Person(“rose”, 20));
hs.add(new Person(“hmm”, 20));
hs.add(new Person(“lilei”, 20));
hs.add(new Person(“jack”, 20));
Iterator it = hs.iterator();
while (it.hasNext()) {
Object next = it.next();
System.out.println(next);
}
}
}
class Person {
private String name;
private int age;
Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
java中index是什么意思
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public int hashCode() {
System.out.println("hashCode:" + this.name);
return this.name.hashCode() + age * 37;
}
@Override
public boolean equals(Object obj) {
System.out.println(this + "---equals---" + obj);
if (obj instanceof Person) {
Person p = (Person) obj;
return this.name.equals(p.name) && this.age == p.age;
} else {
return false;
}
}
@Override
public String toString() {
return "Person@name:" + this.name + " age:" + this.age;
}
}
问题:现在有⼀批数据,要求不能重复存储元素,⽽且要排序。ArrayList 、 LinkedList不能去除重复数据。HashSet可以去除重复,但是是⽆序。
所以这时候就要使⽤TreeSet了
三,TreeSet
案例:使⽤TreeSet集合存储字符串元素,并遍历
public class Demo5 {
public static void main(String[] args) {
TreeSet ts = new TreeSet();
ts.add(“ccc”);
ts.add(“aaa”);
ts.add(“ddd”);
ts.add(“bbb”);
System.out.println(ts); // [aaa, bbb, ccc, ddd]
}
}
—| Itreable 接⼝ 实现该接⼝可以使⽤增强for循环
—| Collection 描述所有集合共性的接⼝
—| List接⼝ 有序,可以重复,有⾓标的集合
—| ArrayList
| LinkedList
—| Set接⼝ ⽆序,不可以重复的集合
—| HashSet 线程不安全,存取速度快。底层是以hash表实现的。
—| TreeSet 红-⿊树的数据结构,默认对元素进⾏⾃然排序(String)。如果在⽐较的时候两个对象返回值为0,那么元素重复。
红-⿊树
红⿊树是⼀种特定类型的⼆叉树
红⿊树算法的规则: 左⼩右⼤。
既然TreeSet可以⾃然排序,那么TreeSet必定是有排序规则的。
1:让存⼊的元素⾃定义⽐较规则。
2:给TreeSet指定排序规则。
⽅式⼀:元素⾃⾝具备⽐较性
元素⾃⾝具备⽐较性,需要元素实现Comparable接⼝,重写compareTo⽅法,也就是让元素⾃⾝具备⽐较性,这种⽅式叫做元素的⾃然排序也叫做默认排序。
⽅式⼆:容器具备⽐较性
当元素⾃⾝不具备⽐较性,或者⾃⾝具备的⽐较性不是所需要的。那么此时可以让容器⾃⾝具备。需要定义⼀个类实现接⼝Comparator,重写compare⽅法,并将该接⼝的⼦类实例对象作为参数传递给TreeMap集合的构造⽅法。
注意:当Comparable⽐较⽅式和Comparator⽐较⽅式同时存在时,以Comparator的⽐较⽅式为主;
注意:在重写compareTo或者compare⽅法时,必须要明确⽐较的主要条件相等时要⽐较次要条件。(假设姓名和年龄⼀直的⼈为相同的⼈,如果想要对⼈按照年龄的⼤⼩来排序,如果年龄相同的⼈,需要如何处理?不能直接return 0,因为可能姓名不同(年龄相同姓名不同的⼈是不同的⼈)。此时就需要进⾏次要条件判断(需要判断姓名),只有姓名和年龄同时相等的才可以返回0.)
通过return 0来判断唯⼀性。
问题:为什么使⽤TreeSet存⼊字符串,字符串默认输出是按升序排列的?因为字符串实现了⼀个接⼝,叫做Comparable 接⼝.字符串重写了该接⼝的compareTo ⽅法,所以String对象具备了⽐较性.那
么同样道理,我的⾃定义元素(例如Person类,Book类)想要存⼊TreeSet集合,就需要实现该接⼝,也就是要让⾃定义对象具备⽐较性.
存⼊TreeSet集合中的元素要具备⽐较性.
⽐较性要实现Comparable接⼝,重写该接⼝的compareTo⽅法
TreeSet属于Set集合,该集合的元素是不能重复的,TreeSet如何保证元素的唯⼀性
通过compareTo或者compare⽅法中的来保证元素的唯⼀性。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。