`
xiao1zhao2
  • 浏览: 38723 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

java中内部类的使用

阅读更多
1.定义

 

内部类是指在一个外部类的内部在定义一个类.内部类作为外部类的一个成员,并且依附于外部类而存在的.内部类可为静态,可用protected和private修饰(外部类只能是public和缺省的包访问权限).内部类主要有以下几种:成员内部类,局部内部类,静态内部类,匿名内部类.

 

2.为什么需要内部类

 

典型情况是:内部类继承自某个类或实现某个接口,内部类的代码操作创建其外部类的对象.所以你可以认为内部类提供了某种进入其外部类的窗口.使用内部类最吸引人的原因是:每个内部类都能独立的继承自接口的一个实现,无论外部类是否已经继承了这个实现对于内部类都没有影响.如果没有内部类提供的可以继承多个具体的或抽象的类的能力,一些设计与编程问题很难解决.从这个角度看,内部类是的多重继承的解决方案变得完整:接口解决了部分问题,内部类有效的实现了"多继承".

 

3.成员内部类

 

作为外部类的一个成员存在,与外部类的属性和方法并列:
>>在外部类访问内部类时,需创建一个内部类的对象.
>>每一个内部类的对象都会链接到一个外部类的对象(除非是一个静态内部类,下面会说到).
>>在内部类中,可以访问所有外部类的成员,但自身不能定义静态成员.
>>在内部类中,可以定义与外部类相同名称的实例变量,在内部类访问自己的变量和与外部类的非同名变量可以直接使用;访问外部类的同名变量时,通过"外部类名.this.变量名"使用.
>>内部类是一个编译时的概念,一旦编译成功,会成为完全不同的两类.本例中编译完成后会出现Outer3.class和Outer3$Inner3.class两个文件.
例:

public class Outer3 {
	private static int i = 10;
	private int j = 20;
	private int k = 30;

	public static void outer_method1() {

	}

	public void outer_method2() {

	}

	class Inner3 {
		// static int inner_i;
		private int inner_j = 200;
		private int k = 300;

		public void inner_method() {
			System.out.println(inner_j);
			System.out.println(this.k);
			System.out.println(Outer3.this.k);
			System.out.println(j);
			outer_method1();
			outer_method2();
		}
	}

	public void outer_method3() {
		this.new Inner3().inner_method();
	}

	public static void outer_method4() {
		Outer3 outer = new Outer3();
		Inner3 inner = outer.new Inner3();
		inner.inner_method();
	}

	public static void main(String[] args) {
		outer_method4();
	}
}

 

 

4.局部内部类

 

在方法中定义的内部类,与局部变量类似.除了有和成员内部类一样的特性外:
>>局部内部类不能有访问修饰符,因为它不是外部类的一部分.
>>可以访问作用域内的变量,但是在访问时隐式的将变量当做final处理.
例:

public class Outer4 {

	public void outer_method() {
		int i = 10;
		// i++;
	
		class Inner4 {
			void inner_method() {
				System.out.println(i);
			}
		}
	}
}

 

 

5.静态内部类

 

如果不需要内部来的对象与外部类的对象之间有联系,可以将内部类声明为static.非静态内部类对象都隐含的保存了一个引用,指向外部类的对象,但是静态内部类就不是这样了:
>>可以直接通过外部类创建内部类的对象.
>>不能在内部类的对象中访问非静态的外部类对象.
>>内部类可以定义静态和非静态成员.
>>内部类无法访问外部类的非静态成员.
例:

public class Outer5 {

	private static int i = 10;
	private int j = 20;
	private int k = 30;

	public static void outer_method1() {

	}

	public void outer_method2() {

	}

	static class Inner5 {
		private static int inner_i;
		private int inner_j = 200;
		private int k = 300;

		public void inner_method() {
			System.out.println(inner_j);
			System.out.println(k);
			// System.out.println(j);
			outer_method1();
			// outer_method2();
		}
	}

	public void outer_method3() {
		System.out.println(new Inner5().inner_j);
	}

	public static void outer_method4() {
		new Inner5().inner_method();
	}

	public static void main(String[] args) {
		outer_method4();
	}
}

 

6.匿名内部类

 

匿名内部类就是没有类名的内部类,一般适用的条件有:
>>只用到类的一个实例.
>>类在定义后马上用到.
>>类非常的小(推荐是4行代码以下).
>>给类命名并不会使你的代码更容易理解.
在使用匿名内部类时,需要记住的原则:
>>匿名类不能有构造方法.
>>匿名类内部不能定义任何静态成员(属性,方法,类).
>>匿名类不能是public,protected,private,static.
>>只能创建一个实例.
>>一定是在new关键字的后面,用其隐含实现一个接口或类.
>>其他内部类的特性.
例:

import java.util.List;
import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;

public class Outer6 {

	public static void main(String[] args) {
		List<String> name = Arrays.asList("James", "Kobe", "Yao");
		Collections.sort(name, new Comparator<String>() {
			public int compare(String s1, String s2) {
				return s1.compareTo(s2);
			}
		});
	}
}

 

静态方法Collections.sort接收一个list和一个Comparator作为输入参数,实现对输入的list中的元素进行比较并排序. 看上去我们直接new了一个Comparator,实际上是定义了一个实现Comparator的匿名类,通过new创建了这个匿名类的一个对象.在本例中编译成功后会出现Outer6$1.class文件,代表匿名类通过编译.

 

7.多层嵌套类

 

在多层嵌套类中,一个内部类被嵌套多少层并不重要,它能访问所有他所嵌入的外部类的成员(即使是private的).例:

public class Outer7 {
	private int i = 10;

	private void outer_method() {

	}

	class A {
		int j = 20;

		void a() {

		}

		class B {
			void b() {
				outer_method();
				a();
				System.out.println(i);
				System.out.println(j);
			}
		}
	}

	public static void main(String[] args) {
		new Outer7().new A().new B().b();
	}
}

 

8.内部类重载

 

如果创建了一个类,继承了外部类并重新定义了次内部类时,是否内部类会被重载呢?但是"重载"内部类好像它是外部类的一个方法,并不起什么作用.例:

public class Outer8 {
	private Inner inner;

	class Inner {
		public Inner() {
			System.out.println("Outer8.Inner()");
		}
	}

	public Outer8() {
		System.out.println("new Outer8()");
		inner = new Inner();
	}
}

 

public class SubOuter8 extends Outer8 {
	private Inner subinner;

	class Inner {
		public Inner() {
			System.out.println("SubOuter8.Inner()");
		}
	}

	public SubOuter8() {
		System.out.println("new SubOuter8()");
		subinner = new Inner();
	}

	public static void main(String[] args) {
		new SubOuter8();
	}
}

 

运行结果:
new Outer8()
Outer8.Inner()
new SubOuter8()
SubOuter8.Inner()

由此可见,当继承外部类的时候,内部类并没有什么特别的变化,这两个内部类完全是相对独立的两个实体,各自在自己的命名空间内.当然也可以明确的继承某个内部类.例:

public class Outer8 {
	private Inner inner;

	class Inner {
		public Inner() {
			System.out.println("Outer8.Inner()");
		}
	}

	public Outer8() {
		System.out.println("new Outer8()");
		inner = new Inner();
	}
}

 

public class SubOuter8 extends Outer8 {
	private Inner subinner;

	class Inner extends Outer8.Inner {
		public Inner() {
			System.out.println("SubOuter8.Inner()");
		}
	}

	public SubOuter8() {
		System.out.println("new SubOuter8()");
		subinner = new Inner();
	}

	public static void main(String[] args) {
		new SubOuter8();
	}
}

 

运行结果:
new Outer8()
Outer8.Inner()
new SubOuter8()
Outer8.Inner()
SubOuter8.Inner()

和前面的结果相比多输出了一个"Outer8.Inner()",说明子类的内部类在实例化时先创建了父类内部类的对象.

 

小结:不同环境下的内部类有着不同的特性,成员内部类与类的其他成员特性相似;局部内部类与局部变量特性相似;静态内部类与静态变量特性相似;匿名内部类经常在为了某个功能而实现某个接口时使用.同时内部类也是类的一种表现形式.

1
0
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics