美文网首页Java学习笔记
java基本功16课:(2)对象与内存控制

java基本功16课:(2)对象与内存控制

作者: 蓝汝丶琪 | 来源:发表于2017-06-17 01:54 被阅读27次

    前言

    上一节提到了实例变量和实例对象的区别,那么这节就继续深入一下变量以及对象。
    主要内容:

    • 实例变量和类变量
    • 父,子类实例变量和内存分配机制
    • final 的注意事项

    1. 实例变量和类变量

    1.1 定义

    • 实例变量:在类里面没有使用static修饰的变量,例如:int i;也称为非静态变量
    • 类变量:在类里面使用了static修饰的变量,例如:static int i; 也称为静态变量

    1.2 区别

    • 首先在同一个JVM内每个类对应一个Class对象,也就是说,只会给该类划分一次内存空间,因此类变量才会只有一次初始化。相对于实例变量,当每次建立对象(实例),都会划分一块新的内存空间。
      所以,所有的该类的实例对象,都是共用一个类变量。

    1.3 初始化

    • 首先看一段代码
    class test{
        //1.weight 初始化的值是多小
        double weight=2.3;
        {
            weight=4.3;
        }   
        test(){
            this.weight=4.6;
        }
    
    
        // 2.向前引用(为什么)
        int num1=num2+2;
        static int num2=4;
        
    
        // 3.请说出name 初始化的值应该是什么
        static{
            name="我爱java";
        }
        static String name="给我丶鼓励";
    
    
    }
    
    
    • 有答案了吗?有疑问吗?
      • 首先得把该类的初始化过程理解清楚。第一步是建立划分一个类对象的内存空间,然后把变量初始化(优先是把类变量给初始化,然后再到实例变量初始化)。第二步,再对变量进行赋值,在赋值的过程中,是按照代码顺序由上往下赋值的。
      • 在了解完类初始化过程之后,第一个问题weight=4.6就很显然了。但是有趣的事,在编译的时候,是可以看到,其实编辑器是把weight=2.3,和非静态代码块放在构造方法里面了。顺序是由上往下排,所以结果还是4.6而已。
      • 为什么可以向前引用呢?那是因为在类初始化的时候,类变量是优先与实例变量初始化的,所以才可以向前引用。
      • name=“给我丶鼓励”,为什么呢?因为这里就很好地解释了初始化其实是两步。变量先初始化默认的值,例如int 是0,然后再把变量给赋值。所以name=null,然后name="我爱java",然后name="给我丶鼓励"。

    2. 父,子类实例变量和内存分配机制

    2.1 继承的初始化

    • 先看代码
    class father{
        int age=44;
    
       //构造方法
        father(){
            System.out.println("父类的构造输出"+this.age);
            this.say();
        }
        public void say(){
            System.out.println("父类的say"+this.age);
        }
    }
    
    class son extends father{
        int age=33;
       //构造方法
        son(){
          System.out.println("子类构造方法");
        }
        public void say(){
            System.out.println("子类的say:"+this.age);
        }
    }
    
    public class second{
        public static void main(String[] ages){
           son myson=new son();
    
        }
    }
    
    • 你觉得会输出什么?
      • 答案是
    父类的构造输出44
    子类的say:0
    子类构造方法
    
    • 为什么同一个this。第一行输出的是父类的44而第二行是子类的方法,而且是0呢? 这个首先涉及继承的初始化问题,以及继承变量的方法和变量的区别了。
    • 首先在初始化的时候,是优先把父类给初始化,因为在子类构造函数中,是先调用super()方法,此方法是把父类初始化。所以是整个初始化过程是 fater->son。所以首先输出的是:父类的构造方法输出44;
    • 为什么第二句是子类的say:0呢?
    "因为子类son把父类的say()重写了,因此,当调用的时候,是调用子类的say()的。"
    

    恩~其实这个说话不正确的。首先需要理解这个this是谁。其实这个this是son。不信?

    System.out.println(this.getClass());
    

    输出结果是:class son 。
    所以,this.say()调用的自然是son的say()方法了。
    那为什么输出是0呢?
    因为值此时赋值只是到父类,子类的age还没赋值,还是初始值0呢。

    2.2 继承的变量和方法的区别以及内存控制

    • 一句话,变量没有重写,方法有重写。
      • 用起来有什么区别呢? 还是刚刚的son类和fater类:
           son myson=new son();
           father myfather=myson;
    
           System.out.println(myson.age);
           myson.say();
           System.out.println(myfather.age);
           myfather.say(); 
          //将会输出什么
    
    33
    子类的say:33
    44
    子类的say:33
    
    • 为什么?因为,变量没有重写,方法有重写。所以父类还是可以调用自己的变量,而方法是调用子类的方法。

    • 细心的同学注意到,为什么

    father myfather=myson
    

    明明是new 子类,但却可以输出父类的age呢?

    • 那是因为在建立子类对象的时候,在划分地址的时候,也有一块内存地址是存父类的变量,因此。子类中的super(),也是可以调用父类的变量的值。(可以试试噢~)

    3. final 的注意事项

    3.1 final方法不能被重写,只能被程序显式地赋值一次。
    3.2 final最大作用是"宏替换"
    3.3 在内部类中的局部变量,需要用final。

    相关文章

      网友评论

        本文标题:java基本功16课:(2)对象与内存控制

        本文链接:https://www.haomeiwen.com/subject/mihvqxtx.html