new 接口{内部实现方法} 转为 匿名内部类

一、内部类

1、内部类初识

一般,一个类里主要包含类的方法和属性,但在Java中还提出在类中继续定义类(内部类)的概念。

内部类的定义:类的内部定义类

先来看一个实例

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. public class Out {  
  2.     private int a=1;  
  3.     //定义内部类  
  4.     class Inner  
  5.     {  
  6.         private int a=12;  
  7.         public void show()  
  8.         {  
  9.             System.out.println(a);  
  10.         }  
  11.     }  
  12.       
  13.     public static void main(String[] args) {  
  14.         Out.Inner in=new Out().new Inner();//实例化内部类  
  15.         in.show();  
  16.           
  17.     }  

     如Out类中定义一个a变量,再定义一个Inner内部类,该类可直接调用输出a;同时为Out设定的show方法内部类也可直接调用。内部类的提出,唯一的好处就在于可直接使用外部类(Out类)的所有属性,不再需要通过get、set操作数据。但这种方便是以牺牲类的基本结构为代价的。这里需要注意的是对内部类的实例化语法。

2、 内部类对外部类的操作:内部类中的变量访问形式

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. public class Out {  
  2.     private int a=1;  
  3.     //定义内部类  
  4.     class Inner  
  5.     {  
  6.         private int a=12;  
  7.         public void show()  
  8.         {  
  9.             int a=122;  
  10.             System.out.println("局部变量:"+a);  
  11.             System.out.println("内部类变量:"+this.a);  
  12.             System.out.println("外部类变量:"+Out.this.a);  
  13.         }  
  14.     }  
  15.       
  16.     public static void main(String[] args) {  
  17.         Out.Inner in=new Out().new Inner();//实例化内部类  
  18.         in.show();        
  19.     }  
依次输出122,12,1 ,这个实例主要是提醒大家在对具有内部类的类成员变量、内部类变量调用的时候需要注意加上对应的关键字。

 二、内部类的常用类型

1、常规内部类

常规内部类没有用static修饰且定义在在外部类类体中。
  1.常规内部类中的方法可以直接使用外部类的实例变量和实例方法。
  2.在常规内部类中可以直接用内部类创建对象。

2、静态内部类

用static修饰内部类

1.如果使用static修饰内部类,该内部类只能访问外部类的静态成员变量。

3、局部内部类

 内部类前加上private修饰,使得该内部类只有当前外部类方法才可调用。其他类无实例化权限。

4、匿名内部类

这里着重介绍匿名内部类的使用。匿名内部类:没有名字的内部类,它的特点是只能使用一次,通常用于简化代码编写。使用匿名内部类必须继承一个父类或实现一个接口。

   声明和构造匿名内部类的一般格式如下:
   new ClassOrInterfaceName(){ //抽象类或接口

    /*类体*/ }

1)匿名内部类在抽象类上的应用

  不使用匿名内部类实现抽象类方法实例

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. public abstract class InnerClass2 {  
  2.     public abstract void show();  
  3.       
  4.     public static void main(String[] args) {  
  5.         InnerClass2 in=new Child(); //多态,实例化子类,调用子类show方法  
  6.         in.show();  
  7.     }  
  8. }  
  9.   
  10. public class Child extends InnerClass2{  
  11.         public void show()  
  12.         {  
  13.             System.out.println("jdfklj");  
  14.         }  
  15. }  
   使用匿名类实现思路

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. public abstract class Out2 {  
  2.     public abstract void show();  
  3.       
  4.     public static void main(String[] args) {  
  5.         Out2 out=new Out2()  
  6.         { //匿名类-直接实例化了out2和show 方法,在调用out.show()  
  7.             public void show()  
  8.             {  
  9.                 System.out.println("jdljsalkf");  
  10.             }  
  11.         };  
  12.         out.show();  
  13.     }     
  14. }  

    在注释出就是使用了一个匿名类,对抽象类Out2的show实例,最后out.show()调用即可。
2)匿名内部类在接口上的应用

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. public interface Out3 {//接口  
  2.     void show();  
  3.       
  4.     public static void main(String[] args) {  
  5.         Out3 out=new Out3()//new Out3  
  6.         {  
  7.             public void show()  
  8.             {  
  9.                 System.out.println("lkjdfaj");  
  10.             }  
  11.         };  
  12.     }  
  13. }  
   这里new Out3是 本片博客出生的源头,直接new 抽象类、new 接口??how could it be? one Answer:cause it is a Anonymous Inner Class.只要是一个抽象类或接口,那就可以使用匿名内部类来实现。最常用的是在多线程中继承Thread类或实现Runnable接口。

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. public static void main(String[] args) {  
  2.         Thread thread =new Thread()  
  3.         {  
  4.             public void run()  
  5.             {  
  6.                 System.out.println("kdjfka");  
  7.             }  
  8.         };  
  9.         thread.start();  
  10.           
  11.         //接口  
  12.         Thread thread2=new Thread(new Runnable(){  
  13.             public void run()  
  14.             {  
  15.                 System.out.println("22222222");  
  16.             }  
  17.               
  18.         });   
  19.         thread2.start();  
  20.     }  
<think>嗯,用户问的是Java中将类强制转换成接口的情况。我得先回想一下Java的类型转换机制。首先,Java中的类型转换分为向上转型和向下转型。向上转型是自动的,比如把子类对象赋值给父类引用,而向下转型需要显式转换,可能会有ClassCastException的风险。 用户提到的强制转换到接口,这可能涉及到两种情况:要么这个类本身已经实现了该接口,要么没有。如果类实现接口,那么即使没有显式地声明,也可以通过强制转换将类的实例转为接口类型。例如,假设有一个类A实现接口I,那么A的实例可以赋值给I类型的变量,或者进行强制转换。但这时候其实不需要强制转换,因为这是向上转型,会自动完成。可能用户有特殊需求,比如在方法参数中明确要求接口类型,或者在某些泛型上下文中需要显式转换。 另一种情况是,如果类没有实现接口,这时候进行强制转换会导致运行时异常。比如,类B没有实现接口I,但尝试将B的实例转换为I,编译时可能不会报错(如果I和B在类型体系中没有直接关系的话),但运行时会抛出ClassCastException。不过,这种情况是否符合编译条件呢?假设B和I之间没有继承关系,直接强制转换在编译时应该会报错,因为编译器会检查类型是否兼容。例如,如果B没有实现I,那么B和I之间没有继承关系,强制转换是不允许的,编译器会报错。所以这种情况下,用户可能在代码中遇到了编译错误,或者运行时异常,需要明确类是否实现了目标接口。 再想想,可能用户遇到了某些特殊情况,比如使用匿名内部类或者动态代理,这时候可能有隐式的接口实现。或者是在使用某些框架时,框架生成代理类实现接口,而用户尝试将原始类转换为接口,这时候可能也会有问题。例如,Spring的AOP代理,如果一个类没有被代理包裹,直接转换可能会失败。 需要给用户解释清楚,当类实现接口时,强制转换其实是多余的,可以直接赋值。而当类没有实现接口时,强制转换会导致错误。同时,要强调instanceof操作符的重要性,用于在转换前检查类型是否兼容,避免ClassCastException。 可能用户的实际问题是在代码中遇到了类型转换异常,或者想了解如何安全地将类转换为接口类型。因此,需要给出示例代码,说明正确和错误的转换情况,以及如何正确使用接口和类型转换。此外,可能需要提到接口的多态性,即通过接口类型引用对象,可以调用接口定义的方法,而不需要关心具体实现类。 总结回答的结构应该是:先解释类型转换的基本概念,然后分情况讨论类是否实现接口时的转换情况,强调编译时和运行时的不同行为,给出示例代码,最后总结注意事项和最佳实践,比如使用instanceof进行检查,避免不必要的强制转换,以及接口在Java多态中的应用。</think>在Java中,将类强制转换为接口是一种类型转换操作,其本质是通过接口类型引用对象。以下是逐步解释: --- ### 1. **前提条件** - **类必须实现接口**(直接或间接实现),否则会在运行时抛出`ClassCastException`。 - 转换目的是通过接口类型调用类实现接口方法,而非访问类的特有方法。 --- ### 2. **转换示例** #### 场景:类实现接口 ```java interface Animal { void eat(); } class Dog implements Animal { public void eat() { System.out.println("Dog eats"); } public void bark() { System.out.println("Dog barks"); } } ``` ```java Dog dog = new Dog(); Animal animal = (Animal) dog; // 显式强制转换(实际可省略,因自动向上转型) animal.eat(); // 合法:调用接口方法 // animal.bark(); // 错误:接口类型无法调用Dog特有方法 ``` --- ### 3. **不实现接口的强制转换** 若类未实现接口,编译可能通过(若存在继承关系欺骗编译器),但运行时报错: ```java class Cat { /* 未实现Animal接口 */ } Cat cat = new Cat(); Animal animal = (Animal) cat; // 编译可能通过(如Animal是Cat父接口),但运行时报ClassCastException ``` --- ### 4. **编译时与运行时的区别** - **编译时检查**:仅验证类型间是否可能存在继承关系(如接口是否在类继承树中)。 - **运行时检查**:实际判断对象是否真正实现了目标接口。 --- ### 5. **正确使用建议** 1. **优先向上转型**:无需显式强制转换: ```java Animal animal = new Dog(); // 自动转型 ``` 2. **使用`instanceof`检查**: ```java if (dog instanceof Animal) { Animal animal = (Animal) dog; } ``` 3. **避免无意义转换**:仅在需要接口的泛型、方法参数等场景使用。 --- ### 6. **接口多态的核心意义** 通过接口类型操作对象,实现**解耦**和**多态**: ```java void feed(Animal animal) { animal.eat(); // 无需关心具体是Dog还是其他实现类 } ``` --- ### 总结 - **合法转换**:类必须实现目标接口。 - **本质**:通过接口类型限制操作范围,强调行为抽象。 - **风险**:未实现接口时强制转换会导致运行时异常,需谨慎使用`instanceof`验证。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值