【JavaSE进阶】05-集合

集合继承结构图

集合继承结构图_Collection部分

注:泛化关系即继承关系,is a;关联关系,has a;实现关系,like a

1 接口java.util.Collection<E>:

接口Collection是集合中的超级父接口

Iterator it = "Collection 对象".iterator();

Collection接口继承了Iterable接口,调用Iterable接口的iterator()方法。it是迭代器对象,可以用来迭代集合(迭代即遍历,把集合元素一个个取出来)。

所有集合继承Iterable的含义是:所有集合都是可迭代的。


2 接口java.util.List<E>:

List集合存储元素特点:有序可重复,存储的元素有下标。

有序:指的是存进去是这个顺序,取出来还是这个顺序。这里的顺序不是说按照大小排序,而是下标排序。有序是因为List集合都有下标,下标从0开始,以1递增。

可重复:存进去的元素值可以相同,比如存了两个zhangsan,但它们的下标不同

比如说:先存zhangsan,再存lisi,再存wangwu。对应的下标就是0,1,2,则取出来也按照先zhangsan,后lisi,最后wangwu

3 接口java.util.Set<E>:

Set集合存储元素特点:无序不可重复,存储的元素无下标。

无序:指的是存进去是这个顺序,取出来就不一定是这个顺序了。另外Set集合中的元素没有下标

不可重复:Set集合中的元素不可重复


接口java.util.List<E>的实现类:

4 类java.util.ArrayList<E>:

ArrayList集合底层采用了数组数据结构,ArrayList集合是非线程安全的。

1、ArrayList集合初始化容量是10

2、ArrayList集合底层是Object类型的数组Object[]

3、扩容到原容量的1.5倍。

4、建议给定一个预估计的初始化容量,减少数组的扩容次数,这是ArrayList集合比较重要的优化策略。

5、数组的优点:检索效率比较高

6、数组的缺点:随机增删元素效率比较低

7、但是需要注意的是:向数组末尾添加元素,效率还是很高的

5 类java.util.LinkedList<E>:

LinkedList集合底层采用了双向链表数据结构

1、LinkedList集合是双向链表。

2、对于链表数据结构来说,随机增删效率较高。检索效率较低。

3、链表中的元素在空间存储上,内存地址不连续。

6 类java.util.Vector<E>:

Vector集合底层采用了数组数据结构

Vector集合是线程安全的。Vector所有的方法都有synchronized关键字修饰所以线程安全,但是效率较低,现在保证线程安全有别的方案,所以Vector使用较少了


接口java.util.Set<E>的实现类:

7 类java.util.HashSet<E>:

HashSet集合在new的时候,底层实际上new了一个HashMap集合。向HashSet集合中存储元素,实际上是存储到HashMap集合中了。HashMap集合是一个哈希表数据结构。HashSet集合初始化容量16。初始化容量建议是2的倍数。

扩容:扩容之后是原容量2倍

8 接口java.util.SortedSet<E>:

SortedSet集合存储元素特点:无序不可重复,存储的元素无下标。

SortedSet集合继承了Set集合,所以它的特点也是无序不可重复。但是放在SortedSet集合中的元素可以自动排序。我们称为可排序集合:放到该集合中的元素是自动按照大小顺序排序的。

9 接口java.util.SortedSet<E>下的类java.util.TreeSet<E>:

TreeSet集合底层实际上是TreeMap,new TreeSet集合的时候,底层实际上new了一个TreeMap集合。向TreeSet集合中存储数据,实际上是将数据存储到TreeMap集合中了。TreeMap集合底层采用了二叉树数据结构

集合继承结构图_Map部分

1 接口java.util.Map<K,V>:

1、Map集合和Collection集合没有关系

2、Map集合以key和value的这种键值对的方式存储元素

3、key和value都是存储java对象的内存地址

4、所有Map集合的key特点:无需不可重复

Map集合的key和Set集合存储元素特点相同


2 类java.util.HashMap<K,V>:

HashMap集合底层是哈希表数据结构,是非线程安全的。

在JDK8之后,如果哈希表单向链表中元素超过8个,单向链表这种数据结构会变成红黑树数据结构。当红黑树上的节点数量小于6时,会重新把红黑树变成单向链表数据结构。这种方式也是为了提高检索效率,二叉树的检索会再次缩小扫描范围,提高效率。初始化容量16。默认加载因子0.75

扩容:扩容之后的容量是原容量的2倍

HashMap集合的key和value允许null


3 类java.util.Hashtable<K,V>:

Hashtable集合底层也是哈希表数据结构,是线程安全的。其中所有的方法都带有synchronized关键字,效率低,现在使用较少了,因为控制线程安全有其它更好的方案。

Hashtable集合初始化容量11

Hashtable集合扩容是:原容量*2+1

Hashtable集合的key和value不允许null

4 类java.util.Hashtable<K,V>的子类java.util.Properties<String,String>

Properties是线程安全的,因为继承Hashtable,另外Properties存储元素的时候也是采用key和value的形式存储,并且key和value只支持String类型,不支持其它类型。Properties被称为属性类


5 接口java.util.SortedMap<K,V>:

SortedMap集合的key存储元素的特点:无序不可重复

SortedMap集合key部分就是SortedSet,会自动按照大小顺序排序,称为可排序的集合

6 接口java.util.SortedMap<K,V>下的类java.util.TreeMap<K,V>:

TreeMap集合底层的数据结构是一个二叉树

总结

总结(所有的实现类):

ArrayList:底层是数组。

LinkedList:底层是双向链表。

Vector:底层是数组,线程安全的,效率较低,使用较少

HashSet:底层是HashMap,放到HashSet集合中的元素等同于放到HashMap集合key部分了。

TreeSet:底层是TreeMap,放到TreeSet集合中的元素等同于放到TreeMap集合key部分了。

HashMap:底层是哈希表。

Hashtable:底层也是哈希表,只不过线程安全的,效率较低,使用较少

Properties:是线程安全的,并且key和value只能存储字符串String。

TreeMap:底层是二叉树。TreeMap集合的key可以自动按照大小顺序排序。


List集合存储元素的特点:

有序可重复

有序:存进去的顺序和取出的顺序相同,每一个元素都有下标。

可重复:存进去1,可以再存储一个1.

Set(Map)集合存储元素的特点:

无序不可重复

无序:存进去的顺序和取出的顺序不一定相同。另外Set集合中元素没有下标。

不可重复:存进去1,不能再存储1了。

SortedSet(SortedMap)集合存储元素特点:

首先是无序不可重复的,但是SortedSet集合中的元素是可排序的。

无序:存进去的顺序和取出的顺序不一定相同。另外Set集合中元素没有下标。

不可重复:存进去1,不能再存储1了。

可排序:可以按照大小顺序排列。


PS:

Map集合的key,就是一个Set集合。

往Set集合中放数据,实际上放到了Map集合的key部分。

1 内容

1.1 主要集合概述

1.2 Collection和Iterator

1.2.1 代码笔记_Collection接口常用方法

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Collection;/*
关于java.util.Collection接口中常用的方法。1、Collection中能存放什么元素?没有使用“泛型”之前,Collection中可以存储Object的所有子类型。使用了“泛型”之后,Collection中只能存储某个具体的类型。集合后期我们会学习“泛型”语法。目前先不用管。Collection中什么都能存,只要是Object的子类型就行。(集合中不能直接存储基本数据类型,也不能存java对象,只是存储java对象的内存地址。)2、Collection中的常用方法boolean add(Object e) 向集合中添加元素int size()  获取集合中元素的个数void clear() 清空集合boolean contains(Object o) 判断当前集合中是否包含元素o,包含返回true,不包含返回falseboolean remove(Object o) 删除集合中的某个元素。boolean isEmpty()  判断该集合中元素的个数是否为0Object[] toArray()  调用这个方法可以把集合转换成数组。【作为了解,使用不多。】*/
public class CollectionTest01 {public static void main(String[] args) {// 创建一个集合对象//Collection c = new Collection(); // 接口是抽象的,无法实例化。// 多态Collection c = new ArrayList();// 测试Collection接口中的常用方法c.add(1200); // 自动装箱(java5的新特性。),实际上是放进去了一个对象的内存地址。Integer x = new Integer(1200);c.add(3.14); // 自动装箱c.add(new Object());c.add(new Student());c.add(true); // 自动装箱// 获取集合中元素的个数System.out.println("集合中元素个数是:" + c.size()); // 5// 清空集合c.clear();System.out.println("集合中元素个数是:" + c.size()); // 0// 再向集合中添加元素c.add("hello"); // "hello"对象的内存地址放到了集合当中。c.add("world");c.add("浩克");c.add("绿巨人");c.add(1);// 判断集合中是否包含"绿巨人"boolean flag = c.contains("绿巨人");System.out.println(flag); // trueboolean flag2 = c.contains("绿巨人2");System.out.println(flag2); // falseSystem.out.println(c.contains(1)); // trueSystem.out.println("集合中元素个数是:" + c.size()); // 5// 删除集合中某个元素c.remove(1);System.out.println("集合中元素个数是:" + c.size()); // 4// 判断集合是否为空(集合中是否存在元素)System.out.println(c.isEmpty()); // false// 清空c.clear();System.out.println(c.isEmpty()); // true(true表示集合中没有元素了!)c.add("abc");c.add("def");c.add(100);c.add("helloworld!");c.add(new Student());// 转换成数组(了解,使用不多。)Object[] objs = c.toArray();for(int i = 0; i < objs.length; i++){// 遍历数组Object o = objs[i];System.out.println(o);}}
}class Student{}

1.2.2 代码笔记_Collection集合迭代

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;/*** 关于集合遍历/迭代专题。(重点:五颗星*****)*/
public class CollectionTest02 {public static void main(String[] args) {// 注意:以下讲解的遍历方式/迭代方式,是所有Collection通用的一种方式。// 在Map集合中不能用。在所有的Collection以及子类中使用。// 创建集合对象Collection c = new ArrayList(); // 后面的集合无所谓,主要是看前面的Collection接口,怎么遍历/迭代。// 添加元素c.add("abc");c.add("def");c.add(100);c.add(new Object());// 对集合Collection进行遍历/迭代// 第一步:获取集合对象的迭代器对象IteratorIterator it = c.iterator();// 第二步:通过以上获取的迭代器对象开始迭代/遍历集合。/*以下两个方法是迭代器对象Iterator中的方法:boolean hasNext()如果仍有元素可以迭代,则返回 true。Object next() 返回迭代的下一个元素。*/while(it.hasNext()){Object obj = it.next();System.out.println(obj);}// 一直取,不判断,会出现异常:java.util.NoSuchElementException/*while(true){Object obj = it.next();System.out.println(obj);}*//*boolean hasNext = it.hasNext();System.out.println(hasNext);if(hasNext) {// 不管你当初存进去什么,取出来统一都是Object。Object obj = it.next();System.out.println(obj);}hasNext = it.hasNext();System.out.println(hasNext);if(hasNext) {Object obj = it.next();System.out.println(obj);}hasNext = it.hasNext();System.out.println(hasNext);if(hasNext) {Object obj = it.next();System.out.println(obj);}hasNext = it.hasNext();System.out.println(hasNext);if(hasNext) {Object obj = it.next();System.out.println(obj);}hasNext = it.hasNext();System.out.println(hasNext);if(hasNext) {Object obj = it.next();System.out.println(obj);}*/}
}

