首页 > 编程语言 > 详细

JAVASE-DAY10&DAY11

时间:2020-02-25 22:24:58      阅读:92      评论:0      收藏:0      [点我收藏+]

DAY10&DAY11

1.容器

一般通过“容器”来容纳和管理数据。

数组就是一种容器,可以在其中放置对象或基本类型数据。

数组的优势:是一种简单的线性序列,可以快速地访问数组元素,效率高。如果从效率和类型检查的角度讲,数组是最好的。

数组的劣势:不灵活。容量需要事先定义好,不能随着需求的变化而扩容。

基于数组并不能满足我们对于“管理和组织数据的需求”,所以我们需要一种更强大、更灵活、容量随时可扩的容器来装载我们的对象。

容器的接口层次结构图:

技术分享图片

2.泛型

泛型是JDK1.5以后增加的,它可以帮助我们建立类型安全的集合。在使用了泛型的集合中,遍历时不必进行强制类型转换。JDK提供了支持泛型的编译器,将运行时的类型检查提前到了编译时执行,提高了代码可读性和安全性。

泛型的本质就是“数据类型的参数化”。 我们可以把“泛型”理解为数据类型的一个占位符(形式参数),即告诉编译器,在调用泛型时必须传入实际类型。

可以在类的声明处增加泛型列表,如:<T,E,V>。

此处,字符可以是任何标识符,一般采用这3个字母。

泛型E像一个占位符一样表示“未知的某个数据类型”,我们在真正调用的时候传入这个“数据类型”。

容器相关类都定义了泛型,我们在开发和工作中,在使用容器类时都要使用泛型。这样,在容器的存储数据、读取数据时都避免了大量的类型判断,非常便捷。

通过阅读源码,我们发现Collection、List、Set、Map、Iterator接口都定义了泛型。因此,我们在使用这些接口及其实现类时,都要使用泛型。

技术分享图片

注意:我们只是强烈建议使用泛型。事实上,不使用编译器也不会报错!

3.Collection接口

Collection 表示一组对象,它是集中、收集的意思。Collection接口的两个子接口是List、Set接口。

Collection接口中定义的方法:

技术分享图片

由于List、Set是Collection的子接口,意味着所有List、Set的实现类都有上面的方法。

4 List特点和常用方法

List是有序、可重复的容器。

有序:List中每个元素都有索引标记。可以根据元素的索引标记(在List中的位置)访问元素,从而精确控制这些元素。

可重复:List允许加入重复的元素。更确切地讲,List通常允许满足 e1.equals(e2) 的元素重复加入容器。

除了Collection接口中的方法,List多了一些跟顺序(索引)有关的方法,参见下表:

技术分享图片

List接口常用的实现类有3个:ArrayList、LinkedList和Vector。

4.1 ArrayList特点和底层实现

ArrayList底层是用数组实现的存储。

特点:查询效率高,增删效率低,线程不安全。我们一般使用它。

查看源码:

技术分享图片

ArrayList底层使用Object数组来存储元素数据。所有的方法,都围绕这个核心的Object数组来开展。

我们知道,数组长度是有限的,而ArrayList是可以存放任意数量的对象,长度不受限制,那么它是怎么实现的呢?本质上就是通过定义新的更大的数组,将旧数组中的内容拷贝到新数组,来实现扩容。

ArrayList的Object数组初始化长度为10,如果我们存储满了这个数组,需要存储第11个对象,就会定义新的长度更大的数组,并将原数组内容和新的元素一起加入到新数组中。新的数组长度一般为原数组的1.5倍。

源码如下:

技术分享图片

4.2 LinkedList特点和底层实现

LinkedList底层用双向链表实现的存储。

特点:查询效率低,增删效率高,线程不安全。

双向链表也叫双链表,是链表的一种,它的每个数据节点中都有两个指针,分别指向前一个节点和后一个节点。 所以,从双向链表中的任意一个节点开始,都可以很方便地找到所有节点。

技术分享图片

每个节点都应该有3部分内容:

