1.类加载和初始化
加载——>连接(验证->准备->解析)——>初始化。
若父类未初始化,则先初始化父类;
初始化一个类(或接口),不会先初始化它所实现的接口(或父接口);
当使用的静态变量或方法不在当前类或接口中定义,而在父类或接口中定义,则不会初始化当前类而只初始化父类;
总之,主动使用的才初始化。
2.类加载器
加载器间的父子关系实际上是加载器对象间的包装关系,而不是类之间的继承关系;
类加载器的命名空间由加载器及所有父加载器所加载的类组成,同一命名空间,类的完整名不同,不同命名空间,类的完整名可相同;
运行时包指加载器相同并且包全名也相同;
父加载器加载的类不能看见子加载器加载的类,但子加载器加载的类却能看见父加载器加载的类;
两个不同命名空间内的类相互不可见,可采用Java反射机制访问实例的属性和方法;
Class对象不再被引用,表示它生命周期结束,可以卸载类,Java虚拟机自带的类加载器(即根加载器、扩展加载器和系统加载器)所加载的类始终不卸载,用户自定义加载器所加载的类可以被卸载,静态变量的声明周期取决于Class对象的生命周期;
Class实例与它的类加载器相互双向关联;
new——>可触及状态<—(finalize)—>可复活状态——>不可触及状态—(垃圾回收)—>生命终止
3.线程
处理线程未捕获的异常UncaughtExceptionHandler以及ThreadLocal<T>;
线程运行机制:程序计数器PC,方法调用栈(局部变量区,操作数和栈数据区);
方法区和堆区为当前进程所有线程共享;
直接调用run方法,是主线程执行,start线程则是当前线程,所以不同;
不能重写start方法,即使重写也要在第一句调用super.start();
一个线程只能被start一次,否则抛出java.lang.IllegalThreadStateException异常;
线程结束或异常不会影响其他线程,主线程发生未处理的异常就会退出程序;
线程优先级不仅取决于Java虚拟机,还依赖于操作系统,若希望程序能移植到不同操作系统,应只用MAX_PRIORITY、MIN_PRIORITY和NORMAL_PRIORITY 3个级别;
4.异常
没有找到处理异常的代码块,若是主线程则调用异常对象的printStackTrace后终止应用程序,非主线程则终止当前线程而其他线程仍然正常运行;
只有发生异常时才对性能造成影响,主要是搜索方法调用栈寻找异常处理代码块,不应该用异常处理机制控制程序的正常流程,仅在可能出现异常的地方用try...catch,将异常处理代码块放在适当的层次,最好放在当前方法中;
try后可只跟finally而不跟catch;
catch从子类向父类排序,父类在子类前编译错误,因为子类异常永远执行不到;
方法有可能出现受检查异常,要么try...cacth,要么throws异常,运行时异常可不用这样;
throws语句后不能紧跟其他语句,因为执行不到;
finally块唯一不执行的情况是先执行了终止应用程序的exit或者强行终止进程;
在finally块中用return/throws/break/continue等改变方法出口的语句,会覆盖catch中的return语句,还可能会导致catch中的异常丢失;
异常转译指捕获原始异常,转换成新异常抛出;
处理多样化的异常,可以自定义异常类,用一个集合成员存储各种异常;
保持异常的原子性方法:检查参数,确保异常发生时没改变对象的初始状态;编写一段恢复代表;临时拷贝上操作,成功后用临时拷贝的内容覆盖原来的对象;
5.模式
标识类型模式,如 Serializable;
接口常量模式;
6.其他
this语句调用其他构造方法须满足:位于构造方法中;第一句;不能用构造方法名直接调用构造函数;
super语句类似;
禁止继承的方法:final;private构造方法;
为保证final类属性也不可变,可采用保护性拷贝clone方法;
finalize出现异常,垃圾回收器不报告异常,程序正常运行,不会中断;
通过对象实例而不是类去访问内部类;
实例内部类不能定义静态成员;
静态内部类能定义静态成员;
局部内部类不能定义静态成员;
内部类不存在覆盖;
基本数据类型,低位可以自动转换为高位,反之不能,如short a=1,b=2;short c=a+b;因为a+b是int型,不能转换为short型;
接口可以继承多个接口;