做网站的标准流程视觉设计师作品集
news/
2025/9/22 20:16:37/
文章来源:
做网站的标准流程,视觉设计师作品集,免费空间列表,培训加盟文章目录1、static存在的主要意义2、static的独特之处3、static应用场景4、静态变量和实例变量的概念5、静态变量和实例变量区别【重点常用】6、访问静态变量和实例变量的两种方式7、static静态方法8、static静态代码块9、static变量与普通变量区别10、静态内部类11、静态导包1…
文章目录1、static存在的主要意义2、static的独特之处3、static应用场景4、静态变量和实例变量的概念5、静态变量和实例变量区别【重点常用】6、访问静态变量和实例变量的两种方式7、static静态方法8、static静态代码块9、static变量与普通变量区别10、静态内部类11、静态导包12、static注意事项13、final与static的藕断丝连提到static关键字相信大家都不陌生这是相对比较难以理解的一个关键字相信各位也都能深深感受的到本篇文章将好好总结一下static这个关键字。
在开始讲static之前我想让各位看一段有意思的代码
public class Test {static{System.out.println(test static 1);}static{System.out.println(test static 2);}public static void main(String[] args) {}
}看完程序小白童鞋发话了啥玩意main方法中啥都没有能运行啥博主你个星星星…
运行结果
test static 1
test static 2小白童鞋:那啥…那啥…博主我说啥了我啥都没说…
其实上面的代码懂的自然懂不懂的自然就不懂了因为上面的代码涉及到JVM的类加载了当然不在本篇博客文章的范畴内如果有兴趣理解上面的程序这篇文章可能会对你有所帮助
这篇文章绝对让你深刻理解java类的加载以及ClassLoader源码分析
1、static存在的主要意义
static的主要意义是在于创建独立于具体对象的域变量或者方法。以致于即使没有创建对象也能使用属性和调用方法
static关键字还有一个比较关键的作用就是 用来形成静态代码块以优化程序性能。static块可以置于类中的任何地方类中可以有多个static块。在类初次被加载的时候会按照static块的顺序来执行每个static块并且只会执行一次。
为什么说static块可以用来优化程序性能是因为它的特性:只会在类加载的时候执行一次。因此很多时候会将一些只需要进行一次的初始化操作都放在static代码块中进行。
2、static的独特之处
1、被static修饰的变量或者方法是独立于该类的任何对象也就是说这些变量和方法不属于任何一个实例对象而是被类的实例对象所共享。 怎么理解 “被类的实例对象所共享” 这句话呢就是说一个类的静态成员它是属于大伙的【大伙指的是这个类的多个对象实例我们都知道一个类可以创建多个实例】所有的类对象共享的不像成员变量是自个的【自个指的是这个类的单个实例对象】…我觉得我已经讲的很通俗了你明白了咩 2、在该类被第一次加载的时候就会去加载被static修饰的部分而且只在类第一次使用时加载并进行初始化注意这是第一次用就要初始化后面根据需要是可以再次赋值的。
3、static变量值在类加载的时候分配空间以后创建类对象的时候不会重新分配。赋值的话是可以任意赋值的
4、被static修饰的变量或者方法是优先于对象存在的也就是说当一个类加载完毕之后即便没有创建对象也可以去访问。
3、static应用场景
因为static是被类的实例对象所共享因此如果某个成员变量是被所有对象所共享的那么这个成员变量就应该定义为静态变量。
因此比较常见的static应用场景有 1、修饰成员变量 2、修饰成员方法 3、静态代码块 4、修饰类【只能修饰内部类也就是静态内部类】 5、静态导包 以上的应用场景将会在下文陆续讲到…
4、静态变量和实例变量的概念
静态变量 static修饰的成员变量叫做静态变量【也叫做类变量】静态变量是属于这个类而不是属于是对象。
实例变量 没有被static修饰的成员变量叫做实例变量实例变量是属于这个类的实例对象。
还有一点需要注意的是static是不允许用来修饰局部变量不要问我问什么因为java规定的
5、静态变量和实例变量区别【重点常用】
静态变量 静态变量由于不属于任何实例对象属于类的所以在内存中只会有一份在类的加载过程中JVM只为静态变量分配一次内存空间。
实例变量 每次创建对象都会为每个对象分配成员变量内存空间实例变量是属于实例对象的在内存中创建几次对象就有几份成员变量。
6、访问静态变量和实例变量的两种方式
我们都知道静态变量是属于这个类而不是属于是对象static独立于对象。
但是各位有木有想过静态成员变量虽然独立于对象但是不代表不可以通过对象去访问所有的静态方法和静态变量都可以通过对象访问【只要访问权限足够允许就行】不理解没关系来个代码就理解了
public class StaticDemo {static int value 666;public static void main(String[] args) throws Exception{new StaticDemo().method();}private void method(){int value 123;System.out.println(this.value);}}猜想一下结果我猜你的结果是123哈哈是咩其实
运行结果 666当然肯定有一些基础非常扎实的大佬会问为什么会有同学认为输出是123 。里面定义的value123只是一个很普通的局部变量而已 。和成员变量没有半毛钱关系 。和静态变量也没有关系。是的确实如这位大佬说的一样只是博主我说的同学是小白同学站在小白童鞋的角度上所以还望大佬理解见谅我举的这个StaticDemo例子主要目的是说明一下this也是可以访问static的变量的value123只是一个跑龙套的配角旨在让小白同学认清其中的关系。
回过头再去品味一下上面的那段话你就能非常客观明了了这个思想概念要有只是这种用法不推荐
因此小结一下访问静态变量和实例变量的两种方法
静态变量 类名.静态变量 对象.静态变量(不推荐) 静态方法 类名.静态方法 对象.静态方法(不推荐) 7、static静态方法
static修饰的方法也叫做静态方法不知道各位发现咩有其实我们最熟悉的static静态方法就是main方法了小白童鞋喔好像真的是哦。由于对于静态方法来说是不属于任何实例对象的this指的是当前对象因为static静态方法不属于任何对象所以就谈不上this了。
还有一点就是构造方法不是静态方法
8、static静态代码块
先看个程序吧看看自个是否掌握了static代码块下面程序代码继承关系为 BaseThree—— BaseTwo—— BaseOne
BaseOne类
package com.gx.initializationblock;public class BaseOne {public BaseOne() {System.out.println(BaseOne构造器);}{System.out.println(BaseOne初始化块);System.out.println();}static {System.out.println(BaseOne静态初始化块);}}BaseTwo类
package com.gx.initializationblock;public class BaseTwo extends BaseOne {public BaseTwo() {System.out.println(BaseTwo构造器);}{System.out.println(BaseTwo初始化块);}static {System.out.println(BaseTwo静态初始化块);}
}BaseThree 类
package com.gx.initializationblock;public class BaseThree extends BaseTwo {public BaseThree() {System.out.println(BaseThree构造器);}{System.out.println(BaseThree初始化块);}static {System.out.println(BaseThree静态初始化块);}
}测试demo2类
package com.gx.initializationblock;/*注这里的ABC对应BaseOne、BaseTwo、BaseThree * 多个类的继承中初始化块、静态初始化块、构造器的执行顺序在继承中先后执行父类A的静态块父类B的静态块最后子类的静态块然后再执行父类A的非静态块和构造器然后是B类的非静态块和构造器最后执行子类的非静态块和构造器*/
public class Demo2 {public static void main(String[] args) {BaseThree baseThree new BaseThree();System.out.println(-----);BaseThree baseThree2 new BaseThree();}
}运行结果
BaseOne静态初始化块
BaseTwo静态初始化块
BaseThree静态初始化块
BaseOne初始化块BaseOne构造器
BaseTwo初始化块
BaseTwo构造器
BaseThree初始化块
BaseThree构造器
-----
BaseOne初始化块BaseOne构造器
BaseTwo初始化块
BaseTwo构造器
BaseThree初始化块
BaseThree构造器至于static代码块运行结果不是很清晰的童鞋详细讲解请看这篇Static静态代码块以及各代码块之间的执行顺序
以上仅仅是让各位明确代码块之间的运行顺序显然还是不够的静态代码块通常用来对静态变量进行一些初始化操作比如定义枚举类代码如下
public enum WeekDayEnum {MONDAY(1,周一),TUESDAY(2, 周二),WEDNESDAY(3, 周三),THURSDAY(4, 周四),FRIDAY(5, 周五),SATURDAY(6, 周六),SUNDAY(7, 周日);private int code;private String desc;WeekDayEnum(int code, String desc) {this.code code;this.desc desc;}private static final MapInteger, WeekDayEnum WEEK_ENUM_MAP new HashMapInteger, WeekDayEnum();// 对map进行初始化static {for (WeekDayEnum weekDay : WeekDayEnum.values()) {WEEK_ENUM_MAP.put(weekDay.getCode(), weekDay);}}public static WeekDayEnum findByCode(int code) {return WEEK_ENUM_MAP.get(code);}public int getCode() {return code;}public void setCode(int code) {this.code code;}public String getDesc() {return desc;}public void setDesc(String desc) {this.desc desc;}
} 当然不仅仅是枚举这一方面还有我们熟悉的单例模式同样也用到了静态代码块如下
public class Singleton {private static Singleton instance;static {instance new Singleton();}private Singleton() {}public static Singleton getInstance() {return instance;}
}9、static变量与普通变量区别
static变量也称作静态变量静态变量和非静态变量的区别是静态变量被所有的对象所共享在内存中只有一个副本它当且仅当在类初次加载时会被初始化。而非静态变量是对象所拥有的在创建对象的时候被初始化存在多个副本各个对象拥有的副本互不影响。
还有一点就是static成员变量的初始化顺序按照定义的顺序进行初始化。
10、静态内部类
静态内部类与非静态内部类之间存在一个最大的区别我们知道非静态内部类在编译完成之后会隐含地保存着一个引用该引用是指向创建它的外围内但是静态内部类却没有。没有这个引用就意味着
1、它的创建是不需要依赖外围类的创建。 2、它不能使用任何外围类的非static成员变量和方法。
代码举例静态内部类实现单例模式
public class Singleton {// 声明为 private 避免调用默认构造方法创建对象private Singleton() {}// 声明为 private 表明静态内部该类只能在该 Singleton 类中被访问private static class SingletonHolder {private static final Singleton INSTANCE new Singleton();}public static Singleton getUniqueInstance() {return SingletonHolder.INSTANCE;}
}
当 Singleton 类加载时静态内部类 SingletonHolder 没有被加载进内存。只有当调用 getUniqueInstance()方法从而触发 SingletonHolder.INSTANCE 时 SingletonHolder 才会被加载此时初始化 INSTANCE 实例并且 JVM 能确保 INSTANCE 只被实例化一次。
这种方式不仅具有延迟初始化的好处而且由 JVM 提供了对线程安全的支持。
11、静态导包
静态导包格式import static
这两个关键字连用可以指定导入某个类中的指定静态资源并且不需要使用类名调用类中静态成员可以直接使用类中静态成员变量和成员方法
// Math. --- 将Math中的所有静态资源导入这时候可以直接使用里面的静态方法而不用通过类名进行调用
// 如果只想导入单一某个静态方法只需要将换成对应的方法名即可import static java.lang.Math.;
// 换成import static java.lang.Math.max;具有一样的效果public class Demo {public static void main(String[] args) {int max max(1,2);System.out.println(max);}
}静态导包在书写代码的时候确实能省一点代码可以直接调用里面的静态成员但是会影响代码可读性所以开发中一般情况下不建议这么使用。
12、static注意事项
1、静态只能访问静态。 2、非静态既可以访问非静态的也可以访问静态的。
13、final与static的藕断丝连
到这里文章本该结束了的但是static的使用始终离不开final字眼二者可谓藕断丝连常常繁见我觉得还是很有必要讲讲那么一起来看看下面这个程序吧。
package Demo;class FinalDemo {public final double i Math.random();public static double t Math.random();
}public class DemoDemo {public static void main(String[] args) {FinalDemo demo1 new FinalDemo();FinalDemo demo2 new FinalDemo();System.out.println(final修饰的 i demo1.i);System.out.println(static修饰的 t demo1.t);System.out.println(final修饰的 i demo2.i);System.out.println(static修饰的 t demo2.t);System.out.println(t1 demo2.t );
// System.out.println( demo2.i );//编译失败}
}
运行结果final修饰的 i0.7282093281367935static修饰的 t0.30720545678577604final修饰的 i0.8106990945706758static修饰的 t0.30720545678577604t1 1.307205456785776
static修饰的变量没有发生变化是因为static作用于成员变量只是用来表示保存一份副本其不会发生变化。怎么理解这个副本呢其实static修饰的在类加载的时候就加载完成了初始化而且只会加载一次也就是说初始化一次所以不会发生变化
至于final修饰的反而发生变化了是不是巅覆你对final的看法关于final详细讲解博主也准备好了一篇文章程序员你真的理解final关键字吗
ok,文章就先到这里了希望这篇文章能够帮助到你对static的认识若有不足或者不正之处希望谅解并欢迎批评指正
参考 《java编程思想》 http://baijiahao.baidu.com/s?id1601254463089390982wfrspiderforpc https://blog.csdn.net/qq_34337272/article/details/82766943 https://www.cnblogs.com/dolphin0520/p/3799052.html
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/910245.shtml
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!