我们来比较Java的class
和interface
的区别:
- 可以实例化
class
(非abstract
); - 不能实例化
interface
。
所有interface
类型的变量总是通过向上转型并指向某个实例的:
CharSequence cs = new StringBuilder();
有没有可能不编写实现类,直接在运行期创建某个interface
的实例呢?
这是可能的,因为Java标准库提供了一种动态代理(Dynamic Proxy)的机制:可以在运行期动态创建某个interface
的实例。
什么叫运行期动态创建?听起来好像很复杂。所谓动态代理,是和静态相对应的。我们来看静态代码怎么写:
定义接口:
public interface Hello { void morning(String name); }
编写实现类:
public class HelloWorld implements Hello { public void morning(String name) { System.out.println("Good morning, " + name); } }
创建实例,转型为接口并调用:
Hello hello = new HelloWorld(); hello.morning("Bob");
这种方式就是我们通常编写代码的方式。
还有一种方式是动态代码,我们仍然先定义了接口Hello
,但是我们并不去编写实现类,而是直接通过JDK提供的一个Proxy.newProxyInstance()
创建了一个Hello
接口对象。这种没有实现类但是在运行期动态创建了一个接口对象的方式,我们称为动态代码。JDK提供的动态创建接口对象的方式,就叫动态代理。
一个最简单的动态代理实现如下:
1 import java.lang.reflect.*; 2 public class Demo12{ 3 /* 4 在运行期动态创建一个interface实例的方法如下: 5 6 1.定义一个InvocationHandler实例,它负责实现接口的方法调用; 7 2.通过Proxy.newProxyInstance()创建interface实例,它需要3个参数: 8 1.使用的ClassLoader,通常就是接口类的ClassLoader; 9 2.需要实现的接口数组,至少需要传入一个接口进去; 10 3.用来处理接口方法调用的InvocationHandler实例。 11 3.将返回的Object强制转型为接口。 12 */ 13 14 public static void main(String[] args) throws Exception{ //记住要抛出异常 15 InvocationHandler handler = new InvocationHandler(){ 16 @Override 17 public Object invoke(Object proxy,Method method,Object[] args){ 18 System.out.println(method); 19 if(method.getName().equals("morning")){ 20 System.out.println("Good morning," + args[0]); 21 } 22 return null; 23 } 24 }; 25 26 Hello h= (Hello)Proxy.newProxyInstance(Hello.class.getClassLoader(),new Class[]{Hello.class},handler); 27 28 h.morning("单身狗"); 29 } 30 } 31 32 interface Hello{ 33 void morning(String name); 34 }
动态代理实际上是JVM在运行期动态创建class字节码并加载的过程,它并没有什么黑魔法
小结
Java标准库提供了动态代理功能,允许在运行期动态创建一个接口的实例;
动态代理是通过Proxy
创建代理对象,然后将接口方法“代理”给InvocationHandler
完成的。