关键词搜不到我的网站,为企业规划一个网站,有什么做网站好用的软件,网站建设昆明网络公司目录 1、ConCurrentHashMap为什么不允许key为null#xff1f; 2、ThreadLocal会出现内存泄露吗#xff1f; 3、AQS理解 4、lock 和 synchronized的区别 1、ConCurrentHashMap为什么不允许key为null#xff1f; 底层 putVal方法 中 如果key || value为空 抛出… 目录 1、ConCurrentHashMap为什么不允许key为null 2、ThreadLocal会出现内存泄露吗 3、AQS理解 4、lock 和 synchronized的区别 1、ConCurrentHashMap为什么不允许key为null 底层 putVal方法 中 如果key || value为空 抛出空指针异常 其实是为了避免在多线程并发场景下的歧义问题 在获取key 返回结果为null 无法判断是 putkv的时候 value本身是null值还是key本身不存在。还是key存在 value为空 。这种不确定性造成线程安全问题 而ConCurrentHashMap本身是线程安全的集合固key不能为null 2、ThreadLocal会出现内存泄露吗 ThreadLocal是用来解决线程安全性的一个工具让每个线程都开辟了一块内存空间用来存储共享变量的一个副本然后每个线程只需要去访问和操作自己的共享变量的副本去避免多线程竞争同一个共享资源。 每个线程都有一个成员变量 ThreadLocalMap当线程访问ThreadLocalMap修饰的成员变量时候 保存数据副本 key为 ThreadLocal 是弱引用 value 保存的是共享数据的副本 每个线程都有一个副本 不存在对于共享数据的并发操作 解决了线程安全问题。 弱引用 成员变量ThreadLocal 允许在引用关系存在的情况下被GC回收 一旦回收 key 的引用就会变为null 就会导致内存永远无法被访问造成内存泄漏 ThreadLocal设计是一定存在内存泄漏的 。 为了避免问题 在进行数据读写的时候 ThreadLocal默认会做一些清理动作找到并清理entry里面的key为null的数据 具体方法 1、每次用完ThreadLocal后主动调用 remove() 方法移除数据 最好方法 2、 ThreadLocal声明为全局变量使得无法被GC回收如果后续线程不再访问这个key也会造成内存泄漏 最终回答 不恰当的使用threadlocal会造成内存泄漏主要原因threadlocalmap里面的key是一个弱引用弱引用特性是 不管是否存在直接引用的关系当threalocal 没有其他的强引用关系的时候对象就会被GC回收掉从而导致key可能变为null造成内存无法被访问 3、AQS理解 AQS是多线程同步器提供了两种锁的机制 3.1、排它锁 存在多个线程去竞争同一共享资源的时候同一时刻只允许一个线程去访问这样一个共享资源也就是多个线程中只能有一个线程去获得一个锁资源 比如 lock中的 ReentrantLock 重入锁就是用到了AQS 中排它锁的功能 3.2、共享锁 也称为读锁同一时刻允许多个线程同时获得锁的资源 3.3、 AQS 作为互斥锁 需要解决三个核心问题 3.3.1、互斥变量的设计如何保证多线程同时更新互斥变量的时候线程的安全性 采用int 类型的互斥变量 state用来记录锁竞争的状态 0--没有 1 有线程持有。 线程获取锁资源 先会判断 state 是否为0 如果是 更新状态为1 表示占有到锁。 如果多个线程同时做一个操作导致线程安全性问题AQS 采用CAS机制保证state 互斥变量更新的一个原子性 3.3.2、未竞争到锁资源的线程等待 以及竟遭到锁的资源释放锁之后的唤醒 未获取到的线程 通过unsafe类中的park方法去进行阻塞把阻塞的线程按照先进先出的原则去加入到一个双向链表的一个结构中。当获取到锁资源的线程释放锁之后会从双向链表的头部去唤醒下一个等待的线程再去竞争锁 3.3.3、锁竞争的公平性和非公共性 公平锁先会判断双向链表是否有阻塞的线程如果有就会排队等候 非公平锁不管是否有等待的线程都会直接尝试更改state 去竞争锁 4、lock 和 synchronized的区别 特性 synchronized 是java中的同步关键字。 两种方法控制锁定力度1、把synchronized关键字修饰在方 法层面。2 、修饰在代码块上。 通过synchronized加锁对象的生命周期来控制锁定作用范 围。比如锁对象是静态对象 或者是类对象锁就属于全局锁。锁对象是普通实例对 象锁定范围取决于实例生命周期。 Lock 是J.U.C包提供的接口实现类 ReentrantLock重入锁的实现。 Lock中锁的力度是 通过 unlock() 方法和 lock() 方法来决定的锁定作用域取决于Lock实例的生命周期 相同点 功能都是java中用来解决线程安全的问题一个工具 不同点 1、Lock比synchronized 灵活性更高 Lock可以自主决定什么时候加锁什么时候释放锁只需要调用lock /unlock方法。Lock还提供了非阻塞的竞争锁方法trylock方法 通过返回true /false 来告诉当前线程是否已经存在有其他线程正在使用锁 synchronized 是关键字。无法实现非阻塞竞争锁的方法 synchronized 锁的释放被动的只有当synchronized 同步代码快 执行结束以后或者代码出现异常 才会被释放。 2、Lock 提供了公平锁、非公平锁机制。 synchronized 只提供了非公平锁的实现 3、性能差别 synchronized 引入偏向锁、轻量级锁、重量级锁、以及锁升级的机制去实现锁的优化 Lock 用到了自旋锁的方法实现性能优化