当前位置: 首页 > news >正文

java基础 关键字static

static

  • static使用简介
  • static结合类的生命周期
    • 1.加载
    • 2.链接
      • (1) 验证(Verification)
      • (2) 准备(Preparation)
      • (3) 解析(Resolution)
    • 3. 初始化
    • 4.使用
    • 5.卸载
    • 总结
  • staic作用总结
    • 静态变量
    • 静态代码块
    • 静态方法
    • 静态内部类

static使用简介

作用域:static(静态)可以声明 内部类 方法 变量 代码块

	// static 修饰变量
    static int b = 20;

	// static 代码块
    static {
        System.out.println("静态代码块执行");
    }

	// static 方法
    static int initA() {
        return 10;
    }
    
    // 静态内部类
	static class StaticInner {}

static结合类的生命周期

想完整了解static功能,需要先了解类的生命周期包括
1.加载(Loading)
2.链接(Linking):又分验证(Verification)准备(Preparation)和解析(Resolution)三个子阶段
3.初始化(Initialization)
4.使用(Using)
5.卸载(Unloading)

1.加载

类加载简化过程:.java类文件-->java.exec编译器-->.class文件-->类加载器ClassLoader加载
						-->.class 字节码文件加载到内存,生成 Class 对象(存储在方法区)

将 .class 字节码文件加载到内存,生成Class 对象(存储在方法区),注意是存储在jvm方法区
触发条件:需要主动使用类时触发(如 new、访问静态成员、反射等调用,虚拟机入口类)

// 首次new对象时触发加载
MyClass obj = new MyClass();

通过调用Class.forName(ClassName)方法,其中ClassName是类的全限定名

String className = "com.example.MyClass";
try {
    Class<?> clazz = Class.forName(className);
    // 现在你可以使用clazz对象来操作MyClass类了
} catch (ClassNotFoundException e) {
    e.printStackTrace();
}

访问静态成员

    public static int staticVariable = 10;

    public static void main(String[] args) {
        // 访问静态变量
        System.out.println("Static variable value: " + staticVariable);
    }

虚拟机入口类:main方法

public class Demo{
    public static void main(String[] args) {
        System.out.println("main 方法执行");// 并没有创建类对象实例,但已经进行类加载
    }
}

初始化子类,父类被加载,这个不进行代码示例了,类加载的触发条件说白了,需要使用到该类再进行加载。类加载的原则:延迟加载,能不加载就不加载

2.链接

(1) 验证(Verification)

目标:确保字节码符合 JVM 规范,防止恶意代码或错误格式等

(2) 准备(Preparation)

目标:为类的静态变量分配内存并赋默认值(零值)

static int value = 123;  // 准备阶段 value = 0,等到初始化阶段赋值为 123

(3) 解析(Resolution)

目标:常量化: 将常量池中(final修饰)的符号引用(如类名、方法名)替换为直接引用(内存地址)

String s = "Hello";  // 符号引用 "Hello" 转为实际字符串对象地址

3. 初始化

目标:执行类的初始化代码(静态变量赋值和静态代码块)
关键过程:
①JVM 生成 方法,合并所有静态变量赋值静态代码块
②若存在父类,先初始化父类(接口不会触发父接口初始化)

public class MyClass {
    static {
        System.out.println("静态代码块执行");
    }
    static int value = 20;
    
    public static void main(String[] args) {
        // 运行会发现 value为20  并且 静态代码块里面的内容执行了
        System.out.println("Static variable value: " + value);
    }
}

注意:
1.这个时候还没有对象实例,对象实例是在栈中,而这里静态变量没有对象可以依靠,存在方法区
2.初始化这里进行的是 给静态变量赋值 和 静态代码块,静态方法并不会自动初始化执行

静态方法的执行规则
1.静态方法不会自动执行:除非被显式调用(如通过类名调用 MyClass.staticMethod())或在初始化阶段通过静态变量赋值或静态代码块间接调用

2.若静态方法未被任何代码触发:它永远不会执行

public class Test {
    static {
        System.out.println("静态代码块执行");
    }
    // 静态方法不会自动执行
    public static void staticMethod() {
        System.out.println("静态方法被调用");
    }
    
    public static void main(String[] args) {
        System.out.println("程序启动");
    }
}

输出打印

