public class NullAS{ private static void a(){ System.out.println("hello word"); } public static void main(String[] args) { NullAS N = null; N.a(); } }
这段代码最终的输出结果是:
hello Word
有些人可能会有疑问,使用null对象调用所属类的静态方法,没有报错,还可以正常输出
这其中就涉及到了Java的类,类成员和static
类成员包括类变量,类方法等
让我们来看看类变量的定义:
“类变量属于整个整个类,当系统第一次准备使用该类时,系统会为该类变量分配内存空间,类变量开始生效,直到该类被卸载,该类的类变量所占用的内存才被系统的垃圾回收机制回首。类变量生存范围几乎等同于该类的生存范围。当类初始化完成后,类变量也初始化完成”
当我们看一下上面代码的分配内存情况可知
当系统创建该类的对象时,系统不会再为类变量分配内存,也不会再次对垒变量进行初始化,也就是说,对象根本不拥有对应类的类变量。通过对象访问类变量只是一种假象。当通过对象来访问类变量时,系统会在底层转换为该类来访问类变量。
由于对象实际上并不持有类变量,类变量时由该类持有的,同一个类的所有对象访问类变量时,实际上访问的都是该类的类变量。从程序运行表面来看,即可看到同一类的所有势力的类变量共享同一块内存区。
类方法也是类成员的一种,与类变量相似。
当使用实例来访问类成员时,实际上依然是委托给该类来访问类成员,因此,即使某个实例为null,它也可以访问它所属类的类成员。
当然,如果一个null对象访问实例成员,那可是会报NullPointerException异常的
类初始模块也是类成员的一种,类初始化块用于执行类初始化动作,在类的初始化阶段,系统会调用该类的类初始化块来对类进行初始化。一旦该类初始化结束后,类初始化块讲永远不会获得执行的机会。
对于static而已,类成员不能访问实例成员。因为类成员是属于类的,类成员的作用域比实例成员的作用域更大,完全可能出现类成员以及初始化完成,但实例成员还不曾初始化的情况,如果允许成员访问实例成员将会引发大量错误
所以,实际上Java和很多语言一样,对象只能访问实例变量,类变量通过类来访问,只是它给使用者造成了一种假象
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。