Java实例变量和类变量

Java实例变量和类变量Java 程序的变量大体可分为成员变量和局部变量 其中局部变量可分为如下 3 类 形参 在方法签名中定义的局部变量 由方法调用者负责为其赋值 随方法的结束而消亡 方法内的局部变量 在方法内定义的局部变量 必须在方法内对其进行显示初始化 这种类型的局部变量从初始化完成后开始生效 随方法的结束而消亡 代码块的局部变量 在代码块内定义的局部变量 必须在代码块内对其进行显式初始化 这种类型的局部变量从初始化

Java程序的变量大体可分为成员变量和局部变量。其中局部变量可分为如下3类。

  • 形参:在方法签名中定义的局部变量,由方法调用者负责为其赋值,随方法的结束而消亡。
  • 方法内的局部变量:在方法内定义的局部变量,必须在方法内对其进行显示初始化。这种类型的局部变量从初始化完成后开始生效,随方法的结束而消亡。
  • 代码块的局部变量:在代码块内定义的局部变量,必须在代码块内对其进行显式初始化,这种类型的局部变量从初始化完成后开始生效,随代码的结束而消亡。

局部变量的作用时间很短暂,它们都被存储在方法的栈内存中。类体内定义的变量被称为成员变量(英文是Field)。如果定义该成员变量时没有使用static修饰,该成员变量又被称为非静态变量或实例变量;如果使用了static修饰,则该成员变量又可被称为静态变量或类变量

对于static关键字而言,从词义上来看,它是“静态”的意思。但从Java程序的角度来看,static的作用就是将实例成员变为类成员。static只能修饰在类里定义的成员部分,包括成员变量、方法、内部类、初始化块、内部枚举类。如果没有使用static修饰这里类里的成员,这里成员属于该类的实例;如果使用了static修饰,这些成员就属于类本身。从这个意义上看,static只能修饰类里的成员,不能修饰外部类,不能修改局部变量、局部内部类。

表明上看,Java类里定义成员变量时没有先后顺序,但实际上Java要求定义成员变量时必须采用合法的前后引用。示例如下

public class ErrorDef { 
    int num1 = num2 + 5; //Cannot reference a field before it is defined int num2 = 20; }
public class ErrorDef { 
    static int num1 = num2 + 5; //Cannot reference a field before it is defined static int num2 = 20; }

但如果一个是实例变量,一个是类变量,则实例变量总是可以引用类变量,示例如下

public class RightDef { 
    int num1 = num2 + 5; static int num2 = 20; }

上面程序中num1是一个实例变量,而num2是一个类变量。虽然num2位于num1之后被定义,但nun1的初始值却可根据num2计算得到。这是因为,num2变量是一个类变量,num1是实例变量,而类变量的初始化时机总是处于实例变量的初始化时机之前。所以,虽然源代码中先定义了num1,再定义了num2,但num2的初始化时机总是位于num1之前,因此num1变量的初始化可根据num2的值计算得到

实例变量和类变量的属性

使用static修饰的成员变量是类变量,属于该类本身;没有使用static修饰的成员变量是实例变量,属于该类的实例。在同一个JVM内,每个类只对应一个Class对象,但每个类可以创建多个Java对象。

由于同一个JVM内每个类只对应一个Class对象,因此同一个JVM内的一个类的类变量只需一块内存空间;但对于实例变量而言,该类每创建一次实例,就需要为实例变量分配一块内存空间。也就是说,程序中有几个实例,实例变量就需要几块内存空间。

下面程序可以很好地表现出来实例变量属于对象,而类变量属于类的特性。

