Spring源码中使用到很多Holder类 ,列如BeanDefinitionHolder,那么作用是什么呢?
Holder类的作用
public class Demo {
public static void main(String[] args) {
int a = 1;
change(a);
System.out.println("a之后的值: " + a);
System.out.println("-----------------我是分割线--------------------");
Integer b = 1;
change2(b);
System.out.println("b之后的值: " + b);
}
public static void change(int aa){
System.out.println("aa初始值: " + aa);
aa = 10;
System.out.println("aa之后的值: " + aa);
}
public static void change2(Integer bb){
System.out.println("bb初始值: " + bb);
bb = 10;
System.out.println("bb之后的值: " + bb);
}
}
我们执行一下看看:
很明显,这两个方法都没能改变原始的a、b变量。上面这段代码执行时的内存单元变化大致是这样的:
从这个图可以看出,执行change方法时,aa变量先是在另一个内存单元中复制了一份a变量的东西,接着在方法中一直改变的都是aa变量,因此很明显a变量从始至终都没有发生改变;执行change2方法时,同样bb变量也是在另一个内存单元中复制了一份b变量的东西,接着方法中改变了bb变量具体指向的值(即:堆内存中的 1 改变成了 10),很明显这个过程跟b变量也没有关系,因此在change2方法执行完毕之后b变量也没有发生改变
那么,我在最上面提到的Holder技术是怎么实现在一个方法中改变对象的值的呢?
public class IntegerHolder {
public Integer value;
public IntegerHolder() {
}
public IntegerHolder(Integer value) {
this.value = value;
}
}
public class Demo2 {
public static void changeInteger(IntegerHolder ii){
ii.value = new Integer(10);
}
public static void main(String[] args) {
IntegerHolder i = new IntegerHolder(1);
System.out.println("初始值: " + i.value);
changeInteger(i);
System.out.println("改变之后的值: " + i.value);
}
}
执行后效果:
发现没有,没有使用return返回值但是却让值发生了改变。如果还是用内存单元图来表示的话,那么大致上是这样的:
从上面的图可以看出,虽然在changeInteger方法中,变量 ii 的地址跟变量 i 的地址不一样,但是它们存储的内容(PS:ii.value这个Integer对象的地址)却从始至终都没有发生改变。相反,在changeInteger方法中改变的是ii.value这个Integer对象真正的值,ii.value的内存地址并没有发生改变,仍然和 i.value的内存地址一样。因此在这个方法结束之后 i.value对应的值也相应发生了改变。这就是Holder技术的实现原理