java内部类详解

内部类:一个类定义在另外一个类的内部,那么该类就称作为内部类。

内部类的class文件名: 外部类$内部类.  好处:便于区分该class文件是属于哪个外部类的。


 内部类的应用场景: 我们在描述A事物的时候,发现描述的A事物内部还存在另外一个比较复杂的事物B时候,而且这个比较复杂事物B还需要访问A事物的属性等数据,那么这时候我们就可以使用内部类描述B事物。


内部类的类别:

1. 成员内部类:

成员内部类的访问方式:
方式一:在外部类提供一个方法创建内部类的对象进行访问。
方式二:在其他类直接创建内部类的对象。 格式:外部类.内部类  变量名 = new 外部类().new 内部类();

注意: 如果是一个静态内部类,那么在其他类创建 的格式:
外部类.内部类  变量名 = new 外部类.内部类();




                内部类的好处:内部类可以直接访问外部类的所有成员。

内部类要注意的细节:
1. 如果外部类与内部类存在同名的成员变量时,在内部类中默认情况下是访问内部类的成员变量。
  可以通过"外部类.this.成员变量名" 指定访问外部类的 成员。
2. 私有的成员内部类只能在外部类提供一个方法创建内部类的对象进行访问,不能在其他类创建对象了。
3. 成员内部类一旦出现了静态的成员,那么该类也必须 使用static修饰。
原因:静态数据不需要依赖对象存在,设置为静态内部类不必依赖外部类存在,符合静态数据的设计原则


//外部类
class Outer{
	
	//成员变量
	int x = 100; // Outer.class文件被加载到内存的时候存在内存中。  静态的成员数据是不需要对象存在才能访问。

	//成员内部类
	static	class Inner{      
		
		static	int i = 23;

		public static void print(){
			System.out.println("这个是成员内部类的静态print方法!"+i);
		}
	}

	
	//在外部的方法中创建了内部类的对象,然后调用内部 方法。
	public void instance(){
		Inner inner = new Inner();
		inner.print();
	}

}

class Outer2{
	
	//成员变量
	int x = 100; // Outer2.class文件被加载到内存的时候存在内存中。  静态的成员数据是不需要对象存在才能访问。

	//成员内部类
		class Inner{      
		
			int i = 10;

		public void print(){
			System.out.println("这个是成员内部类的print方法!"+i);
		}
	}

	
	//在外部的方法中创建了内部类的对象,然后调用内部 方法。
	public void instance(){
		Inner inner = new Inner();
		inner.print();
	}

}

//其他类
class Demo4 
{
	public static void main(String[] args) 
	{
		//访问内部类的静态数据
		System.out.println(Outer.Inner.i);
		//非静态内部类
		Outer2.Inner inner2 = new Outer2().new Inner();
		inner2.print();


		//静态内部类
		Outer.Inner inner = new Outer.Inner();
		inner.print();
	}
}
运行结果:


2.局部内部类:

 在一个类 的方法内部定义另外一个类,那么另外一个类就称作为局部内部类。


局部内部类要注意的细节:
1. 如果局部 内部类访问了一个局部变量,那么该局部变量必须使用final修饰原因见代码注释

class  Outer{

	String name= "外部类的name";

	public void test(){
		//局部变量
		final	int y =100;  // y 什么时候从内存中消失? 方法执行完毕之后y消失。

		//局部内部类
		class Inner{     /*
							当test方法执行完毕之后,那么y马上从内存中消失,而Inner对象在方法
							执行完毕的时候还没有从内存中消失,而inner对象的print方法还在访问着
							y变量,这时候的y变量已经消失了,那么就给人感觉y的生命变量已经被延长了
							.

							解决方案: 如果一个局部内部类访问一个局部变量的时候,那么就让该局部内部类
							访问这个局部 变量 的复制品。			加final防止复制品被修改	
						*/
			int x = 10;

			public void print(){
				System.out.println("这个是局部内部类的print方法.."+y);
			}	
		}
		
		Inner inner = new Inner();  //这个inner对象什么时候消失?  Inner对象的生命周期比局部变量y的生命周期要长。
		inner.print();
	}


}





class Demo5 
{
	public static void main(String[] args) 
	{
		Outer outer = new Outer();
		outer.test();
	}
}
运行结果:


3.匿名内部类:

没有类名的类就称作为匿名内部类。

匿名内部类的好处:简化书写。

匿名内部类的使用前提:必须存在继承或者实现关系才能使用。

匿名内部类一般是用于实参。


abstract class Animal{
	
	public abstract Animal run();

	public abstract void sleep();
}


class Outer{

	public void print(){
		//需求: 在方法内部定义一个类继承Animal类,然后调用run方法与sleep()。
		
		/*
		//局部内部类
		class Dog extends Animal{
			
			public void run(){
				System.out.println("狗在跑..");
			}

			public void sleep(){
				System.out.println("狗趴在睁开眼睛睡..");
			}
		}
		
		//创建对象
		Dog d = new Dog();
		d.run();	
		d.sleep();
		*/	
		//匿名内部类 :匿名内部类只是没有类名,其他的一概成员都是具备的。
		// 匿名内部类与Animal是继承 的关系。  目前是创建Animal子类的对象. 
	Animal	a = new Animal(){  //多态
		
			//匿名内部的成员 
			public Animal run(){
				System.out.println("狗在跑..");
				return this;
			}
			
			public void sleep(){
				System.out.println("狗趴在睁开眼睛睡..");
			}

			//特有的方法
			public void bite(){
				System.out.println("狗在咬人..");
			}
	
		};
	
		//a.run().bite();必须用局部内部类
		a.run();
		a.sleep();
		
	}
}



class Demo6 
{
	public static void main(String[] args) 
	{
		//System.out.println("Hello World!");
		
		Outer outer = new Outer();
		outer.print();
	
	}
}

有特有方法时,匿名内部类不适合使用

运行结果:



对接口运用匿名内部类

//实现关系下匿名内部类
interface Dao{

	public void add();
}
class Demo7 
{
	public static void main(String[] args) 
	{
		test(new Dao(){
			
			public void add(){
				System.out.println("添加员工成功");
			}
		});


	}

	//调用这个方法...
	public static void  test(Dao d){  //形参类型是一个接口引用..
		d.add();
	}
}
运行结果


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值