finalize

前言

今天又看了下java编程思想,听实习期间“老大”的,这本书需要看三年,这也是我买这本书的第三年了(没有完整的看完过),每年都会看下,但是每次看感觉都不一样,第一年看是头大,第二年看是勉强能看懂一部分,第三年看会带着工作中的实践去思考,渐入佳境,时间过的太快了我还是没看完,今年必须得看完了,明年再看一遍就不再看了。

今天看到finalize有一个实例一直搞不出来,问了下我们公司得架构师,架构师说我怎么还在看这个,说我方向可能错了搞得太深了,应该去看框架源码、jvm内核啥得,我问的问题实际工作遇到太少了。但是我总觉得连java本身一些基础的知识都不了解(这里只是了解并没有深入)的比较全面会阻碍我未来思考问题的广度,源码我也是看的,但是是遇到问题或者感兴趣才去看,至于jvm也一直在看也是断断续续的,这可能和我自己的性格也是有关系的,总是想知道的比较清楚。

主题

finalize在实际工作中确实没有遇到过,但是还是要先了解下,书上是说这个是用来做一些除了new出来对象的清除,比如一些通过本地方法开辟出来的内存消耗,但是调用这个方法是对对象产生了gc回收的时候,等下一次才会真正回收该对象,同时也可以做一些gc回收之前对对象的检验,比如根据对象的字段值检验下对象是否能被回收,不能回收可以打印日志或者抛出异常等。下面看例子

  static class Person {
        boolean checkedOut = false;

        Person(boolean checkedOut) {
            this.checkedOut = checkedOut;
        }

        void checkIn() {
            checkedOut = false;
        }

        @Override
        protected void finalize() throws Throwable {
            if (checkedOut) {
                System.out.println("Error:checked out");
            }
            super.finalize();
        }
    }

    public static void main(String[] args) throws InterruptedException {
        Person p1 = new Person(true);
        p1.checkIn();
        Person p2 = new Person(true);
          
        System.gc();
    }

执行结果

"C:\Program Files\Java\jdk1.8.0_60\bin\java" ...

Process finished with exit code 0

并没有打印出来

后来折腾了下,改为:

 static class Person {
        boolean checkedOut = false;

        Person(boolean checkedOut) {
            this.checkedOut = checkedOut;
        }

        void checkIn() {
            checkedOut = false;
        }

        @Override
        protected void finalize() throws Throwable {
            if (checkedOut) {
                System.out.println("Error:checked out");
            }
            super.finalize();
        }
    }

    public static void main(String[] args) throws InterruptedException {
        Thread t1=new Thread() {
            @Override
            public void run() {
                Person p1 = new Person(true);
                p1.checkIn();
                Person p2 = new Person(true);

            }
        };
        t1.start();
        Thread.sleep(10);
        System.gc();
        Thread.sleep(10);
    }

就可以了

"C:\Program Files\Java\jdk1.8.0_60\bin\java" ...
Error:checked out

Process finished with exit code 0

解释下改动点:

  • 1:new对象和发送消息用开启了新的线程,不和gc回收都放在主线程是因为都在本方法提虚拟机怕后面还会使用到对象。
  • 2:第一个Thread.sleep(10),因为开启新线程了,需要等new对象和发送消息完成了才进行gc。
  • 3:第二个Thread.sleep(10),因为gc后面没有其他要执行的了,主线程都已经结束看不到执行finalize的打印。

开始并没有反应过来还是因为对jvm的执行过程不够理解,这也充分说明了基础可以不牢固,同样可以写业务代码,但是有一天发生了一些不可解释的问题时总归结于“神学”或者重启电脑,哈哈哈。

你可能感兴趣的:(java复习笔记)