1. 定义位置不同
- 成员变量:定义在类中方法之外。成员变量可以是实例变量或类变量,类变量使用
static
关键字修饰,而实例变量则不使用。它们属于类的一部分,用于描述类的属性或状态。 - 局部变量:定义在方法内部或方法的参数列表中。局部变量包括形参、方法局部变量和代码块局部变量。它们的作用域仅限于定义它们的方法或代码块内。
2. 存储位置不同
- 成员变量:存储在堆内存中。堆内存用于存储对象实例,因此成员变量是随着对象的创建而分配内存的。
- 局部变量:存储在栈内存中。栈内存主要用于存储方法调用的参数、局部变量等,随着方法的调用和返回而创建和销毁。
- 从变量在内存中的存储方式来看,如果成员变量是使用
static
修饰的,那么这个成员变量是属于类的,如果没有使用static
修饰,这个成员变量是属于实例的。而对象存在于堆内存,局部变量则存在于栈内存。
3. 生命周期不同
- 成员变量:成员变量的生命周期与对象的生命周期相同。它们随着对象的创建而创建,随着对象的销毁而销毁。对于静态成员变量,其生命周期则贯穿整个程序运行期。
- 局部变量:局部变量的生命周期与方法的执行期相同。当方法被调用时,局部变量被创建;当方法执行完毕并返回时,局部变量被销毁。
- 生存时间:从变量在内存中的生存时间上看,成员变量是对象的一部分,它随着对象的创建而存在,而局部变量随着方法的调用而自动生成,随着方法的调用结束而消亡。
4. 初始化值不同
- 成员变量:在创建对象时,系统会自动为成员变量分配内存空间,并为其指定初始值。对于基本数据类型的成员变量,初始值为该类型的默认值(如int为0,boolean为false等);对于对象类型的成员变量,初始值为null。
- 局部变量:局部变量在定义后必须显式初始化才能使用,系统不会为局部变量自动分配初始值。如果尝试使用未初始化的局部变量,编译器将报错。
5. 修饰符和访问权限
- 成员变量:可以被
public
、protected
、private
、static
等修饰符修饰,以控制其访问权限和存储方式。 - 局部变量:不能被访问控制修饰符(如
public
、protected
、private
)及static
修饰。局部变量仅在其所在的方法或代码块内部可见和可访问。
6. 命名冲突与优先级
- 成员变量和局部变量可以同名。在方法内部使用时,如果局部变量与成员变量同名,则局部变量具有更高的优先级。如果需要访问同名的成员变量,可以使用
this
关键字来引用。
- 语法形式:从语法形式上看,成员变量是属于类的,而局部变量是在代码块或方法中定义的变量或是方法的参数;成员变量可以被
public
,private
,static
等修饰符所修饰,而局部变量不能被访问控制修饰符及static
所修饰;但是,成员变量和局部变量都能被final
所修饰。
默认值:从变量是否有默认值来看,成员变量如果没有被赋初始值,则会自动以类型的默认值而赋值(一种情况例外:被 final
修饰的成员变量也必须显式地赋值),而局部变量则不会自动赋值。
为什么成员变量有默认值?
-
先不考虑变量类型,如果没有默认值会怎样?变量存储的是内存地址对应的任意随机值,程序读取该值运行会出现意外。
-
默认值有两种设置方式:手动和自动,根据第一点,没有手动赋值一定要自动赋值。成员变量在运行时可借助反射等方法手动赋值,而局部变量不行。
-
对于编译器(javac)来说,局部变量没赋值很好判断,可以直接报错。而成员变量可能是运行时赋值,无法判断,误报“没默认值”又会影响用户体验,所以采用自动赋默认值。
成员变量与局部变量代码示例:
public class VariableExample {
// 成员变量
private String name;
private int age;
// 方法中的局部变量
public void method() {
int num1 = 10; // 栈中分配的局部变量
String str = "Hello, world!"; // 栈中分配的局部变量
System.out.println(num1);
System.out.println(str);
}
// 带参数的方法中的局部变量
public void method2(int num2) {
int sum = num2 + 10; // 栈中分配的局部变量
System.out.println(sum);
}
// 构造方法中的局部变量
public VariableExample(String name, int age) {
this.name = name; // 对成员变量进行赋值
this.age = age; // 对成员变量进行赋值
int num3 = 20; // 栈中分配的局部变量
String str2 = "Hello, " + this.name + "!"; // 栈中分配的局部变量
System.out.println(num3);
System.out.println(str2);
}
}