成员变量又称为成员属性,它是描述对象状态的数据,是类中很重要的组成成分。本节详细讨论如何来定义成员变量、成员变量的访问权限,以及静态成员变量与实例成员变量之间的区别。
成员变量的定义
定义成员变量的语法如下:
[变量修饰符] 类型说明符变量名
类的成员变量和在方法中所声明的局部变量都是用户标识符,它们的命名规则相同。变量修饰符是可选项,一个没有变量修饰符的变量定义如下:
public class Cuber{ double width,height; int number; }
成员变量的类型可以是Java中的任意数据类型,包括基本类型、数组、类和接口。在一个类中,成员变量应该是唯一的,但是成员变量的名字可以和类中某个方法的名字相同,例如:
public class Point{ int x, y; int x(){ return x; } }
其中,方法x()和变量x具有相同的名字,但笔者不赞成这样写,因为这会引起不必要的混淆。
可以用成员变量修饰符来规定变量的相关属性,这些属性包括:
● 成员变量的访问权限。一共有4种访问权限可供选择,在3.3.2节将详细介绍。
● 成员变量是否为静态。默认情况下,成员变量是实例成员,在外部需要通过对象才能操作。如果用static修饰,就成为了静态成员,也称为类变量,无需通过对象就可以操作。
● 是否为常量。默认的是变量,如果前面加上final关键字,它就是一个常量。
这些修饰符可以任意组合使用。加上修饰符的成员变量如下所示:
public class Cuber{ private double width,height; //定义两个私有的成员变量 public static int count; //定义一个公共的静态类变量 public static final int COLORE=1; //定义一个公共的整型静态常量 }
虽然Java并没有规定,成员变量必须定义在类的开始部分,不过在实际编程中,多数程序员将成员变量定义在成员方法的前面。
3.3.2 成员变量的访问权限
访问权限修饰符声明了成员变量的访问权限。Java提供的显示的访问权限修饰符有3种,分别是:私有(private)、保护(protected)和公共(public)。除此之外,还有一种默认的访问权限:friendly,它并不是Java的关键字,只有当变量前面没有写明任何访问权限修饰符时,就默认以friendly作为访问权限。为了表达上的方便,省略了其中“成员”两字,将被这些修饰符所修饰的变量分别称为私有变量、保护变量和公共变量。下面分别讨论各个修饰符的用法。
1.公共变量
凡是被public修饰的成员变量,都称为公共变量,它可以被任何类所访问。即允许该变量所属的类中所有方法访问,也允许其他类在外部访问。如例3.2所示。
【例3.2】 公共变量使用示例。
//-----------文件名declarePublic.java,程序编号3.2------------------
public class declarePublic{
public int publicVar=10; //定义一个公共变量
}
在类declarePublic中声明了一个公共变量publicVar,它可以被任何类所访问。下面这段程序中,类otherClass可以合法地修改变量publicVar的值,而无论otherClass位于什么地方。
1 //-----------文件名otherClass.java,程序编号3.3------------------ 2 public class otherClass{ 3 void change(){ 4 declarePublic ca=new declarePublic(); //创建一个declarePublic对象 5 ca.publicVar=20; //通过对象名来访问它的公共变量,正确 6 } 7 }
用public修饰的变量,允许任何类在外部直接访问,这破坏了封装的原则,造成数据安全性能下降,所以除非有特别的需要,否则不要使用这种方案。
2.私有变量
凡是被private修饰的成员变量,都称为私有变量。它只允许在本类的内部访问,任何外部类都不能访问它。
【例3.3】 私有变量使用示例。
1 //-----------文件名declarePrivate.java,程序编号3.4------------------ 2 public class declarePrivate{ 3 private int privateVar=10; //定义一个私有变量 4 void change(){ 5 privateVar=20; //在本类中访问私有变量,合法 6 } 7 }
如果企图在类的外部访问私有变量,编译器将会报错。
1 //-----------文件名otherClass.java,程序编号3.5------------------ 2 public class otherClass{ 3 void change(){ 4 declarePrivate ca=new declarePrivate(); //创建一个declarePrivate对象 5 ca.privateVar=20; //企图访问私有变量,非法 6 } 7 }
为了让外部用户能够访问某些私有变量,通常类的设计者会提供一些方法给外部调用,这些方法被称为访问接口。下面是一个改写过的declarePrivate类。
1 //-----------文件名declarePrivate.java,程序编号3.6------------------ 2 public class declarePrivate{ 3 private int privateVar=10; //定义一个私有变量 4 void change(){ 5 privateVar=20; 6 } 7 public int getPrivateVar(){ //定义一个接口,返回私有变量privateVar的值 8 return privateVar; 9 } 10 public boolean setPrivateVar(int value){ //定义一个接口,可以设置privateVar 11 的值 12 //可以在这里先检测value是否在允许的范围内,然后再执行下面的语句。 13 privateVar = value; 14 return true; 15 } 16 }
私有变量很好地贯彻了封装原则,所有的私有变量都只能通过程序员设计的接口来访问,任何外部使用者都无法直接访问它,所以具有很高的安全性。但是,在下面这两种情况下,需要使用Java另外提供的两种访问类型:
● 通过接口访问私有变量,将降低程序的性能,在程序性能比较重要的情况下,需要在安全性和效率间取得一个平衡。
● 私有变量无法被子类继承,当子类必须继承成员变量时,需要使用其他的访问类型。
3.保护变量
凡是被protected修饰的变量,都被称为保护变量。除了允许在本类的内部访问之外,还允许它的子类以及同一个包中的其他类访问。子类是指从该类派生出来的类。包是Java中用于管理类的一种松散的集合。二者的详细情况将在第4章介绍。下面是一个简单的 例子。
【例3.4】 保护变量使用示例。
下面这个程序先定义一个名为onlyDemo的包,declarProtected类就属于这个包。
1 //-----------文件名declareProtected.java,程序编号3.7------------------ 2 package onlyDemo; 3 public class declareProtected{ 4 protected int protectedVar=10; //定义一个保护变量 5 void change(){ 6 protectedVar=20; //合法 7 } 8 }
%说明:读者编译这个文件时,需要用这个命令(下同):
javac -d . 文件名
下面这个otherClass类也定义在onlyDemo包中,与declareProtected类同属于一个包。
1 //-----------文件名otherClass.java,程序编号3.8------------------ 2 package onlyDemo; 3 public class otherClass{ //它也在包onlyDemo中 4 void change(){ 5 declareProtected ca=new declareProtected(); 6 ca.protectedVar=20; //合法 7 } 8 }
下面这个deriveClass类是declareProtected的子类,它并不在onlyDemo包中。它也可以访问保护变量protectedVar,但是只能通过继承的方式访问。
1 //-----------文件名declareProtected.java,程序编号3.9------------------ 2 import onlyDemo.declareProtected; //引入需要的包 3 public class deriveClass extends declareProtected{ //定义一个子类 4 void change(){ 5 //合法,改变的是deriveClass从declarProtected中所继承的protectedVar值 6 protectedVar=30; 7 } 8 }
%说明:import是Java中的关键字,用于引入某个包。这将在4.13节中详细介绍。
子类如果不在父类的同一包中,是无法通过“对象名.变量名”的方式来访问protected类型的成员变量,比如下面这种访问是非法的:
1 //-----------文件名deriveClass.java,程序编号3.10----------------- 2 import onlyDemo.declareProtected; 3 public class deriveClass extends declareProtected{ //定义一个子类 4 void change(){ 5 declareProtected ca=new declareProtected(); 6 ca.protectedVar=30; //错误,不允许访问不在同一包中的保护变量 7 } 8 }
4.默认访问变量
如果在变量前不加任何访问权修饰符,它就具有默认的访问控制特性,也称为friendly变量。它和保护变量非常像,它只允许在同一个包中的其他类访问,即便是子类,如果和父类不在同一包中,也不能继承默认变量(这是默认访问变量和保护变量的唯一区别)。因为它限定了访问权限只能在包中,所以也有人称默认访问权限为包访问权限。
【例3.5】 默认访问变量使用示例。
1 //-----------文件名declareDefault.java,程序编号3.11------------------ 2 package onlyDemo; //本类定义在包中 3 public class declareDefault{ 4 int defaultVar=10; //定义一个默认访问变量 5 void change(){ 6 defaultVar=20; //合法 7 } 8 } 9 onlyDemo包中的其他类,可以访问defaultVar变量: 10 //-----------文件名otherClass.java,程序编号3.12------------------ 11 package onlyDemo; 12 public class otherClass{ //它也在包onlyDemo中 13 void change(){ 14 declareDefault ca=new declareDefault(); 15 ca.defaultVar=20; //合法 16 } 17 }
下面是它的子类,也在onlyDemo包中。它除了可以像包中其他类那样通过“对象名.变量名”来访问默认变量,还可以通过继承的方式来访问。
1 //-----------文件名deriveClass.java,程序编号3.13------------------ 2 package onlyDemo; 3 public class deriveClass extends declareDefault{ //定义一个子类 4 void change(){ 5 //合法,改变的是deriveClass从declarDefault中所继承的defaultVar值 6 defaultVar=30; 7 } 8 } 9 如果子类不在onlyDemo包中,就不会继承默认变量,也就无法像上面那样来访问。 10 //-----------文件名deriveClass.java,程序编号3.14------------------ 11 import onlyDemo.declareDefault; 12 public class deriveClass extends declareDefault{ //定义一个子类 13 void change(){ 14 defaultVar=30; //非法,这个变量没有继承下来 15 } 16 }
3.3.3 实例成员变量和静态成员变量
1.实例成员变量
在3.3.2节中,所有的对象都是实例成员变量。它们的最大特色是:如果所属的对象没有被创建,它们也就不存在。如果在类的外部使用它,需要先创建一个对象,然后通过“对象名.变量名”来访问。前面所有的例子都遵循了这一规则。在类的内部,实例成员方法也可以直接访问实例成员变量,比如例3.5,具体原因,将在3.5节中讲述。
不同的对象,拥有不同的实例成员变量,它们互不干扰。
【例3.6】 不同对象的实例成员变量使用示例。
1 //-----------文件名instanceVar.java,程序编号3.15------------------ 2 public class instanceVar{ 3 protected int instVar=0; //定义一个实例成员变量 4 } 5 下面这个类showInstVar用两个对象来访问它的实例成员变量。 6 //-----------文件名showInstVar.java,程序编号3.16------------------ 7 public class showInstVar{ 8 public static void main(String args[]){ 9 instanceVar one = new instanceVar(); //创建对象one 10 instanceVar two = new instanceVar(); //创建对象two 11 //分别为这两个对象的成员变量赋值 12 one.instVar = 100; 13 two.instVar = 200; 14 //分别显示这两个对象的成员变量值 15 System.out.println("one.instVar="+one.instVar); 16 System.out.println("two.instVar="+two.instVar); 17 } 18 }
程序3.16输出的结果如下:
one.instVar=100
two.instVar=200
从本例中明显地看出,不同对象的成员变量是不相同的,它们互不干涉。
2.静态成员变量
在某些情况下,程序员希望定义一个成员变量,可以独立于类的任何对象,即所有的对象都共用同一个成员变量。由于Java中不能像C一样定义全局变量,因此,Java中引入了静态成员变量。
在成员变量前加上static标识符就可以定义一个静态成员变量。相对于实例成员变量,静态成员变量具有以下特点:
● 它被类的所有对象共享,因此又被称为类变量。
● 它不是属于某个具体对象,也不是保存在某个对象的内存区域中,而是保存在类的公共存储单元。因此,可以在类的对象被创建之前就能使用。
● 它既可以通过“对象名.变量名”方式访问,也可以通过“类名.变量名”的方式访问。它们是完全等价的。
【例3.7】 静态成员变量使用示例。
1 //-----------文件名staticVar.java,程序编号3.17------------------ 2 public class staticVar{ 3 protected static int stat=0; //定义一个静态成员变量 4 } 5 下面这个程序使用不同的方法来访问这个静态变量。 6 //-----------文件名showStaticVar.java,程序编号3.18------------------ 7 public class showStaticVar{ 8 public static void main(String args[]){ 9 staticVar.stat=100; //通过类名.变量名访问静态变量,无需创建对象 10 System.out.println("staticVar.stat="+staticVar.stat); 11 staticVar one = new staticVar(); //创建对象one 12 staticVar two = new staticVar(); //创建对象two 13 //分别为这两个对象的静态成员变量赋值 14 one.stat = 200; 15 two.stat = 300; 16 //分别显示这两个对象的静态成员变量值 17 System.out.println("one.stat="+one.stat); 18 System.out.println("two.stat="+two.stat); 19 //再通过类来显示静态变量的值 20 System.out.println("staticVar.stat="+staticVar.stat); 21 } 22 }
程序3.18输出结果如下:
staticVar.stat=100
one.stat=300
two.stat=300
staticVar.stat=300
从上述结果中可以看到,静态变量stat是一个公共变量,无论哪个对象改变了它的值,对其他所有该类对象都有效。静态变量的一个重要作用是当作同类各个对象之间传递信息使用,类似于C语言中的全局变量。但这样破坏了数据的封装原则,往往会留下隐患,所以使用这类变量时需要万分谨慎。
静态变量的另一个用途是定义静态常量,比如:
public static double PI = 3.1415926;
这样的静态常量可以无需创建对象就直接使用,省略了创建对象的步骤,类似于C语言中用define定义的常量。这样定义常量,不仅使用方便,而且节省内存空间。在JDK中,存在着大量的这种静态常量。
%说明:本节中所有的成员变量的类型都是基本类型,其实它们也都可以是复合类型,比如数组、类、接口等类型。