内部类:一个类定义在另外一个类的内部,那么该类就称作为内部类。
class Outer{ // Outer&Inner.class int i ; // 成员变量 class Inner0{ // 成员内部类 int j ; // 成员局部变量 public void test() { class Inner1{ // 局部内部类 } } } }
成员内部类的访问方式:
方式一:在外部类提供一个方法创建内部类的对象进行访问。
class Outer{ // Outer&Inner.class int x=10; class Inner1{ // 成员内部类 int i=10; public void print() { System.out.println("成员内部类"); } } public void instance() { //在外部类的方法中创建内部类的对象,调用内部类的方法 System.out.println("x="+x); Inner1 inner1 =new Inner1(); inner1.print(); } } public class out_in { public static void main(String[] args) { // TODO Auto-generated method stub Outer outer=new Outer(); outer.instance(); } }
方式二:在其他类直接创建内部类的对象。 格式:外部类.内部类 变量名 = new 外部类().new 内部类();
Outer.Inner1 inner1 = new Outer().new Inner1(); // 右边 new Outer()之后,Inner1()对象加载进内存;然后New 一个Inner1() // 左边 外部的内部对象 Outer.Inner1 inner1 inner1.print();
内部类的好处:内部类可以直接访问外部类的所有成员
应用场景:内部类的应用场景: 我们在描述A事物的时候,发现描述的A事物内部还存在另外一个比较复杂的事物B时候,而且这个比较复杂事物B还需要访问A事物的属性等数据,那么这时候我们就可以使用内部类描述B事物。(访问内部属性很方便)
比如: 人--->心脏
class 人{ 血... 氧气... 等.... class 心脏{ .... } }
内部类要注意的细节:
1. 如果外部类与内部类存在同名的成员变量时,在内部类中默认情况下是访问内部类的成员变量。
可以通过"外部类.this.成员变量名" 指定访问外部类的成员。
2. 私有的成员内部类只能在外部类提供一个方法创建内部类的对象进行访问,不能在其他类创建对象了。
3. 成员内部类一旦出现了静态的成员,那么该类也必须使用static修饰。
为什么“成员内部类一旦出现了静态的成员,那么该类也必须使用static修饰”?
答: 因为 “static int i=10”是在 Outer.class文件被加载到内存的时候存在内存中。 静态的成员数据是不需要对象存在才能访问。------ 需要好好思考
注意: 如果是一个静态内部类,那么在其他类创建 的格式: 外部类.内部类 变量名 = new 外部类.内部类();
* * 内部类:一个类定义在另外一个类的内部,就叫做内部类 * 内部类的class文件名: 外部类$内部类. * 好处:便于区分该class文件是属于哪个外部类的。 * * */ class Outer{ // Outer&Inner.class int x=10; private static class Inner1{ // 成员内部类 //内部成员类假如私有化之后,就只能用外部类创建内部类的方法了。 int x=20; // 内外部有同名的变量 static int i=10; //成员内部类一旦出现了静态的成员,那么该类也必须使用static修饰。 public void print() { System.out.println("成员内部类"+i+"..."+x); // 内外部有同名的变量,访问内部类,如果没有则访问外部 //System.out.println("成员内部类"+i+"..."+Outer.this.x); // 内外部有同名的变量,想访问外部的,可以通过"外部类.this.成员变量名" 指定访问外部类的成员。 } } public void instance() { //在外部类的方法中创建内部类的对象,调用内部类的方法 //System.out.println("x="+x); Inner1 inner1 =new Inner1(); inner1.print(); } } public class out_in { public static void main(String[] args) { // TODO Auto-generated method stub Outer outer=new Outer(); //内部成员类假如私有化之后,就只能用外部类创建内部类的方法了。 outer.instance(); // Outer.Inner1 inner1 = new Outer().new Inner1(); //右边 new Outer()之后,Inner1()对象加载进内存;然后New 一个Inner1() // //这句话其实是创建了两个对象,所以用this时候要做区分,所以在访问外部变量时,用Outer.this.x // // 左边 外部的内部对象 Outer.Inner1 inner1 // inner1.print(); } }
局部内部类:在一个类的的方法内部定义一个类,那么另一个类就叫做局部内部类;
class outer_01{ public void test() { final int y = 100; // 局部变量 // y 什么时候从内存中消失? 方法执行完毕之后y消失。 class inner_01{ // 局部内部类 int x=10; public void print() { System.out.println("局部内部类print()方法"+y); // 内部类访问了一个局部变量,那么该局部变量必须使用final修饰。 } } inner_01 inner_01=new inner_01(); // 这个inner对象什么时候消失? Inner对象的生命周期比局部变量y的生命周期要长。 inner_01.print(); } } public class out_int_02 { public static void main(String[] args) { // TODO Auto-generated method stub outer_01 outer_01=new outer_01(); outer_01.test(); } }
内部类访问了一个局部变量,那么该局部变量必须使用final修饰?
y 什么时候从内存中消失? 方法执行完毕之后y消失。
inner_01对象什么时候从内存中消失? 当test方法执行完毕之后,那么y马上从内存中消失,而Inner对象在方法执行完毕的时候还没有从内存中消失,而inner对象的print方法还在访问着y变量,这时候的y变量已经消失了,那么就给人感觉y的生命变量已经被延长了。
解决方案: 如果一个局部内部类访问一个局部变量的时候,那么就让该局部内部类访问这个局部变量的复制品。
匿名内部类:没有类名的类就称作为匿名内部类。
匿名内部类的好处:简化书写。
匿名内部类的使用前提:必须存在继承或者实现关系才能使用。
匿名内部类一般是用于实参。
需求1: 在方法内部定义一个类,继承Animal类,然后调用run方法。----可以使用局部内部类解决
abstract class Animal_in{ public abstract void run(); } class outer_02{ //需求1: 在方法内部定义一个类,继承Animal类,然后调用run方法。 public void print(){ class Dog_01 extends Animal_in{ public void run() { System.out.println("...狗在跑..."); } } //创建对象 Dog_01 dog_01 =new Dog_01(); dog_01.run(); } } public class anonymous_inner_class { public static void main(String[] args) { // TODO Auto-generated method stub outer_02 outer_02= new outer_02(); outer_02.print(); } }
需求2: 在方法内部定义一个类,继承Animal类,然后调用run方法。 -----匿名内部类解决
abstract class Animal_in{ public abstract void run(); } class outer_02{ //需求1: 在方法内部定义一个类,继承Animal类,然后调用run方法。 -----匿名内部类解决 public void print(){ //匿名内部类 :匿名内部类只是没有类名,其他的一概成员都是具备的. // 匿名内部类与Animal是继承的关系。 // 目前是创建Animal子类的对象。 new Animal_in() { public void run() { System.out.println("...狗在跑..."); } }.run(); //.run之前都是一个对象的意思 } } public class anonymous_inner_class { public static void main(String[] args) { // TODO Auto-generated method stub outer_02 outer_02= new outer_02(); outer_02.print(); } }
加入内部特有方法
abstract class Animal_in{ public abstract void run(); } class outer_02{ //需求1: 在方法内部定义一个类,继承Animal类,然后调用run方法。 -----匿名内部类解决 public void print(){ //匿名内部类 :匿名内部类只是没有类名,其他的一概成员都是具备的. // 匿名内部类与Animal是继承 的关系。 目前是创建Animal子类的对象. new Animal_in() { public void run() { System.out.println("...狗在跑..."); } //特有方法 public void bite() { System.out.println("...狗在咬人..."); } }.bite(); //.bite之前都是一个对象的意思 } } public class anonymous_inner_class { public static void main(String[] args) { // TODO Auto-generated method stub outer_02 outer_02= new outer_02(); outer_02.print(); } }
需求3: 在方法内部定义一个类,继承Animal类,然后调用run方法与sleep()。。 -----匿名内部类解决(但是很恶心)
abstract class Animal_in{ public abstract Animal_in run(); public abstract void sleep(); } class outer_02{ //需求2: 在方法内部定义一个类,继承Animal类,然后调用run方法与sleep()。 -----匿名内部类解决 public void print(){ //匿名内部类 :匿名内部类只是没有类名,其他的一概成员都是具备的. // 匿名内部类与Animal是继承 的关系。 目前是创建Animal子类的对象. new Animal_in() { public Animal_in run() { System.out.println("...狗在跑..."); return this; } public void sleep() { System.out.println("...狗在睡觉..."); } }.run().sleep(); //.run之前都是一个对象的意思 } } public class anonymous_inner_class { public static void main(String[] args) { // TODO Auto-generated method stub outer_02 outer_02= new outer_02(); outer_02.print(); } }
需求4: 在方法内部定义一个类,继承Animal类,然后调用run方法与sleep()。 -----匿名内部类解决---多态
abstract class Animal_in{ public abstract Animal_in run(); public abstract void sleep(); } class outer_02{ //需求2: 在方法内部定义一个类,继承Animal类,然后调用run方法与sleep()。。 -----匿名内部类解决 public void print(){ //匿名内部类 :匿名内部类只是没有类名,其他的一概成员都是具备的. // 匿名内部类与Animal是继承 的关系。 目前是创建Animal子类的对象. Animal_in a = new Animal_in() { //可以用多态解决“.run().sleep()”这个问题 public Animal_in run() { System.out.println("...狗在跑..."); return this; } public void sleep() { System.out.println("...狗在睡觉..."); } }; a.run(); //多态解决这个问题 a.sleep(); } } public class anonymous_inner_class { public static void main(String[] args) { // TODO Auto-generated method stub outer_02 outer_02= new outer_02(); outer_02.print(); } }
实现关系下匿名内部类
interface Dao_02{ public void add(); } class Outer_03{ public void print(){ //创建一个匿名内部类的对象 new Dao_02(){ // 不是接口不能创建对象吗?怎么现在又可以了? // 现在其实不是创建接口对象,是创建匿名类的对象 public void add(){ System.out.println("添加成功"); } }.add(); } } public class anonymous_inner_class_02 { public static void main(String[] args) { // TODO Auto-generated method stub Outer_03 Outer_03 =new Outer_03(); Outer_03.print(); } }