Java基础知识
一.final 与 Static
- public static final 类常量,在类内部多个方法中可用,若为public 在其他类当中也可用
- 静态数据区不属于堆,当然更不属于对象,在静态数据区
- 如果编译器能够在编译阶段确定某变量的值,那么编译器就会把该变量当做编译期常量来使用
- 静态方法在类加载的时候就存在了,它不依赖于任何实例
- 只能访问所属类的静态字段和静态方法,方法中不能有 this 和 super 关键字
二.java缓存池
1. 涉及知识点
- 自动装箱与拆箱
- 声明方法及值的比较
- 整型(I,S,L,Character,Byte有缓存池,浮点型没有
- 缓存池的大小
2. 参考文章
三.String 以及 StringPool
1.涉及知识点
- 生成的过程,new的话若pool里没有,会自动添加到pool但是返回的还是堆里的地址
- 向string pool添加的过程
- 常量计算
- intern方法的使用
- 不会被GC回收
- 字符串是常量,字符串池中的每个字符串对象只有唯一的一份,可以被多个引用所指向,避免了重复创建内容相同的字符串;通过字面值赋值创建的字符串对象存放在字符串池中,通过关键字new出来的字符串对象存放在堆中。
String a = "hello";
string b = "lo";
System.out.println(a == "hel"+b); //false 在运行时通过连接计算出的字符串是新创建的,因此是不同的
System.out.println(a == "hel"+"lo");//true 由常量表达式计算出的字符串是在编译时进行计算,然后被当作常量
/**
* string 与 final
* final修饰的b相当于常量,是在编译时就确定的,所以b和d指向的是StringPool当中的同一个字符串
* 所以在计算c的时候,相当于常量表达式九三得到的值,所以也是StringPool当中的“hello2”,与a相同
* 而e则是在栈中通过连接新建的字符串
*/
public static void name() {
String a = "hello2";
final String b = "hello";
String d = "hello";
String c = b + 2;
String e = d + 2;
System.out.println((a == c));
System.out.println((a == e));
}
- string 不可变,线程安全
- stringBuffer 线程安全,内部使用 synchronized 进行同步
- stringBuider 线程不安全
2.参考文章
四.参数传递
1. 涉及知识点
- java的传参是call by value
- 对于基本数据类型,是在栈里复制一份
- 对于对象的引用,在栈里复制的是对象的引用,形参可以修改对象当中的参数,当指向新的对象,则与原实无关
2. 参考文章
五. 继承
1.涉及知识点
- super调用父类同名变量及方法,显示调用父类构造器
- 父类的构造器调用以及初始化过程一定在子类的前面
- 父类和子类的class文件都载入到内存,当父类和子类有Static时。先初始化Static,再初始化子类的Static,再初始化父类的其它成员变量->父类构造方法->子类其它成员变量->子类的构造方法。
- 如果是final方法,private 方法,static方法,则不会进行动态绑定,知道调用哪一个方法
- 其他方法会进行动态绑定,则会根据类型选择合适的方法
- 成员变量进行初始化过程如下:默认初始化-> 显示初始化 -> 构造方法初始化
- 静态代码块 > 构造代码块 > 构造方法
- 先进行父类初始化,然后进行子类初始化
- 父类(静态变量、静态语句块)
- 子类(静态变量、静态语句块)
- 父类(实例变量、普通语句块)
- 父类(构造函数)
- 子类(实例变量、普通语句块)
- 子类(构造函数)
2.参考文章
六.重写与重载
1.知识点
- this.show(O)
- super.show(O)
- this.show((super)O)
- super.show((super)O)
- 其中若B继承A,然后有A a2 = new B(); 则a2保存的是B的引用,但是a2的类型是A,this指向的也是A
2.文章
习题解答
隐藏与覆盖
隐藏与覆盖最好的解释版本
七.equals()与hashCode()
1.知识点
- 对于基本类型,== 判断两个值是否相等,基本类型没有 equals() 方法。
- 对于引用类型,== 判断两个变量是否引用同一个对象,而 equals() 判断引用的对象是否等价。
- 检查是否为同一个对象的引用,如果是直接返回 true;
- 检查是否是同一个类型,如果不是,直接返回 false;
- 将 Object 对象进行转型;
- 判断每个关键域是否相等。
- 在覆盖 equals() 方法时应当总是覆盖 hashCode() 方法,保证等价的两个对象散列值也相等。
7.clone
1.知识点
- 克隆方法用于创建对象的拷贝,为了使用clone方法,类必须实现java.lang.Cloneable接口重写protected方法clone,如果没有实现Clonebale接口会抛出CloneNotSupportedException.
- 在克隆java对象的时候不会调用构造器
- java提供一种叫浅拷贝(shallowcopy)的默认方式实现clone,创建好对象的副本后然后通过赋值拷贝内容,意味着如果你的类包含引用类型,那么原始对象和克隆都将指向相同的引用内容,这是很危险的,因为发生在可变的字段上任何改变将反应到他们所引用的共同内容上。为了避免这种情况,需要对引用的内容进行深度克隆。
- 使用 clone() 方法来拷贝一个对象即复杂又有风险,它会抛出异常,并且还需要类型转换。Effective Java 书上讲到,最好不要去使用 clone(),可以使用拷贝构造函数或者拷贝工厂来拷贝一个对象。
2. 文章
clone讲解
八.反射
1.知识点
- 反射的核心是 JVM 在运行时才动态加载类或调用方法/访问属性,它不需要事先(写代码的时候或编译期)知道运行对象是谁
- Java反射的主要功能
- 在运行时判断一个对象所属的类,instanceof
- 在运行时构造任意类的对象,使用Class对象的newinstance方法构造Object对象
- 判断一个类所具有的成员变量和方法,getMethods,getDeclaredMethods,getFields,getDeclaredMethods
- 在运行时调用任意一个对象的方法
九.泛型
1.知识点
- 什么是泛型以及泛型的好处:操作的数据类型被指定为一个参数,这种参数类型可以用在类、接口和方法中,分别被称为泛型类、泛型接口、泛型方法。
- 类型擦除: Java 的泛型在编译器有效,在运行期被删除,也就是说所有泛型参数类型在编译后都会被清除掉
- 泛型数组初始化时不能声明泛型类型
- instanceof()不允许使用泛型参数
- 泛型的class都是相同的,例如:
List<String> ls = new ArrayList<String>();
List<Integer> li = new ArrayList<Integer>();
System.out.println(ls.getClass() == li.getClass()); //true
- 类型参数并不能用来创建对象或是作为静态变量的类型
- 泛型的限定通配符以及非限定通配符 PECS原则,带有超类的通配符可以向泛型当中写入,带有子类型限定的通配符可以从泛型对象读取
- 利用类型参数创建实例的做法编译器不会通过
2.相关文章
十.注解
1.知识点
- 定义该Annotation时使用了@Retention和@Target两个系统元注释,其中@Retention注释指定Test注释可以保留多久,@Target注释指定Test注释能修饰的目标(只能是方法)
- 仅仅使用注释来标识程序元素对程序是不会有任何影响的,这也是Java注释的一条重要原则
-
标记Annotation 程序通过判断该Annotation来决定是否运行指定方法,没有任何成员变量
-
元数据Annotation 简化事件编程
- 注解更像是一种特殊的接口,注解的属性定义方式就和接口中定义方法的方式一样,而应用了注解的类可以认为是实现了这个特殊的接口
2.相关文章