探秘static——类不需实例化就能用?

简介: 一般情况下,需要用到某个类的时候,都必须先实例化这个类,才能对其进行调用。在编程过程中发现,有些类不用实例化就可以直接拿来使用,使用它的字段、方法等等。

      一般情况下,需要用到某个类的时候,都必须先实例化这个类,才能对其进行调用。在编程过程中发现,有些类不用实例化就可以直接拿来使用,使用它的字段、方法等等。


      这时候,靠的就是static作用。static英文意思为“静止的,静态的”,在OOP中可以作为修饰符,类、字段、属性、方法等被static修饰后,变为静态类、静态字段、静态属性、静态方法……


      static修饰的类成为静态类,静态类中只能包含静态成员(被static修饰的字段、属性、方法),不能被实例化,不能被继承;非静态中可以包含静态成员。  


     1、被调用时必须先实例化的情况:


      被调用成员为非静态成员(此时它所属的类肯定为非静态类)。如下小例子:


    public class ClassA      //ClassA类(非静态类)
    {
        public ClassA() { }  //构造函数
        public void Fun() { }//ClassA类中的方法(非静态方法)
    }
    public class ClassB     //需要调用ClassA类中方法的ClassB类
    {
        public ClassB() { }  //构造函数
        public void Fun()  
        {
            ClassA a = new ClassA();//调用ClassA类中的方法需要先实例化
            a.Fun();
        }
    }


     说明:ClassA类为非静态类,其中的方法Fun()也为非静态方法,所以在ClassB中调用时需要先实例化ClassA类。


      2、被调用时不需要实例化的情况:

      被调用成员为静态成员(此时它所属的类为静态类或非静态类)。如下小例子:


      (1)被调用类为非静态类:

    public class ClassA      //ClassA类(非静态类)
    {
        public ClassA() { }  //构造函数
        public static void Fun() { }//ClassA类中的方法(静态方法)
    }
    public class ClassB     //需要调用ClassA类中方法的ClassB类
    {
        public ClassB() { }  //构造函数
        public void Fun()  
        {
            ClassA.Fun();  //调用ClassA类中的方法直接调用:类名.成员
        }
    }


      说明:ClassA类为非静态类,但其中的方法Fun()为非静态方法,所以在ClassB中调用时不实例化ClassA类(而且不能实例化),直接调用其成员,语法为“类名.成员”。


      (2)被调用类为静态类:


    public static class ClassA      //ClassA类(静态类)
    {
        //当然静态类中不能存在构造函数
        public static void Fun() { }//ClassA类中的方法(静态方法)
    }
    public class ClassB     //需要调用ClassA类中方法的ClassB类
    {
        public ClassB() { }  //构造函数
        public void Fun()  
        {
            ClassA.Fun();  //调用ClassA类中的方法直接调用:类名.成员
        }
    }

     说明:ClassA类为静态类,其中的方法也Fun()为非静态方法,所以在ClassB中调用时不实例化ClassA类(而且不能实例化),直接调用其成员,语法为“类名.成员”。


      3、static修饰符(拓展):


       (1) 用来修饰类或类的成员,这时不需要创建实例就能访问(而且不能实例化),在被调用的时候自动实例化,且在内存中产生一个实例。当含有静态成员的非静态类实例化出对象后,这些对象公用这些静态成员,通过类名或对象名都能访问它们。

       在网上看到两个有趣的小例子:

       人是一个非静态类。人有脑子,这是一个用static修饰的属性。

这个属性是针对所有的人,是人的共同特征。而不是某个特点对象所特有的(比如张三有脑子,李四没脑子),因为只要是个人,他都有脑子(虽然卡洛斯·罗德里格兹只有半个脑子,还有些人比较脑残→_→,那也算是有脑子),既然是人类共有的,那么他就不能被实例化。


32.jpg


      另外一个能被实例化的例子,还是人这个类,人的身高,就是一个非static的属性。因为每个人的身高是不同的。比如我身材魁梧,高达1.55米,这个1.55米是描述我的身高,是跟我这个特定的对象有联系的。姚明才2.26米,这是姚明这个对象的数据。不管是1.55还是2.26,这都和特定的对象有联系,而不是人类所共有的特征。所以非static的可以被实例化,而static不能被实例化。


33.jpg


      (2) 修饰方法内部的静态变量:

      我们的代码都是从硬盘加载到内存中才能运行的,在内存中主要分为三个区域,来存放我们的代码,分别是堆、栈和静态存储区。堆中存放的是代码中的引用类型变量,如类的实例、数组等;栈中存放的是代码中的值类型,如整型、浮点型、布尔型等;静态存储区中存放静态变量和全局变量、常亮。


      整个程序运行时,代码都是共用静态存储区中的静态变量的,例如定义存款余额为静态变量,同一个银行卡号,无论你是在银行取钱、在ATM机取钱,还是网银消费,用的都是这个静态的余额。


      所以,方法内部的静态变量,执行完静态变量值不消失,再次执行此对象的方法时,值仍存在,它不是在堆和栈中分配的,是在静态区非配的, 这是与局部变量最大的区别。关于内存分配问题可参考《静态存储区、堆和栈的区别》。


