wordpress网站添加背景音乐,个人或主题网站建设,wordpress 好不好,wordpress增加连接池1. 常用四种引用 快速记忆法#xff1a;“硬(俗称的强引用) -- 软(SoftReference) -- 弱(WeakReference) -- 虚(PhantomReference)” 此处将常说的“强引用”记忆成“硬引用”可以对应到次席的“软引用”#xff08;反义词#xff1a;硬-软#xff09;这样更容…1. 常用四种引用 快速记忆法“硬(俗称的强引用) -- 软(SoftReference) -- 弱(WeakReference) -- 虚(PhantomReference)” 此处将常说的“强引用”记忆成“硬引用”可以对应到次席的“软引用”反义词硬-软这样更容易记住 a. 强引用
平常我们代码中写到的引用类型都是强引用类型比如Object obj new Object();, Object实例就有一个强引用类型指向它在GC过程中即使发生OOM该Object实例都不会被回收。
b. 软引用 - SoftReference
定义方式SoftReference sr new SoftReference(new Object()); 一个对象的实例被一个软引用实例指向那么在GC过程中发生OOM之前该Object对象实例会被回收掉在内存充足的情况下是不会被回收的。同时可以将一个引用队列关联到该软引用上在软引用指向的对象被回收后该软引用会被加入到关联的引用队列中。我们可以通过Reference的get()方法获取到该软引用指向的对象实例。
c. 弱引用 - WeakReference
弱引用基本上同上面的软引用类似WeakReference wr new WeakReference(new Object());但是特殊点就是在它被创建后的下一次GC时候其指向的对象实例会被回收掉不管内存是不是充足反正就是活不过一次GC。JDK中的WeakHashMap就是使用到WeakReference其Key就是被包装成WeakReference。
d. 虚引用 - PhantomReference
定义方式PhantomReference pr new PhantomReference(new Object(), new ReferenceQueue())虚引用对象再被定义时必须指定一个引用队列实例。JDK文档中介绍它主要用于对象被回收前资源的释放操作替换finalize()方法。它和前面的两个软引用和弱引用不同的地方有两点 2. 验证 虚引用在对象被回收之前添加到引用队列中同时需要手动处理它指向的对象才会被回收 思路
a. 创建一个虚引用对象然后发起一次GC操作查看其指向的对象实例是否被回收
b. 通过检测它关联的引用队列取出加入的虚引用对象查看此时其指向的对象实例是否被回收
c. 调用虚引用对象的clear方法之后查看其指向的对象实例是否被回收
2.1. 代码
public class ReferenceApp1 {public static void main(String[] args) throws Exception {phantom();}/*** 验证PhantomReference* throws Exception*/private static void phantom() throws Exception {// 步骤1. 定义一个InnerPhantomRefObj对象实例InnerPhantomRefObj innerPhantomRefObj new InnerPhantomRefObj();innerPhantomRefObj.setName(InnerPhantomRefObj-1);ReferenceQueueObject referenceQueue new ReferenceQueue();// 步骤2. 定义一个虚引用对象PhantomReferenceObject phtRef new PhantomReference(innerPhantomRefObj, referenceQueue);// 移除InnerPhantomRefObj对象实例上的强引用不然后面操作不会被回收innerPhantomRefObj null;System.err.println(before gc | get PhantomReference referent phtRef.get());// 步骤3. 发起GC操作System.gc();int i 0;Reference tmp null;while(true) {System.out.println(phantom iteration i);Thread.sleep(5000);if(i 1) {// 步骤4. 从引用队列中取出虚引用对象tmp referenceQueue.poll();if(tmp ! null) {System.err.println(get PhantomReference from ReferenceQueue);}// 发起一次GC操作其实此时InnerPhantomRefObj对象不会被回收System.gc();}if(i 5) {if(tmp ! null) {System.err.println(after gc | get PhantomReference referent tmp.get());// 步骤5. 调用虚引用上的clear方法让下一次GC操作回收掉InnerPhantomRefObj对象tmp.clear();// 或者让GC操作释放PhantomReference对象实例
// tmp null;
// phtRef null;System.err.println(clear PhantomReference);// 发起一次GC操作System.gc();}}if(i 10) {break;}}}Dataprivate static class InnerPhantomRefObj {private String name;}
}2.2. 观察VisualVM中实例个数变化判断是否被回收
a. 从上述代码的步骤1到步骤5之间的实例统计截图如(实例个数为1没有被回收)
b. 执行步骤5调用PhantomReference的clear方法之后的实例统计截图如(实例个数为0已被被回收) 在Java 8以及之前的版本中在虚引用回收后虚引用指向的对象才会回收。在Java 9以及更新的版本中虚引用不会对对象的生存产生任何影响。 3. 类比
看到一篇英文博客中用一个例子来类比软、弱、虚引用三者之间的差别非常好在此借用一下
比如一个快餐店中桌子座位有限服务员会随时清理桌子座位你进去点单找到一个座位坐下会存在下面几种情况
a. 然后后面有很多人过来点单时当座位不够时你会让出座位但在此之前每次服务员过来清理座位时你都没有让出座位这种情况就像就像软引用
b. 第一次服务员过来清理桌子座位时你就让出座位这种情况就像弱引用
c. 第一次服务员过来清理桌子座位时你可以随时准备让出座位其实这时候你并没有让出位置但是后面服务员说出一句让你让出座位时你才会让出座位这种情况就像虚引用