作者:蜗牛 | 来源:互联网 | 2024-11-13 18:24
Java初学者必备的六大核心知识点
本文总结了Java初学者需要掌握的六大核心知识点,帮助你更好地理解和应用Java编程。无论你是刚刚入门还是希望巩固基础,这些知识点都是必不可少的。
问题四:final关键字的真正含义是什么?
final关键字使得被修饰的变量不可变,但需要注意的是,对于对象型变量,final只保证引用本身的不变,而不是引用指向的对象不变。
引用本身的不变:
final StringBuffer a = new StringBuffer("immutable");
final StringBuffer b = new StringBuffer("not immutable");
a = b; // 编译期错误
引用指向的对象不变:
final StringBuffer a = new StringBuffer("immutable");
a.append(" broken!"); // 编译通过
可见,final只对引用的“值”(即它所指向的对象的内存地址)有效,它迫使引用只能指向初始指向的那个对象,改变它的指向会导致编译期错误。至于它所指向的对象的内容变化,final是不负责的。这类似于==操作符:==操作符只负责引用的“值”相等,至于这个地址所指向的对象内容是否相等,==操作符是不管的。
理解final问题有很重要的含义。许多程序漏洞都基于此——final只能保证引用永远指向固定对象,不能保证那个对象的状态不变。在多线程的操作中,一个对象会被多个线程共享或修改,一个线程对对象无意识的修改可能会导致另一个使用此对象的线程崩溃。一个错误的解决方法就是在此对象新建的时候把它声明为final,意图使得它“永远不变”。其实这是徒劳的。
问题五:如何正确初始化变量?
本问题讨论变量的初始化,首先来看一下Java中有哪些种类的变量。
1. 类的属性,或者叫成员变量
2. 方法里的局部变量
3. 方法的参数
对于第一种变量,Java虚拟机会自动进行初始化。如果给出了初始值,则初始化为该初始值。如果没有给出,则把它初始化为该类型变量的默认初始值。
int类型变量默认初始值为0
float类型变量默认初始值为0.0f
double类型变量默认初始值为0.0
boolean类型变量默认初始值为false
char类型变量默认初始值为0(ASCII码)
long类型变量默认初始值为0L
所有对象引用类型变量默认初始值为null,即不指向任何对象。注意数组本身也是对象,所以没有初始化的数组引用在自动初始化后其值也是null。
对于两种不同的类属性,static属性与instance属性,初始化的时机是不同的。instance属性在创建实例的时候初始化,static属性在类加载,也就是第一次用到这个类的时候初始化,对于后来的实例的创建,不再次进行初始化。这个问题会在以后的系列中进行详细讨论。
对于第二种变量,必须明确地进行初始化。如果在没有初始化之前就试图使用它,编译器会报错。如果初始化的语句在try块中或if块中,也必须要让它在第一次使用前一定能够得到赋值。也就是说,把初始化语句放在只有if块的条件判断语句中编译器会报错,因为执行的时候可能不符合if后面的判断条件,如此一来初始化语句就不会被执行了,这就违反了局部变量使用前必须初始化的规定。但如果在else块中也有初始化语句,就可以通过编译,因为无论如何,总有至少一条初始化语句会被执行,不会发生使用前未被初始化的事情。对于try-catch也是一样,如果只有在try块里才有初始化语句,编译不通过。如果在catch或finally里也有,则可以通过编译。总之,要保证局部变量在使用之前一定被初始化了。所以,一个好的做法是在声明它们的时候就初始化它们,如果不知道要初始化成什么值好,就用上面的默认值吧!
其实第三种变量和第二种本质上是一样的,都是方法中的局部变量。只不过作为参数,肯定是被初始化过的,传入的值就是初始值,所以不需要初始化。
问题六:instanceof关键字的作用是什么?
instanceof是Java的一个二元操作符,类似于==, >, <等操作符。由于它是由字母组成的,所以也是Java的保留关键字。它的作用是测试它左边的对象是否是它右边的类的实例,返回boolean类型的数据。举个例子:
String s = "I AM an Object!";
boolean isObject = s instanceof Object;
我们声明了一个String对象引用,指向一个String对象,然后用instanceof来测试它所指向的对象是否是Object类的一个实例,显然是真的,所以返回true,也就是isObject的值为true。
instanceof有一些用处。例如,在编写一个处理账单的系统时,有这样三个类:
public class Bill {}
public class PhoneBill extends Bill {}
public class GasBill extends Bill {}
在处理程序中有一个方法,接受一个Bill类型的对象,计算金额。假设两种账单计算方法不同,而传入的Bill对象可能是两种中的任何一种,所以可以用instanceof来判断:
public double calculate(Bill bill) {
if (bill instanceof PhoneBill) {
// 计算电话账单
}
if (bill instanceof GasBill) {
// 计算燃气账单
}
}
这样就可以用一个方法处理两种子类。然而,这种做法通常被认为是没有很好地利用面向对象中的多态性。实际上,上述功能要求用方法重载完全可以实现,这是面向对象编程应有的做法,避免回到结构化编程模式。只要提供两个名字和返回值都相同,但接受参数类型不同的方法就可以了:
public double calculate(PhoneBill bill) {
// 计算电话账单
}
public double calculate(GasBill bill) {
// 计算燃气账单
}