 class Person { static int eyeNum; String name; int age; public void info() { System.out.println("Name: " + name + ", Age: " + age); } } public class FieldTest { 
    public static void main(String[] args) { // 类变量属于该类本身,只要改类初始化完成,程序即可使用类变量 Person.eyeNum = 2; // ① // 通过Person类访问 eyeNum类变量 System.out.println("Person的eyeNum属性:" + Person.eyeNum); // 创建第一个Person对象 Person p1 = new Person(); p1.name = "zhangsan"; p1.age = 22; System.out.println("通过p1变量访问eyeNum类变量:" + p1.eyeNum); // ② p1.info(); // 创建第二个Person对象 Person p2 = new Person(); p2.name = "lisi"; p2.age = 30; p2.info(); // 通过p2修改Person类的eyeNum类变量 p2.eyeNum = 4; // ③ System.out.println("通过p1变量访问eyeNum类变量:" + p1.eyeNum); System.out.println("通过p2变量访问eyeNum类变量:" + p2.eyeNum); System.out.println("通过Person类访问eyeNum类变量:" + Person.eyeNum); } }

上面程序中①行代码直接对Person类的eyeNum类变量赋值。这没任何问题,因为eyeNum类变量是属于Person类的,当Person类初始化完成后,eyeNum类变量也随之初始化完成。因此,程序即可对该类变量赋值,也可访问该类变量的值。

执行①行代码之后,程序的内存分配如图所示。

这里写图片描述

一旦Person类初始化完成,程序即可通过Person类访问eyeNum类变量。除此之外,Java还允许通过Person类的任意实例来访问eyeNum类变量

虽然Java允许通过Person对象来访问Person类的eyeNum类变量,但由于Person对象本身并没有eyeNum类变量(只有实例变量才属于Person实例),因此程序通过Person对象来访问eyeNum类变量时,底层依然会转换为通过Person访问eyeNum类变量。也就是说,不管通过哪个Person对象来访问eyeNum类变量,都与通过Person类访问eyeNum类变量的效果完全相同。因此,在②行代码处通过p1来访问eyeNum变量将再次输出2.

执行完②行代码后,程序的内存分配如图

这里写图片描述

当Person类初始化完成之后,类变量也随之初始化完成,以后不管程序创建多少个Person对象,系统不再为eyeNum类变量分配内存,但程序每创建一个Person对象,系统将再次为name、age实例变量分配内存,并执行初始化。

当程序执行完③行代码之后,内存中再次增加了一个Person对象。当程序通过p2对eyeNum类变量进行赋值时,实际上依然是对Person类的eyeNum类变量进行赋值。此时程序的内存分配如图所示。

这里写图片描述

当Person类的eyeNum类变量被改变之后,程序通过p1、p2、Person类访问eyeNum类变量都将输出4。这是由于,不管通过哪个Person对象来访问eyeNum类变量,底层都将转换为通过Person来访问eyeNum类变量。由于p1和p2两个变量指向不同的Java对象,当通过它们访问实例变量时,程序将输出不同的结果。

实例变量的初始化时机

  • 定义实例变量时指定初始值
  • 非静态初始化块中对实例变量指定初始值
  • 构造器中对实例变量指定初始值

其中第1、2种方式(定义时指定的初始值和非静态初始化块中指定的初始值)比第3种方式(构造器中指定初始值)更早执行,但第1、2中方式的执行顺序与他们在源程序中的排列顺序相同。

 class Cat { String name; int age; public Cat(String name, int age) { System.out.println("执行非静态初始化块"); this.name = name; this.age = age; } { System.out.println("执行构造器"); weight = 3.0; } double weight = 2.5; public String toString() { return "Name: " + name + ", Age: " + age + "Weight: " + weight; } } public class InitTst { 
    public static void main(String[] args) { Cat cat1 = new Cat("tom", 3); // ① System.out.println(cat1); Cat cat2 = new Cat("jiafei", 2); // ② System.out.println(cat2); } }

每当程序调用指定构造器来创建Java对象时,该构造器必然会获得执行的机会。除此之外,该类所包含的非静态初始化块将会获得执行的机会,而且总是在构造器执行之前获得执行。

当程序执行①行代码创建第一个Cat对象的时候,程序将会先执行Cat类的非静态初始化块,再调用该Cat类的构造器来初始化该Cat实例。执行完①行代码后的内存分配如图所示。

这里写图片描述

从图中可以看出,该Cat对象的weight实例变量的值为2.5,二不是初始化块中指定的。这是因为,初始化块中指定初始值,定义weight时指定初始值,都属于对该实例变量执行的初始化操作,他们的执行顺序与它们的顺序相同。在本程序中,初始化块中对weight的赋值位于定义weight语句之前,因此程序将先执行初始化块中的初始化操作,执行完成后weight实例变量的值为3.0,然后再执行定义weight时指定的初始值,执行完成后weight实例变量的值为2.5。 从这个意义上来看,初始化块中对weight所指定的初始化值每次都将被2.5所覆盖。

当执行②行代码再次创建一个Cat对象时,程序将再一次调用非静态初始化块、相应的构造器来初始化Cat对象。

执行完②行代码后,程序的内存分配如图所示。

这里写图片描述

类变量的初始化时机

  • 定义类变量时指定初始值
  • 静态初始化块中对类变量指定初始值。

这两种方式的执行顺序与它们在源程序中的排列顺序相同。

public class StaticInitTest { 
    // 定义count类变量,定义时指定初始值 static int count = 2; // 通过静态初始化块为name类变量指定初始值 static { System.out.println("StaticInitTest的静态初始化块"); name = "hello"; } // 定义name类变量时指定初始值 static String name = "itmyhome"; public static void main(String[] args) { System.out.println("count类变量的值" + StaticInitTest.count); System.out.println("name类变量的值" + StaticInitTest.name); } }

静态初始化块中为类变量指定初始值,每次运行该程序,系统将会对StaticInitTest类执行初始化:先为所有类变量分配内存空间,再按源代码中的排序执行静态初始化块中所指定的初始值和定义类变量时所指定的初始值.

对于本例程序而已,静态初始化夸中对name变量的指定初始值位于定义name变量时指定初始值之前,因此系统先将name类变量赋值为“hello”,然后再将该name类变量赋值为“itmyhome”。每运行该程序一次,这个初始化过程只执行一次,因此运行上面程序将看到输出name类变量的值为“itmyhome”.

下面程序更清楚地表现了类变量的初始化过程。首先定义了Price类,该Price类里有一个静态的initPrice变量,用于代表初始价格。每次创建Price实例时,系统会以initPrice为基础,减去当前打折价格(由discount参数代表)即得到该Price的currentPrice变量值

 class Price { // 类成员是Price实例 final static Price INSTANCE = new Price(2.8); // 再定义一个类变量 static double initPrice = 20; // 定义该Price的currentPrice实例变量 double currentPrice; public Price(double distinct) { // 根据静态变量计算实例变量 currentPrice = initPrice - distinct; } } public class PriceTest { 
    public static void main(String[] args) { // 通过Price的INSTANCE访问currentPrice实例变量 System.out.println(Price.INSTANCE.currentPrice); // ① // 创建Price实例 Price p = new Price(2.8); // 通过创建的Price实例访问currentPrice实例变量 System.out.println(p.currentPrice); // ② } }

上面程序中①、②行代码都访问Price实例的currentPrice实例变量,而且程序都是通过new Price(2.8)来创建Price实例的。表面上看,程序输出两个Price的currentPrice都应该返回17.2(由20减去2.8得到),但实际上运行程序并没有输出两个17.2,而是输出-2.8和17.2

如果仅仅停留在代码表面来看这个问题,往往很难得到正确的结果,下面从内存角度来分析这个程序。第一次用到Price类时,程序开始对Price类进行初始化,初始化分成以下2个阶段。

  • (1)系统为Price的两个类变量分配内存空间。
  • (2)按初始化代码(定义时指定初始化值和初始化块中执行初始值)的排列顺序对类变量执行初始化。

初始化第一阶段,系统先为INSTANCE、initPrice两个类变量分配内存空间,此时INSTANCE、initPrice的默认值null和0.0。截止初始化进入第二个阶段,程序按顺序依次为INSTANCE、initPrice进行复制。对INSTANCE赋值时要调用Price(2.8),创建Price实例,此时立即执行程序中③代码为currentPrice进行赋值,此时initPrice类变量的值为0,因此赋值的结果是currentPrice等于-2.8。接着,程序再次将initPrice赋为20,但此时对INSTANCE的currentPrice实例变量以及不起作用了。

当Price类初始化完成后,INSTANCE类变量引用到一个currentPrice为-2.8的Price实例,而initPrice类变量的值为20.0。当再次创建Price实例时,该Price实例的currentPrice实例变量的值才等于20.0

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/198675.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月26日 下午2:43
下一篇 2026年3月26日 下午2:43


相关推荐

  • 【Cover Letter 】SCI 投稿加分必备,手把手教你写 投稿Cover Letter

    【Cover Letter 】SCI 投稿加分必备,手把手教你写 投稿Cover LetterCoverLetter是我们投稿时,与手稿一同发送给编辑的投稿信。一封内容简洁的投稿信,会让编辑对你文章的第一印象加分不少。今天,我们一起来学习一下,一篇给文章加分的投稿信应该怎么写。投稿信的内容主要包含文章的标题、类型,没有一稿多投的声明,文章的主要内容及亮点,还有通讯作者的信息。敲黑板!一篇投稿信最重要的两点,是语言简洁和符合期刊标准。所以,要想写好投稿信,我们务必要了解各…

    2022年4月30日
    60
  • 第二课 如何在WINDOWS环境下搭建以太坊开发环境

    第二课 如何在WINDOWS环境下搭建以太坊开发环境

    2021年6月16日
    92
  • get请求关于url长度过长问题_get请求关于url长度过长问题

    get请求关于url长度过长问题_get请求关于url长度过长问题今天在写一个php相应jsonp请求的功能时,发现当url中包含的请求参数过长时会返回414错误。如下图414Request-URITooLarge414Request-URITooLargenginx在网上查询之后,浏览器和服务器对url长度都有限制,现总结如下。1、IEIE浏览器(MicrosoftInternetExplorer)对url长度限制是2083(2K+53)…

    2022年8月24日
    15
  • sqlserver快速删除大批量数据_sqlserver删除数据

    sqlserver快速删除大批量数据_sqlserver删除数据如何使用dropdownlist绑定数据库?首先我们要了解一些dropdownlist的一些属性。DataSource:为数据源DataBind:数据源绑定DataTextField:要显示的文本值DataValueField:显示文本值的编号其中DataTextField主要是给用户看的,我们程序员要用的是DataValueField。下面是代码展示:这里我就建立了一个部门表插入了三条数据:gocreatetableDepartment(DepI

    2022年10月8日
    6
  • Fibonacci数列 C语言

    Fibonacci数列 C语言Fibonacci 数列定义如下 第 1 第 2 个数均为 1 从第 3 个数开始 该数是其前面两个数之和 Fibonacci 数列为 1 1 2 3 5 8 13 编写递归函数 求 Fibonacci 数列的第 n 个数 并编写主函数 调用该递归函数 输出数列前 20 个数的值 intfibonacci intsum intmain void inti for i 1 i lt 20 i printf d

    2026年3月17日
    2
  • Linux学习——浅谈读写锁的使用

    Linux学习——浅谈读写锁的使用一、读写锁是什么?读写锁其实还是一种锁,是给一段临界区代码加锁,但是此加锁是在进行写操作的时候才会互斥,而在进行读的时候是可以共享的进行访问临界区的。读写锁和互斥量(互斥锁)很类似,是另一种线程同步机制,但不属于POSIX标准,可以用来同步同一进程中的各个线程。当然如果一个读写锁存放在多个进程共享的某个内存区中,那么还可以用来进行进程间的同步,读写锁的使用规则:只要没有写模式下的加锁,任…

    2022年8月12日
    8

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号