class  Node {
       Node  previous;     //前一个节点
       Object  element;    //本节点保存的数据
       Node  next;         //后一个节点
}

查看LinkedList的源码,可以看到里面包含了双向链表的相关代码:

技术分享图片

注意事项:

entry在英文中表示“进入、词条、条目”的意思。在计算机英语中一般表示“项、条目”的含义。

4.3 Vector接口

Vector底层是用数组实现的List,相关的方法都加了同步检查,因此“线程安全,效率低”。

比如,indexOf方法就增加了synchronized同步标记。

技术分享图片

如何选用ArrayList、LinkedList、Vector?

  1. 需要线程安全时,用Vector。

  2. 不存在线程安全问题时,并且查找较多用ArrayList(一般使用它)。

  3. 不存在线程安全问题时,增加或删除元素较多用LinkedList。

5.Map接口

Map就是用来存储“键(key)-值(value) 对”的。 Map类中存储的“键值对”通过键来标识,所以“键对象”不能重复。

Map 接口的实现类有HashMap、TreeMap、HashTable、Properties等。

Map接口中常用的方法:

技术分享图片

5.1 HashMap和HashTable

HashMap采用哈希算法实现,是Map接口最常用的实现类。由于底层采用了哈希表存储数据,我们要求键不能重复,如果发生重复,新的键值对会替换旧的键值对。HashMap在查找、删除、修改方面都有非常高的效率。

HashTable类和HashMap用法几乎一样,底层实现几乎一样,只不过HashTable的方法添加了synchronized关键字确保线程同步检查,效率较低。

HashMap与HashTable的区别

  1. HashMap: 线程不安全,效率高。允许key或value为null。

  2. HashTable: 线程安全,效率低。不允许key或value为null。

5.2 HsahMap底层实现详解

数据结构中由数组和链表来实现对数据的存储,他们各有特点。

(1) 数组:占用空间连续。寻址容易,查询速度快。但是,增加和删除效率非常低。

(2) 链表:占用空间不连续。寻址困难,查询速度慢。但是,增加和删除效率非常高。

“哈希表”结合数组和链表的优点(即查询快,增删效率也高)。哈希表的本质就是“数组+链表”。

5.2.1 HashMap基本结构

哈希表的基本结构就是“数组+链表”。

打开HashMap源码,发现有如下两个核心内容:

技术分享图片

其中的Entry[] table 就是HashMap的核心数组结构,我们也称之为“位桶数组”。Entry的源码如下:

技术分享图片

一个Entry对象存储了:

  1. key:键对象

  2. value:值对象

  3. next:下一个节点

  4. hash:键对象的hash值

显然每一个Entry对象就是一个单向链表结构。

Entry对象存储结构图:

技术分享图片

Entry数组存储结构图(HashMap结构图):

技术分享图片

5.2.2 存储数据过程put(key,value)

此处的核心是如何产生hash值,该值用来对应数组的存储位置。

HashMap存储数据过程示意图:

技术分享图片

我们的目的是将”key-value两个对象”成对存放到HashMap的Entry[]数组中。参见以下步骤:

  1. 获得key对象的hashcode

    首先调用key对象的hashcode()方法,获得hashcode。

  2. 根据hashcode计算出hash值(要求在[0, 数组长度-1]区间)

    hashcode是一个整数,我们需要将它转化成[0, 数组长度-1]的范围。我们要求转化后的hash值尽量均匀地分布在[0,数组长度-1]这个区间,减少“hash冲突”。

    • 一种极端简单和低下的算法是:

      hash值 = hashcode/hashcode;

      也就是说,hash值总是1。意味着,键值对对象都会存储到数组索引1位置,这样就形成一个非常长的链表。相当于每存储一个对象都会发生“hash冲突”,HashMap也退化成了一个“链表”。

    • 一种简单和常用的算法是(相除取余算法):

      hash值 = hashcode%数组长度;

      这种算法可以让hash值均匀的分布在[0,数组长度-1]的区间。早期的HashTable就是采用这种算法。但是,这种算法由于使用了“除法”,效率低下。JDK后来改进了算法。首先约定数组长度必须为2的整数幂,这样采用位运算即可实现取余的效果:hash值 = hashcode&(数组长度-1)。

  3. 生成Entry对象

    如上所述,一个Entry对象包含4部分:key对象、value对象、hash值、指向下一个Entry对象的引用。我们现在算出了hash值。下一个Entry对象的引用为null。

  4. 将Entry对象放到table数组中

    如果本Entry对象对应的数组索引位置还没有放Entry对象,则直接将Entry对象存储进数组;如果对应索引位置已经有Entry对象,则将已有Entry对象的next指向本Entry对象,形成链表。

    总结如上过程:

    当添加一个元素(key-value)时,首先计算key的hash值,以此确定插入数组中的位置,但是可能存在同一hash值的元素已经被放在数组同一位置了,这时就添加到同一hash值的元素的后面,他们在数组的同一位置,就形成了链表,同一个链表上的Hash值是相同的,所以说数组存放的是链表。JDK8中,当链表长度大于8时,链表就转换为红黑树,这样又大大提高了查找的效率。

    5.2.3 取数据过程get(key)

我们需要通过key对象获得“键值对”对象,进而返回value对象。明白了存储数据过程,取数据就比较简单了,参见以下步骤:

  1. 获得key的hashcode,通过hash()散列算法得到hash值,进而定位到数组的位置。

  2. 在链表上挨个比较key对象。调用equals()方法,将key对象和链表上所有节点的key对象进行比较,直到碰到返回true的节点对象为止。

  3. 返回equals()为true的节点对象的value对象。

hashcode()和equals()方法的关系:

Java中规定,两个内容相同(equals()为true)的对象必须具有相等的hashCode。因为如果equals()为true而两个对象的hashcode不同,那在整个存储过程中就发生了悖论。

5.2.4 扩容问题

HashMap的位桶数组,初始大小为16。实际使用时,显然大小是可变的。如果位桶数组中的元素达到(0.75*数组 length),就重新调整数组大小变为原来2倍大小。

扩容很耗时。扩容的本质是定义新的更大的数组,并将旧数组内容挨个拷贝到新数组中。

5.2.5 JDK8将链表在大于8情况下变为红黑二叉树

JDK8中,HashMap在存储一个元素时,当对应链表长度大于8时,链表就转换为红黑树,这样又大大提高了查找的效率。

5.3 二叉树和红黑二叉树

不太懂==以后专门学数据结构再看

平衡二叉树(AVL)

在平衡二叉树中任何节点的两个子树的高度最大差别为1,所以它也被称为高度平衡树。 增加和删除节点可能需要通过一次或多次树旋转来重新平衡这个树。

节点的平衡因子是它的左子树的高度减去它的右子树的高度(有时相反)。带有平衡因子1、0或 -1的节点被认为是平衡的。带有平衡因子 -2或2的节点被认为是不平衡的,并需要重新平衡这个树。

比如,我们存储排好序的数据【3,4,8,12,13,14,16,23】,增加节点如果出现不平衡,则通过节点的左旋或右旋,重新平衡树结构,最终平衡二叉树如下图所示:

技术分享图片

平衡二叉树追求绝对平衡,实现起来比较麻烦,每次插入新节点需要做的旋转操作次数不能预知。

红黑二叉树

红黑二叉树(简称:红黑树),它首先是一棵二叉树,同时也是一棵自平衡的排序二叉树。

红黑树在原有的排序二叉树增加了如下几个要求:

  1. 每个节点要么是红色,要么是黑色。

  2. 根节点永远是黑色的。

  3. 所有的叶节点都是空节点(即 null),并且是黑色的。

  4. 每个红色节点的两个子节点都是黑色。(从每个叶子到根的路径上不会有两个连续的红色节点)

  5. 从任一节点到其子树中每个叶子节点的路径都包含相同数量的黑色节点。

    这些约束强化了红黑树的关键性质:从根到叶子的最长的可能路径不多于最短的可能路径的两倍长。这样就让树大致上是平衡的。

    红黑树是一个更高效的检索二叉树,JDK 提供的集合类 TreeMap、TreeSet 本身就是一个红黑树的实现。

技术分享图片

红黑树的基本操作:插入、删除、左旋、右旋、着色。 每插入或者删除一个节点,可能会导致树不在符合红黑树的特征,需要进行修复,进行 “左旋、右旋、着色”操作,使树继续保持红黑树的特性。

5.4 TreeMap的使用和底层实现

TreeMap是红黑二叉树的典型实现。打开TreeMap的源码,发现里面有一行核心代码:

private transient Entry<K,V> root = null;

root用来存储整个树的根节点。Entry(TreeMap的内部类)的底层源码:

技术分享图片

可以看到里面存储了本身数据、左节点、右节点、父节点、以及节点颜色。TreeMap的put()/remove()方法大量使用了红黑树的理论。

TreeMap和HashMap实现了同样的接口Map,因此,用法对于调用者来说没有区别。HashMap效率高于TreeMap;在需要排序的Map时才选用TreeMap。

6.Set接口

Set接口继承自Collection,Set接口中没有新增方法,方法和Collection保持完全一致。

Set容器特点:无序、不可重复。无序指Set中的元素没有索引,只能遍历查找;不可重复指不允许加入重复的元素。更确切地讲,新元素如果和Set中某个元素通过equals()方法对比为true,则不能加入;甚至,Set中也只能放入一个null元素,不能多个。

Set常用的实现类有:HashSet、TreeSet等,一般使用HashSet。

6.1 HashSet底层实现

HashSet是采用哈希算法实现,底层实际是用HashMap实现的(HashSet本质就是一个简化版的HashMap),因此,查询效率和增删效率都比较高。

HashSet的底层源码:

技术分享图片

我们发现里面有个map属性,这就是HashSet的核心秘密。

再看add()方法,发现增加一个元素就是在map中增加一个键值对,键对象就是这个元素,值对象是名为PRESENT的Object对象。说白了,就是“往set中加入元素,本质就是把这个元素作为key加入到了内部的map中”。

由于map中key都是不可重复的,因此,Set天然具有“不可重复”的特性。

6.2 TreeSet的使用和底层实现

TreeSet底层实际是用TreeMap实现的,内部维持了一个简化版的TreeMap,通过key来存储Set的元素。 TreeSet内部需要对存储的元素进行排序,因此,我们对应的类需要实现Comparable接口。这样,才能根据compareTo()方法比较对象之间的大小,才能进行内部排序。

【示例9-10】TreeSet和Comparable接口的使用

public class Test {
   public static void main(String[] args) {
       User u1 = new User(1001, "高淇", 18);
       User u2 = new User(2001, "高希希", 5);
       Set<User> set = new TreeSet<User>();
       set.add(u1);
       set.add(u2);
  }
}

class User implements Comparable<User> {
   int id;
   String uname;
   int age;

   public User(int id, String uname, int age) {
       this.id = id;
       this.uname = uname;
       this.age = age;
  }
   /**
    * 返回0 表示 this == obj 返回正数表示 this > obj 返回负数表示 this < obj
    */
   @Override
   public int compareTo(User o) {
       if (this.id > o.id) {
           return 1;
      } else if (this.id < o.id) {
           return -1;
      } else {
           return 0;
      }
  }
}

使用TreeSet要点:

  1. 由于是二叉树,需要对元素做内部排序。如果要放入TreeSet中的类没有实现Comparable接口,则会抛出异常:java.lang.ClassCastException。

  1. TreeSet中不能放入null元素。

7.迭代器Iterator

迭代器是遍历集合的一种方式,是依赖于集合而存在的。

Iterator iterator()         //迭代器,集合的专用遍历方式
Object next() //获取元素,并移动到下一个位置
NoSuchElementException //没有这样的元素,因为已经到最后了
boolean hasNext() //如果仍有元素可以迭代,则返回true
Collection c = new ArrayList();
c.add("hello");
c.add("world");
Iterator it = c.iterator();
while(it.hasNext()){
   String s = it.next();
   System.out.printlin(s);
}

