• java 重新学习 (二)


    一.栈内存里的引用变量并未真正存储对象的成员变量,对象的成员变量数据实际存放在堆内存中,而引用变量只是指向该堆内存里的对象。

    二.堆内存里的对象可以有多个引用,若果堆内存中没有变量指向该对象,程序无法再次访问该对象,该对象就变为垃圾等待回收。

    三.  普通代码块:在方法或语句中出现的{}就称为普通代码块。普通代码块和一般的语句执行顺序由他们在代码中出现的次序决定--“先出现先执行”

       构造块:直接在类中定义且没有加static关键字的代码块称为{}构造代码块。 构造代码块在创建对象时被调用每次创建对象都会被调用,并且构造代码块的执行次序优先于类构造函数

       //静态代码块:在java中使用static关键字声明的代码块。静态块用于初始化类,为类的属性初始化。

       //每个静态代码块只会执行一次。由于JVM在加载类时会执行静态代码块,所以静态代码块先于主方法执行。
       //如果类中包含多个静态代码块,那么将按照"先定义的代码先执行,后定义的代码后执行"。
       //注意:1 静态代码块不能存在于任何方法体内。2 静态代码块不能直接访问静态实例变量和实例方法,需要通过类的实例对象来访问。

       //   系统在类初始化阶段执行静态初始化块时,不仅会执行本类的静态初始化块,还会一直追溯到其父类的静态初始化块……最后才执行本类的静态初始化块。

        对象初始化阶段,会执行最顶端的父类初始化块,对顶端的父类构造器,然后依次向下,直到执行当前类的初始化块,当前类的构造器。

        

    class Root {
            static {
                System.out.println("Root静态初始化块");
            }
            {
                System.out.println("Root普通初始化块");
            }
            public Root() {
                System.out.println("Root无参构造器");
            }
        }
    
        class Mid extends Root {
            static {
                System.out.println("Mid静态初始化块");
            }
            {
                System.out.println("Mid普通初始化块");
            }
            public Mid() {
                System.out.println("Mid无参构造器");
            }
        }
    
        class Leaf extends Mid {
            static {
                System.out.println("Leaf静态初始化块");
            }
            {
                System.out.println("Leaf普通初始化块");
            }
            public Leaf() {
                System.out.println("Leaf无参构造器");
            }
        }
    View Code

        如果初始化Leaf 对象:输出结果为

     Root静态初始化块 - Mid静态初始化块-Leaf静态初始化块   -    Root普通初始化块 -Root无参构造器 - Mid普通初始化块-Mid无参构造器--Leaf普通初始化块-Leaf无参构造器。如果在初始化依次Leaf对象将不再执行父类与本身的静态代码块,输出结果为上述去掉静态代码块部分。

    四. 方法参数传递机制:值传递,引用传递 看堆栈分配区分

    五.java1.5 形参个数可变 在方法里最后一个形参的类型加上三个点(...)表明该形参可以接受多个参数值。public void test(String...test);

    六.初始化对象时会先给类变量(类)分配内存空间(分配到堆内存中)。不过也可以通过P1 和P2 调用类变量,但是本质还是通过Person类即通过Person类的内存访问的。为了避免歧义尽量不要用对象调用类变量。

       

    七。局部变量不属于任何实例和类,因此它总是保存在其所在方法的栈内存中,如果局部变量时基本类型的变量,则直接把这个变量的值保存在该变量对应的内存中。

    如果局部变量时引用类型的变量,则这个变量里存放的是地址,通过该地址引用到该变量实际引用的对象或数组。栈内存中的变量无需系统垃圾回收,往往随着方法或代码块的运行结束而结束。

    八.封装的目的:隐藏类的实现细节;可进行数据检查,从而有利保证对象信息的完整性;便于修改,提高代码可维护性;将对象的成员变量和实现隐藏起来,不允许外界直接访问;把方法暴露出来,让方法来控制对这些成员变量进行安全的访问和操作。

    private :同一个类中

    default :同一个类中;同一个包中

    protected:同一个类中,同一个包中,子类中

    public:同一个类中,同一个包中,子类中,全局范围内;

    九. 继承

      当程序创建一个子类对象时,系统不仅会为该类中定义的实例变量分配内存,也会为他从父类继承得到的所有是咧变量分配空间。

      子类构造器执行体重既没有super调用,也没有this调用,系统会在执行子类构造器之前,隐式调用父类无参的构造器。

      创造任何对象总是从该类所在继承树最顶层类的构造器开始执行。然后依次向下执行,最后才执行本类的构造器

    十.多肽

      引用变量在编译阶段只能调用器编译时类型锁具有的方法,但运行是则执行它运行时类型所具有的方法。eg:Object c = new  Person(); 只能调用object类的方法

      但是 运行时的方法 是子类的方法 表现为多肽  

      通过引用变量来访问其包含的实例变量时,系统总是试图访问它编译时类型所定义的成员变量,而不是它运行时所定义的成员变量。

       eg:父类有个字段a=1  子类有个相同名字的字段a=2 .   Base b = new Sub();     System.out.println(b.a); 输出结果为 1

  • 相关阅读:
    【ECMAScript】循环
    【VUE】新建子应用时修改文件
    es 的reindex迁移索引
    js compare 函数参数的位置
    OpenShift 与 OpenStack:让云变得更简单
    Golang 常见设计模式之装饰模式
    悲报, GIF 之父因新冠去世
    rbac模型
    nginx reload无效解决方法
    设计模式【消息事件篇】
  • 原文地址:https://www.cnblogs.com/wenbuzhu/p/9869806.html
Copyright © 2020-2023  润新知