java的jdk动态代理,这里的动态是怎么体现的?

CaptainDing 2016-11-26 10:24:16
  1. import java.lang.reflect.InvocationHandler;
  2. import java.lang.reflect.Method;
  3. import java.lang.reflect.Proxy;

  4. public class ProxyHandler implements InvocationHandler {

  5. private Object obj;

  6. public Object bind(Object obj){
  7. this.obj = obj;
  8. return Proxy.newProxyInstance(obj.getClass().getClassLoader(),
  9. obj.getClass().getInterfaces(),
  10. this);
  11. }
  12. @Override
  13. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

  14. Object result = null;
  15. System.out.println("before invoke!*****");
  16. result = method.invoke(obj, args);
  17. System.out.println("after invoke!-----");
  18. return result;
  19. }

  20. }


  21. public class RealSubject implements Subject {

  22. @Override
  23. public void dosomething() {
  24. System.out.println(" call something! ");

  25. }
  26. }

  27. public interface Subject {

  28. public void dosomething();
  29. }

  30. public class SubjectProxy implements Subject {

  31. @Override
  32. public void dosomething() {

  33. }
  34. }


  35. public class TestProxy {

  36. public static void main(String args[]){
  37. ProxyHandler proxy = new ProxyHandler();
  38. //绑定该类实现的所有接口
  39. Subject sub = (Subject) proxy.bind(new RealSubject());
  40. sub.dosomething();
  41. }
  42. }


具体这个代理类,java底层是如何实现的?
...全文
给本帖投票
377 1 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
1 条回复
切换为时间正序
请发表友善的回复…
发表回复
bichir 2016-11-26
  • 打赏
  • 举报
回复
import java.lang.reflect.Field;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Proxy;

import com.sun.xml.internal.ws.org.objectweb.asm.ClassReader;
import com.sun.xml.internal.ws.org.objectweb.asm.ClassWriter;
 
public class ProxyHander implements InvocationHandler {
 
    private Object obj;
 
    public Object bind(Object obj){
        this.obj  = obj;
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(),
                obj.getClass().getInterfaces(),
                this);
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    	
        Object result = null;
        System.out.println("before invoke!*****");
        result = method.invoke(obj, args);
        System.out.println("after invoke!-----");       
        return result;
    }
 
    public static void main(String args[]){
    	RealSubject rs = new RealSubject();
    	ProxyHander proxy = new ProxyHander();
        //绑定该类实现的所有接口
        Subject obj = (Subject)proxy.bind(rs);
        obj.dosomething();
        
        Class<?> c = obj.getClass();
        Field[] fs = c.getDeclaredFields();
        
        //定义可变长的字符串,用来存储属性  
        StringBuffer sb = new StringBuffer();  
        //通过追加的方法,将每个属性拼接到此字符串中  
        //最外边的public定义  
        sb.append(Modifier.toString(c.getModifiers()) + " class " + c.getSimpleName() );  
        Class<?>[] interfaces = c.getInterfaces();
        if(interfaces.length>0){
        	sb.append(" implements ");
        }
        for(Class<?> i:interfaces){
        	sb.append(i.getName()+" ");
        }
        sb.append("{\n");
        //里边的每一个属性  
        for(Field field:fs){  
        	field.setAccessible(true);
            sb.append("\t");//空格  
            sb.append(Modifier.toString(field.getModifiers())+" ");//获得属性的修饰符,例如public,static等等  
            sb.append(field.getType().getSimpleName() + " ");//属性的类型的名字  
            sb.append(field.getName()+";\n");//属性的名字+回车  
        }  
        
        Method[] m = c.getDeclaredMethods();
        
        for(Method method:m){
        	method.setAccessible(true);
        	sb.append("\t");//空格  
            sb.append(Modifier.toString(method.getModifiers())+" ");//获得属性的修饰符,例如public,static等等  
            sb.append(method.getReturnType().getName()+" ");
            Class<?>[] param = method.getParameterTypes();//获取方法类返回型
            sb.append(" "+method.getName()+" (");
            for(int i=0;i<param.length;i++){
            	sb.append(param[i].getName()+" "+"arg"+i);
            	if(i!=param.length-1){
            		sb.append(",");
            	}
            }
            sb.append(")\n");
        }
        
        sb.append("}");  
  
        System.out.println(sb);  
        
        
   }
}
 
 
class RealSubject implements Subject {
 
    @Override
    public void dosomething() {
        System.out.println(" call something! ");
         
    }
    
}
 
interface Subject {
 
    public void dosomething();
}
 
执行结果
before invoke!*****
 call something! 
after invoke!-----
public final class $Proxy0 implements Subject {
	private static Method m1;
	private static Method m3;
	private static Method m0;
	private static Method m2;
	public final boolean  equals (java.lang.Object arg0)
	public final java.lang.String  toString ()
	public final int  hashCode ()
	public final void  dosomething ()
}
从上面反射后的class内容可以知道他会用classwrite重新建一个Proxyxx的类并实现你的超类接口,你的dosometing会被重写。你可以把接口换成类试一下,这个代理就会失效了。所以框架注入时都会要求用户写接口也是这样来的。但现在的cglib 可以不写接口也能实现注入

62,635

社区成员

发帖
与我相关
我的任务
社区描述
Java 2 Standard Edition
社区管理员
  • Java SE
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧

手机看
关注公众号

关注公众号

客服 返回
顶部