Java 中的静态字段和静态方法 

目录
  • 一、静态变量
    • 静态变量声明
    • 静态变量测试
  • 二、静态方法
    • 静态方法测试
    • 静态方法是如何工作的
    • 为什么 Java Main 方法是静态方法
  • 三、静态常量
  • 四、总结

先看看下面这一串代码:

public class Main {
    public static void main(String[] args) {
        System.out.println("Hello, WOrld!");
    }
}

我们的 ​​main​​ 方法被标记了 ​​static​​ 修饰符,那么 ​​static​​ 这个修饰符的含义是啥呢?

一、静态变量

如果一个字段定义为 static,那么每个类只有一个这样的字段。

先来看看没有静态静态遍历的类:

class Student {
  int stuId;
  String name;
  String school = "HY No.1 High School";
}

假设高中有 1500 名学生,上述代码现在所有的实例数据成员在每次创建对象时都会得到内存。

所有的学生都有其独特的 stuId 和名字,这些实例数据成员在这种情况下是正确的,毕竟都是唯一的。

但是,这里的 "school " 是所有对象的共同属性。如果不声明为静态变量的话,也会占用多个内存。但如果我们把它变成静态的,这个字段将只获得一次内存。

静态变量声明

class Student {
  int stuId;  // 实例变量
  String name;
  static String school = "HY No.1 High School"; // 静态变量
}

如果你将任何变量声明为静态,它就被称为静态变量。

  • 静态变量可以用来指代所有对象的共同属性(对每个对象来说不是唯一的),例如,雇员的公司名称,学生的学院名称等。
  • 静态变量只在类加载时在类区获得一次内存。

静态变量测试

代码测试:

package com.yuzhou1su.RelearnJava;

class Student {
    int stuId; 
    String name;
    static String school = "HY No.1 High School";
  
    Student(int id, String n) {
        stuId = id;
        name = n;
    }

    void display() {
        System.out.println("Student id:" + stuId + ", Name:" + name + " is from " + school);
    }
}

public class TestVariable {

    public static void main(String[] args) {
        // TODO Auto-generated method stub
        Student s1 = new Student(001, "Karsa");
        Student s2 = new Student(002, "Ellen");

        s1.display();
        s2.display();
    }

}

执行结果:

Student id:1, Name:Karsa is from HY No.1 High School
Student id:2, Name:Ellen is from HY No.1 High School

静态变量只会获得一次内存,如果任何对象更改了静态变量的值,它将保留其值。

看如下代码:

package com.yuzhou1su.RelearnJava;

public class StaticVariableCount {
  
  static int count = 0;
  
  StaticVariableCount() {
    count++;
    System.out.println(count);
  }

  public static void main(String[] args) {
    // TODO Auto-generated method stub
    StaticVariableCount svc1 = new StaticVariableCount();
    StaticVariableCount svc2 = new StaticVariableCount();
    StaticVariableCount svc3 = new StaticVariableCount();
    
  }
}

测试结果:

1
2
3

二、静态方法

Java中的静态方法是一个属于类的方法,但不被认为是该类的实例;相反,Java中的静态方法可以很容易地被创建和实现,而不需要任何实例的调用。静态方法可以访问类中的任何数据成员,可以对数据成员进行任何操作,也可以将任何数值作为输入,尽管要访问的成员变量在类中应该有变量的范围,而方法只能是静态的。

public static void syntax_ex (String_name) {
    Body of the program for execution.
}
  • public。该类的访问修饰语是public。
  • static。方法的范围是静态的,这意味着所有的成员变量和返回类型都在静态的范围内。
  • void。语法流程中的这个关键字表示在当前方法中没有处理任何返回类型。
  • syntax_ex。类的名称,表示静态方法是当前定义的类的一部分,后面是字符串名称。
  • body。它包括整个核心逻辑或业务逻辑(如果需要在静态模式下)。

如果你在任何方法上使用静态关键字,它就被称为静态方法,

静态方法:

  • 静态方法属于类,而不是属于类的对象。
  • 静态方法可以被调用而不需要创建一个类的实例。
  • 静态方法可以访问静态数据成员,并可以改变它的值。

静态方法测试

如果我们想更改学习名字的操作呢?就可以声明一个静态方法。

package com.yuzhou1su.RelearnJava;

class Student {
    int stuId; 
    String name;
    static String school = "HY No.1 High School";
    
    static void changeSchool() {
        school = "HY No.5 High School";
    }
  
    Student(int id, String n) {
        stuId = id;
        name = n;
    }

    void display() {
        System.out.println("Student id:" + stuId + ", Name:" + name + " is from " + school);
    }
}

public class TestVariable {

    public static void main(String[] args) {
        // TODO Auto-generated method stub
        
        Student.changeSchool();
        Student s1 = new Student(001, "Karsa");
        Student s2 = new Student(002, "Ellen");

        s1.display();
        s2.display();
    }

}

测试结果:

Student id:1, Name:Karsa is from HY No.5 High School
Student id:2, Name:Ellen is from HY No.5 High School

静态方法是如何工作的

  • 静态方法和实例方法是 Java 中的两种方法,它们在程序员中造成了一些混乱,但这仅仅是一种误解。静态方法和实例方法都有很大的区别。让我们看看静态方法在 Java 中是如何工作的。Java 中的静态方法是一种驻留在类中的方法,即使没有创建对象或没有进行实例化,也可以访问。在类的名称后面加上方法的名称并传递参数,就可以访问该类的任何实例。
  • 它可以被表示为​​ClassName.methodName(arguments)​​ 。此外,这些方法的组成有一个目标,即该方法应可与类中的所有成员变量和每个人的对象共享,其范围由修改器 ​​static​​ 定义。这些方法没有任何重载的能力;相反,它们可以在编译时使用编译器的静态绑定来重载,每当程序员需要在类的所有实例、对象或成员变量之间共享一个共同的代码片段时,静态方法就会成为救星,因为它通过创建一个共同的静态范围来为所有成员、对象和变量创造一种共享的规定。
  • 一个类的所有静态字段都可以使用静态字段作为一个类的静态方法的一部分进行访问。另外,静态方法也与内存分配功能有关,并且也是可以支持的。它将静态方法字段和变量的一部分与一些永久生成的堆存储在内存中,用于关联值。内存分配不支持对象作为静态方法堆的创建,或者方法本身不支持实例化。但接下来的问题是,静态方法是如何通过共享和创建所有成员的范围作为类的一部分来工作的。

为什么 Java Main 方法是静态方法

这是因为对象不需要调用静态方法。如果是非静态方法,JVM 会先创建一个对象,然后调用 ​​main()​​ 方法,这会导致额外内存分配的问题。

main 方法不对任何对象进行操作,事实上,在启动程序时还没有任何对象。静态的 main 方法将执行并构造程序所需要的对象。

三、静态常量

静态变量使用的比较少,但静态常量却很常用。比如 Math 类中定义一个静态常量:

public class Math {
  public static final double PI = 3.14159265358979;
}

然后在程序中,可以用 Math.PI 来访问这个常量。

四、总结

Java 中为什么需要静态变量?

答:每当我们希望为一个类的所有对象拥有一个公共属性时,我们就使用一个类级别的变量,即静态变量。

在类加载时,此变量仅在内存中加载一次。 由于它不是在Java中按对象定义的,因此可以节省内存。

为什么用 Java 创建静态变量不是一个好习惯?

答:静态变量是类的所有对象共有的。 如果创建了新对象,则无需测试静态变量的值。 使用静态变量的任何代码都可以处于任何状态。 它可以在新对象内或在类级别。 因此,静态变量的范围在Java类中是开放式的。

如果我们希望对范围进行更严格的控制,则变量应在对象创建级别创建。

同样,定义静态变量也不是一个好习惯,因为它们违反了面向对象编程的原理。

Java 中静态方法的目的?

答:Java 提供了静态方法的功能来在类级别创建行为。 静态方法是类的所有对象所共有的。 我们不需要创建类的任何对象来调用静态方法。 因此,它提供了不创建用于调用它的对象的便利。

静态方法也可以访问和修改静态数据成员。 这也有助于在类级别上保持行为和状态。

为什么在 Java 中将 main 方法标记为静态方法?

答:Java 中的 main 方法被标记为静态,因此 JVM 可以调用它来启动程序。 如果 main 方法不是静态的,那么Java进程将调用哪个构造函数?

因此,在Java中将主要方法标记为静态是众所周知的约定。 但是,如果我们去除static,那将会有歧义。 Java进程可能不知道要调用哪个类的方法来启动程序。

因此,此约定有助于 Java 进程识别类中作为参数传递给 Java 进程的程序的启动代码。

在什么情况下我们使用静态块?

答:有时,有一个具有静态成员变量的类。 这些变量需要一些复杂的初始化。 这时,静态块可作为初始化复杂静态成员变量初始化的工具。静态块甚至在执行main之前执行。有时,我们也可以用静态的类方法替换静态块。

是否可以在不定义main()方法的情况下执行程序?

