JAVA学习day13

首先先简单提一下昨天没有说的线程池

image.png

线程池代码示例,实际开发过程中,一般程序猿用的并不多

锁和死锁

锁是java中用来保证线程操作原子性的一种机制
比较一下,锁是数据库中用来保证事务操作原子性的一种机制
原子性:如果把一个事务看作是一个程序,它要么完整的被执行,要么完全不执行。这种特性就叫原子性。

用到锁的环节

有数据脏读,根据代码应该有10000个线程,但实际并没有达到
image.png
添加synchronized关键字上锁,给代码块上锁。这样就能保证数据每次结果都是10000
image.png
或者直接在方法签名上添加synchronized直接锁方法,锁方法比锁代码块范围更大
image.png

java中的锁有synchronized和Lock锁

synchronized是关键字,可以锁方法和代码块,Lock是类(官方推荐)只能锁代码块

lock锁代码示例

image.png

数据类型分为线程安全类型和线程不安全类型

如果一个数据类型需要我们手动加锁来保证其操作的原子性,那么他就是不安全线程
如果一个数据类型能够自己在方法中加锁来保证其操作的原子性,那么它就是线程安全的
简单列举几个常见的数据类型


image.png

以ArrayList为例

image.png

只有在多线程共享数据的前提下,才需要考虑加锁的问题,一般程序中都不需要用到线程安全

产生死锁的必要条件

互斥条件:

锁要具有排他性(一个资源只能被一个线程占用),在同一时刻锁只能被一个线程持有

请求与保持条件:

一个线程因为请求其他资源被阻塞时,对已获取的资源保持不释放

不剥夺条件:

一个线程没有主动释放资源之前,不能被其他线程强行剥夺的

循环等待条件:

A线程持有资源a的锁,B线程持有资源b的锁,在互相不释放自己资源的前提下,去请求对方持有的锁,这时候会形成双方循环等待,造成永久堵塞

产生死锁的原因

两个线程手上各自持有一个对象的锁,在没有释放的情况下,去请求对方手上的锁,这时候就产生了死锁

如何解决死锁

破坏任意一个条件即可


image.png
1.破坏互斥条件:

用共享锁,在同一时刻,锁可以被多个线程使用

2.破坏请求与保持条件:

一次性申请所有资源

3.破坏不剥夺条件:

一个线程因为请求其他资源被阻塞时,主动释放已获取资源

4.破坏循环等待条件:

所有的线程按照同样的顺序请求资源

集合容器整理

image.png

1.collection 和map的区别

(1)collection和map是官方提供的集合容器的两大体系的顶层接口
(2)collection代表单元素集合体系
(3)map代表kv键值对集合体系
(4)collection体系继承了iterable迭代器接口,所有的子类都提供了迭代器的实现,而map体系没有

2.List,Set,Queue的区别

(1)List,Set,Queue都是collection体系下的子接口,分别代表三个子体系
(2)List体系特点有序不唯一,Set体系特点是无序且唯一,Queue体系特点是先入先出

3.队列和栈的区别

(1)队列是一种FIFO(First in First out)先入先出的结构
(2)栈是一种FILO(First in Last out)先入后出的结构
Java集合体系中的LinkedList可以实现队列和栈结构
在链表头部插入尾部取出或者尾部插入头部取出,就是队列。(插入和取出在不同方向上进行)
在链表头部插入头部取出或者尾部插入尾部取出就是栈(插入和取出在相同方向进行)

4.ArrayList和Array的区别

Array是数组,ArrayList是类
Array是定长的(需要手动扩容),ArrayList长度可变(使用过程中自动扩容)
ArrayList的底层是Array

5.ArrayList和LinkedList的区别

