Skip to content

Latest commit

 

History

History
269 lines (214 loc) · 9.21 KB

File metadata and controls

269 lines (214 loc) · 9.21 KB

文章出处

文章出自:安卓进阶学习指南

作者:Cloud9527

审核者: shixinzhang Struggle Alex_赵

完稿日期:

深入浅出Java内部类

内部类的概念

定义在一个类中或者方法中的类称作为内部类。

内部类又可以具体细分为成员内部类局部内部类匿名内部类静态内部类

成员内部类

成员内部类就是最普通的内部类,它定义在一个类的内部中,就如同一个成员变量一样。如下面的形式

class OutClass { // 外部类
    private int i = 1;
    public static String str = "outclass";

    class InnerClass { // 成员内部类

        public void innerMethod() {
            System.out.println("i=" + i);
            System.out.println("str=" + str);

        }
    }
}

public class TestClass {

    public static void main(String[] args) {
        //先创建外部类对象
        OutClass outClass = new OutClass(); 
        //创建内部类对象
        OutClass.InnerClass in = outClass.new InnerClass();
        //内部类对象调用自己的方法
        in.innerMethod();
    }
} 

因为内部类依附于外部类存在,所以需要外部类的实例来创建内部类;成员内部类可以无条件的访问外部类的成员属性和成员方法(包括private成员和static成员),但是外部类想要调用内部类的方法或者属性必须先要创建内部类的实例才可以。

注:如果内部类的属性和方法和外部类的属性和方法同名,如果直接调用这个属性或者方法则是默认调用内部类的。如果想调用外部类的属性和方法则需要如下写法 :

外部类.this.属性

外部类.this.方法

成员内部类就如同外部类的成员一样,同样可以被public、protected、private、缺省(default)这些修饰符来修饰,下面是这些修饰符的权限范围:

修饰符 当前类 同一包内 子孙类 其他包
public Y Y Y Y
protected Y Y Y N
default Y Y N N
private Y N N N

局部内部类

它和成员内部类的区别就是:局部内部类的作用域只能在其所在的代码块或者方法内,在其它地方是无法创建该类的对象。

public class OutClass {

    {
        class PartClassOne { // 局部内部类
            private void method() {
                System.out.println("PartClassone");
            }
        }
        new PartClassOne().method();
    }

    public void testMethod() {
        class PartClassTwo { // 局部类内部类
            private void method() {
                System.out.println("PartClassTwo");
            }
        }
        new PartClassTwo().method();
    }
}

匿名内部类

先让我们来看一段最常见的代码

Car jeep=new Car();

在Java中**“操纵的标识符实际是指向一个对象的引用”**也就是说jeep是一个指向Car类对象的引用,而右面的 new Car()才是真正创建对象的语句。这可以将jeep抽象的理解为Car类对象的“名字”,而匿名内部类顾名思义可以抽象的理解为没有“名字”内部类.

button.setOnClickListener(new OnClickListener() {
    @Override
    public void onClick(View v) {
    // TODO Auto-generated method stub
    }
});

上面代码是Android中最常见的设置button的点击事件,其中new OnClickListener() {…} 就是一个匿名内部类,在这里没有创建类对象的引用,而是直接创建的类对象。这个就是最典型的匿名内部类。大部分匿名类用于接口回调。

静态内部类

使用static关键字修饰的内部类就是静态内部类,静态内部类和外部类没有任何关系,可以看作是和外部类平级的类。静态内部类只能访问外部类的静态属性和静态方法,外部类只能访问内部类的静态属性和静态方法

public class OutClassThree {
    private String str = "out";
    private static String str2 = "static_str";

    public void print() {
        System.out.println("outclass:" + str);
        System.out.println("outclass:" + str2);
          //外部类可以直接调用静态内部类的静态属性
        System.out.println("outclass:" + StaticInnerClass.instr2); 

        StaticInnerClass innerClass = new StaticInnerClass();
        innerClass.print();
    }

    public static class StaticInnerClass {
        private String instr = "out";
        private static String instr2 = "static_str";

        public void print() {
            // 静态内部类不能访问外部类的非静态属性和方法
            // System.out.println("InnerClass:" + OutClassThree.this.str);
            // 静态内部类可以直接访问外部类的静态属性 外部类.静态属性
            System.out.println("InnerClass:" + OutClassThree.str2);
            System.out.println("InnerClass:" + instr2);
        }
    }
}

Java内部类的使用场景

上面介绍了Java中4种内部类的定义,如果只看玩这些想必大家还是不清楚内部类的应用场景,下面就由我介绍一些内部类的使用的典型场景。

一、利用private内部类可以完全禁止其他人依赖类型编码,并可将具体的实现细节完全隐藏。

在Java中普通类(非内部类)是不可以设为private或者protected,只能设置成public default。如果想要一类完全禁止其他人依赖类型编码,就只能使用private内部类。 举个例子: 在我公司的项目中有一个公共的支付页面,它要支持多种不同的业务,如飞机票、酒店、火车票等。但是每种业务在这个页面又有其特有的信息要展示,我是这样来定义这个页面需要的实体的

class PublicPay{
	int orderId; //公共订单id
	double price; //订单金额
	private class FlightPayInfo{
		String fligtNumber; //航班号
	}
	private class HotelPayInfo{
	String hotelName;//酒店名称
	}
	private class TrainPayInfo{
	String trainCode;//车次
	}
}

我先在PublicPay类中定义了所有业务类型都需要的订单id和订单金额,然后又分别定义了三个私有的内部类,他们都拥有自己特有的订单信息。当然你可以说这个三个内部类也可以定义成普通类呀,确实普通类也同样能满足需求,但是我要求的这三个业务的支付信息类只是在这个公共支付信息页面才用到,我想要这几个不同业务的信息类在使用后就进入不可见或不可用的状态,这个时候内部类就能满足我的需求。

二、匿名内部类的使用

在Android中我们写一个View的点击事件通常是这样写的

public class ActivityOne extends AppCompatActivity implements View.OnClickListener{
    Button mButton1;
    Button mButton2;
    Button mButton3;
    Button mButton4;
    Button mButton5;
    Button mButton6;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main2);
        mButton1 = (Button)findViewById(R.id.btn1);
		.
		.
		.
		mButton1.setOnClickListener(this);
		.
		.
		.
    }

    @Override
    public void onClick(View view) {
        switch (view.getId()){
            case R.id.btn1:
                break;
				 .
				 .
				 .
			 case R.id.btn6:
				 break;
        }
    }
}

onClick()方法中我们看到了大量的switch语句块,这样的结构不是很好的面向对象设计,我们可以使用内部类作为很好的替代方法

public class ActivityTwo extends AppCompatActivity {
    Button mButton1;
    Button mButton2;
    Button mButton3;
    Button mButton4;
    Button mButton5;
    Button mButton6;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main2);
        mButton1 = (Button)findViewById(R.id.btn1);
        .
        .
        .

        
        mButton1.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {

            }
        });
        mButton2.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {

            }
        });
        .
        .
        .
        mButton6.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {

            }
        });
    }

}

这些代码和上面的代码对比,没用使用公共的onClick()方法,而是每个button都单独通过匿名内部类设置了自己的点击事件。这可能导致许多内部类。但是,我们可以避免大的switch语句,并有封装逻辑的额外好处。这就是Android中最典型的匿名内部类的使用。

参考资料

《Java编程思想》

http://blog.csdn.net/qq7342272/article/details/6671433

http://www.cnblogs.com/latter/p/5665015.html

https://www.javaworld.com/article/2077411/core-java/inner-classes.html