答:不,从Java 7开始,您需要main()方法来执行程序。 在Java的早期版本中,有一种解决方法可用于使用静态块执行。 但是现在这个差距已经缩小。

当main方法的签名中未提及static修饰符时会发生什么?

答:根据Java规范,main方法必须标记为静态。 它只需要一个字符串数组的参数即可。

程序可以使用非静态方法进行编译。 但是在执行时会给出NoSuchMethodError。

Java中的静态方法和实例方法有什么区别?

答:通常,需要为不依赖于对象成员变量的类定义行为。 这种行为是通过静态方法捕获的。 如果存在依赖于对象成员变量的行为,则我们不会将其标记为静态,而是将其保留为实例方法。

要调用为静态方法,我们不需要创建对象。 我们只用类名来称呼它。 但是要调用实例方法,我们需要先创建/获取一个对象。

实例成员变量不能通过静态方法访问。 但是实例方法可以调用实例变量和静态变量。

到此这篇关于Java 中的静态字段和静态方法 的文章就介绍到这了,更多相关Java 静态方法内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java8 接口默认方法和静态方法

    目录 1.默认接口方法 2.静态接口方法 3.两者的场景差异 前些日有同学问我Java接口中为什么要增加一个默认方法.在Java 8 中不单单增加了默认接口方法还增加了静态接口方法.今天就来聊聊这两个方法. 1.默认接口方法 java中的默认接口方法是这样定义的: public interface NewInterface { ​ void otherMethod(); default void doSomething() { System.out.println(" do something

  • JAVA构造器是否为静态方法你知道吗

    在thinking in java中,第96页提到,“即使没有显示地使用static关键字,构造器实际上也是静态方法“,对此我产生了疑惑,于是找相关资料进行确认,在一篇大神的博客中得到了答案,那就是构造器不是静态方法. 主要是以下几点原因: Java虚拟机规范第二版中定义了四种不同的字节码指令来处理Java程序中不同种类的方法的调用: invokestatic - 用于调用类(静态)方法 invokespecial - 用于调用实例方法,特化于super方法调用.private方法调用与构造器调

  • Java的动态代理和静态代理详解

    目录 0.代理模式 1.静态代理 2.加深理解 3.动态代理 动态代理的例子 总结 0.代理模式 为什么要学习代理模式?这是SpringAOP的底层[SpringAOP和SpringMVC] 代理模式的分类: 静态代理 动态代理 1.静态代理 静态代理中,我们对目标对象的每个方法的增强都是手动完成的(后面会具体演示代码_),非常不灵活(比如接口一旦新增加方法,目标对象和代理对象都要进行修改)且麻烦(_需要对每个目标类都单独写一个代理类). 实际应用场景非常非常少,日常开发几乎看不到使用静态代理的

  • Java 静态代理与动态代理解析

    目录 一.代码实践 静态代理 动态代理 二.常见的动态代理场景 Retrofit中的动态代理 使用动态代理实现onClick注入 三.源码探索Jdk中的动态代理 生成代理类 四.总结 静态代理: 由我们开发者自己手动创建或者在程序运行前就已经存在的代理类,静态代理通常只代理一个类,动态代理是代理一个接口下的多个实现类. 动态代理: 在程序运行时,运用java反射机制动态创建而成,静态代理事先知道要代理的是什么,而动态代理不知道要代理什么东西,只有在运行时才知道,通常动态代理实现方式是通过实现 j

  • JAVA代理,静态,动态详解

    目录 代理 静态代理 动态代理 JDK动态代理 CGLib动态代理 总结 代理 为其他对象提供一种代理以控制这个对象的访问,在某些情况下一个对象不能直接访问那个对象时,代理就起到了客户端和被代理对象 (委托类) 中介作用. 按照代理的创建时期,代理类可以分为两种: 静态:由程序员创建代理类或特定工具自动生成源代码再对其编译.在程序运行前代理类的.class文件就已经存在了. 动态:在程序运行时运用反射机制动态创建而成. 静态代理 Subject: 代理类和被代理类实现同样的接口 Proxy:代理

  • Java 中的静态字段和静态方法 

    目录 一.静态变量 静态变量声明 静态变量测试 二.静态方法 静态方法测试 静态方法是如何工作的 为什么 Java Main 方法是静态方法 三.静态常量 四.总结 先看看下面这一串代码: public class Main {     public static void main(String[] args) {         System.out.println("Hello, WOrld!");     } } 我们的 ​​main​​ 方法被标记了 ​​static​​ 修

  • php面向对象中static静态属性和静态方法的调用

    本文实例讲述了php中static静态属性和静态方法的调用.分享给大家供大家参考.具体如下: 这里分析了php面向对象中static静态属性和静态方法的调用.关于它们的调用(能不能调用,怎么样调用),需要弄明白了他们在内存中存放位置,这样就非常容易理解了.静态属性.方法(包括静态与非静态)在内存中,只有一个位置(而非静态属性,有多少实例化对象,就有多少个属性). 实例: <?php header("content-type:text/html;charset=utf-8"); c

  • java中的静态代码块、构造代码块、构造方法详解

    运行下面这段代码,观察其结果: package com.test; public class HelloB extends HelloA { public HelloB() { } { System.out.println("I'm B class"); } static { System.out.println("static B"); } public static void main(String[] args) { new HelloB(); } } cla

  • JAVA中的静态代理、动态代理以及CGLIB动态代理总结

    代理模式是java中最常用的设计模式之一,尤其是在spring框架中广泛应用.对于java的代理模式,一般可分为:静态代理.动态代理.以及CGLIB实现动态代理. 对于上述三种代理模式,分别进行说明. 1.静态代理 静态代理其实就是在程序运行之前,提前写好被代理方法的代理类,编译后运行.在程序运行之前,class已经存在. 下面我们实现一个静态代理demo: 静态代理 定义一个接口Target package com.test.proxy; public interface Target { p

  • JAVA中堆、栈,静态方法和非静态方法的速度问题

    一.堆和栈的速度性能分析 堆和栈是JVM内存模型中的2个重要组成部分,自己很早以前也总结过堆和栈的区别,基本都是从存储内容,存储空间大小,存储速度这几个方面来理解的,但是关于堆和栈的存储速度,只知道堆存储速度慢,栈存储速度快,至于为什么堆比栈的存取速度慢,并没有特别深入的研究,从网上也找了很多资料,但很多理由并不太认同,这里也列举一些,并结合自己的理解来分析,如果不正确欢迎指正. 1.从分配的角度分析 java中栈的大小和生命周期在编译期间就确定了的(可以参考之前写的一篇JVM内存模型中的分析,

  • Java中static静态变量的初始化完全解析

    静态变量初始化顺序 1.简单规则 首先先看一段最普遍的JAVA代码: public class Test { public static Test1 t = new Test1(); public static int a = 0; public static int b; public static void main(String[] arg) { System.out.println(Test.a); System.out.println(Test.b); } } class Test1

  • 深入解析java中的静态代理与动态代理

    java编码中经常用到代理,代理分为静态代理和动态代理.其中动态代理可以实现spring中的aop. 一.静态代理:程序运行之前,程序员就要编写proxy,然后进行编译,即在程序运行之前,代理类的字节码文件就已经生成了 被代理类的公共父类 复制代码 代码如下: package staticproxy;public abstract class BaseClass {    public abstract void add();} 被代理类 复制代码 代码如下: package staticpro

  • Kotlin使用静态变量与静态方法详解

    前言 在日常开发过程中,静态变量和  静态方法 是我们常见的用法,Java中相信大家并不陌生了,那么在 Kotlin 中该如何使用呢? 其实很简单,只需要一个将变量和方法包含在 companion object 域中即可,比如这样: class Constant { companion object { // 接口根地址 const val BASE_URL = "http://xxxx.xxx.xxx/" // 友盟 const val UMENG_APP_KEY = "x

  • 基于Java中对域和静态方法的访问不具有多态性(实例讲解)

    1.将方法调用同方法主体关联起来被称为 2.编译期绑定(静态)是在程序编译阶段就确定了引用对象的类型 3.运行期绑定(动态绑定)是指在执行期间判断所引用对象的实际类型,根据其实际的类型调用其相应的方法 4.除了static方法和final方法(private方法属于final方法),其他所有方法都是后期绑定,Java中所有的方法都是通过动态绑定来实现多态 5.访问某个域的行为不具有多态性 package polymorphism; class SuperField { public int fi

  • 深入浅析Java中普通代码块、构造代码块与静态代码块

    //执行顺序:(优先级从高到低.) 静态代码块>mian方法>构造代码块>构造方法. 其中静态代码块只执行一次.构造代码块在每次创建对象是都会执行. 1.普通代码块 public static void main(String[] args) { /*普通代码块: *直接定义在在方法或语句中出现"{普通代码的执行语句}"的就称为普通代码块. *普通代码块执行顺序由他们在代码中出现的次序决定--"先出现先执行" * */ { System.out.p

随机推荐