1.底层数据结构实现︰ArrayList底层数据结构是动态数组,而 LinkedList的底层数据结构是双向链表
2.随机访问(即读)效率∶ArrayList比LinkedList在随机访问的时候效率要高,因为ArrayList底层是数组,可以通过索引号快速访问,LinkedList是通过二分查找法遍历链表节点进行查找的
3.增加和删除效率∶在非首尾的增加和删除操作,LinkedList 要比 ArrayList 效率要高,因为ArrayList增删操作需要大量的前移或后移,这个过程中涉及到大量的赋值操作比较耗时间,LinkedList只需要修改节点对象的左右指针即可。
4.内存空间占用:LinkedList 比 ArrayList更占内存,因为 LinkedList的节点除了存储数据,还存储了两个引用,一个指向前一个元素,一个指向后一个元素。
5.综合来说,在需要频繁读取集合中的元素时,更推荐使用ArrayList,而在插入和删除操作较多时,更推荐使用 LinkedList。

6.ArrayList和Vector的区别(可以简单提一下CopyOnWriteArrayList)

ArrayList 是线程不安全的,Vector 是线程安全的,ArrayList中所有的方法都没有加同步锁,Vector中所有方法都加了synchronized同步锁,官方在jdk1.5版本中又推出了一个CopyOnWriteArrayList,使用Lock锁实现线程安全,然后弃用了Vector,因为Lock锁的性能比synchronized锁性能更好。
在并发编程中,如果多个线程共享一个ArrayList,那么必须考虑线程安全问题,可以自己对ArrayList操作的代码加锁或者直接用线程安全的CopyOnWriteArrayList类,在不考虑线程安全的环境下,用ArrayList更好。因为开锁和关锁是很耗性能的

7.Array怎么转换为ArrayList,ArrayList怎么转换为Array

Array就是数组,ArrayList是底层为数组的容器类
(1)官方提供的数组工具类Arrays,提供了一个静态方法asList()可以把数组转换为List,参数是数组,返回值是List
(2)ArrayList类中提供了toArray方法,可以吧ArrayList转换为Array后进行返回

8.HashSet和TreeSet

HashSet和TreeSet都是Set接口下的子类
HashSet的底层是HashMap,他将数据存储在HashMap的key中
HashSet是无序的唯一的,因为HashMap的key是无序唯一的
TreeSet的底层是TreeMap,他将数据存储在TreeMap的key中
HashSet是有序的,唯一的,因为TreeMap的key是有序唯一的

9.Map接口体系

HashMap和HashTable区别(带一下ConcurrentHashMap)

HashMap是线程不安全的,HashTable是线程安全的,HashMap中所有的方法都没有加同步锁,HashTable中所有方法都加了synchronize同步锁,官方在jdk1.5版本中又推出了一个ConcurrentHashMap,使用Lock锁实现线程安全,然后弃用了HashTable,因为Lock锁的性能比synchronized锁性能更好。
在并发编程中,如果多个线程共享一个HashMap,那么必须考虑线程安全问题,可以自己对HashMap操作的代码加锁或者直接用线程安全的ConcurrentHashMap类,在不考虑线程安全的环境下,用HashMap更好。因为开锁和关锁是很耗性能的

对Null key支持:HashMap支持key为null,但只能为一,HashTable不支持k为空,会直接抛出NPE,HashMap和HashTable支持value为空,不限制个数,ConcurrentHashMap的key和value都不支持为空

HashMap在1.8以后,设置了阈值=8,当链表长度超过阈值的时候,会把链表转换成红黑树,以此减少检索时间,HashTable被弃用了,没有更新

10.初始容量大小和扩容容量大小的区别

HashMap默认初始容量是16,扩容策略是原来长度的2倍
HashTable默认初始容量是11,扩容策略是原来是2n+1
HashMap如果手动指定了初始容量,不是2的n次方,他也会找到最紧接的一个2的n次方作为初始容量;HashTable如果手动指定了初始容量,会直接诶使用给定的大小;
Hashtable如果手动指定了初始容量,会直接使用给定的消失
Hashtable采用的锁全表机制,ConcurrentHashMap采用了分段锁的设计,锁粒度更细,性能更好

11.Hashmap和Treemap的区别

HashMap的底层是数组+链表/红黑树,key是无序的,唯一的
TreeMap的底层是红黑树,key是有序的唯一的
Hashmap的性能比TreeMap更好

12.HashMap的底层原理(数据结构+put()流程 + resize()扩容策略)

