Integer 类
1、特点
public static void main(String[] args) {
Integer a = 1;
Integer b = 2;
System.out.println("交换前:" + "a = " + a + " " + "b = " + b);
swap(a, b);
System.out.println("交换后:" + "a = " + a + " " + "b = " + b);
}
private static void swap(Integer numA, Integer numB) {
Integer temp = numA;
numA = numB;
numB = temp;
}
输出结果:
交换前:a = 1 b = 2
交换后:a = 1 b = 2
产生这样结果的解释:
(1)Integer a = 1 实际上等价于 Integer a = Integer.valueOf(1) ,这就是自动装箱。尽管 a 和 b 都是 Integer 类型的对象,但是由于 Integer 这个类本身就跟 String 一样,是用 final 修饰的,并且 value 属性也是 final 修饰的,所以无论是调用方法重新赋值或者直接在 main 方法重新赋值,都会产生一个新的对象,原来的那个对象不会做任何改变,表面上都是同一个变量名,实际上却是一个新的对象 。
(2)关于值传递和引用传递的问题 。这里要特殊考虑String,以及Integer、Double等基本类型包装类,它们的类前面都有final修饰,为不可变的类对象,每次操作(new或修改值)都是新生成一个对象,对形参的修改时,实参不受影响,与值传递的效果类似,但实际上仍是引用传递。
在这里补充
2、值传递和引用传递的区别
(1)Java方法基本数据类型是传值,对象类型传引用,这是千真万确的。
(2)当参数是对象时,无论方法体内进行了何种操作,都不会改变实参对象的引用。
(3)当参数是对象时,只有在方法内部改变了对象的内容时,才会改变实参对象内容。
(4)按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。
(5)按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。
(6)“在Java里面参数传递都是按值传递”这句话的意思是:按值传递是传递的值的拷贝,按引用传递其实传递的是引用的地址值,所以统称按值传递。
(7)在Java里面只有基本类型和按照下面这种定义方式的String是按值传递,其它的都是按引用传递。就是直接使用双引号定义字符串方式:String str = “Java”;
说白了就是对象为参数时,传递的是参数本身,而值(如int , char , 或者String,Integer等等)为参数时,传递的是一个副本。
3、自动封装(共享设计模式的应用)
(1) 实例演示:
package com.heima.wrapclass;
public class Demo05_Integer {
public static void main(String[] args) {
Integer i1 = new Integer(97);
Integer i2 = new Integer(97);
System.out.println(i1 == i2); //false
System.out.println(i1.equals(i2)); //true
System.out.println("-----------");
Integer i3 = new Integer(197);
Integer i4 = new Integer(197);
System.out.println(i3 == i4); //false
System.out.println(i3.equals(i4)); //true
System.out.println("-----------");
Integer i5 = 127;
Integer i6 = 127;
System.out.println(i5 == i6); //true
System.out.println(i5.equals(i6)); //true
System.out.println("-----------");
Integer i7 = 128;
Integer i8 = 128;
System.out.println(i7 == i8); //false
System.out.println(i7.equals(i8)); //true
/*
* -128到127是byte的取值范围,如果在这个取值范围内,自动装箱就不会新创建对象,而是从常量池中获取
* 如果超过了byte取值范围就会再新创建对象
*
* public static Integer valueOf(int i) {
assert IntegerCache.high >= 127;
if (i >= IntegerCache.low && i <= IntegerCache.high) //i>= -128 && i <= 127
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
*/
}
}
解释:
(1)"==" 比较的是两个对象的是否指向同一个对象,即对象在内存中的地址一样;而equal()比较的是两个对象的值是否相同。
(2)new 是在java对分配内存,自然会创建不同的对象;
(3)-128到127是byte的取值范围,如果在这个取值范围内,自动装箱就不会创建新对象了,而从常量池中获取超过了byte的取值范围就会在创建新对象
(2) 自动装箱其底层会调用valueOf()方法,简单源码分析(JDK1.8):
public final class Integer extends Number implements Comparable<Integer> {
public static Integer valueOf(int i) {
//当 i >= -128 且 i <= 127 时,会直接将取缓冲区中的对象
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);//超过了byte取值范围会在堆内存创建
}
//内部类充当缓冲区
private static class IntegerCache {
static final int low = -128;
static final int high;
static final Integer cache[];
static {
// high value may be configured by property
int h = 127;
String integerCacheHighPropValue =
sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
if (integerCacheHighPropValue != null) {
try {
int i = parseInt(integerCacheHighPropValue);
i = Math.max(i, 127);
// Maximum array size is Integer.MAX_VALUE
h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
} catch( NumberFormatException nfe) {
// If the property cannot be parsed into an int, ignore it.
}
}
high = h;
cache = new Integer[(high - low) + 1];
int j = low;
for(int k = 0; k < cache.length; k++)
cache[k] = new Integer(j++);
// range [-128, 127] must be interned (JLS7 5.1.7)
assert IntegerCache.high >= 127;
}
private IntegerCache() {}
}
}