静态代码块执行
程序启动

说明静态方法在初始化时候并不会主动执行,下面显式调用方式

public class StaticTest {
    static int a = initA();         // 静态变量 调用静态方法赋值

    static {
        System.out.println("静态代码块");
        System.out.println(initA()); // 静态代码块 调用静态方法
    }

    static int initA() {
        System.out.println("静态方法 initA 被调用");
        return 10;
    }


    public static void main(String[] args) {
        System.out.println("main 方法执行");
        StaticTest.initA();  // 显式调用静态方法
    }
}

打印

静态方法 initA 被调用
静态代码块
静态方法 initA 被调用
10
main 方法执行
静态方法 initA 被调用

4.使用

目标:通过实例化、方法调用等方式使用类

public class MyClass {
    { 
      System.out.println("非静态代码块"); 
    }
    
    public MyClass() {
     System.out.println("构造函数"); 
    }
	
	public  void method() {
        System.out.println("非静态method方法运行");
    }

    public static void main(String[] args) {
       System.out.println("main运行");// 仅运行入口main方法 不创建对象实例
    }    

    public static void main(String[] args) {
        MyClass obj = new MyClass();  // 输出:非静态代码块 → 构造函数
    }
}

关键行为:
首先实例化:调用构造函数 ,这个时候进行初始化实例变量和非静态代码块

new对象--》设定个类名 通过类名找Class对象 -->初始化实例变量和非静态代码块 -->分配内存

其次方法调用显示调用执行实例方法或静态方法。这点一样 方法都需要显示调用执行
最后字段访问:读写静态或实例变量,在创建对象实例后,静态或非静态变量都初始化完成都可以访问了,只不过静态变量在类加载就进行初始化,而实例变量是在创建对象实例后才进行初始化

5.卸载

目标:从内存中移除不再使用的类

触发条件(需同时满足)
1.类的所有实例已被回收。
2.类的 Class 对象无引用(如无反射访问)。

加载该类的 ClassLoader 已被回收,即Class对象不在了,如果再次需要使用该类,那么上述的从加载到卸载流程再来一遍

总结

1.类加载只发生一次,实例化可多次进行,实例化对象(Object)需依靠类加载(需要引用Class对象)

2.类加载生成Class对象,如果类中包含内部类,会生成多个Class对象,Class对象即类本身,一个类对应一个Class对象,存在jvm的方法区

3.静态代码块在初始化阶段执行,静态变量在链接 准备阶段赋值默认值,在初始化阶段重新赋静态值

4.静态方法的执行需要显示调用,具体的运行阶段取决于执行者

staic作用总结

静态变量

上面我们了解到 静态变量在链接 准备阶段赋值默认值,在初始化阶段重新赋静态值,由于类加载只进行一次,内存中只有独一份的静态变量值,所以类所有实例共享同一份数据(内存中只有一份拷贝)

静态代码块

类加载时执行一次,用于初始化静态资源,也仅执行一次

静态方法

注意Static的使用注意规则
1、静态只能访问静态。
2、非静态既可以访问非静态的,也可以访问静态的。

①需要显示调用才会运行,这点和实例方法没有区别,可直接通过类名调用
②由于静态只能访问静态,所以在类初始化时,可以通过静态变量或静态代码块调用 静态方法,从而让静态方法也可以在类初始化阶段运行,这点实例方法做不到,因为静态不能访问非静态的
③静态方法不能访问实例成员(非静态)(无 this 上下文)

静态内部类

这里特别需要说明的是 静态内部类,在Java中,将一个类的定义放在另外一个类的定义内部,这就是内部类,添加了静态修饰的内部类就是静态内部类

public class TopLevelClass {  
    // 静态内部类
    static class StaticInner { 
    	..................................
    }

	//调用静态内部类的方式
	TopLevelClass.StaticInner staticInner = new TopLevelClass.StaticInner();
	//或者
	StaticInner staticInner = new StaticInner();
}

考虑几个问题
1.为什么static修饰类只能修饰内部类,不能修饰外层类,如上TopLevelClass
2.为什么通过staic修饰内部类,即静态内部类的好处在哪里
3.静态内部类的加载运行 以及实例化 消亡时机都是什么

