java ref 应用类型_Java引用类型原理剖析

问题

在分析前,先抛几个问题?

1.网上大多数文章对于软引用的介绍是:在内存不足的时候才会被回收,那内存不足是怎么定义的?什么才叫内存不足?

2.网上大多数文章对于虚引用的介绍是:形同虚设,虚引用并不会决定对象的生命周期。主要用来跟踪对象被垃圾回收器回收的活动。真的是这样吗?

3.虚引用在Jdk中有哪些场景下用到了呢?

Reference

我们先看下Reference.java中的几个字段public abstract class Reference {    //引用的对象

private T referent;

//回收队列,由使用者在Reference的构造函数中指定

volatile ReferenceQueue super T> queue;    //当该引用被加入到queue中的时候,该字段被设置为queue中的下一个元素,以形成链表结构

volatile Reference next;    //在GC时,JVM底层会维护一个叫DiscoveredList的链表,存放的是Reference对象,discovered字段指向的就是链表中的下一个元素,由JVM设置

transient private Reference discovered;

//进行线程同步的锁对象

static private class Lock { }    private static Lock lock = new Lock();    //等待加入queue的Reference对象,在GC时由JVM设置,会有一个java层的线程(ReferenceHandler)源源不断的从pending中提取元素加入到queue

private static Reference pending = null;

}

一个Reference对象的生命周期如下:

AAffA0nNPuCLAAAAAElFTkSuQmCC

image

主要分为Native层和Java层两个部分。

Native层在GC时将需要被回收的Reference对象加入到DiscoveredList中(代码在referenceProcessor.cpp中process_discovered_references方法),然后将DiscoveredList的元素移动到PendingList中(代码在referenceProcessor.cpp中enqueue_discovered_ref_helper方法),PendingList的队首就是Reference类中的pending对象。 具体代码就不分析了,有兴趣的同学可以看看这篇文章。

看看Java层的代码private static class ReferenceHandler extends Thread {

...        public void run() {            while (true) {

tryHandlePending(true);

}

}

}

static boolean tryHandlePending(boolean waitForNotify) {

Reference r;

Cleaner c;        try {            synchronized (lock) {                if (pending != null) {

r = pending;                    //如果是Cleaner对象,则记录下来,下面做特殊处理

c = r instanceof Cleaner ? (Cleaner) r : null;                    //指向PendingList的下一个对象

pending = r.discovered;

r.discovered = null;

} else {                   //如果pending为null就先等待

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java中的MetaObjectHandler是Mybatis框架中用于处理实体类的公共字段的接口,它提供了在插入和更新操作时自动填充实体类的公共字段的方法。我们来看一下它的源码实现。 MetaObjectHandler接口的定义如下: ```java public interface MetaObjectHandler { void insertFill(MetaObject metaObject); void updateFill(MetaObject metaObject); } ``` 该接口中定义了两个抽象方法insertFill和updateFill,这两个方法分别用于在插入和更新操作时自动填充实体类的公共字段。 MetaObject是Mybatis框架中用于封装Java对象的一个通用类,通过反射获取Java对象的属性和方法,从而实现对Java对象的读写操作。MetaObjectHandler中的insertFill和updateFill方法参数中的MetaObject对象就是Java对象的封装类,可以通过该对象来获取Java对象的属性和方法,从而实现对实体类的公共字段的填充操作。 下面是一个示例代码,演示了如何在实体类中自动填充创建时间和更新时间: ```java public class MyMetaObjectHandler implements MetaObjectHandler { @Override public void insertFill(MetaObject metaObject) { LocalDateTime now = LocalDateTime.now(); this.setFieldValByName("createTime", now, metaObject); this.setFieldValByName("updateTime", now, metaObject); } @Override public void updateFill(MetaObject metaObject) { LocalDateTime now = LocalDateTime.now(); this.setFieldValByName("updateTime", now, metaObject); } } ``` 在上述示例代码中,我们实现了MetaObjectHandler接口,并重写了其中的insertFill和updateFill方法。在insertFill方法中,我们获取当前时间,并通过setFieldValByName方法将其赋值给实体类的createTime和updateTime字段;在updateFill方法中,我们同样获取当前时间,并将其赋值给实体类的updateTime字段。 最后,我们需要在Mybatis的配置文件中配置MyMetaObjectHandler: ```xml <!-- 配置自定义的MetaObjectHandler --> <bean id="myMetaObjectHandler" class="com.example.MyMetaObjectHandler"/> <!-- 配置全局的MetaObjectHandler --> <mybatis:configuration> <mybatis:defaultScriptingLanguage type="org.apache.ibatis.scripting.xmltags.XMLLanguageDriver"> <mybatis:property name="metaObjectHandler" ref="myMetaObjectHandler"/> </mybatis:defaultScriptingLanguage> </mybatis:configuration> ``` 在上述配置文件中,我们通过id为myMetaObjectHandler的bean来配置自定义的MetaObjectHandler,然后通过属性metaObjectHandler将其注入到Mybatis的配置中。这样,我们就可以在插入和更新操作时自动填充实体类的公共字段了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值