zoukankan      html  css  js  c++  java
  • 动态代理

     

    1.什么是代理(中介)

    目标对象/被代理对象 ------ 房主:真正的租房的方法

    代理对象 ------- 黑中介:有租房子的方法(调用房主的租房的方法)

    执行代理对象方法的对象 ---- 租房的人

     

    流程:我们要租房----->中介(租房的方法)------>房主(租房的方法)

    抽象:调用对象----->代理对象------>目标对象

    2.动态代理

    动态代理:不用手动编写一个代理对象,不需要一一编写与目标对象相同的方法,这个过程,在运行时 的内存中动态生成代理对象。------字节码对象级别的代理对象

     

    动态代理的API:

    在jdk的API中存在一个Proxy中存在一个生成动态代理的的方法newProxyInstance

    static Object

    newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)

     

    返回值:Object就是代理对象

    参数:loader:代表与目标对象相同的类加载器-------目标对象.getClass().getClassLoader()

    interfaces:代表与目标对象实现的所有的接口字节码对象数组

    h:具体的代理的操作,InvocationHandler接口

     

    注意:JDK的Proxy方式实现的动态代理 目标对象必须有接口 没有接口不能实现jdk版动态代理

     如下:

    package com.proxy;
    
    public class Target implements TargetInterface{
    
    	@Override
    	public void method1() {
    		System.out.println("method1 running...");
    	}
    
    	@Override
    	public String method2() {
    		System.out.println("method2 running...");
    		return "method2";
    	}
    
    	@Override
    	public int method3(int x) {
    		return x;
    	}
    
    	
    	
    }
    

      

    package com.proxy;
    
    public interface TargetInterface {
    
    	public void method1();
    	public String method2();
    	public int method3(int x);
    }
    package com.proxy;
    
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    
    public class ProxyTest2 {
    
    	public static void main(String[] args) {
    		
    		final Target target = new Target();
    		
    		//动态创建代理对象
    		
    		TargetInterface proxy = (TargetInterface) Proxy.newProxyInstance(
    				target.getClass().getClassLoader(), 
    				target.getClass().getInterfaces(), 
    				new InvocationHandler() {
    					@Override
    					//被执行几次?------- 看代理对象调用方法几次
    					//代理对象调用接口相应方法 都是调用invoke
    					/*
    					 * proxy:是代理对象
    					 * method:代表的是目标方法的字节码对象
    					 * args:代表是调用目标方法时参数
    					 */
    					public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    						//反射知识点
    						Object invoke = method.invoke(target, args);//目标对象的相应方法
    						//retrun返回的值给代理对象
    						return invoke;
    					}
    				}
    			);
    		
    		proxy.method1();//调用invoke---Method:目标对象的method1方法  args:null  返回值null
    		String method2 = proxy.method2();//调用invoke---Method:目标对象的method2方法  args:null  返回值method2
    		int method3 = proxy.method3(100);////调用invoke-----Method:目标对象的method3方法 args:Object[]{100}  返回值100
    		
    		System.out.println(method2);
    		System.out.println(method3);
    		
    	}
    	
    }
    

      

      还可以用来增强方法、拦截  (不需要创建动态代理对象)

    package com.web.filter;
    
    import java.io.IOException;
    import java.io.UnsupportedEncodingException;
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    
    import javax.servlet.Filter;
    import javax.servlet.FilterChain;
    import javax.servlet.FilterConfig;
    import javax.servlet.ServletException;
    import javax.servlet.ServletRequest;
    import javax.servlet.ServletResponse;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletRequestWrapper;
    
    public class EncodingFilter implements Filter{
    
    	
    	@Override
    	public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
    			throws IOException, ServletException {
    		
    		final HttpServletRequest req = (HttpServletRequest) request;
    		
    		//使用动态代理完成全局编码
    		HttpServletRequest enhanceRequset = (HttpServletRequest) Proxy.newProxyInstance(
    				req.getClass().getClassLoader(), 
    				req.getClass().getInterfaces(), 
    				new InvocationHandler() {
    					@Override
    					public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    						//对getParameter方法进行增强
    						String name = method.getName();//获得目标对象的方法名称
    						if("getParameter".equals(name)){
    							String invoke = (String) method.invoke(req, args);//乱码
    							//转码
    							invoke = new String(invoke.getBytes("iso8859-1"),"UTF-8");
    							return invoke;
    						}
    						return method.invoke(req, args);
    					}
    				}
    					
    			);
    		
    		
    		chain.doFilter(enhanceRequset, response);
    		
    		
    		
    				
    	}
    
    	@Override
    	public void destroy() {
    		
    	}
    	
    	@Override
    	public void init(FilterConfig filterConfig) throws ServletException {
    		
    	}
    
    }
    
    
    

      

  • 相关阅读:
    17.天堂和地狱只在你的一念之间
    14.事情原来可能更糟
    如何评价一个网站的人气(Link Popularity Check)
    1.人生何必一定要成功
    4.何不看开一点
    字符串到枚举的转换代码
    广州泳场一览表
    收集开源项目
    7.把“失去”当作“抛弃”
    经常使用的Oracle监控语句
  • 原文地址:https://www.cnblogs.com/wuxu/p/10929372.html
Copyright © 2011-2022 走看看