【Java】static关键字,内部类
一. static 关键字
在Java中,被static修饰的成员,称之为静态成员,也可以称为类成员,其不属于某个具体的对象,是所有对象所共享的。
1. static修饰成员变量
static修饰的成员变量,称为静态成员变量
【静态成员变量特性】:
• 不属于某个具体的对象,是类的属性,所有对象共享的,不存储在某个对象的空间中
• 既可以通过对象引用访问(不推荐使用),也可以通过类名访问,但一般更推荐使用类名访问
• 类变量存储在方法区当中
• 生命周期伴随类的一生(即:随类的加载而创建,随类的卸载而销毁)
public class Student{
public String name;
public String gender;
public int age;
public double score;
public static String classRoom = "rj2104";
public Student(String name, String gender, int age, double score) {
this.name = name;
this.gender = gender;
this.age = age;
this.score = score;
}
// ...
public static void main(String[] args) {
// 静态成员变量可以直接通过类名访问
System.out.println(Student.classRoom);
Student s1 = new Student("Li leilei", "男", 18, 3.8);
Student s2 = new Student("Han MeiMei", "女", 19, 4.0);
Student s3 = new Student("Jim", "男", 18, 2.6);
// 也可以通过对象访问:但是classRoom是三个对象共享的
System.out.println(s1.classRoom);
System.out.println(s2.classRoom);
System.out.println(s3.classRoom);
}
}
2. static修饰成员方法
一般类中的数据成员都设置为private,而成员方法设置为public,
Java中,被static修饰的成员方法称为静态成员方法,是类的方法,不是某个对象所特有的。
静态成员一般是通过静态方法来访问的。
public class Student2{
// ...
private static String classRoom = "rj2104";
// ...
public static String getClassRoom(){
return classRoom;
}
}
class TestStudent {
public static void main(String[] args) {
System.out.println(Student2.getClassRoom());
}
}
【静态方法特性】:
• 不属于某个具体的对象,是类方法
• 可以通过对象调用,也可以通过 类名.静态方法名(…) 方式调用,更推荐使用后者
• 不能在静态方法中访问任何非静态成员变量和非静态成员方法;因为非静态方法中默认有this参数,但在静态方法中调用时候无法传递this引用;除非在静态方法中新new一个对象,再通过对象引用去访问此对象
• 静态方法无法重写,不能用来实现多态
3. static成员变量的初始化
静态成员变量一般不会放在构造方法中来初始化,构造方法中初始化的是与对象相关的实例属性
静态成员变量的初始化分为两种:就地初始化 和 静态代码块初始化。
1.就地初始化:在定义时直接给出初始值
public class Student2{
// ...
//就地初始化
private static String classRoom = "rj2104";
//...
}
2.用静态代码块完成初始化
public class Student2{
// ...
private static String classRoom;
//静态代码块初始化
static {
classRoom = "rj2104";
}
// ...
}
二. 内部类
在 Java 中,可以将一个类定义在另一个类或者一个方法的内部, 前者称为内部类,后者称为外部类。内部类也是封装的一种体现。
内部类和外部类共用同一个java源文件,但是经过编译之后,内部类会形成单独的字节码文件, 一般形成的字节码文件文件名为:外部类名字$内部类名字.class
public class OutClass {
class InnerClass{
}
}
// OutClass是外部类
// InnerClass是内部类
根据内部类定义的位置不同,一般可以分为以下几种形式:
1.成员内部类(普通内部类)
实例内部类:未被static修饰的成员内部类
静态内部类:被static修饰的成员内部类
2.局部内部类
3.匿名内部类
1. 实例内部类
即未被static修饰的成员内部类。
【注意事项】:
1.外部类中的任何成员都可以在实例内部类方法中直接访问
2.实例内部类当中不能有静态的成员变量;非要定义,那么只能是被static final修饰的静态常量,常量是在程序编译的时候就确定的
3.实例内部类所处的位置与外部类成员位置相同,因此也受public、private等访问限定符的约束
4.实例内部类对象必须在先有外部类对象前提下才能创建
5.实例内部类的非静态方法中默认包含了一个指向外部类对象的引用和一个指向自身实例内部类对象的引用
6.在实例内部类方法中访问同名的成员时,优先访问自己的,如果要访问外部类同名的成员,必须:外部类名称.this.同名成员 来访问
7.外部类中,不能直接访问实例内部类中的成员,如果要访问必须先要创建内部类的对象。
public class OutClass {
private int a;
static int b;
int c;
public void methodA() {
a = 10;
System.out.println(a);
}
public static void methodB() {
System.out.println(b);
}
// 实例内部类:未被static修饰
class InnerClass {
int c;
//实例内部类当中 不能有静态的成员变量. 非要定义,那么只能是被static final修饰的
public static final int d = 6;
public void methodInner() {
// 在实例内部类中可以直接访问外部类中:任意访问限定符修饰的成员
a = 100;
b = 200;
methodA();
methodB();
System.out.println(d);
// 如果外部类和实例内部类中具有相同名称成员时,优先访问的是内部类自己的
c = 300;
System.out.println(c);
// 如果要访问外部类同名成员时候,必须:外部类名称.this.同名成员名字
OutClass.this.c = 400;
System.out.println(OutClass.this.c);
}
}
public static void main(String[] args) {
// 外部类:对象创建 以及 成员访问
OutClass outClass = new OutClass();
System.out.println(outClass.a);
System.out.println(outClass.b);
System.out.println(outClass.c);
outClass.methodA();
outClass.methodB();
System.out.println("=============实例内部类的访问=============");
// 要访问实例内部类中成员,必须要创建实例内部类的对象
// 而普通内部类定义与外部类成员定义位置相同,因此创建实例内部类对象时必须借助外部类
// 创建实例内部类对象
OutClass.InnerClass innerClass1 = new OutClass().new InnerClass();
innerClass1.methodInner();
// 上述语法比较怪异,也可以先将外部类对象先创建出来,然后再创建实例内部类对象
OutClass.InnerClass innerClass2 = outClass.new InnerClass();
innerClass2.methodInner();
}
}
2. 静态内部类
被static修饰的内部成员类称为静态内部类。、
【注意事项】:
1.在静态内部类中只能访问外部类中的静态成员,除非在内部类当中new一个外部类的对象,通过外部类对象的引用去访问其中的非静态成员。
2.创建静态内部类对象时,不需要先创建外部类对象
public class OuterClass2 {
public int data1 = 1;
int data2 = 2;
public static int data3 = 3;
public void test() {
System.out.println("out::test()");
}
// 静态内部类:被static修饰的成员内部类
static class InnerClass2 {
public int data4 = 4;
int data5 = 5;
public static int data6 = 6;
public void func() {
System.out.println("out::func()");
//test();
// 编译失败,在静态内部类中不能直接访问外部类中的非静态成员
//System.out.println(data1);
//System.out.println(data2);
//外部类的非静态成员,需要通过外部类的对象的引用才能访问。
OuterClass2 outerClass = new OuterClass2();
System.out.println(outerClass.data1);
System.out.println(outerClass.data2);
outerClass.test();
// 在静态内部类中只能访问外部类的静态成员
System.out.println(data3);
System.out.println(data4);
System.out.println(data5);
System.out.println(data5);
System.out.println(data6);
}
}
public static void main(String[] args) {
// 静态内部类对象创建 和 成员访问
OuterClass2.InnerClass2 innerClass2 = new OuterClass2.InnerClass2();
innerClass2.func();
}
}
3. 局部内部类
定义在外部类的方法体或者{ }中,一般使用的非常少。
【注意事项】
1.局部内部类只能在所定义的方法体内部使用
3.不能被public、static等修饰符修饰
3.局部内部类生成的字节码文件稍有区别:外部类名字$数字内部类名字.class
ppublic class OutClass {
int a = 10;
public void method(){
int b = 10;
// 局部内部类:定义在方法体内部
// 不能被public、static等访问限定符修饰
class InnerClass{
public void methodInnerClass(){
System.out.println(a);
System.out.println(b);
}
}
// 只能在该方法体内部使用,其他位置都不能用
InnerClass innerClass = new InnerClass();
innerClass.methodInnerClass();
}
public static void main(String[] args) {
// OutClass.InnerClass innerClass = null; 编译失败
}
}
4. 匿名内部类
匿名内部类,就是没有名字的一种嵌套类
匿名内部类形成的字节码文件文件名为:外部类名字$数字.class
4.1 使用匿名内部的好处与演示
在实际开发中,我们会遇到下面的情况:
一个接口/类的方法的某个执行过程在程序中只会执行一次,但为了使用它,我们需要创建它的实现类/子类去实现/重写方法。
代码中为了这一次的使用去创建一个类,未免太过麻烦,此时就可以使用匿名内部类来解决这个问题
首先来看我们正常的实现逻辑,假设有一个接口,接口当中只有一个方法
public interface Interface {
void show();
}
为了使用该接口的show方法,我们需要去创建一个实现类,重写show方法的具体实现
public class Test implements Interface{
@Override
public void show() {
System.out.println("只执行一次show()");
}
}
public class Main {
public static void main(String[] args) {
Test test = new Test();
test.show();
}
}
如果实现类Test在程序中只使用一次,那么为了这一次的使用去创建一个类太过繁琐,这种情况下就可以用匿名内部类来实现,无需创建新的类,减少代码冗余,
看下面代码:
class Main {
public static void main(String[] args) {
//写法一
Interface in = new Interface() {
@Override
public void show() {
System.out.println("匿名内部类中重写show()");
}
};
//调用接口方法
in.show();
//写法二
new Interface() {
@Override
public void show() {
System.out.println("匿名内部类中重写show()");
}
}.show();//调用接口方法
}
}
4.2 匿名内部类的定义格式和使用
定义格式1:
接口名称 引用名 = new 接口名称() {
// 覆盖重写所有抽象方法
};
引用名.方法调用
定义格式2:
new 接口名称() {
// 覆盖重写所有抽象方法
}.方法调用;
对格式“new 接口名称() {…}”的理解:
1.new代表创建一个新的对象对象
2.接口名称就是匿名内部类需要实现哪个接口
3.{…}中是匿名内部类的内容
【注意事项】:
• 匿名内部类,在【创建对象】的时候,只能使用唯一 一次。
• 匿名对象,在【调用方法】的时候,只能调用唯一 一次。
• 匿名内部类是省略了【实现类/子类名称】,但是匿名对象是省略了【对象名称】
• 匿名内部类可以用在具体类、抽象类、接口上,且对方法个数没有要求。
public class Class {
public void show(String s){
System.out.println("Class::show()");
}
}
public abstract class AbstractClass {
abstract void show(String s);
}
public interface Interface {
void show(String s);
}
public class TestDome {
public static void main(String[] args) {
//重写普通类的方法
new Class(){
@Override
public void show(String s) {
System.out.println(s);
}
}.show("普通类");
//重写抽象类的抽象方法
new AbstractClass(){
@Override
void show(String s) {
System.out.println(s);
}
}.show("抽象类");
//实现接口的抽象方法
new Interface(){
@Override
public void show(String s) {
System.out.println(s);
}
}.show("接口");
}
}
执行结果: