Java语言中的修饰符—static

Java
230
0
0
2023-07-04

没有经过全文修饰的文章,读起来就会显得苍白无力,而善于用贴切修饰词的文章则能更形象地呈现主题,而在Java编程中也提供了一些修饰语,它们可以修饰类、变量和方法。对修饰符的灵活使用将大大提高软件的重用性、安全性和运行性能。下面的表格列出了类、构建方法、成员方法、成员变量和各种局部变量可用的修饰符,其中标注“√”表示可修饰(表中的类指顶级类,不包括内部类)。

修饰符



成员方法


构造方法


成员变量


局部变量


abstract(抽象的)







static(静态的)







public(公共的)







protected(受保护的)







private (私有的)







synchronized (同步的)







native(本地的)







transient(暂时的)







volatile(易失的)







final(不可改变的)







概念及用法讲解

static可以修饰类的成员变量、成员方法和代码块,被 static 修饰的成员变量和成员方法归某个类所有,不依赖于特定实例,被类的所有实例共享,只要这个类被加载, JVM 就可以根据类名在运行时数据区内定位它们,接下来我将分开进行讲解。

static变量(静态变量)

在 Java 语言中,类的成员变量分为两种:一种是被static修饰的变量,叫静态变量;另外一种就是没有被static修饰的成员变量,叫实例变量,二者区别如下:

· 静态变量 在内存中只进行一次备份,当运行时,JVM只为静态变量分配一次内存,可以直接通过类名进行访问静态变量。

·实例变量,则是我们每创建一个实例,就会为实例变量分配一次内存,实例变量在内存中的多个备份互不影响。

对于静态变量,在类的内部,可以在任何方法内直接访问,而在其他类中,可以直接通过某个类的类名来进行访问,示例参考如下代码:

 public class Sample{
    public static int number;//定义静态变量number
    public void method(){
        int x=number;//类的内部直接访问 number 静态变量
    }
}
public class Sample{
    public void method(){
        int x=Sample.number;//通过Sample1的类名来访问number静态变量  
    }
}复制代码类型:[java] 

同时,在main()方法中,我们可以采用多种方式来访问静态变量,示例参考如下代码:

 public class Scope{
    static int a;
    public static void main(String []args){
        a++;//直接访问静态变量a
         Scope  s=new Scope();
        s.a++;//通过引用变量s1来访问静态变量a
        Scope.a++;//通过Scpe类名访问静态变量a  
        System.out.println(a);//打印
    }
}

对于大多学过 c语言 的读者而言,静态变量的功能类似于全局变量,其主要实现两个功能:1、实现实例之间的数据共享;2、如果所有实例包括一个相同的常量属性,我们便可把这个常量定义为一个静态常量,从而节省内存空间。

static方法(静态方法)

同成员变量,成员方法也分为静态方法(类方法)和实例方法,并且静态方法不需创建实例,通过类名直接进行访问。,简单示例代码如下:

 public class Sample{
    public static int add(int x,int y){
        return x+y;
    }
}
public class Sample{
    public void method{
        int result=Sample.add(1,2);//通过类名调用类的静态方法  
        System.out.println("result="+result);
    }
}

对于静态方法而言,不能使用this关键字以及直接访问所属类的实例变量和实例方法,可以直接访问所属类的静态变量和静态方法,那么我们该如何理解这种访问限制呢?如下代码所示,假设student中的information方法无误,那么我们在执行student.information()时,JVM会很容易从student的方法区找到number静态变量,但是JVM却无法判断name是属于哪个对象的,其只会到方法区去寻找该变量,不会去往存放对象的堆区寻找,因此最终JVM无法找到name变量。

 public class student{
    public static int number;
    public String name;
    public static void information(){
        System.out.println("number"+number);//编译正确
        System.out.println("name"+name);//编译错误
    }
}
public class school{
    public static void main(String []args){
        student.information();//静态方法不允许访问非静态变量  
    }
}

若我们进行一些代码修改,使得student类information()方法中通过student对象的引用来访问name属性,则可以编译通过,例如:

 public static void information(){
    System.out.println("number"+number);//编译正确
    for(student stu:students){//遍历students集合,从集合中依次取出每个student对象  
        System.out.println("name"+name);//打印这个student对象的name属性  
    }
} 

对于静态方法而言,其必须要被实现,也就是说,静态方法不能被定义为抽象,static与 abstract 处于一种“有你没我”的对立局面。

static代码块(静态代码块)

在static讲解的最后一块,笔者将介绍在类中不存在于任何方法体中的静态代码块,JVM在加载类时,会按照静态代码块出现的顺序来执行它们,并且每个静态代码块只会执行一次,代码示例如下(同样的,静态代码块不可以直接访问类的实例变量和实例方法,而必须用实例的引用来进行访问):

 public class Sample{
    static int i=;
    static{
        System.out.println("static"+ i++);
    }
    static{
        System.out.println("static"+ ++i);
    }
    public static void main(String []args){
        Sample s=new Sample();
        Sample s=new Sample();//输出结果为“static1    1,static2   3”
    }
}

我们都知道,构造函数可用于初始化类的实例,而对于静态代码块而言,我们可以用来对静态变量赋初始值,增加了代码的可读性。