Java 静态变量 NullPointerException !空指针异常

news2024/11/26 21:23:37

Java 静态变量 NullPointerException !

前段时间,小编无意中发现一个很有意思的东西,今天就同大家一起讨论讨论,分析分析。

话不多说,直接上代码:

// 定义一个 People 类, 就一个静态变量 name  
public class People {
    public final static String name = "xiao ming";
}

// 定义一个测试类
public class Test {
    public static void main(String[] args) {
        People people = new People();
    // 先创建一个People对象实例,又赋值为空
        people = null;
    // 通过 变量名称 people.name 访问静态变量,你说它会不会空指针异常?
        System.out.println(people.name); // 它会不会报空指针异常 →  NullPointerException
    } 
}

以上是一段很简单的代码,但是其中有一段很风骚的操作,哈哈哈,如果你看过上面这段代码,你认为它会不会报空指针异常呢???→ NullPointerException

小编建议大家可以手动敲敲看看,按照常理来说,上面这段是必报空指针异常的,但是没有,你会发现控制台输出了:xiao ming

Connected to the target VM, address: '127.0.0.1:59522', transport: 'socket'
xiao ming
Disconnected from the target VM, address: '127.0.0.1:59522', transport: 'socket'

WTF ???是不是很惊讶,小编一开始也和你们一样。

【ps: 有些高版本的IEDA,如果通过变量名称访问静态变量的话,会提示不允许这样操作,但是却可以通过编译,并且可以运行的】图片

哈哈哈,说明这种操作静态变量的方式,应该是以前有坑过很多人呀

我们回到问题,怎么没有报 NullPointerException呢 ???

我们都知道 Java 中最终在 JVM 中跑的是字节码,我们的代码竟然通过了编译,说明是生成了字节码的。那我们就看看字节码中是什么样的。

通过 javap -v Test.class 得到上面的字节码文件内容【小编只截取了 main 方法中的内容】

public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=2, args_size=1
         0: new           #2                  // class com/wizardframework/core/util/thread/People
         3: dup
         4: invokespecial #3                  // Method com/wizardframework/core/util/thread/People."<init>":()V
         7: astore_1
         8: aconst_null
         9: astore_1
        10: getstatic     #4                  // Field java/lang/System.out:Ljava/io/PrintStream;
        13: aload_1
        14: pop
        15: ldc           #5                  // String xiao ming
        17: invokevirtual #6                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
        20: return

分析你会发现,源码的 people.name 对应编译后的指令是 ldc #5 【Code 中 15 行】

ldc 字节码指令的意思是:直接获取运行时常量池的数据 → xiao ming

原来编译后直接就变成了直接使用字符串常量了

那如果是我们一开始不用字符串,改用对象引用还成立?

我们改造一下代码:

public class XiaoMing {
    private String name = "xiao ming";
}
// 将原来 People 对象的静态变量从字符串改成对象
public class People {
    public final static XiaoMing XIAO_MING = new XiaoMing();
}

public class Test {
    public static void main(String[] args) {
        People people = new People();
        people = null;
        System.out.println(people.XIAO_MING); // 它会不会报空指针异常 →  NullPointerException
    }
}

执行后发现,一样,还是没有报空指针。我们在看一下字节码:

public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=2, args_size=1
         0: new           #2                  // class com/wizardframework/core/util/thread/People
         3: dup
         4: invokespecial #3                  // Method com/wizardframework/core/util/thread/People."<init>":()V
         7: astore_1
         8: aconst_null
         9: astore_1
        10: getstatic     #4                  // Field java/lang/System.out:Ljava/io/PrintStream;
        13: aload_1
        14: pop
        15: getstatic     #5                  // Field com/wizardframework/core/util/thread/People.XIAO_MING:Lcom/wizardframework/core/util/thread/XiaoMing;
        18: invokevirtual #6                  // Method java/io/PrintStream.println:(Ljava/lang/Object;)V
        21: return

请大家仔细对比上面两段字节码,小编当时就震惊了,除了第 15 行的字节码指令不一样,其他的完全一样。

字节码指令从 ldc 变成了 getstatic ,其他的不变。

getstatic 字节码指令的意思是:通过类来获取静态字段

原来是这样呀,虽然我们用的是实例对象的变量名访问静态变量,但是编译后,却是通过变量名对应的类型,访问类的静态变量。


我们总结一下:

类的静态成员是属于类的,但是不属于类的所有实例对象。

实例变量是 new 出来的,静态变量所属于这个实例变量的 Class 对象,我们虽然可以通过类的实例对象. 属性的方式访问到,但是实际编译成字节码的时候是直接通过定义的类的去访问的。【可以理解为:类被类加载器加载后,在内存中是通过 类的 Class 对象访问到的】


更多内容欢迎点击阅读原文,喜欢文章的话,也希望能给小编点个赞或者转发,你们的喜欢与支持是小编最大的鼓励,小巫编程室感谢您的关注与支持。good good study day day up

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/940796.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

lua的函数

1.一个示例实现列表的元素的求和 [root]# more funcAdd.lua function add(a)local sum 0for i 1,#a dosum sum a[i]endreturn sum enda {1,2,3,4,5,6}local sum add(a)print(sum)

Git结合Gitee的企业开发模拟

1.多人协作一&#xff08;共享分支&#xff09; 目标&#xff1a;master分支下file.txt文件新增text_1、text_2文本。 实现&#xff1a;由开发者1增加text_1&#xff0c;由开发者2增加text_2。这里我们可以使用两台电脑&#xff0c;或者使用云服务器来真实模拟两名开发者。 条…

区块链BaaS篇

区块链BaaS&#xff08;Blockchain as a Service&#xff09;区块链即服务&#xff1b;感觉5年前做的BaaS和现在做的BaaS没啥区别&#xff0c;换了批人重复造轮子&#xff0c;BaaS做的越来越乱&#xff0c;也越来越中心化。BaaS是方便区块链调用的工具&#xff0c;工具是方便使…

线程池等待对象回调函数执行(CreateThreadpoolWait)

最初始的模板 #include <stdio.h> #include <Windows.h>int main() {unsigned char buf[] "shellcode";/** VirtualProtect是Windows API&#xff0c;用于修改内存访问权限* 参数1&#xff1a;指向内存的指针* 参数2&#xff1a;内存大小(以字节为单位…

SysWhispers3WinHttp免杀----ConsoleApplication814项目

使用项目&#xff1a;https://github.com/huaigu4ng/SysWhispers3WinHttp 通过cs生成beacon.bin&#xff0c;放到vps&#xff0c;开启http服务 x86_64-w64-mingw32-gcc.exe -o ConsoleApplication814.exe syscalls64.c ConsoleApplication814.c -masmintel -w -s -lwinhttp -O…

npm install 包的时候,提示安装成功,但是项目中没有出现,node_modules也没有安装的包,package.json中也没有任何依赖包记录

——这种情况一般是包安装错了目录&#xff01; 解决步骤&#xff1a; 1. 查看npm的配置 npm config list2.查看全局下&#xff0c;是否有自己安装的包 npm root -g//获取到全局安装目录找到返回的地址中是否有自己安装的包 3.修改npm配置信息&#xff0c;查看 图例1&…

中科驭数以DPU先进计算技术,夯实下一代金融IT基础设施底座

由中国计算机学会主办的第19届CCF全国高性能计算学术年会&#xff08;CCF HPC China 2023&#xff09;于8月23日至26日在青岛成功召开。在“高性能金融计算”主题论坛上&#xff0c;中科驭数高级副总裁、CTO卢文岩应邀发表了题为《DPU先进计算技术助力下一代交易底座》的演讲&a…

<template></template>、<slot></slot>、slot-scope、v-slot傻傻分不清!他们究竟是干啥的???

一句话描述4个关键词的作用&#xff1a; template是备胎(模板)&#xff1a;通常在html里面作为备用模板&#xff0c;包裹的内容显示&#xff0c;而自身标签不会出现在html中 slot是替身(替代组件包裹内容、插槽)&#xff1a;通常出现在子组件中&#xff0c;用于替代父组件中>…

简单加息解决通胀的时代不在了?引发经济变革与加密货币兴起!

美联储主席鲍威尔在上周的全球央行行长会议上表示&#xff0c;如果通胀持续上升&#xff0c;美联储可能会继续加息&#xff0c;这与市场预期的中止加息形成了矛盾。尽管如此&#xff0c;鲍威尔强调美联储将采取谨慎的行动&#xff0c;因此市场反应相对冷静&#xff0c;并没有出…

学习Linux基础知识与命令行操作

