培训还有最后一天,也就是今天是倒数第二天,后天考核,但是不怎么会,我决定明天再看考核的内容。今天呢,老师讲了些东西,可是我没听,有那么点小尴尬,因为自己有一些不懂的东西,想弄懂。
今天也会说一些知识点,其中就包括昨天没有解答上来的那个问题----数值在 -128到127之间的Integer类型对象创建的问题。
下面就说一下我今天总结的内容,以反问的形式。
一.局部变量的修饰符问题
1.局部变量能使用权限修饰符(private,dafault,protected,public)修饰吗?
答:权限修饰符可以修饰成员变量和成员方法,但是不可以修饰局部变量。因为局部变量定义在方法体中,它的作用范围只在该方法的方法体内有效。方法体在类中,就算最小的权限private修饰的成员在本类也是可以被访问的。所以我们可以这样理解,方法体的 “权限” 比private还要小,因为只要在本类中就可以访问private修饰的成员,而想访问局部变量,只能在这个方法体内进行访问,出了方法体就访问不到这个局部变量了。所以既然方法体的 “权限” 比private还小,那么在局部变量前添加访问权限修饰符显然是没有意义的。
2.局部变量能够被static来修饰吗?
答:是不能的,因为static修饰的东西说明这个东西是属于类的。而局部变量是属于代码块的,出了代码块它就不能被访问到。即使创建对象,它也不能通过对象. 的方法访问,也不能通过类名. 的方式访问。所以说明它不属于类(不能属于类),也就不能被static来修饰了。
3.局部变量可以使用final来修饰吗?
答:当然可以了,如果不用final来修饰局部变量,那么这个局部变量就是一个局部变量,但是如果用final来修饰,那么这个局部变量就变成了一个局部常量,这并不影响什么。final的作用是,在修饰基本数据类型变量时,一旦变量初始化后,将不允许在次赋值,在修饰引用数据类型时,一旦引用指向了一个对象时,那么这个引用只能指向这个对象,而不能在次指向其他的对象。
4.局部变量一定要初始化吗,为什么成员变量就可以不初始化?
答:Java中的变量分为两种,一种是局部变量,一种是成员变量。其实对于变量,不管是局部变量还是成员变量,我们在使用它们之前都要进行初始化,因为例如输出语句,想打印一个局部变量a,在你没初始化的时候,它是没有值的,jvm不知道输出什么,你应该先给它进行初始化,告诉jvm你想要输出的确切值。当然了,如果你不使用局部变量,那么即使局部变量放在那里不初始化编译也是可以通过的。
局部变量需要我们手动初始化,因为当我们不给它初始化的时候,它就真的没有初始化。而成员变量则不同,对于基本数据类型,即使我们没有给它进行初始化,jvm也会按照对应的基本类型来给它赋值,进行初始化。对于引用数据类型,如果我们不进行初始化,那么jvm会给它初始化为空。
二.Integer对象的问题
这个问题也就是昨天我没有回答上来的问题,今天看了一些博客,了解了一下。我们先来看一段代码。
package com.java.text;
public class TextInnerClass {
public static void main(String[] args) {
TextInnerClass a=new TextInnerClass();
a.text1();
}
public void text1() {
// Integer f1 = 100, f2 = 100, f3 = 150, f4 = 150; 和下面的代码等同
Integer a = 100;
Integer b = 100;
Integer c = 225;
Integer d = 225;
System. out.println( a == b); //true
System. out.println( c == d); //false
}
}
为什么第一个为true,第二个为false呢?
当我们通过表达式创建一个Integer对象时,也就是我们给一个Integer赋予一个int类型的时候会调用Integer类的静态方法valueOf()(自动装箱)。上面的代码我也可以写成下面这样。
这里我们可以看一下静态方法 valueOf() 的源码:
这里看到如果传进来的int值出了这个边界,就new 一个Integer对象。而如果在这个边界内,就从cache数组中取出值来给这个Integer类型的引用,也就是这个 Integer引用指向这个数组的某一个值。那么边界到底是多少呢?
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() {}
}
我对这部分代码也不是很懂,但是我知道它将 -128到127这个区间的值放到了cache数组中。
通过这两个代码块我们就可以解释上面的问题了,当传入的int值为100时,这个int值在这个区间内,所以直接将cache数组中的100值取出来给这个索引a,也就是这个索引a指向这个cache数组的这个100元素所在的地址,同理b也一样,所以a==b为true。而当闯入的int值为255时,不在这个区间内,那么就不从这个数组中取值,而是通过new 操作符在堆中创建一个新的对象,所以c和d其实指向的是不同的对象,当然是false了。
我们在来看一段代码:
这个值应该是什么呢?
不管是通过new操作符创建的对象,还是通过表达式创建的对象,它们都是对象包装类Integer类型,所以当Integer类型和int类型进行比较时,我们先将 Integer 拆装为int类型,然后在比较值的大小。所以两个输出结果都是true。
java中还有与Integer类似的是Long类型,它也有一个缓存,在区间[-128,127]范围内获取缓存的值,而Long与long比较的时候先转换成long类型再做值的比较。而Double类型,它没有缓存,但是当Double与double比较的时候会先转换成double类型,再做值的比较。
三.数组和集合的存放位置
1.数组?
答:数组在定义的时候就已经定义了要存放的数据类型,所以一种数组只能存放一种数据类型。按照这个分,数组可以分为两种:一个是基本数据类型数组,另一种是引用数据类型数组。当一个对象使用关键字“new”创建时,会在堆上分配内存空间,然后返回地址给这个对象的引用,这对数组来说也是一样的,因为数组也是一个对象;
2.数组对象及其引用存放在内存中的哪里?
答:如上代码,让我们来调用方法m1,看看发生了什么:当m1被调用的时候,栈帧Frame-1被创建并push到栈中,同时局部变量i也在栈帧Frame-1内创建。然后,m2方法在m1方法内部被调用,栈帧Frame-2被创建并push到栈中,在m2方法中,一个新的对象A在堆中被创建,而它的引用则被put到栈帧Frame-2里;现在内存中堆和栈的大致情况如下图:
数组同样是对象,所以数组对象以及引用在内存中的分布如上所示 ;
链接:http://www.programcreek.com/2013/04/what-does-a-java-array-look-like-in-memory/
3.数组是什么数据结构?
答:从逻辑上说是线性表中的顺序存储结构,从物理上说是在物理内存里分配出一块连续的空间按顺序存储。
4.ArrayList在内存中的存储方式(图解)?
例如要存入集合的对象如下:
内存图解如下:
首先,在堆中创建一个集合对象(默认初始容量为10),地址指向 al ,在集合中添加元素并不是在集合中直接添加,
而是在堆内存中重新为添加的对象分配空间, 其对象的地址保存在集合容器中。 这种存储方法类似于“拉链法”, 迭代器的原理也是如此。
四. default
default可以说是包访问权限,是一种默认的权限,在同一个包中可以访问default修饰的东西。以前知道这个,但是今天忽然发现了一个问题,就是在属性前(变量),方法,类前我们如果不写修饰符,那么 属性 ,方法,和类就是默认是default修饰的,也就是包访问权限。它们可以不写修饰符,但是不能写default(显示的写出)来修饰它们,这样会报错。
我现在怎么觉得数据结构挺重要的呢。。。。。。。