相关文章
|
存储 编解码 缓存
webgl系列之抗锯齿和深度缓存
前言 大家好我是Fly 哥, 这是今年webgl 系列的第三篇文章, 如果你之前的两篇文章没看的话,建议先看一下,然后再来看这一篇文章 Webgl 系列之buffer的使用 webgl系列之对光栅化的理解 上一篇文章,任何虚拟3维世界的转换到二维屏幕中通过「采样」 也就判断屏幕上的每个像素中心点是不是在三角形内部的得到了 下面这幅图: 图片 走样之前 这时候有同学问, 这不像三角形哇, 这个其实用个专业的词—— 「锯齿」 , 我的理解 一个三角形经过光栅化后, 得到屏幕上每一个像素点 组成的像素点的集合。那到底是经过什么样的处理得到下面这张图: 图片 final 反走样 其实出现上面
webgl系列之抗锯齿和深度缓存
|
7月前
|
虚拟化
【2025最新】VMware Workstation Pro 虚拟机基础配置教程,方便你的神操作!
VMware Workstation Pro 是一款强大的虚拟化工具,本文简要介绍其基本配置与操作。安装后,用户可以通过“编辑菜单 > 首选项”设置虚拟机保存路径、是否显示托盘图标等全局配置。新建虚拟机时,需选择操作系统类型、配置处理器、内存、网络和硬盘等参数。建议根据主机硬件性能合理分配资源,避免过度占用。创建完成后,用户可加载 ISO 镜像文件安装所需操作系统。通过这些步骤,您可以轻松搭建并管理多个虚拟环境。
1042 8
【2025最新】VMware Workstation Pro 虚拟机基础配置教程,方便你的神操作!
|
存储 C语言
数据结构中的线性表链式存储介绍及其基本操作
链式存储是线性表的一种重要存储方式,它通过节点和指针的结构,实现了灵活的动态存储管理。本文介绍了单向链表的基本操作,并提供了相应的C语言代码示例。理解和掌握链表的操作对学习和应用数据结构具有重要意义。希望这篇博客能帮助你更好地理解线性表的链式存储。
313 2
|
关系型数据库 MySQL Apache
实时计算 Flink版操作报错之mysql整库同步到doris连接器报错,如何解决
在使用实时计算Flink版过程中,可能会遇到各种错误,了解这些错误的原因及解决方法对于高效排错至关重要。针对具体问题,查看Flink的日志是关键,它们通常会提供更详细的错误信息和堆栈跟踪,有助于定位问题。此外,Flink社区文档和官方论坛也是寻求帮助的好去处。以下是一些常见的操作报错及其可能的原因与解决策略。
|
数据采集 存储 安全
数据治理的重要性与实施策略
【8月更文挑战第23天】数据治理是企业在数字化时代面临的重要挑战和机遇。通过加强数据治理工作,企业可以提升数据质量、保障数据安全、促进数据共享与流通以及符合法规要求。为了实施有效的数据治理工作,企业需要制定明确的数据治理战略、建立完整的数据治理框架、引入先进的技术工具、加强员工培训和意识提升以及建立监督和评估机制。只有这样,企业才能充分发挥数据的价值,为企业的决策和业务创新提供有力支持。
|
12月前
|
存储 安全 NoSQL
Cookie、Session、Token 解析
Cookie、Session、Token 解析
451 1
|
Java 测试技术 数据库连接
maven依赖详解
maven依赖详解
429 1
|
存储 缓存 JavaScript
提升Blazor应用性能的探索之旅:深入解析关键技巧与最佳实践
【8月更文挑战第31天】在开发现代Web应用时,性能与用户体验至关重要。Blazor作为一款使用.NET构建交互式Web UI的框架,提供了诸多便利。为了充分发挥其潜力并优化体验,掌握一些性能提升技巧十分必要。本文将分享几个实用的Blazor性能优化方法,包括减少不必要的服务器端调用、使用懒加载以及优化DOM操作。通过这些技巧,可以显著提升应用性能,为用户提供更流畅的体验。以下是具体方法及示例代码。
234 0
|
关系型数据库 MySQL Docker
构建MySQL8.0.26镜像和容器
MySQL8.0.26 percona-toolkit Dockerfile
321 3
|
存储 NoSQL MongoDB
【MongoDB 专栏】如何高效使用 MongoDB 的索引
【5月更文挑战第10天】MongoDB的索引是提升查询性能的关键,它基于B树结构,分为单字段、复合、多键和文本索引。创建索引可通过`createIndex()`或管理工具,适用于频繁查询、排序分组和连接操作。优化策略包括选择合适字段、避免过度索引和定期评估。注意索引影响写入性能、大小限制及可能的失效情况。通过案例分析,应根据业务需求合理创建和使用索引,以实现最佳性能。
219 1
【MongoDB 专栏】如何高效使用 MongoDB 的索引