0
点赞
收藏
分享

微信扫一扫

final 、finally、finalize有什么区别

九点韶留学 2022-06-23 阅读 61


final 是可以用来修饰类、方法、变量、分别不同的意义。 

  • final 修饰的类class代表不可以继承
  • final 修饰的变量表示不可修改
  • final 修饰的方法表示不可重写(override)

finall 是Java中保证代码一定要被执行的一种机制,我们可以使用 try-finally 或者 try-catch-finally 来进行类似关闭JDBC,unlock 锁等动作。

finalize 是基础类 java.lang.object 的一个方法,他的设计目的是保证对象在垃圾收集前完成特定资源的回收。finalize机制不推荐使用。JDK9 被标记为 deprecated。



final使用有什么好处?


final 变量产生了某种程序的不可变效果,可以用来保护只读数据。尤其在并发编程中,可以明确地不能赋值 final 变量,有利于减少额外的同步开销。


try{
// do something
System.out.println("do Something");
// system.exit(0):正常退出,程序正常执行结束退出
// system.exit(1):是非正常退出,就是说无论程序正在执行与否,都退出,
System.exit(1);
}finally{
System.out.println("Print from finally");
}


这个比较特殊 上面 finally 里面的代码不会被执行。


如何实现一个不可变类

将class 本身声明为 final ,这样就不能被继承扩展

将所有成员变量定义为 private 和 final ,并且不要实现 setter

通常构造对象是,成员变量使用深度拷贝来初始化,而不是直接赋值。因为你无法确定输入对象不被其他人修改。

如果确实需要实现getter 方法。获知其他可能返回内部状态的方法,使用 copy-on-write (写时复制)原则。Java 有 CopyOnWriteArrayLIst 实现


栗子:

public boolean add(T e) {
final ReentrantLock lock = this.lock;
lock.lock();
try {

Object[] elements = getArray();

int len = elements.length;
// 复制出新数组

Object[] newElements = Arrays.copyOf(elements, len + 1);
// 把新元素添加到新数组里

newElements[len] = e;
// 把原数组引用指向新数组

setArray(newElements);

return true;

} finally {

lock.unlock();

}

}
final void setArray(Object[] a) {
array = a;
}


为啥说不推荐使用  finalize?


Java 平台正在使用 java.lang.Cleaner 类替换 finalize 实现 ,Cleaner 的实现使用 虚引用(PhantomRefrence),这个是一种常见的所谓 post-mortem 清理机制。


主要原因是 finalize 会掩盖资源回收时的出错信息 看 java.lang.ref.Finalizer 源码


private void runFinalizer(JavaLangAccess jla) {
synchronized (this) {
if (hasBeenFinalized()) return;
remove();
}
try {
Object finalizee = this.get();
if (finalizee != null && !(finalizee instanceof java.lang.Enum)) {
jla.invokeFinalize(finalizee);

/* Clear stack slot containing this variable, to decrease
the chances of false retention with a conservative GC */
finalizee = null;
}
} catch (Throwable x) { }
super.clear();
}



Throwable 生吞了一切已成,及时出现异常或者出错,也得不到任何有效信息。


JDK9 平台使用了 Clearner 来替换原来的 finalize 实现


public class CleaningExample implements AutoCloseable {
// A cleaner, preferably one shared within a library
private static final Cleaner cleaner = <cleaner>;

static class State implements Runnable {

State(...) {
// initialize State needed for cleaning action
}

public void run() {
// cleanup action accessing State, executed at most once
}
}

private final State;
private final Cleaner.Cleanable cleanable

public CleaningExample() {
this.state = new State(...);
this.cleanable = cleaner.register(this, state);
}

public void close() {
cleanable.clean();
}
}



Cleaner 采用 虚引用技术,但是我们平时还是尽量不怎么用,这个只是最后的保障。并不能完全依赖 Cleaner做内存回收。












举报

相关推荐

0 条评论