 迭代集合的原理:

 迭代原理:

 1.2.3 代码笔记_Iterator迭代器

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Collection;
import java.util.HashSet;
import java.util.Iterator;/*
关于集合的迭代/遍历*/
public class CollectionTest03 {public static void main(String[] args) {// 创建集合对象Collection c1  = new ArrayList(); // ArrayList集合:有序可重复// 添加元素c1.add(1);c1.add(2);c1.add(3);c1.add(4);c1.add(1);// 迭代集合Iterator it = c1.iterator();while(it.hasNext()){// 存进去是什么类型,取出来还是什么类型。Object obj = it.next();/*if(obj instanceof Integer){System.out.println("Integer类型");}*/// 只不过在输出的时候会转换成字符串。因为这里println会调用toString()方法。System.out.println(obj);}// HashSet集合:无序不可重复Collection c2 = new HashSet();// 无序:存进去和取出的顺序不一定相同。// 不可重复:存储100,不能再存储100.c2.add(100);c2.add(200);c2.add(300);c2.add(90);c2.add(400);c2.add(50);c2.add(60);c2.add(100);Iterator it2 = c2.iterator();while(it2.hasNext()){System.out.println(it2.next());}}
}

1.2.4 代码笔记_contains方法

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Collection;/*
深入Collection集合的contains方法:boolean contains(Object o)判断集合中是否包含某个对象o如果包含返回true, 如果不包含返回false。contains方法是用来判断集合中是否包含某个元素的方法,那么它在底层是怎么判断集合中是否包含某个元素的呢?调用了equals方法进行比对。equals方法返回true,就表示包含这个元素。*/
public class CollectionTest04 {public static void main(String[] args) {// 创建集合对象Collection c = new ArrayList();// 向集合中存储元素String s1 = new String("abc"); // s1 = 0x1111c.add(s1); // 放进去了一个"abc"String s2 = new String("def"); // s2 = 0x2222c.add(s2);// 集合中元素的个数System.out.println("元素的个数是:" + c.size()); // 2// 新建的对象StringString x = new String("abc"); // x = 0x5555// c集合中是否包含x?结果猜测一下是true还是false?System.out.println(c.contains(x)); //判断集合中是否存在"abc" true}
}

Collection的contains方法,contains()底层调用了equals 

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Collection;/*
测试contains方法
测试remove方法。
结论:存放在一个集合中的类型,一定要重写equals方法。*/
public class CollectionTest05 {public static void main(String[] args) {// 创建集合对象Collection c = new ArrayList();// 创建用户对象User u1 = new User("jack");// 加入集合c.add(u1);// 判断集合中是否包含u2User u2 = new User("jack");// 没有重写equals之前:这个结果是false//System.out.println(c.contains(u2)); // false// 重写equals方法之后,比较的时候会比较name。System.out.println(c.contains(u2)); // truec.remove(u2);System.out.println(c.size()); // 0/*Integer x = new Integer(10000);c.add(x);Integer y = new Integer(10000);System.out.println(c.contains(y)); // true*/// 创建集合对象Collection cc = new ArrayList();// 创建字符串对象String s1 = new String("hello");// 加进去。cc.add(s1);// 创建了一个新的字符串对象String s2 = new String("hello");// 删除s2cc.remove(s2); // s1.equals(s2) java认为s1和s2是一样的。删除s2就是删除s1。// 集合中元素个数是?System.out.println(cc.size()); // 0}
}class User{private String name;public User(){}public User(String name){this.name = name;}// 重写equals方法// 将来调用equals方法的时候,一定是调用这个重写的equals方法。// 这个equals方法的比较原理是:只要姓名一样就表示同一个用户。public boolean equals(Object o) {if(o == null || !(o instanceof User)) return false;if(o == this) return true;User u = (User)o;// 如果名字一样表示同一个人。(不再比较对象的内存地址了。比较内容。)return u.name.equals(this.name);}}

1.2.5 代码笔记_remove方法

重点:当集合的结构发生改变时,迭代器必须重新获取,如果还是用以前老的迭代器,会出现异常:java.util.ConcurrentModificationException

重点:在迭代集合元素的过程中,不能调用集合对象的remove方法,删除元素:c.remove(o); 迭代过程中不能这样。会出现:java.util.ConcurrentModificationException

重点:在迭代元素的过程当中,一定要使用迭代器Iterator的remove方法,删除元素,不要使用集合自带的remove方法删除元素。

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;/*
关于集合元素的remove重点:当集合的结构发生改变时,迭代器必须重新获取,如果还是用以前老的迭代器,会出现异常:java.util.ConcurrentModificationException重点:在迭代集合元素的过程中,不能调用集合对象的remove方法,删除元素:c.remove(o); 迭代过程中不能这样。会出现:java.util.ConcurrentModificationException重点:在迭代元素的过程当中,一定要使用迭代器Iterator的remove方法,删除元素,不要使用集合自带的remove方法删除元素。*/
public class CollectionTest06 {public static void main(String[] args) {// 创建集合Collection c = new ArrayList();// 注意:此时获取的迭代器,指向的是那是集合中没有元素状态下的迭代器。// 一定要注意:集合结构只要发生改变,迭代器必须重新获取。// 当集合结构发生了改变,迭代器没有重新获取时,调用next()方法时:java.util.ConcurrentModificationExceptionIterator it = c.iterator();// 添加元素c.add(1); // Integer类型c.add(2);c.add(3);// 获取迭代器//Iterator it = c.iterator();/*while(it.hasNext()){// 编写代码时next()方法返回值类型必须是Object。// Integer i = it.next();Object obj = it.next();System.out.println(obj);}*/Collection c2 = new ArrayList();c2.add("abc");c2.add("def");c2.add("xyz");Iterator it2 = c2.iterator();while(it2.hasNext()){Object o = it2.next();// 删除元素// 删除元素之后,集合的结构发生了变化,应该重新去获取迭代器// 但是,循环下一次的时候并没有重新获取迭代器,所以会出现异常:java.util.ConcurrentModificationException// 出异常根本原因是:集合中元素删除了,但是没有更新迭代器(迭代器不知道集合变化了)//c2.remove(o); // 直接通过集合去删除元素,没有通知迭代器。(导致迭代器的快照和原集合状态不同。)// 使用迭代器来删除可以吗?// 迭代器去删除时,会自动更新迭代器,并且更新集合(删除集合中的元素)。it2.remove(); // 删除的一定是迭代器指向的当前元素。System.out.println(o);}System.out.println(c2.size()); //0}
}

1.3 List接口

1.3.1 List接口概述

1.3.2 代码笔记_List接口特有方法

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;/*
测试List接口中常用方法1、List集合存储元素特点:有序可重复有序:List集合中的元素有下标。从0开始,以1递增。可重复:存储一个1,还可以再存储1.2、List既然是Collection接口的子接口,那么肯定List接口有自己“特色”的方法:以下只列出List接口特有的常用的方法:void add(int index, Object element)Object set(int index, Object element)Object get(int index)int indexOf(Object o)int lastIndexOf(Object o)Object remove(int index)以上几个方法不需要死记硬背,可以自己编写代码测试一下,理解一下,以后开发的时候,还是要翻阅帮助文档。*/
public class ListTest01 {public static void main(String[] args) {// 创建List类型的集合。//List myList = new LinkedList();//List myList = new Vector();List myList = new ArrayList();// 添加元素myList.add("A"); // 默认都是向集合末尾添加元素。myList.add("B");myList.add("C");myList.add("C");myList.add("D");//在列表的指定位置插入指定元素(第一个参数是下标)// 这个方法使用不多,因为对于ArrayList集合来说效率比较低。myList.add(1, "KING");// 迭代Iterator it = myList.iterator();while(it.hasNext()){Object elt = it.next();System.out.println(elt);}// 根据下标获取元素Object firstObj = myList.get(0);System.out.println(firstObj);// 因为有下标,所以List集合有自己比较特殊的遍历方式// 通过下标遍历。【List集合特有的方式,Set没有。】for(int i = 0; i < myList.size(); i++){Object obj = myList.get(i);System.out.println(obj);}// 获取指定对象第一次出现处的索引。System.out.println(myList.indexOf("C")); // 3// 获取指定对象最后一次出现处的索引。System.out.println(myList.lastIndexOf("C")); // 4// 删除指定下标位置的元素// 删除下标为0的元素myList.remove(0);System.out.println(myList.size()); // 5System.out.println("====================================");// 修改指定位置的元素myList.set(2, "Soft");// 遍历集合for(int i = 0; i < myList.size(); i++){Object obj = myList.get(i);System.out.println(obj);}}
}/*
计算机英语:增删改查这几个单词要知道:增:add、save、new删:delete、drop、remove改:update、set、modify查:find、get、query、select*/

1.3.3 代码笔记_ArrayList集合

ArrayList集合初始化容量及扩容

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.List;/*
ArrayList集合:1、默认初始化容量10(底层先创建了一个长度为0的数组,当添加第一个元素的时候,初始化容量10。)2、集合底层是一个Object[]数组。3、构造方法:new ArrayList();new ArrayList(20);4、ArrayList集合的扩容:增长到原容量的1.5倍。ArrayList集合底层是数组,怎么优化?尽可能少的扩容。因为数组扩容效率比较低,建议在使用ArrayList集合的时候预估计元素的个数,给定一个初始化容量。5、数组优点:检索效率比较高。(每个元素占用空间大小相同,内存地址是连续的,知道首元素内存地址,然后知道下标,通过数学表达式计算出元素的内存地址,所以检索效率最高。)6、数组缺点:随机增删元素效率比较低。另外数组无法存储大数据量。(很难找到一块非常巨大的连续的内存空间。)7、向数组末尾添加元素,效率很高,不受影响。8、面试官经常问的一个问题?这么多的集合中,你用哪个集合最多?答:ArrayList集合。因为往数组末尾添加元素,效率不受影响。另外,我们检索/查找某个元素的操作比较多。7、ArrayList集合是非线程安全的。(不是线程安全的集合。)*/
public class ArrayListTest01 {public static void main(String[] args) {// 默认初始化容量是10// 数组的长度是10List list1 = new ArrayList();// 集合的size()方法是获取当前集合中元素的个数。不是获取集合的容量。System.out.println(list1.size()); // 0// 指定初始化容量// 数组的长度是20List list2 = new ArrayList(20);// 集合的size()方法是获取当前集合中元素的个数。不是获取集合的容量。System.out.println(list2.size()); // 0list1.add(1);list1.add(2);list1.add(3);list1.add(4);list1.add(5);list1.add(6);list1.add(7);list1.add(8);list1.add(9);list1.add(10);System.out.println(list1.size());// 再加一个元素list1.add(11);System.out.println(list1.size()); // 11个元素。/*int newCapacity = ArraysSupport.newLength(oldCapacity,minCapacity - oldCapacity,oldCapacity >> 1);*/// 100 二进制转换成10进制: 00000100右移一位 00000010 (2)  【4 / 2】// 原先是4、现在增长:2,增长之后是6,增长之后的容量是之前容量的:1.5倍。// 6是4的1.5倍}
}

二进制位运

package com.bjpowernode.javase.collection;/*
位运算符 >>*/
public class BinaryTest {public static void main(String[] args) {// 5// >> 1 二进制右移1位。// >> 2 二进制右移2位。// 10的二进制位是:00001010  【10】// 10的二进制右移1位是:00000101  【5】System.out.println(10 >> 1); // 右移1位就是除以2// 二进制位左移1位// 10的二进制位是:00001010  【10】// 10的二进制左移1位:00010100 【20】System.out.println(10 << 1);}
}

ArrayList集合有参构造方法

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Collection;
import java.util.HashSet;
import java.util.List;/*
集合ArrayList的构造方法*/
public class ArrayListTest02 {public static void main(String[] args) {// 默认初始化容量10List myList1 = new ArrayList();// 指定初始化容量100List myList2 = new ArrayList(100);// 创建一个HashSet集合Collection c = new HashSet();// 添加元素到Set集合c.add(100);c.add(200);c.add(900);c.add(50);// 通过这个构造方法就可以将HashSet集合转换成List集合。List myList3 = new ArrayList(c);for(int i = 0; i < myList3.size(); i++){System.out.println(myList3.get(i));}}
}

 1.3.4 代码笔记_链表数据结构

单向链表:

双向链表:

package com.bjpowernode.javase.danlink;/*
单链表中的节点。
节点是单向链表中基本的单元。
每一个节点Node都有两个属性:一个属性:是存储的数据。另一个属性:是下一个节点的内存地址。*/
public class Node {// 存储的数据Object data;// 下一个节点的内存地址Node next;public Node(){}public Node(Object data, Node next){this.data = data;this.next = next;}
}
package com.bjpowernode.javase.danlink;/*
链表类。(单向链表)*/
public class Link<E> {public static void main(String[] args) {Link<String> link = new Link<>();link.add("abc");// 类型不匹配。//link.add(123);}// 头节点Node header;int size = 0;public int size(){return size;}// 向链表中添加元素的方法(向末尾添加)public void add(E data){//public void add(Object data){// 创建一个新的节点对象// 让之前单链表的末尾节点next指向新节点对象。// 有可能这个元素是第一个,也可能是第二个,也可能是第三个。if(header == null){// 说明还没有节点。// new一个新的节点对象,作为头节点对象。// 这个时候的头节点既是一个头节点,又是一个末尾节点。header = new Node(data, null);}else {// 说明头不是空!// 头节点已经存在了!// 找出当前末尾节点,让当前末尾节点的next是新节点。Node currentLastNode = findLast(header);currentLastNode.next = new Node(data, null);}size++;}/*** 专门查找末尾节点的方法。*/private Node findLast(Node node) {if(node.next == null) {// 如果一个节点的next是null// 说明这个节点就是末尾节点。return node;}// 程序能够到这里说明:node不是末尾节点。return findLast(node.next); // 递归算法!}// 删除链表中某个数据的方法public void remove(Object obj){}// 修改链表中某个数据的方法public void modify(Object newObj){}// 查找链表中某个元素的方法。public int find(Object obj){return 1;}
}
package com.bjpowernode.javase.danlink;public class Test {public static void main(String[] args) {// 创建了一个集合对象Link link = new Link();// 往集合中添加元素link.add("abc");link.add("def");link.add("xyz");// 获取元素个数System.out.println(link.size());}
}

 链表优点和缺点:

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;/*
链表的优点:由于链表上的元素在空间存储上内存地址不连续。所以随机增删元素的时候不会有大量元素位移,因此随机增删效率较高。在以后的开发中,如果遇到随机增删集合中元素的业务比较多时,建议使用LinkedList。链表的缺点:不能通过数学表达式计算被查找元素的内存地址,每一次查找都是从头节点开始遍历,直到找到为止。所以LinkedList集合检索/查找的效率较低。ArrayList:把检索发挥到极致。(末尾添加元素效率还是很高的。)LinkedList:把随机增删发挥到极致。加元素都是往末尾添加,所以ArrayList用的比LinkedList多。*/
public class LinkedListTest01 {public static void main(String[] args) {// LinkedList集合底层也是有下标的。// 注意:ArrayList之所以检索效率比较高,不是单纯因为下标的原因。是因为底层数组发挥的作用。// LinkedList集合照样有下标,但是检索/查找某个元素的时候效率比较低,因为只能从头节点开始一个一个遍历。List list = new LinkedList();list.add("a");list.add("b");list.add("c");for(int i = 0; i <list.size(); i++){Object obj = list.get(i);System.out.println(obj);}// LinkedList集合有初始化容量吗?没有。// 最初这个链表中没有任何元素。first和last引用都是null。// 不管是LinkedList还是ArrayList,以后写代码时不需要关心具体是哪个集合。// 因为我们要面向接口编程,调用的方法都是接口中的方法。//List list2 = new ArrayList(); // 这样写表示底层你用了数组。List list2 = new LinkedList(); // 这样写表示底层你用了双向链表。// 以下这些方法你面向的都是接口编程。list2.add("123");list2.add("456");list2.add("789");for(int i = 0; i < list2.size(); i++){System.out.println(list2.get(i));}}
}

1.3.5 代码笔记_ LinkedList源码分析

LinkedList内存图:

   public boolean add(E e) {linkLast(e);return true;}void linkLast(E e) {final Node<E> l = last;final Node<E> newNode = new Node<>(l, e, null);last = newNode;if (l == null)first = newNode;elsel.next = newNode;size++;modCount++;}

1.3.6 代码笔记_Vector集合源码分析

package com.bjpowernode.javase.collection;import java.util.*;/*
Vector:1、底层也是一个数组。2、初始化容量:103、怎么扩容的?扩容之后是原容量的2倍。10--> 20 --> 40 --> 804、ArrayList集合扩容特点:ArrayList集合扩容是原容量1.5倍。5、Vector中所有的方法都是线程同步的,都带有synchronized关键字,是线程安全的。效率比较低,使用较少了。6、怎么将一个线程不安全的ArrayList集合转换成线程安全的呢?使用集合工具类:java.util.Collections;java.util.Collection 是集合接口。java.util.Collections 是集合工具类。*/
public class VectorTest {public static void main(String[] args) {// 创建一个Vector集合List vector = new Vector();//Vector vector = new Vector();// 添加元素// 默认容量10个。vector.add(1);vector.add(2);vector.add(3);vector.add(4);vector.add(5);vector.add(6);vector.add(7);vector.add(8);vector.add(9);vector.add(10);// 满了之后扩容(扩容之后的容量是20.)vector.add(11);Iterator it = vector.iterator();while(it.hasNext()){Object obj = it.next();System.out.println(obj);}// 这个可能以后要使用!!!!List myList = new ArrayList(); // 非线程安全的。// 变成线程安全的Collections.synchronizedList(myList); // 这里没有办法看效果,因为多线程没学,你记住先!// myList集合就是线程安全的了。myList.add("111");myList.add("222");myList.add("333");}
}

1.4 Set接口

1.4.1 哈希表

1.4.2 HashSet

HashSet中的数据是无序的不可重复的。HashSet按照哈希算法存取数据的,具有非常好的性能。它的工作原理是这样的:当向HashSet中插入数据的时候,他会调用对象的hashCode得到该对象的哈希码,然后根据哈希码计算出该对象插入到集合中的位置。

1.4.3 Comparable和Comparator的区别

一个类实现了 Camparable 接口则表明这个类的对象之间是可以相互比较的,这个类对象组成 的集合就可以直接使用 sort 方法排序。

Comparator 可以看成一种算法的实现,将算法和数据分离,Comparator 也可以在下面两种环境 下使用:

1、类的没有考虑到比较问题而没有实现 Comparable,可以通过 Comparator 来实现排序而不必 改变对象本身

2、可以使用多种排序标准,比如升序、降序等

1.4.4 代码笔记_演示HashSet集合特点

package com.bjpowernode.javase.collection;import java.util.HashSet;
import java.util.Set;/*
HashSet集合:无序不可重复。*/
public class HashSetTest01 {public static void main(String[] args) {// 演示一下HashSet集合特点Set<String> strs = new HashSet<>();// 添加元素strs.add("hello3");strs.add("hello4");strs.add("hello1");strs.add("hello2");strs.add("hello3");strs.add("hello3");strs.add("hello3");strs.add("hello3");// 遍历/*hello1hello4hello2hello31、存储时顺序和取出的顺序不同。2、不可重复。3、放到HashSet集合中的元素实际上是放到HashMap集合的key部分了。*/for(String s : strs){System.out.println(s);}}
}

1.4.5 代码笔记_演示TreeSet集合特点

package com.bjpowernode.javase.collection;import java.util.Set;
import java.util.TreeSet;/*
TreeSet集合存储元素特点:1、无序不可重复的,但是存储的元素可以自动按照大小顺序排序!称为:可排序集合。2、无序:这里的无序指的是存进去的顺序和取出来的顺序不同。并且没有下标。*/
public class TreeSetTest01 {public static void main(String[] args) {// 创建集合对象Set<String> strs = new TreeSet<>();// 添加元素strs.add("A");strs.add("B");strs.add("Z");strs.add("Y");strs.add("Z");strs.add("K");strs.add("M");// 遍历/*ABKMYZ从小到大自动排序!*/for(String s : strs){System.out.println(s);}}
}

1.5 Map接口

1.5.1 代码笔记_Map接口常用方法

package com.bjpowernode.javase.collection;import java.util.Collection;
import java.util.HashMap;
import java.util.Map;/*
java.util.Map接口中常用的方法:1、Map和Collection没有继承关系。2、Map集合以key和value的方式存储数据:键值对key和value都是引用数据类型。key和value都是存储对象的内存地址。key起到主导的地位,value是key的一个附属品。3、Map接口中常用方法:V put(K key, V value) 向Map集合中添加键值对V get(Object key) 通过key获取valuevoid clear()    清空Map集合boolean containsKey(Object key) 判断Map中是否包含某个keyboolean containsValue(Object value) 判断Map中是否包含某个valueboolean isEmpty()   判断Map集合中元素个数是否为0V remove(Object key) 通过key删除键值对int size() 获取Map集合中键值对的个数。Collection<V> values() 获取Map集合中所有的value,返回一个CollectionSet<K> keySet() 获取Map集合所有的key(所有的键是一个set集合)Set<Map.Entry<K,V>> entrySet()将Map集合转换成Set集合假设现在有一个Map集合,如下所示:map1集合对象key             value----------------------------1               zhangsan2               lisi3               wangwu4               zhaoliuSet set = map1.entrySet();set集合对象1=zhangsan 【注意:Map集合通过entrySet()方法转换成的这个Set集合,Set集合中元素的类型是 Map.Entry<K,V>】2=lisi     【Map.Entry和String一样,都是一种类型的名字,只不过:Map.Entry是静态内部类,是Map中的静态内部类】3=wangwu4=zhaoliu ---> 这个东西是个什么?Map.Entry*/
public class MapTest01 {public static void main(String[] args) {// 创建Map集合对象Map<Integer, String> map = new HashMap<>();// 向Map集合中添加键值对map.put(1, "zhangsan"); // 1在这里进行了自动装箱。map.put(2, "lisi");map.put(3, "wangwu");map.put(4, "zhaoliu");// 通过key获取valueString value = map.get(2);System.out.println(value);// 获取键值对的数量System.out.println("键值对的数量:" + map.size());// 通过key删除key-valuemap.remove(2);System.out.println("键值对的数量:" + map.size());// 判断是否包含某个key// contains方法底层调用的都是equals进行比对的,所以自定义的类型需要重写equals方法。System.out.println(map.containsKey(new Integer(4))); // true// 判断是否包含某个valueSystem.out.println(map.containsValue(new String("wangwu"))); // true// 获取所有的valueCollection<String> values = map.values();// foreachfor(String s : values){System.out.println(s);}// 清空map集合map.clear();System.out.println("键值对的数量:" + map.size());// 判断是否为空System.out.println(map.isEmpty()); // true}
}

静态内部类: 

package com.bjpowernode.javase.collection;import java.util.HashSet;
import java.util.Set;public class MyClass {// 声明一个静态内部类private static class InnerClass {// 静态方法public static void m1(){System.out.println("静态内部类的m1方法执行");}// 实例方法public void m2(){System.out.println("静态内部类中的实例方法执行!");}}public static void main(String[] args) {// 类名叫做:MyClass.InnerClassMyClass.InnerClass.m1();// 创建静态内部类对象MyClass.InnerClass mi =  new MyClass.InnerClass();mi.m2();// 给一个Set集合// 该Set集合中存储的对象是:MyClass.InnerClass类型Set<MyClass.InnerClass> set = new HashSet<>();// 这个Set集合中存储的是字符串对象。Set<String> set2 = new HashSet<>();Set<MyMap.MyEntry<Integer, String>> set3 = new HashSet<>();}
}class MyMap {public static class MyEntry<K,V> {}
}

 1.5.2 代码笔记_遍历Map集合

第一种方式:获取所有的key,通过遍历key,来遍历value

第二种方式:Set<Map.Entry<K,V>> entrySet()

package com.bjpowernode.javase.collection;import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;/*
Map集合的遍历。【非常重要】*/
public class MapTest02 {public static void main(String[] args) {// 第一种方式:获取所有的key,通过遍历key,来遍历valueMap<Integer, String> map = new HashMap<>();map.put(1, "zhangsan");map.put(2, "lisi");map.put(3, "wangwu");map.put(4, "zhaoliu");// 遍历Map集合// 获取所有的key,所有的key是一个Set集合Set<Integer> keys = map.keySet();// 遍历key,通过key获取value// 迭代器可以/*Iterator<Integer> it = keys.iterator();while(it.hasNext()){// 取出其中一个keyInteger key = it.next();// 通过key获取valueString value = map.get(key);System.out.println(key + "=" + value);}*/// foreach也可以for(Integer key : keys){System.out.println(key + "=" + map.get(key));}// 第二种方式:Set<Map.Entry<K,V>> entrySet()// 以上这个方法是把Map集合直接全部转换成Set集合。// Set集合中元素的类型是:Map.EntrySet<Map.Entry<Integer,String>> set = map.entrySet();// 遍历Set集合,每一次取出一个Node// 迭代器/*Iterator<Map.Entry<Integer,String>> it2 = set.iterator();while(it2.hasNext()){Map.Entry<Integer,String> node = it2.next();Integer key = node.getKey();String value = node.getValue();System.out.println(key + "=" + value);}*/// foreach// 这种方式效率比较高,因为获取key和value都是直接从node对象中获取的属性值。// 这种方式比较适合于大数据量。for(Map.Entry<Integer,String> node : set){System.out.println(node.getKey() + "--->" + node.getValue());}}
}

 Map集合转换成Set集合entrySet()方法:

 1.5.3 代码笔记_哈希表数据结构

哈希表或者散列表数据结构:

为什么哈希表的随机增删以及查询效率都很高?

增删是在链表上完成;查询也不需要全部扫描,只需要部分扫描。

重点:通过讲解可以得出HashMap集合的key,会先后调用两个方法,一个方法是hashCode(),一个方法是equals(),那么这两个方法都需要重写

package com.bjpowernode.javase.collection;import java.util.HashMap;
import java.util.Map;
import java.util.Set;/*
HashMap集合:1、HashMap集合底层是哈希表/散列表的数据结构。2、哈希表是一个怎样的数据结构呢?哈希表是一个数组和单向链表的结合体。数组:在查询方面效率很高,随机增删方面效率很低。单向链表:在随机增删方面效率较高,在查询方面效率很低。哈希表将以上的两种数据结构融合在一起,充分发挥它们各自的优点。3、HashMap集合底层的源代码:public class HashMap{// HashMap底层实际上就是一个数组。(一维数组)Node<K,V>[] table;// 静态的内部类HashMap.Nodestatic class Node<K,V> {final int hash; // 哈希值(哈希值是key的hashCode()方法的执行结果。hash值通过哈希函数/算法,可以转换存储成数组的下标。)final K key; // 存储到Map集合中的那个keyV value; // 存储到Map集合中的那个valueNode<K,V> next; // 下一个节点的内存地址。}}哈希表/散列表:一维数组,这个数组中每一个元素是一个单向链表。(数组和链表的结合体。)4、最主要掌握的是:map.put(k,v)v = map.get(k)以上这两个方法的实现原理,是必须掌握的。5、HashMap集合的key部分特点:无序,不可重复。为什么无序? 因为不一定挂到哪个单向链表上。不可重复是怎么保证的? equals方法来保证HashMap集合的key不可重复。如果key重复了,value会覆盖。放在HashMap集合key部分的元素其实就是放到HashSet集合中了。所以HashSet集合中的元素也需要同时重写hashCode()+equals()方法。6、哈希表HashMap使用不当时无法发挥性能!假设将所有的hashCode()方法返回值固定为某个值,那么会导致底层哈希表变成了纯单向链表。这种情况我们称为:散列分布不均匀。什么是散列分布均匀?假设有100个元素,10个单向链表,那么每个单向链表上有10个节点,这是最好的,是散列分布均匀的。假设将所有的hashCode()方法返回值都设定为不一样的值,可以吗,有什么问题?不行,因为这样的话导致底层哈希表就成为一维数组了,没有链表的概念了。也是散列分布不均匀。散列分布均匀需要你重写hashCode()方法时有一定的技巧。7、重点:放在HashMap集合key部分的元素,以及放在HashSet集合中的元素,需要同时重写hashCode和equals方法。8、HashMap集合的默认初始化容量是16,默认加载因子是0.75这个默认加载因子是当HashMap集合底层数组的容量达到75%的时候,数组开始扩容。重点,记住:HashMap集合初始化容量必须是2的倍数,这也是官方推荐的,这是因为达到散列均匀,为了提高HashMap集合的存取效率,所必须的。*/
public class HashMapTest01 {public static void main(String[] args) {// 测试HashMap集合key部分的元素特点// Integer是key,它的hashCode和equals都重写了。Map<Integer,String> map = new HashMap<>();map.put(1111, "zhangsan");map.put(6666, "lisi");map.put(7777, "wangwu");map.put(2222, "zhaoliu");map.put(2222, "king"); //key重复的时候value会自动覆盖。System.out.println(map.size()); // 4// 遍历Map集合Set<Map.Entry<Integer,String>> set = map.entrySet();for(Map.Entry<Integer,String> entry : set){// 验证结果:HashMap集合key部分元素:无序不可重复。System.out.println(entry.getKey() + "=" + entry.getValue());}}
}

1.5.4 笔记代码_ 同时重写hashCode和equals

为什么放在HashMap集合key部分的元素需要重写equals方法呢?

equals如果不重写,调用的是Object类的equals(),默认比较的是两个对象的内存地址,我们应该比较内容。

终极结论:放在HashMap集合key部分的,以及放在HashSet集合中的元素,需要同时重写hashCode方法和equals方法。

package com.bjpowernode.javase.bean;import java.util.HashSet;
import java.util.Set;/*
1、向Map集合中存,以及从Map集合中取,都是先调用key的hashCode方法,然后再调用equals方法!
equals方法有可能调用,也有可能不调用。拿put(k,v)举例,什么时候equals不会调用?k.hashCode()方法返回哈希值,哈希值经过哈希算法转换成数组下标。数组下标位置上如果是null,equals不需要执行。拿get(k)举例,什么时候equals不会调用?k.hashCode()方法返回哈希值,哈希值经过哈希算法转换成数组下标。数组下标位置上如果是null,equals不需要执行。2、注意:如果一个类的equals方法重写了,那么hashCode()方法必须重写。
并且equals方法返回如果是true,hashCode()方法返回的值必须一样。equals方法返回true表示两个对象相同,在同一个单向链表上比较。那么对于同一个单向链表上的节点来说,他们的哈希值都是相同的。所以hashCode()方法的返回值也应该相同。3、hashCode()方法和equals()方法不用研究了,直接使用IDEA工具生成,但是这两个方法需要同时生成。4、终极结论:放在HashMap集合key部分的,以及放在HashSet集合中的元素,需要同时重写hashCode方法和equals方法。5、对于哈希表数据结构来说:如果o1和o2的hash值相同,一定是放到同一个单向链表上。当然如果o1和o2的hash值不同,但由于哈希算法执行结束之后转换的数组下标可能相同,此时会发生“哈希碰撞”。*/
public class HashMapTest02 {public static void main(String[] args) {Student s1 = new Student("zhangsan");Student s2 = new Student("zhangsan");// 重写equals方法之前是false//System.out.println(s1.equals(s2)); // false// 重写equals方法之后是trueSystem.out.println(s1.equals(s2)); //true (s1和s2表示相等)System.out.println("s1的hashCode=" + s1.hashCode()); //284720968 (重写hashCode之后-1432604525)System.out.println("s2的hashCode=" + s2.hashCode()); //122883338 (重写hashCode之后-1432604525)// s1.equals(s2)结果已经是true了,表示s1和s2是一样的,相同的,那么往HashSet集合中放的话,// 按说只能放进去1个。(HashSet集合特点:无序不可重复)Set<Student> students = new HashSet<>();students.add(s1);students.add(s2);System.out.println(students.size()); // 这个结果按说应该是1. 但是结果是2.显然不符合HashSet集合存储特点。怎么办?}
}

1.5.5 笔记代码_HashMap和Hashtable的区别

package com.bjpowernode.javase.bean;import java.util.HashMap;
import java.util.Map;/*
HashMap集合key部分允许null吗?允许但是要注意:HashMap集合的key null值只能有一个。有可能面试的时候遇到这样的问题。*/
public class HashMapTest03 {public static void main(String[] args) {Map map = new HashMap();// HashMap集合允许key为nullmap.put(null, null);System.out.println(map.size()); // 1// key重复的话value是覆盖!map.put(null, 100);System.out.println(map.size()); //1// 通过key获取valueSystem.out.println(map.get(null)); // 100}
}
package com.bjpowernode.javase.bean;import java.util.Hashtable;
import java.util.Map;/*
Hashtable的key可以为null吗?Hashtable的key和value都是不能为null的。HashMap集合的key和value都是可以为null的。Hashtable方法都带有synchronized:线程安全的。
线程安全有其它的方案,这个Hashtable对线程的处理
导致效率较低,使用较少了。Hashtable和HashMap一样,底层都是哈希表数据结构。
Hashtable的初始化容量是11,默认加载因子是:0.75f
Hashtable的扩容是:原容量 * 2 + 1*/
public class HashtableTest01 {public static void main(String[] args) {Map map = new Hashtable();//map.put(null, "123");map.put(100, null);}
}

 1.5.6 笔记代码_属性类Properties类

package com.bjpowernode.javase.collection;import java.util.Properties;/*
目前只需要掌握Properties属性类对象的相关方法即可。
Properties是一个Map集合,继承Hashtable,Properties的key和value都是String类型。
Properties被称为属性类对象。
Properties是线程安全的。*/
public class PropertiesTest01 {public static void main(String[] args) {// 创建一个Properties对象Properties pro = new Properties();// 需要掌握Properties的两个方法,一个存,一个取。pro.setProperty("url", "jdbc:mysql://localhost:3306/bjpowernode");pro.setProperty("driver","com.mysql.jdbc.Driver");pro.setProperty("username", "root");pro.setProperty("password", "123");// 通过key获取valueString url = pro.getProperty("url");String driver = pro.getProperty("driver");String username = pro.getProperty("username");String password = pro.getProperty("password");System.out.println(url);System.out.println(driver);System.out.println(username);System.out.println(password);}
}

1.5.7 笔记代码_TreeSet

TreeSet对String是可排序的:

package com.bjpowernode.javase.collection;import java.util.TreeSet;/*
1、TreeSet集合底层实际上是一个TreeMap
2、TreeMap集合底层是一个二叉树。
3、放到TreeSet集合中的元素,等同于放到TreeMap集合key部分了。
4、TreeSet集合中的元素:无序不可重复,但是可以按照元素的大小顺序自动排序。
称为:可排序集合。*/
public class TreeSetTest02 {public static void main(String[] args) {// 创建一个TreeSet集合TreeSet<String> ts = new TreeSet<>();// 添加Stringts.add("zhangsan");ts.add("lisi");ts.add("wangwu");ts.add("zhangsi");ts.add("wangliu");// 遍历for(String s : ts){// 按照字典顺序,升序!System.out.println(s);}TreeSet<Integer> ts2 = new TreeSet<>();ts2.add(100);ts2.add(200);ts2.add(900);ts2.add(800);ts2.add(600);ts2.add(10);for(Integer elt : ts2){// 升序!System.out.println(elt);}}
}/*
数据库中有很多数据:userid  name     birth-------------------------------------1       zs          1980-11-112       ls          1980-10-113       ww          1981-11-114       zl          1979-11-11编写程序从数据库当中取出数据,在页面展示用户信息的时候按照生日升序或者降序。这个时候可以使用TreeSet集合,因为TreeSet集合放进去,拿出来就是有顺序的。*/

TreeMap中put方法的源码(cpr != null通过实现比较器接口Comparator实现排序; cpr = null通过实现Comparable接口实现排序)

    public V put(K key, V value) {Entry<K,V> t = root;if (t == null) {compare(key, key); // type (and possibly null) checkroot = new Entry<>(key, value, null);size = 1;modCount++;return null;}int cmp;Entry<K,V> parent;// split comparator and comparable pathsComparator<? super K> cpr = comparator;// 有比较器,通过比较器排序if (cpr != null) {do {parent = t;cmp = cpr.compare(key, t.key);if (cmp < 0)t = t.left;else if (cmp > 0)t = t.right;elsereturn t.setValue(value);} while (t != null);}// 无比较器,通过实现Comparable接口排序else {if (key == null)throw new NullPointerException();@SuppressWarnings("unchecked")Comparable<? super K> k = (Comparable<? super K>) key;do {parent = t;cmp = k.compareTo(t.key);if (cmp < 0)t = t.left;else if (cmp > 0)t = t.right;elsereturn t.setValue(value);} while (t != null);}Entry<K,V> e = new Entry<>(key, value, parent);if (cmp < 0)parent.left = e;elseparent.right = e;fixAfterInsertion(e);size++;modCount++;return null;}

TreeSet无法对自定义类型排序(没有实现java.lang.Comparable接口)

package com.bjpowernode.javase.collection;import java.util.TreeSet;/*
对自定义的类型来说,TreeSet可以排序吗?以下程序中对于Person类型来说,无法排序。因为没有指定Person对象之间的比较规则。谁大谁小并没有说明啊。以下程序运行的时候出现了这个异常:java.lang.ClassCastException:class com.bjpowernode.javase.collection.Personcannot be cast to class java.lang.Comparable出现这个异常的原因是:Person类没有实现java.lang.Comparable接口。*/
public class TreeSetTest03 {public static void main(String[] args) {Person p1 = new Person(32);//System.out.println(p1);Person p2 = new Person(20);Person p3 = new Person(30);Person p4 = new Person(25);// 创建TreeSet集合TreeSet<Person> persons = new TreeSet<>();// 添加元素persons.add(p1);persons.add(p2);persons.add(p3);persons.add(p4);// 遍历for (Person p : persons){System.out.println(p);}}
}class Person {int age;public Person(int age){this.age = age;}// 重写toString()方法public String toString(){return "Person[age="+age+"]";}
}

TreeSet集合中元素可排序的第一种方式:自定义类型实现Comparable接口:

package com.bjpowernode.javase.collection;import java.util.TreeSet;public class TreeSetTest04 {public static void main(String[] args) {Customer c1 = new Customer(32);Customer c2 = new Customer(20);Customer c3 = new Customer(30);Customer c4 = new Customer(25);// 创建TreeSet集合TreeSet<Customer> customers = new TreeSet<>();// 添加元素customers.add(c1);customers.add(c2);customers.add(c3);customers.add(c4);// 遍历for (Customer c : customers){System.out.println(c);}}
}// 放在TreeSet集合中的元素需要实现java.lang.Comparable接口。
// 并且实现compareTo方法。equals可以不写。
class Customer implements Comparable<Customer>{int age;public Customer(int age){this.age = age;}// 需要在这个方法中编写比较的逻辑,或者说比较的规则,按照什么进行比较!// k.compareTo(t.key)// 拿着参数k和集合中的每一个k进行比较,返回值可能是>0 <0 =0// 比较规则最终还是由程序员指定的:例如按照年龄升序。或者按照年龄降序。@Overridepublic int compareTo(Customer c) { // c1.compareTo(c2);// this是c1// c是c2// c1和c2比较的时候,就是this和c比较。/*int age1 = this.age;int age2 = c.age;if(age1 == age2){return 0;} else if(age1 > age2) {return 1;} else {return -1;}*///return this.age - c.age; // =0 >0 <0return c.age - this.age;}public String toString(){return "Customer[age="+age+"]";}
}

 比较规则该怎么写:

package com.bjpowernode.javase.collection;import java.util.TreeSet;/*
先按照年龄升序,如果年龄一样的再按照姓名升序。*/
public class TreeSetTest05 {public static void main(String[] args) {TreeSet<Vip> vips = new TreeSet<>();vips.add(new Vip("zhangsi", 20));vips.add(new Vip("zhangsan", 20));vips.add(new Vip("king", 18));vips.add(new Vip("soft", 17));for(Vip vip : vips){System.out.println(vip);}}
}class Vip implements Comparable<Vip>{String name;int age;public Vip(String name, int age) {this.name = name;this.age = age;}@Overridepublic String toString() {return "Vip{" +"name='" + name + '\'' +", age=" + age +'}';}/*compareTo方法的返回值很重要:返回0表示相同,value会覆盖。返回>0,会继续在右子树上找。【10 - 9 = 1 ,1 > 0的说明左边这个数字比较大。所以在右子树上找。】返回<0,会继续在左子树上找。*/@Overridepublic int compareTo(Vip v) {// 写排序规则,按照什么进行比较。if(this.age == v.age){// 年龄相同时按照名字排序。// 姓名是String类型,可以直接比。调用compareTo来完成比较。return this.name.compareTo(v.name);} else {// 年龄不一样return this.age - v.age;}}
}

自平衡二叉树数据结构 : 

TreeSet集合中元素可排序的第二种方式:实现比较器接口Comparator:

package com.bjpowernode.javase.collection;import java.util.Comparator;
import java.util.TreeSet;/*
TreeSet集合中元素可排序的第二种方式:使用比较器的方式。
最终的结论:放到TreeSet或者TreeMap集合key部分的元素要想做到排序,包括两种方式:第一种:放在集合中的元素实现java.lang.Comparable接口。第二种:在构造TreeSet或者TreeMap集合的时候给它传一个比较器对象。
Comparable和Comparator怎么选择呢?当比较规则不会发生改变的时候,或者说当比较规则只有1个的时候,建议实现Comparable接口。如果比较规则有多个,并且需要多个比较规则之间频繁切换,建议使用Comparator接口。Comparator接口的设计符合OCP原则。*/
public class TreeSetTest06 {public static void main(String[] args) {// 创建TreeSet集合的时候,需要使用这个比较器。// TreeSet<WuGui> wuGuis = new TreeSet<>();//这样不行,没有通过构造方法传递一个比较器进去。// 给构造方法传递一个比较器。//TreeSet<WuGui> wuGuis = new TreeSet<>(new WuGuiComparator());// 大家可以使用匿名内部类的方式(这个类没有名字。直接new接口。)TreeSet<WuGui> wuGuis = new TreeSet<>(new Comparator<WuGui>() {@Overridepublic int compare(WuGui o1, WuGui o2) {return o1.age - o2.age;}});wuGuis.add(new WuGui(1000));wuGuis.add(new WuGui(800));wuGuis.add(new WuGui(810));for(WuGui wuGui : wuGuis){System.out.println(wuGui);}}
}// 乌龟
class WuGui{int age;public WuGui(int age){this.age = age;}@Overridepublic String toString() {return "小乌龟[" +"age=" + age +']';}
}// 单独在这里编写一个比较器
// 比较器实现java.util.Comparator接口。(Comparable是java.lang包下的。Comparator是java.util包下的。)
/*
class WuGuiComparator implements Comparator<WuGui> {@Overridepublic int compare(WuGui o1, WuGui o2) {// 指定比较规则// 按照年龄排序return o1.age - o2.age;}
}*/

1.6 Collections工具类

Collections位于java.util包中,提供了一系列实用的方法,如:对集合排序,对集合中的内容查找等

1.6.1 笔记代码_Collections工具类

package com.bjpowernode.javase.collection;import java.util.*;/*
java.util.Collection 集合接口
java.util.Collections 集合工具类,方便集合的操作。*/
public class CollectionsTest {public static void main(String[] args) {// ArrayList集合不是线程安全的。List<String> list = new ArrayList<>();// 变成线程安全的Collections.synchronizedList(list);// 排序list.add("abf");list.add("abx");list.add("abc");list.add("abe");Collections.sort(list);for(String s : list){System.out.println(s);}List<WuGui2> wuGuis = new ArrayList<>();wuGuis.add(new WuGui2(1000));wuGuis.add(new WuGui2(8000));wuGuis.add(new WuGui2(500));// 注意:对List集合中元素排序,需要保证List集合中的元素实现了:Comparable接口。Collections.sort(wuGuis);for(WuGui2 wg : wuGuis){System.out.println(wg);}// 对Set集合怎么排序呢?Set<String> set = new HashSet<>();set.add("king");set.add("kingsoft");set.add("king2");set.add("king1");// 将Set集合转换成List集合List<String> myList = new ArrayList<>(set);Collections.sort(myList);for(String s : myList) {System.out.println(s);}// 这种方式也可以排序。//Collections.sort(list集合, 比较器对象);}
}class WuGui2 implements Comparable<WuGui2>{int age;public WuGui2(int age){this.age = age;}@Overridepublic int compareTo(WuGui2 o) {return this.age - o.age;}@Overridepublic String toString() {return "WuGui2{" +"age=" + age +'}';}
}

1.7 泛型初步

泛型能更早的发现错误,如类型转换错误,通常在运行期才会发现,如果使用泛型,那么在编译期将会发现,通常错误发现的越早,越容易调试,越容易减少成本

1.7.1 代码笔记_泛型机制

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;/*
1、JDK5.0之后推出的新特性:泛型
2、泛型这种语法机制,只在程序编译阶段起作用,只是给编译器参考的。(运行阶段泛型没用!)
3、使用了泛型好处是什么?第一:集合中存储的元素类型统一了。第二:从集合中取出的元素类型是泛型指定的类型,不需要进行大量的“向下转型”!4、泛型的缺点是什么?导致集合中存储的元素缺乏多样性!大多数业务中,集合中元素的类型还是统一的。所以这种泛型特性被大家所认可。*/
public class GenericTest01 {public static void main(String[] args) {/*// 不使用泛型机制,分析程序存在缺点List myList = new ArrayList();// 准备对象Cat c = new Cat();Bird b = new Bird();// 将对象添加到集合当中myList.add(c);myList.add(b);// 遍历集合,取出每个Animal,让它moveIterator it = myList.iterator();while(it.hasNext()) {// 没有这个语法,通过迭代器取出的就是Object//Animal a = it.next();Object obj = it.next();//obj中没有move方法,无法调用,需要向下转型!if(obj instanceof Animal){Animal a = (Animal)obj;a.move();}}*/// 使用JDK5之后的泛型机制// 使用泛型List<Animal>之后,表示List集合中只允许存储Animal类型的数据。// 用泛型来指定集合中存储的数据类型。List<Animal> myList = new ArrayList<Animal>();// 指定List集合中只能存储Animal,那么存储String就编译报错了。// 这样用了泛型之后,集合中元素的数据类型更加统一了。//myList.add("abc");Cat c = new Cat();Bird b = new Bird();myList.add(c);myList.add(b);// 获取迭代器// 这个表示迭代器迭代的是Animal类型。Iterator<Animal> it = myList.iterator();while(it.hasNext()){// 使用泛型之后,每一次迭代返回的数据都是Animal类型。//Animal a = it.next();// 这里不需要进行强制类型转换了。直接调用。//a.move();// 调用子类型特有的方法还是需要向下转换的!Animal a = it.next();if(a instanceof Cat) {Cat x = (Cat)a;x.catchMouse();}if(a instanceof Bird) {Bird y = (Bird)a;y.fly();}}}
}class Animal {// 父类自带方法public void move(){System.out.println("动物在移动!");}
}class Cat extends Animal {// 特有方法public void catchMouse(){System.out.println("猫抓老鼠!");}
}class Bird extends Animal {// 特有方法public void fly(){System.out.println("鸟儿在飞翔!");}
}

1.7.2 代码笔记_自动类型推断机制(又称为钻石表达式)

package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;/*
JDK之后引入了:自动类型推断机制。(又称为钻石表达式)*/
public class GenericTest02 {public static void main(String[] args) {// ArrayList<这里的类型会自动推断>(),前提是JDK8之后才允许。// 自动类型推断,钻石表达式!List<Animal> myList = new ArrayList<>();myList.add(new Animal());myList.add(new Cat());myList.add(new Bird());// 遍历Iterator<Animal> it = myList.iterator();while(it.hasNext()){Animal a = it.next();a.move();}List<String> strList = new ArrayList<>();// 类型不匹配。//strList.add(new Cat());strList.add("http://www.126.com");strList.add("http://www.baidu.com");strList.add("http://www.bjpowernode.com");// 类型不匹配。//strList.add(123);//System.out.println(strList.size());// 遍历Iterator<String> it2 = strList.iterator();while(it2.hasNext()){// 如果没有使用泛型/*Object obj = it2.next();if(obj instanceof String){String ss = (String)obj;ss.substring(7);}*/// 直接通过迭代器获取了String类型的数据String s = it2.next();// 直接调用String类的substring方法截取字符串。String newString = s.substring(7);System.out.println(newString);}}
}

 1.7.3 代码笔记_自定义泛型

package com.bjpowernode.javase.collection;/*
自定义泛型可以吗?可以自定义泛型的时候,<> 尖括号中的是一个标识符,随便写。java源代码中经常出现的是:<E>和<T>E是Element单词首字母。T是Type单词首字母。*/
public class GenericTest03<标识符随便写> {public void doSome(标识符随便写 o){System.out.println(o);}public static void main(String[] args) {// new对象的时候指定了泛型是:String类型GenericTest03<String> gt = new GenericTest03<>();// 类型不匹配//gt.doSome(100);gt.doSome("abc");// =============================================================GenericTest03<Integer> gt2 = new GenericTest03<>();gt2.doSome(100);// 类型不匹配//gt2.doSome("abc");MyIterator<String> mi = new MyIterator<>();String s1 = mi.get();MyIterator<Animal> mi2 = new MyIterator<>();Animal a = mi2.get();// 不用泛型就是Object类型。/*GenericTest03 gt3 = new GenericTest03();gt3.doSome(new Object());*/}
}class MyIterator<T> {public T get(){return null;}
}

 1.7.4 代码笔记_foreach

package com.bjpowernode.javase.collection;/*
JDK5.0之后推出了一个新特性:叫做增强for循环,或者叫做foreach*/
public class ForEachTest01 {public static void main(String[] args) {// int类型数组int[] arr = {432,4,65,46,54,76,54};// 遍历数组(普通for循环)for(int i = 0; i < arr.length; i++) {System.out.println(arr[i]);}// 增强for(foreach)// 以下是语法/*for(元素类型 变量名 : 数组或集合){System.out.println(变量名);}*/System.out.println("======================================");// foreach有一个缺点:没有下标。在需要使用下标的循环中,不建议使用增强for循环。for(int data : arr) {// data就是数组中的元素(数组中的每一个元素。)System.out.println(data);}}
}
package com.bjpowernode.javase.collection;import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;/*
集合使用foreach*/
public class ForEachTest02 {public static void main(String[] args) {// 创建List集合List<String> strList = new ArrayList<>();// 添加元素strList.add("hello");strList.add("world!");strList.add("kitty!");// 遍历,使用迭代器方式Iterator<String> it = strList.iterator();while(it.hasNext()){String s = it.next();System.out.println(s);}// 使用下标方式(只针对于有下标的集合)for(int i = 0; i < strList.size(); i++){System.out.println(strList.get(i));}// 使用foreachfor(String s : strList){ // 因为泛型使用的是String类型,所以是:String sSystem.out.println(s);}List<Integer> list = new ArrayList<>();list.add(100);list.add(200);list.add(300);for(Integer i : list){ // i代表集合中的元素System.out.println(i);}}
}

2 集合主要掌握内容总结

2.1 List

package com.bjpowernode.javase.review;import java.util.ArrayList;
import java.util.Iterator;
import java.util.LinkedList;/*1.1、每个集合对象的创建(new)1.2、向集合中添加元素1.3、从集合中取出某个元素1.4、遍历集合*/
public class ArrayListTest {public static void main(String[] args) {// 创建集合对象//ArrayList<String> list = new ArrayList<>();LinkedList<String> list = new LinkedList<>();// 添加元素list.add("zhangsan");list.add("lisi");list.add("wangwu");// 从集合中取出某个元素// List集合有下标String firstElt = list.get(0);System.out.println(firstElt);// 遍历(下标方式)for(int i = 0; i < list.size(); i++){String elt = list.get(i);System.out.println(elt);}// 遍历(迭代器方式,这个是通用的,所有Collection都能用)Iterator<String> it = list.iterator();while(it.hasNext()){System.out.println(it.next());}// while循环修改为for循环/*for(Iterator<String> it2 = list.iterator(); it2.hasNext(); ){System.out.println("====>" + it2.next());}*/// 遍历(foreach方式)for(String s : list){System.out.println(s);}}
}

2.2 HashSet

package com.bjpowernode.javase.review;import java.util.HashSet;
import java.util.Iterator;
import java.util.Objects;
import java.util.Set;/*1.1、每个集合对象的创建(new)1.2、向集合中添加元素1.3、从集合中取出某个元素1.4、遍历集合1.5、测试HashSet集合的特点:无序不可重复。*/
public class HashSetTest {public static void main(String[] args) {// 创建集合对象HashSet<String> set = new HashSet<>();// 添加元素set.add("abc");set.add("def");set.add("king");// set集合中的元素不能通过下标取了。没有下标// 遍历集合(迭代器)Iterator<String> it = set.iterator();while(it.hasNext()){System.out.println(it.next());}// 遍历集合(foreach)for(String s : set){System.out.println(s);}set.add("king");set.add("king");set.add("king");System.out.println(set.size()); //3 (后面3个king都没有加进去。)set.add("1");set.add("10");set.add("2");for(String s : set){System.out.println("--->" + s);}// 创建Set集合,存储Student数据// (HashMap的key,存储在HashMap集合key的元素需要同时重写hashCode + equals)Set<Student> students = new HashSet<>();Student s1 = new Student(111, "zhangsan");Student s2 = new Student(222, "lisi");Student s3 = new Student(111, "zhangsan");students.add(s1);students.add(s2);students.add(s3);System.out.println(students.size()); // 2// 遍历for(Student stu : students){System.out.println(stu);}}
}class Student {int no;String name;public Student() {}public Student(int no, String name) {this.no = no;this.name = name;}@Overridepublic String toString() {return "Student{" +"no=" + no +", name='" + name + '\'' +'}';}@Overridepublic boolean equals(Object o) {if (this == o) return true;if (o == null || getClass() != o.getClass()) return false;Student student = (Student) o;return no == student.no &&Objects.equals(name, student.name);}@Overridepublic int hashCode() {return Objects.hash(no, name);}
}

2.3 TreeSet(TreeMap类似)

构造TreeSet集合时传入比较器可以改变排序规则

        // 编写比较器可以改变规则。
        TreeSet<Integer> ts = new TreeSet<>(new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o2 - o1; // 自动拆箱
            }
        });
这里Integer实现了Comparable接口,但是源码Integer被final修饰,无法被继承,所以无法重写compareTo方法。所以只能通过new TreeSet<>()时传入比较器的方式来改变比较规则,进而改变排序规则

package com.bjpowernode.javase.review;import java.util.Comparator;
import java.util.Iterator;
import java.util.TreeSet;/*1.1、每个集合对象的创建(new)1.2、向集合中添加元素1.3、从集合中取出某个元素1.4、遍历集合1.5、测试TreeSet集合中的元素是可排序的。1.6、测试TreeSet集合中存储的类型是自定义的。1.7、测试实现Comparable接口的方式1.8、测试实现Comparator接口的方式(最好测试以下匿名内部类的方式)*/
public class TreeSetTest {public static void main(String[] args) {// 集合的创建(可以测试以下TreeSet集合中存储String、Integer的。这些类都是SUN写好的。)//TreeSet<Integer> ts = new TreeSet<>();// 编写比较器可以改变规则。TreeSet<Integer> ts = new TreeSet<>(new Comparator<Integer>() {@Overridepublic int compare(Integer o1, Integer o2) {return o2 - o1; // 自动拆箱}});// 添加元素ts.add(1);ts.add(100);ts.add(10);ts.add(10);ts.add(10);ts.add(10);ts.add(0);// 遍历(迭代器方式)Iterator<Integer> it = ts.iterator();while(it.hasNext()) {Integer i = it.next();System.out.println(i);}// 遍历(foreach)for(Integer x : ts){System.out.println(x);}// TreeSet集合中存储自定义类型TreeSet<A> atree = new TreeSet<>();atree.add(new A(100));atree.add(new A(200));atree.add(new A(500));atree.add(new A(300));atree.add(new A(400));atree.add(new A(1000));// 遍历for(A a : atree){System.out.println(a);}//TreeSet<B> btree = new TreeSet<>(new BComparator());// 匿名内部类方式。TreeSet<B> btree = new TreeSet<>(new Comparator<B>() {@Overridepublic int compare(B o1, B o2) {return o1.i - o2.i;}});btree.add(new B(500));btree.add(new B(100));btree.add(new B(200));btree.add(new B(600));btree.add(new B(300));btree.add(new B(50));for(B b : btree){System.out.println(b);}}
}// 第一种方式:实现Comparable接口
class A implements Comparable<A>{int i;public A(int i){this.i = i;}@Overridepublic String toString() {return "A{" +"i=" + i +'}';}@Overridepublic int compareTo(A o) {//return this.i - o.i;return o.i - this.i;}
}class B {int i;public B(int i){this.i = i;}@Overridepublic String toString() {return "B{" +"i=" + i +'}';}
}// 比较器
class BComparator implements Comparator<B> {@Overridepublic int compare(B o1, B o2) {return o1.i - o2.i;}
}

2.4 HashMap

package com.bjpowernode.javase.review;import java.util.HashMap;
import java.util.Map;
import java.util.Set;/*1.1、每个集合对象的创建(new)1.2、向集合中添加元素1.3、从集合中取出某个元素1.4、遍历集合*/
public class HashMapTest {public static void main(String[] args) {// 创建Map集合Map<Integer, String> map = new HashMap<>();// 添加元素map.put(1, "zhangsan");map.put(9, "lisi");map.put(10, "wangwu");map.put(2, "king");map.put(2, "simth"); // key重复value会覆盖。// 获取元素个数System.out.println(map.size());// 取key是2的元素System.out.println(map.get(2)); // smith// 遍历Map集合很重要,几种方式都要会。// 第一种方式:先获取所有的key,遍历key的时候,通过key获取valueSet<Integer> keys = map.keySet();for(Integer key : keys){System.out.println(key + "=" + map.get(key));}// 第二种方式:是将Map集合转换成Set集合,Set集合中每一个元素是Node// 这个Node节点中有key和valueSet<Map.Entry<Integer,String>> nodes = map.entrySet();for(Map.Entry<Integer,String> node : nodes){System.out.println(node.getKey() + "=" + node.getValue());}}
}

2.5 Properties

package com.bjpowernode.javase.review;import java.util.Properties;public class PropertiesTest {public static void main(String[] args) {// 创建对象Properties pro = new Properties();// 存pro.setProperty("username", "test");pro.setProperty("password", "test123");// 取String username = pro.getProperty("username");String password = pro.getProperty("password");System.out.println(username);System.out.println(password);}
}

3 课堂笔记

day28课堂笔记

1、集合概述

1.1、什么是集合?有什么用?

数组其实就是一个集合。集合实际上就是一个容器。可以来容纳其它类型的数据。

集合为什么说在开发中使用较多?

集合是一个容器,是一个载体,可以一次容纳多个对象。在实际开发中,假设连接数据库,数据库当中有10条记录,那么假设把这10条记录查询出来,在java程序中会将10条数据封装成10个java对象,然后将10个java对象放到某一个集合当中,将集合传到前端,然后遍历集合,将一个数据一个数据展现出来。

1.2、集合不能直接存储基本数据类型,另外集合也不能直接存储java对象

集合当中存储的都是java对象的内存地址。(或者说集合中存储的是引用。)
list.add(100); //自动装箱Integer(集合中存储基本数据类型时,基本数据类型都会自动装箱成一个对象,集合中存储的即为该对象的内存地址)
  注意:
    集合在java中本身是一个容器,是一个对象。
    集合中任何时候存储的都是“引用”

集合中存储的是对象的内存地址:

1.3、在java中每一个不同的集合,底层会对应不同的数据结构。往不同的集合中存储元素,等于将数据放到了不同的数据结构当中。

什么是数据结构?数据存储的结构就是数据结构。不同的数据结构,数据存储方式不同。例如:数组、二叉树、链表、哈希表......以上这些都是常见的数据结构。

  你往集合c1中放数据,可能是放到数组上了。
  你往集合c2中放数据,可能是放到二叉树上了。
  .....
  你使用不同的集合等同于使用了不同的数据结构。

  你在java集合这一章节,你需要掌握的不是精通数据结构。java中已经将数据结构
  实现了,已经写好了这些常用的集合类,你只需要掌握怎么用?在什么情况下选择
  哪一种合适的集合去使用即可。

  new ArrayList(); 创建一个集合,底层是数组。
  new LinkedList(); 创建一个集合对象,底层是链表。
  new TreeSet(); 创建一个集合对象,底层是二叉树。
  .....

1.4、集合在java JDK中哪个包下?
java.util.*;
所有的集合类和集合接口都在java.util包下。    
1.5、为了让大家掌握集合这块的内容,最好能将集合的继承结构图背会!!!
集合整个这个体系是怎样的一个结构,你需要有印象。

1.6、在java中集合分为两大类:
一类是单个方式存储元素:
  单个方式存储元素,这一类集合中超级父接口:java.util.Collection;

一类是以键值对儿的方式存储元素:
  以键值对的方式存储元素,这一类集合中超级父接口:java.util.Map;

2、总结重点:

第一个重点:把集合继承结构图背会。

第二个重点:把Collection接口中常用方法测试几遍。

第三个重点:把迭代器弄明白。

第四个重点:Collection接口中的remove方法和contains方法底层都会调用equals,这个弄明白。

day29课堂笔记

1、List接口中的常用方法。
    List是Collection接口的子接口。所以List接口中有一些特有的方法。
        void add(int index, Object element)
        Object set(int index, Object element)
        Object get(int index)
        int indexOf(Object o)
        int lastIndexOf(Object o)
        Object remove(int index)
2、迭代器迭代元素的过程中不能使用集合对象的remove方法删除元素,
要使用迭代器Iterator的remove方法来删除元素,防止出现异常:
    ConcurrentModificationException

3、ArrayList
    ArrayList集合初始化容量10
    扩容为原容量1.5倍。
    底层是数组。

    数组优点和缺点要能够说出来!
    另外要注意:ArrayList集合末尾增删元素效率还是可以的。

4、链表数据结构
    第一:单向链表和双向链表数据结构要理解。
    第二:链表数据结构的优点和缺点要能够说出来。

5、Vector:
    Vector初始化容量是10.
    扩容为原容量的2倍。
    底层是数组。
    Vector底层是线程安全的。

    怎么得到一个线程安全的List:
        Collections.synchronizedList(list);

6、JDK5.0新特性:泛型
    第一:集合使用泛型来减少向下转型的操作。
    第二:怎么使用泛型?
    第三:怎么自定义泛型?

7、JDK5.0新特性:
    foreach
    对数组怎么遍历?
        for(int i : arr){
            System.out.println(i);
        }
    对集合怎么遍历?
        for(String s : list){
            System.out.println(s);
        }

8、JDK8新特性:钻石表达式
    List<String> list = new ArrayList<>();
    类型自动推断!

day30课堂笔记

1、掌握Map接口中常用方法。

2、遍历Map集合的两种方式都要精通。
    第一种:获取所有key,遍历每个key,通过key获取value.
    第二种:获取Set<Map.Entry>即可,遍历Set集合中的Entry
        调用entry.getKey() entry.getValue()

3、了解哈希表数据结构。

4、存放在HashMap集合key部分和HashSet集合中的元素需要同时重写hashCode和equals。

5、HashMap和Hashtable的区别。
    HashMap:
        初始化容量16,扩容2倍。
        非线程安全
        key和value可以为null。

    Hashtable
        初始化容量11,扩容2倍+1
        线程安全
        key和value都不能是null。

6、Properties类的常用两个方法。
    setProperty
    getProperty

7、了解自平衡二叉树数据结构。
    左小右大原则存储。
    中序遍历方式。

8、TreeMap的key或者TreeSet集合中的元素要想排序,有两种实现方式:
    第一种:实现java.lang.Comparable接口。
    第二种:单独编写一个比较器Comparator接口。

9、集合工具类Collections:
    synchronizedList方法
    sort方法(要求集合中元素实现Comparable接口。)

day31课堂笔记

1、集合这块最主要掌握什么内容?
    1.1、每个集合对象的创建(new)
    1.2、向集合中添加元素
    1.3、从集合中取出某个元素
    1.4、遍历集合
    1.5、主要的集合类:
        ArrayList
        LinkedList
        HashSet (HashMap的key,存储在HashMap集合key的元素需要同时重写hashCode + equals)
        TreeSet
        HashMap
        Properties
        TreeMap

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/817245.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

英特尔推出中国特供版Gaudi 3芯片,性能暴降92%以应对美国出口管制|TodayAI

英特尔近期发布消息&#xff0c;其将在中国市场推出专为该地区定制的“特供版”Gaudi 3 AI芯片&#xff0c;以符合美国对AI芯片的出口管制。这一版本包括HL-328型号的OAM兼容夹层卡&#xff0c;预计将于6月24日发布&#xff1b;以及HL-388型号的PCIe加速卡&#xff0c;计划在9月…

从建表语句带你学习doris_表索引

1、doris建表概述 1.1、doris建表模板 CREATE [EXTERNAL] TABLE [IF NOT EXISTS] [DATABASE.]table_name (column_definition1[,column_deinition2,......][,index_definition1,[,index_definition2,]] ) [ENGINE [olap|mysql|broker|hive]] [key_desc] [COMMENT "tabl…

不花一分钱,四大方法教你免费申请SSL证书

在数字化时代&#xff0c;数据安全与隐私保护的重要性日益凸显。为了确保在线信息传输的机密性和完整性&#xff0c;数字证书&#xff0c;尤其是SSL/TLS证书扮演着至关重要的角色。为个人及企业用户提供了经济、高效的加密解决方案。随着市场对SSL证书的逐渐重视&#xff0c;免…

docker run启动一个开发备忘清单速查表 —— 筑梦之路

docker run -itd --name reference -p 3000:3000 registry.cn-beijing.aliyuncs.com/deanmr/reference:latest包含&#xff1a;运维&#xff0c;前端&#xff0c;后端&#xff0c;工具&#xff0c;命令&#xff0c;数据库 部分截图展示&#xff1a;

【Bugku】sqli-0x1

1.打开靶场&#xff0c;进入实验场景 2.按F12查看源代码&#xff0c;发现有一个/?pls_help路径&#xff0c;在url后加上查看。 3.得到的php源码 首先&#xff0c;代码通过 error_reporting(0) 和 error_log(0) 关闭了错误报告&#xff0c;这可以防止攻击者从错误信息中获取敏…

设计模式之大话西游

8年前深究设计模式&#xff0c;现如今再次回锅&#xff5e; 还是大话设计模式 这本书还是可以的 大话西游经典的台词&#xff1a;“曾经有一份真挚的爱情摆在我面前,我没有珍惜,等我失去的时候,我才后悔莫及,人世间最痛苦的事莫过于此。如果上天能够给我一个再来一次的机会,我会…

【每周精选资讯 | 第 5 期】2024-04-08 ~ 2024-04-14

文章目录 前言内容百度智能云发布干帆大模型一体机&#xff0c;加速企业私有化部署大模型北大开源 aiXcoder-7B 代码大模型&#xff0c;专为企业私有部署设计WPS AI 企业版发布&#xff1a;多个大模型自由切换英特尔发布 AI 芯片 Gaudi 3&#xff0c;称性能超越英伟达 H100马云…

国外站群服务器有哪几种?

国外站群服务器种类繁多&#xff0c;它们各具特色&#xff0c;适用于不同的业务需求和场景。以下将为您科普几种常见的国外站群服务器及其特点。 首先&#xff0c;美国站群服务器以其丰富的IP资源和强大的网络技术著称。作为全球网络技术和数据中心发展的领先者&#xff0c;美国…

数据库SQL语言实战(二)

目录 检索查询 题目一 题目二 题目三 题目四 题目五 题目六 题目七 题目八 题目九&#xff08;本篇最难的题目&#xff09; 分析 实现&#xff08;两种方式&#xff09; 模板 总结 检索查询 按照要求查找数据库中的数据 题目一 找出没有选修任何课程的学…

YesPMP众包平台 | 活动有礼,现金奖励点击领取!

YesPMP众包平台在线发福利啦&#xff0c;4月16日活动火热开启&#xff0c;现金奖励等你来领&#xff0c;最高可领千元&#xff0c;赶快参与将奖励收入囊中&#xff0c;一起来了解活动细节吧&#xff01; 一、活动内容&#xff1a; 活动一&#xff1a;【项目征集令】活动&…

Python Flask-Security- 构建安全而强大的Web应用

Flask-Security是一个基于Flask的安全扩展&#xff0c;为开发者提供了构建安全且强大的Web应用的工具。本文将深入探讨Flask- Security的核心功能、基本用法以及在实际应用中的一些高级特性&#xff0c;通过丰富的示例代码&#xff0c;助您更全面地了解和应用这一用于Web应用安…

2024年【危险化学品经营单位主要负责人】考试题库及危险化学品经营单位主要负责人新版试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 危险化学品经营单位主要负责人考试题库是安全生产模拟考试一点通总题库中生成的一套危险化学品经营单位主要负责人新版试题&#xff0c;安全生产模拟考试一点通上危险化学品经营单位主要负责人作业手机同步练习。2024…

python+playwright 学习-88 禁止加载图片等资源

前言 对于爬虫的小伙伴来说,有时候只需抓取页面的文本,不用加载图片,可以加快操作页面速度,那么我们可以设置禁止加载图片等资源。 禁止图片加载 根据url地址的后缀,图片资源后缀一般是png,jpg,jpeg,gif等格式。 from playwright.sync_api import sync_playwrightwith…

Gradle 实战 - 插件-ApiHug准备-工具篇-015

&#x1f917; ApiHug {Postman|Swagger|Api...} 快↑ 准√ 省↓ GitHub - apihug/apihug.com: All abou the Apihug apihug.com: 有爱&#xff0c;有温度&#xff0c;有质量&#xff0c;有信任ApiHug - API design Copilot - IntelliJ IDEs Plugin | Marketplace ApiHug …

rv1103/buildroot系统中添加包如v4l2

v4l2: rv1103给出的包中已经有v4l,只需要在menuconfig中打开编译选项&#xff0c;步骤如下&#xff1a; 在luckfox的github网站中下载的源代码在~/linux/luckfox/luckfox-pico-main中目录结构如下&#xff1a; 打开编译选项 cd ./sysdrv/source/buildroot/buildroot-2023.02.…

一种驱动器的功能安全架构介绍

下图提供了驱动器实现安全功能的架构 具有如下特点&#xff1a; 1.通用基于总线或者非总线的架构。可以实现ethercat的FSOE&#xff0c;profinet的profisafe&#xff0c;或者伺服本体安全DIO现实安全功能。 2.基于1oo2D架构&#xff0c;安全等级可以达到sil3。 3.高可用性。单…

第十五篇【传奇开心果系列】Python自动化办公库技术点案例示例:深度解读Python 自动化处理图像在各行各业的应用场景

传奇开心果博文系列 系列博文目录Python自动化办公库技术点案例示例系列 博文目录前言一、行业应用场景介绍二、 **计算机视觉研究与开发示例代码**三、人工智能与机器学习示例代码四、医疗健康领域示例代码五、制造业与质量控制示例代码六、农业与环境科学示例代码七、电子商务…

小米汽车值得去吗?最终拒了 offer。

车企选择 今天逛某职场 App 时&#xff0c;无意间看到一篇寻求 offer 抉择意见的帖子&#xff1a; 这位同学刚从加班闻名&#xff08;但 CEO 强调既学华为狼性&#xff0c;也学华为分配&#xff09;的理想汽车离职。 经过了 6 轮面试&#xff0c;收到了小米 offer&#xff0c;但…

【合合TextIn】智能文档处理系列—电子文档解析技术全格式解析

一、引言 在当今的数字化时代&#xff0c;电子文档已成为信息存储和交流的基石。从简单的文本文件到复杂的演示文档&#xff0c;各种格式的电子文档承载着丰富的知识与信息&#xff0c;支撑着教育、科研、商业和日常生活的各个方面。随着信息量的爆炸性增长&#xff0c;如何高…

网络网络层之(2)ARP协议

网络网络层之(2)ARP协议 Author&#xff1a;Once Day Date: 2024年4月1日 漫漫长路&#xff0c;有人对你笑过嘛… 全系列文档可参考专栏&#xff1a;通信网络技术_Once-Day的博客-CSDN博客。 参考文档: 《TCP/IP详解卷一》arp(8) - Linux manual page (man7.org)彻底搞懂系…