Java笔试准备重点拾遗1–基础知识部分
1. 前期绑定与后期绑定
前期绑定指的是编译器在编译时,将调用代码的入口点指向该段代码的绝对地址。
后期绑定/动态绑定指的是编译器在代码执行时,才会决定需要调用的代码。代码地址使用额外的代码段来指定。
和C++不同,Java默认使用此方法来实现多态,而C++需要在基类中显式地声明虚方法(virtual关键字)
2. 基本成员默认值
只有变量作为类的成员使用时,Java才确保给定默认值;某个方法内的局部变量的值仍然可能是任意值。如果没有初始化,程序会报错。
boolean –> false
char –> ‘\u0000′(null)
byte –> (byte)0
short –> (short)0
int –> 0
long –> 0l
float –> 0.0f
double –> 0.0
3. 等价性测试
- 等于号和不等于测试的是值的相等:对于基础类型,测试的是两个值的相等;对于引用类型,测试的是对象地址的相等(也即是否为同一个对象)
-
equals()测试的是两个对象的相等:如果未重载equals()方法,默认比较的是地址的相等;如果重载了equals()方法,按照重载方法的比较方式处理。对于重用的系统提供的类,它们都实现了自己的equals()方法,如String, Integer, Double等等。
-
对于小数及其包装类的比较是非常严格的,如果两个小数差一点都会被认作不等。因此建议使用范围相等比较。
4. 移位运算的截尾
-
对于char、byte、short在移位运算前会被转成int,完成移位运算后截取相应类型的一部分作为结果
-
int、long的移位方式与char、byte、short相同,不过通常不使用类型转换,直接丢弃超出范围的二进制位
-
Java中整数数值的表示法:“有符号的二进制补码”
5. 循环语句
- 循环语句的终止条件如果存在表达式计算,则在每一次片段中都会被执行一遍。提示不要在终止语句的位置随意使用自增自减操作符,尤其是对for循环而言。
6. Foreach语句
Foreach语句的格式通常为:
for (int i: iList) {
dosomething(i);
}
Foreach语句可用于任何Iterable对象,比如数组、List、Set等。String类无法直接进行迭代不过可以使用toCharArray()
转成char数组。
相较For循环,Foreach循环有以下限制:
- Foreach循环无法改变原始值;
- Foreach循环无法直接追踪元素下标,这点与Python有很大不同
- Foreach循环只能从开始元素起正向递增访问,不能反向访问,也不能随机访问;
- Foreach循环不能同时循环两个Iterable对象
7. 使用标号的break和continue语句
Java中不存在goto语句,不过可以通过带有标号的break和continue语句实现一些goto的逻辑。这通常在多层循环的终止中使用。如果我们有一个二层循环且在内层循环中,现在我们希望退出整个循环。在只是用普通break时,我们只能退出内层循环,然后需要一个局部变量帮助我们退出完成循环。使用带有标号的break语句可以一次性退出整个循环体结构。
在此种情况下,标号后面必需紧跟循环语句。
break与continue总结:
- 一般的continue会跳转至本层循环(通常是内层循环)的顶部,继续执行;
- 标号continue会跳转至标签所在位置,继续执行紧跟在标签后面的循环;
- 一般break会跳出当前循环;
- 带标号的break会跳出标签所指定的循环。
- 在for循环中,一般continue会自动完成自增自减操作,而带标号的break与continue不会进行自增自减操作,如果有需要需要自行完成。
8. 传参数时的类型匹配
- 常数:
整数会优先当做int处理,如果没有匹配的方法可以直接提升至long、float、double(注意,按照此顺序提升);
小数会当做double处理,其他情况均需要强制类型转换;
char会优先当做char处理,如果没有匹配的方法会直接升为int、long、float、double(注意,按照此顺序提升);
如果需要向下窄化匹配,需要手动进行强制类型转换。 -
变量:
变量依然按照常量处理的方式,首先看作对应类型的常量,然后依上述方式对变量进行提升,如果提升后有匹配的方法则调用;否则出现错误。
9. 方法重载的判断依据
只有参数列表。两个同名方法的差别可以是参数的个数,参数类型的差异或者是参数类型的顺序(不推荐这么做,这样容易使得程序变得混乱)。返回值并不能用于区分同名方法。
10. 默认构造器
默认构造器是指不含任何参数的构造器。当类中没有任何构造器时,java会为其显式地创造一个;如果存在任何一个程序员定义的构造器,java将不会创造此默认构造器。
11. 使用this显式调用构造器
这种方法可以在一个构造器中调用另一个构造器,以减少代码。注意,这种调用方式仅可以在构造函数中使用
例:
class ThisConstructor {
String message;
int code;
ThisConstructor() {
this("hello", 5); // 使用this()调用另一个构造函数
}
ThisConstructor(String msg, int code) {
message = msg;
this.code = code; // 另一个this的常用场景:区分局部变量与类成员
}
}
12. 终结处理与垃圾回收
finalize()方法
finalize()方法理论上并不需要被执行。它与C++中的析构函数并不一样。官方说明表示finalize()方法在对象被回收的时候完成,但是如果清理工作放在finalize()中可能会引发对象无法被回收从而出现问题。
finalize()通常在Java调用其他语言的代码时用到,它可以释放其他代码所带来的内存分配。另外可以使用finalize()检查对象在释放时是否为期待的值。这将其作为一种逻辑检查的方式。
对象回收的3个要点
- 对象可能不被垃圾回收
- 垃圾回收不等于析构
- 垃圾回收只与内存相关
垃圾回收器的工作方式
- 引用计数
此方法存在于java回收机制的讲解中,然而jvm不是这么干的
- 停止-复制(stop-and-copy)
此方法将暂停程序的运行,然后将所有存活的对象拷贝到一片新的内存中,然后将之前内存的区域全部释放。
找出存活对象的方式:从堆栈与静态存储区出发,遍历所有的引用,找出存活的对象。
优势:拷贝出的对象在内存中连续存储;解决循环引用造成的垃圾无法释放问题;
劣势:浪费空间;如果垃圾很少时,反复拷贝效率很低
- 标记-清扫(mark-and-sweep)
此方法首先找出存活的对象,在找这些对象的过程中,为这些对象进行标记。一轮扫描过后,清扫所有未被标记的对象。
优势:对于短时间出现的垃圾碎片便于回收
劣势:会产生内存空隙
- 分块处理
为了提高内存的利用率,java将内存分为若干块。每个块拥有一个代数(generation count)作为其存活的标志。当jvm准备清理内存时,会整理上次垃圾回收后产生的新内存。另外,通过代数可以监视出对象的活跃情况。如果每个块的代数很稳定,说明运行内存很稳定,没有多少块需要释放。
- 自适应
JVM结合以上的种种方法,通过自适应的方法决定垃圾回收方式:当对象相对稳定时,jvm会切换至“标记-清扫”模式;当对象碎片较多时,jvm会切换至“停止-复制”模式,以清理散落的内存。
提升速度的其他方式
- 惰性评估:编译器在需要编译的时候编译代码,并对代码进行优化。这样使得不会被执行的代码不会被编译,常用的代码编译速度越来越快。
Java编程思想第4版:P87-91
13. 成员初始化
指定初始化
可以使用已经计算出的值为成员赋初值。注意,在赋初值的时候右值必需可被计算出来。
指定初始化会在其他初始化方式之前被执行,初始化的顺序按照成员定义的顺序执行。
例:
class A{
A(int i) {System.out.println("A("+i+")");}
}
class B{
a1 = new A(1);
B() {
System.out.println("B constructor");
A a = new A(4);
}
a2 = new A(2);
void f() {System.out.println("B::f()");}
a3 = new A(3);
}
public class Main{
public static void main(String[] s) {
B b = new B();
b.f();
}
}
输出结果为:
A(1)
A(2)
A(3)
B constructor
A(4)
B::f()
A(1), A(2), A(3)都是使用指定初始化方式初始化的,它们的执行顺序在构造初始化的前面。这三者的顺序依声明的顺序决定。在指定初始化完成后,进行构造初始化,因此输出B constructor与A(4)。(前面这么多都是主方法中B b = new B();这句引发的。)最后B::f()在调用b.f()时输出。
构造器初始化
构造器初始化在构造器中完成,使用赋值语句。如果一个成员已经使用指定初始化方式赋值了,构造器初始化将会覆盖掉已经初始化的值。
静态类型的初始化
静态类型的变量在类第一次加载的时候被初始化,并且只会在第一次加载的时候被初始化。如果一个类没有被加并且类的成员没有被引用过,这个类就不会被创建,内部的static成员也不会被初始化。静态类型的初始化先于普通成员的初始化。
对象创建过程总结:
- 创建类的对象或类的静态方法被访问或类的静态成员被访问时,jvm定位对应的class文件;
- 载入class文件,所有静态初始化的方法和参数都会被加载并初始化;
- 使用new创建对象的时候,jvm在堆上分配足够的空间;
- 按照默认值原则为对象的成员赋“系统”初始值;
- 执行指定初始化
- 执行构造器初始化
指定构造器的另外写法
使用大括号将指定构造的语句写在一起,置于成员声明的后面。这种写法不会影响初始化的顺序,static成员也可以使用此方法。此种方式在内部类的初始化时有重要作用。
class A {
int i;
}
class B {
A a1;
A a2;
static int i;
{
a1 = new A();
a2 = new A();
}
static {
i = 1;
}
}
Java编程思想第4版:P93-98
数组初始化相关
数组可以通过以下方法创建,2、3两种方法还可以进行初始化。这两种方法可以用于参数传递以及返回多个值。数组在创建之后会按照默认值得对应关系为每个元素赋予默认值。
int[] a = new int[3];
int[] a = {1,2,3};
int[] a = new int[]{1,2,3};
可变参数列表
使用数组实现可变参数列表
这种方式下传入的参数是一个数组,由于数组的长度不固定,由此可以实现不定参数的传入机制。参数列表可以指定为通用的Object类型,也可以指定为String、int等特定类型。
class A{
void f(Object[] obj) {
for (Object o: obj) {
System.out.println(o);
}
}
public static void main(String[] s) {
A a = new A();
a.f(new Object[]{1,2,3}); // 同一基本数据类型
a.f(new Object[]{new Integer(1),2.1,"3"}); // 不同类型混合,这是Object的特性
}
}
使用可变参数列表(SE5特性)
可变参数列表的样式与上述数组相似,不过将[]
变为了...
,不过传入参数的本质依然是数组,而且是被java包装过的数组。Java会将一系列参数按照类型自动匹配到参数列表中,对于基本类型可能会转变为对应的包装类。如果传入的参数组已经是个数组,java会自动将数组映射至参数列表中。
class A{
void f(Object... obj) {
for (Object o: obj) {
System.out.println(o);
}
}
void g(int p, Object... obj) {
System.out.println(p);
for (Object o: obj) {
System.out.println(o);
}
}
public static void main(String[] s) {
A a = new A();
a.f(1,2,3); // 同一基本数据类型,数据罗列
a.f(new int[]{1, 2, 3}); // 同一基本数据类型,传入已经组成好的数组
a.f(new Integer(1), 2.1, "3"); // 不同类型混合,这是Object的特性
a.f(new A(), new A(), new A());
a.f(); // 空值也可以
a.g(2, 3, 4); // 2会匹配为p, 3,4会匹配为obj
}
}
可变参数列表造成的重载问题
对于上面的例子,如果将方法g()
改为方法f()
是否可以?答案是不可以,因为对于一列int参数,均满足两个f()
的定义,jvm无法判断。
还是上面的例子,如果将方法g()
改为方法f()
并且方法f()
中的Object类型改为String类型是否可以?答案是可以,因为对于第一个参数,两个f()
接受的类型不同,一个是String,另一个是int。
看以下例子:
class A{
void f(String... obj) {
for (String o: obj) {
System.out.println(o);
}
}
void f(Integer... obj) {
for (Integer o: obj) {
System.out.println(o);
}
}
public static void main(String[] s) {
A a = new A();
a.f("1", "2");
a.f(1, 2);
// a.f() // ?
}
}
主方法中最后一行a.f()
能够被执行吗?答案是不可以,因为一个空参数列表都可以匹配两个f()
,因此会出现错误。不过其他的两个匹配可以完成。
尾随参数
尾随参数是将可变参数列表放在不可变(固定)参数列表的后面,一个方法中最多有一个尾随参数,尾随参数必需放在参数列表的最后。
枚举类型初步(SE5)
创建枚举类型:
public enum ResultCode {
SUCCESS, RESOURCE_NOT_FOUND, DATABASE_ERROR, INTERNAL_ERROR
}
引用枚举类型的一个值:
ResultCode code = ResultCode.SUCCESS;
System.out.println(code); // 输出SUCCESS
使用ordinal()
方法可以返回枚举值按照声明顺序所在的下标,values()
可以返回枚举类型的Iterable对象。
枚举类型可以作为标记值,在switch语句中充当标签。