关于静态static那些事

相信许多初学者甚至是工作了一两年的java程序员,会对static感到迷惑,下面谈谈我对static关键字所了解到的一些知识及个人理解。
static(静态)方法或数据的访问

java的核心思想是面向对象,即万物皆对象。通常我们创建类时会指出那个类的对象的外观与行为。除非用new创建那个类的一个对象,才会正式生产数据存储空间,并可使用相应的方法。但对于static(静态)关键字来说是特殊的,一旦将什么东西设为static,数据或方法就不会同那个类的任何对象实例(比如非静态方法、非静态变量)联系到一起。所以我们可以不用new对象,就可以直接通过类名.调用一个static方法,或访问一些static数据,因此我们不能在静态方法中引用关键字this,也不能调用非静态方法。对于非static(静态)数据和方法,在访问或调用时我们必须知道它们操作的具体对象。所以在访问非静态数据或方法时必须先创建对象。当然我们也可以通过创建对象的方式去访问static(静态)方法或变量,但不建议通过这种方式访问,因为这样不能突出“静态”的本质。

static(静态)变量的初始化

静态变量是独立存在的变量,只是位置放在某个类下,在类加载时被分配到数据区,它在内存中只有一个拷贝,而不会被分配多次,其后的所有的赋值操作都是值改变,地址保持不变,项目或程序一启动运行到该类时就直接常驻内存。也正是因为如此它不与类的任何对象相关联,而且可以不用创建对象直接通过类名.去访问。也就是说静态变量在编译的时候就已经确定了内存的位置。静态变量的生命周期取决于类的生命周期。
非静态变量(实例变量)就是相当于该类的属性,需要先初始化(实例化)该类,就是new 该类后,才可以调用。被分配到栈区,但是如果该类未被再次使用,被垃圾回收器回收后,该实例也将不存在了,就是不在内存中了。

关于static(静态)的注意事项
一、静态变量一定要先声明后赋值,
先看一个列子

这里写图片描述
接下来我们换下位置
这里写图片描述
对你没有看错,你会发现变量i是先使用后赋值。怎么会这样呢?上面说过,静态变量是类加载时被分配到数据区,它在内存中只有一个拷贝,不会被分配多次,其后的所有赋值操作都是值改变,地址则保持不变。我们知道JVM初始化变量是先声明空间,然后再赋值,也就是说
int i = 10;
在JVM中是分开执行,等价于:
int i; //分配地址空间
i = 100; //赋值
静态变量是在初始化时首先被加载的,JVM会去查找类中所有的静态声明,然后分配空间,注意这时只是完成了地址空间的分配,还有有赋值,之后JVM会根据类中静态赋值(包括静态类赋值和静态块赋值)的先后顺序来执行。对于程序来说,就是先声明了int类型的地址空间,并吧地址传递给了i,然后按照类中的先后顺序执行赋值动作,首先执行静态块中i=10,接着执行i=1,所以最终结果是1。

二、不要覆写静态方法

我们知道在java中可以通过覆写(Override)来增强或减弱父类的方法和行为,但覆写是针对非静态方法(也叫实例方法,只有生产实例才能调用的方法)的,不能针对静态方法(static修饰的方法,也叫类方法),先看一个例子:

public class StaticTest {

    public static void main(String[] args) {
        Father f = new Son();
        //调用静态方法
        f.method1();
        //调用非静态方法
        f.method2();

    }
}

class Father{
    public static void method1(){
        System.out.println("我是父类的方法1。。。");
    }
    public void method2(){
        System.out.println("我是父类的方法2。。。");
    }
}

class Son extends Father{
    public static void method1(){
        System.out.println("我是子类的方法1。。。");
    }
    public void method2(){
        System.out.println("我是子类的方法2。。。");
    }
}

这个结果是不是让你感到困惑,为什么同样是调用子类方法,一个执行了子类方法,一个执行了父类方法。
我们知道一个实例对象有两个类型:表名类型和实际类型,表名类型是声明时的类型,实际类型是对象产生时的类型,比如本例中的变量f的表面类型是Father,实际类型是Son.对于非静态方法,它是根据对象的实际类型来执行的,也就是执行了Father类中method2。而对于静态方法来说就比较特殊了,首先静态方法不依赖实例对象,它是通过类名访问的;其次,可以通过对象访问静态方法,如果是通过对象调用静态方法,JVM则会通过对象的表面类型查找到静态方法的入口,继而执行,因此产生了上面的结果。
当然如果我们在子类方法中添加@Override注解的话,子类方法1会编译不通过,所以阿里开发手册上有一条强制规则要求,所有的覆写方法,必须加@Override 注解。


评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值