问题1.为什么static修饰类只能修饰内部类,不能修饰外层类,如上TopLevelClass
外层类是在包下面的,他的创建不依附于其余的类的实例,直接new就可以,本身是顶级结构。而static 表示“不依赖实例”,通过上面类加载是生命周期就可以看出,类没有创建实例之前在加载之后的初始化static修饰的就可以运行工作,所以外层类不依赖其余的类的实例,static 表示“不依赖实例”,外层类再使用static就属于多此一举了

问题2.为什么通过staic修饰内部类,即静态内部类的好处在哪里
这个要从成员内部类说,内部类就是一个类放在另一个类,那么内部类就属于外层类的成员,内部类有很多好处,比如封装性 隐藏性 实现回调函数 等等

成员内部类的就是非静态的内部类

public class Demo {
	// 非静态内部类:成员内部类
    class NoStaticInner {
    ----------------------------------------
    }
	// 调用方式
	Demo.NoStaticInner = new Demo.NoStaticInner();

	// 实际上就相当于
	Demo demo = new Demo();
	Demo.NoStaticInner noStaticInner = demo.new NoStaticInner(); // 通过外部实例.new创建
    }
}

成员内部类的实例化必须依赖外部类的实例,因为它隐含持有外部类对象的引用
1.成员内部类通过 demo.new 实例化,隐含持有外部类实例的引用(Demo.this)
2.好处是 可直接访问外部类的实例成员,因为非static隐含持有外部类实例的引用(Demo.this)

再来看静态内部类

public class Demo {
	// 静态内部类
    static class StaticInner {
    ----------------------------------------
    }
	// 直接通过类名实例化,无需外部类实例
	Demo.StaticInner staticInner= new Demo.StaticInner();

	// 实际上就相当于,和Demo实例化无关
	StaticInner staticInner= new StaticInner();
	// 因为在Demo类中,所以看上去是和成员内部类一样Demo.StaticInner,Demo.StaticInner表示路径
    }
}

静态内部类的实例化不依赖外部类的实例,因为它没有外部类对象的隐含引用
1.静态内部类通过 new Outer.StaticInner() 直接实例化。不持有外部类实例的引用,因此更节省内存
2.只能访问外部类的静态成员(如 staticOuterField)。static不持有外部类实例的引用(Demo.this)

总结下来就是
1.好处解耦了:成员内部类实例化依靠外层类,静态内部类实例化不依靠外层类
2.坏处是引用受限制: static不持有外部类实例的引用(Demo.this)不能访问外层类的非静态内容

问题3.静态内部类的加载运行 以及实例化 消亡时机都是什么
静态内部类也是类,加载运行实例化也和正常类一样,需要主动触发,但属于内部类,所以受外部类的管理,由外部类的类加载器卸载触发,通常发生在动态模块化环境(如热部署)中

相关文章:

  • 导引头是个啥
  • 反射 tcp
  • DrissionPage移动端自动化:从H5到原生App的跨界测试
  • Linux: 线程控制
  • 企业官网的管理后台包含哪些功能模块详细
  • 本地项目提交到gitee
  • uboot启动过程中无法被打断和进入uboot命令行问题解决(基于2017.09版本uboot)
  • 若依前后端分离版本从mysql切换到postgresql数据库
  • [CISSP] [8] 安全模型,设计和能力的原则
  • Kubernetes 节点磁盘空间空了怎么办?解决 containerd overlay 100%问题
  • 如何创建单独的城市活码?活码能永久使用吗?
  • 理解JSON-RPC 2.0 协议
  • 力扣HOT100之链表:24. 两两交换链表中的节点
  • 1. Git 下载和安装
  • docker compose安装智能体平台N8N
  • 蓝桥杯嵌入式按键长按双击
  • 408 计算机网络 知识点记忆(5)
  • Go 微服务框架 | 中间件
  • C++17模板编程与if constexpr深度解析
  • LLM实现模型并行训练:deepspeed 是什么; transformers` 怎么实现模型并行训练吗?
  • WordPress首页站内搜索/百度大数据查询平台
  • 西安短视频拍摄制作公司/网站整站优化公司
  • 做3d打印网站/互联网营销师是干什么
  • 武汉市市政建设集团网站/百度手机软件应用中心
  • 网络营销运营推广方案下载/短视频seo营销系统
  • 做网站的一般多钱/宁波seo搜索引擎优化