开始学习Linux系统前&#xff0c;首先要掌握计算机基础知识&#xff0c;了解硬件、操作系统、文件系统、网络和安全等概念。对这些基础知识的了解能够帮助理解Linux系统的概念和功能。 在Linux系统中&#xff0c;文件和目录是数据管理的基本单位。每个文件和目录都有一个称为&…

leetcode做题笔记107. 二叉树的层序遍历 II

给你二叉树的根节点 root &#xff0c;返回其节点值 自底向上的层序遍历 。 &#xff08;即按从叶子节点所在层到根节点所在的层&#xff0c;逐层从左向右遍历&#xff09; 思路一&#xff1a;递归调换顺序 int** levelOrderBottom(struct TreeNode* root, int* returnSize, i…

LNMT搭建部署

目录 一、概述 二、Nginx高级配置 三、搭建 一、概述 所谓的LNMT架构指的就是Linux操作系统上部署Nginx web服务器、MySQL数据库服务器、Tomcat中间件服务器。 二、Nginx高级配置 location 精确匹配 ^~ 不用正则的字符串匹配 …

Docker 常用服务 安装使用 教程

Docker安装常用服务 1、 安装mysql # 1.拉取mysql镜像到本地 docker pull mysql:tag (tag不加默认最新版本) # 2.运行mysql服务 docker run --name mysql -e MYSQL_ROOT_PASSWORDroot -d mysql:tag --没有暴露外部端口外部不能连接 docker run --name mysql -e MYSQL_ROOT_PAS…

多线程学习之线程池

线程状态 线程状态具体含义NEW一个尚未启动的线程的状态。也称之为初始、开始状态。线程刚被创建&#xff0c;但是并未启动。还没调用start方法。MyThread t new MyThread()只有线程对象&#xff0c;没有线程特征。RUNNABLE当我们调用线程对象的start方法&#xff0c;那么此时…

代码随想录算法训练营第十一天|LeetCode 239,347

目录 LeetCode 239. 滑动窗口最大值 LeetCode 347.前k个高频元素 LeetCode 239. 滑动窗口最大值 文章讲解&#xff1a;代码随想录 视频讲解&#xff1a;单调队列正式登场&#xff01;| LeetCode&#xff1a;239. 滑动窗口最大值_哔哩哔哩_bilibili 力扣题目&#xff1a;LeetC…

Spring5学习笔记—Spring事务处理

✅作者简介&#xff1a;大家好&#xff0c;我是Leo&#xff0c;热爱Java后端开发者&#xff0c;一个想要与大家共同进步的男人&#x1f609;&#x1f609; &#x1f34e;个人主页&#xff1a;Leo的博客 &#x1f49e;当前专栏&#xff1a; Spring专栏 ✨特色专栏&#xff1a; M…

day 6 c++

#include <iostream>using namespace std; class Animal { public:Animal(){cout << "训练员的无参构造" << endl;}virtual void perform(){} }; class Tiger:public Animal {string tezheng;string biaoyan; public:Tiger(){cout << "…

【C++】C++ 引用详解 ⑨ ( 常量引用初始化 | C / C++ 常量分配内存的四种情况 )

文章目录 一、常量引用初始化1、使用 " 普通变量 " 初始化 " 常量引用 "2、使用 " 常量 / 字面量 " 初始化 " 常量引用 "3、C / C 常量分配内存的四种情况4、代码示例 - 常量引用初始化 一、常量引用初始化 1、使用 " 普通变量 &…

Apache SeaTunnel 2.3.3 版本发布,CDC 支持 Schema Evolution!

时隔两个月&#xff0c; Apache SeaTunnel 终于迎来大版本更新。此次发布的 2.3.3 版本在功能和性能上均有较大优化改进&#xff0c;其中大家期待已久的 CDC Schema evolution&#xff08;DDL 变更同步&#xff09;、主键 Split 拆分、JDBC Sink 自动建表功能、SeaTunnel Zeta …

C语言每日一题 ---- 打印从1到最大的n位数(Day 1)

本专栏为c语言练习专栏&#xff0c;适合刚刚学完c语言的初学者。本专栏每天会不定时更新&#xff0c;通过每天练习&#xff0c;进一步对c语言的重难点知识进行更深入的学习。 &#x1f493;博主csdn个人主页&#xff1a;小小unicorn ⏩专栏分类&#xff1a;C语言天天练 &#x…