8.遍历集合的方法总结

遍历List方法一:普通for循环

for(int i=0;i<list.size();i++){//list为集合的对象名
   String temp = (String)list.get(i);
   System.out.println(temp);
}

遍历List方法二:增强for循环(使用泛型!)

for (String temp : list) {
System.out.println(temp);
}

遍历List方法三:使用Iterator迭代器(1)

for(Iterator iter= list.iterator();iter.hasNext();){
   String temp = (String)iter.next();
   System.out.println(temp);
}

遍历List方法四:使用Iterator迭代器(2)

Iterator  iter =list.iterator();
while(iter.hasNext()){
   Object  obj =  iter.next();
   iter.remove();//如果要遍历时,删除集合中的元素,建议使用这种方式!
   System.out.println(obj);
}

遍历Set方法一:增强for循环

for(String temp:set){
System.out.println(temp);
}

遍历Set方法二:使用Iterator迭代器

for(Iterator iter = set.iterator();iter.hasNext();){
   String temp = (String)iter.next();
   System.out.println(temp);
}

遍历Map方法一:根据key获取value

Map<Integer, Man> maps = new HashMap<Integer, Man>();
Set<Integer>  keySet =  maps.keySet();
for(Integer id : keySet){
System.out.println(maps.get(id).name);
}

遍历Map方法二:使用entrySet

Set<Entry<Integer, Man>>  ss = maps.entrySet();
for (Iterator iterator = ss.iterator(); iterator.hasNext();) {
   Entry e = (Entry) iterator.next();
   System.out.println(e.getKey()+"--"+e.getValue());
}

9. Collections工具类

类 java.util.Collections 提供了对Set、List、Map进行排序、填充、查找元素的辅助方法。

1. void sort(List)                  //对List容器内的元素排序,排序的规则是按照升序进行排序。
?
2. void shuffle(List) //对List容器内的元素进行随机排列。
?
3. void reverse(List) //对List容器内的元素进行逆续排列 。
?
4. void fill(List, Object) //用一个特定的对象重写整个List容器。
?
5. int binarySearch(List, Object) //对于顺序的List容器,采用折半查找的方法查找特定对象。

10.总结

  1. Collection 表示一组对象,它是集中、收集的意思,就是把一些数据收集起来。

  1. Collection接口的两个子接口:

  • List中的元素有顺序,可重复。常用的实现类有ArrayList、LinkedList和Vector。

    • ArrayList特点:查询效率高,增删效率低,线程不安全。

    • LinkedList特点:查询效率低,增删效率高,线程不安全。

    • Vector特点:线程安全,效率低,其它特征类似于ArrayList。

  • Set中的元素没有顺序,不可重复。常用的实现类有HashSet和TreeSet。

    • HashSet特点:采用哈希算法实现,查询效率和增删效率都比较高。

    • TreeSet特点:内部需要对存储的元素进行排序。因此,我们对应的类需要实现Comparable接口。这样,才能根据compareTo()方法比较对象之间的大小,才能进行内部排序。

  1. 实现Map接口的类用来存储键(key)-值(value) 对。Map接口的实现类有HashMap和TreeMap等。Map类中存储的键-值对通过键来标识,所以键值不能重复。

  1. Iterator对象称作迭代器,用以方便的实现对容器内元素的遍历操作。

  1. 类 java.util.Collections 提供了对Set、List、Map操作的工具方法。

  1. 如下情况,可能需要我们重写equals/hashCode方法:

  • 要将我们自定义的对象放入HashSet中处理。

  • 要将我们自定义的对象作为HashMap的key处理。

  • 放入Collection容器中的自定义对象后,可能会调用remove、contains等方法时。

  1. JDK1.5以后增加了泛型。泛型的好处:

  • 向集合添加数据时保证数据安全。

  • 遍历集合元素时不需要强制转换。

JAVASE-DAY10&DAY11

原文:https://www.cnblogs.com/winterriver/p/12363898.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!