HashMap的底层是数组+链表/红黑树
1.根据数组中节点为null,创建新的节点对象,把k, v存储在节点对象中,把节点对象存储在数组中。2.如果数组的节点不为null,判断节点的key与插入元素的key是否相等。
1.相等,直接用新的k,v覆盖原节点中的k,V。
2.不相等,判断此时节点是否为红黑树。
1.是红黑树,创建红黑树节点对象存储k,V,插入到红黑树中。
2.不是红黑树,创建链表节点对象存储k,v,插入到链表中,判断链表长度是否大于阈值8。
3.判断++size是否大于阈值,是就扩容。

HashMap默认初始容量是16
resize()方法是在hashmap中的size大于阈值时或者初始化时,就调用resize方法进行扩容每次扩容的时候始终是原数组长度的2倍,即长度永远是2的n次方
扩容后节点对象的位置要么在原位置,要么偏移到两倍的位置

13.HashMap的长度为什么是2的幂次方

HashMap长度为2的幂次方的原因是为了减少Hash碰撞,尽量使Hash算法的结果均匀。

14.什么是哈希碰撞/哈希冲突,怎么解决哈希冲突,HashMap采用的策略

如果有两个字符串通过同样的哈希算法计算出来的哈希码一样,则称他们发生了哈
希碰撞/哈希冲突。
解决Hash冲突方法有:开放定址法、链地址法(拉链法)。HashMap中采用的是链地址法。
链地址法(拉链法),将哈希值相同的元素构成一个同义词的单链表,并将单链表的头指针存放在哈希表的第i个单元中,查找、插入和删除主要在同义词链表中进行。链表法适用于经常进行插入和删除的情况。

15. HashMap为什么不直接使用key的hashCode()函数返回的哈希码作为落槽时的索引号, HashMap是怎么解决的呢

这题也可以这样问"HashMap的底层是如何计算key落槽时的索引的”
hashCode()方法返回的是 int整数类型,其范围为-(231)~(231-1),约有40亿个映射空间,而 HashMap的容量范围是在16(初始化默认值)~2N 30,HashMap通常情况下是取不到最大值的,并且设备上也难以提供这么多的存储空间,从而导致通过hashCode()计算出的哈希值可能不在数组大小范围内,进而无法匹配存储位置;
HashMap自己实现了自己的 hash()方法,通过两次扰动使得它自己的哈希值高低位自行进行异或运算,降低哈希碰撞概率也使得数据分布更平均。

16.==和equals()方法的区别

==和equals()都可以用来进行比较,语法是a ==b或者a.equals(b)
==比较的是内存地址,equals()方法是Object中的方法,可以被任意类继承或者重写,通过看官方Object类原码可以知道equals()方法默认也是用==比较内存地址
如果想要修改equals()方法的比较规则,可以重写equals()方法,String类就重写equals()方法的比较规则,由默认的比较两个字符串对象的内存地址,修改为比较字符串中每个字符是否相等
因为堆区中可能会出现两个一模一样的字符串,但用==比较会返回false,因为其内容一样但内存地址不一样,所以字符串比较必须用equals方法,否则就会出现两个一模一样的字符串,因为地址不一样,比较后出现不相等的情况

17.为什么重写了hashCdoe()方法,必须重写equals()方法(反过来问也一样)

如果两个对象equals相等,其hashcode必然相等,如果equals相等,hashcode不相等就不能保证唯一性
HashMap底层采用了key的hashcode()来计算数组的索引index
如果数组[index]为空说明key不存在,直接落槽插入
如果数组[index]不为空,说明该位置有key存在,但不能一定说明一存在的key与要插入的key重复,因为可能会发生哈希碰撞,此时应该进一步用equals方法比较已存在的key与要插入的key是否相等。如果相等就说明一定是重复的,应该覆盖,如果不相等,说明发生了哈希碰撞,应该插入链表中
重写equals方法的目的是为了不去比较两个对象的内存地址,改为比较对象的内容,如果一个类重写了equals没有hashcode就可能出现两个地址不同的对象,equals比较相等但是hashcode比较不相等,这样会违反hashmap的唯一性,因此重写了equals方法,必须也要重写hashcode方法,且必须满足两个对象equals相等,hashcode也必须相等

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

推荐阅读更多精彩内容