小小理解动态代理

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class DynamicProxyForInterfaces {

	public static void main(String[] args) {
		//创建具体的委托类
		Mp3 mp3 = new Mp3();
		Mobile mobile = new Mobile();
		Mp3 sonyMp3 = new SonyMp3();
		Mobile iphone = new Iphone();
		//1.使用音乐播放处理器获取mp3的服务
		/*
		 * 参数说明:
		 * 第一个参数是类加载器,一般使用你需要的服务的那个接口的加载器即可
		 * 第二参数是你想获取的服务的列表,即接口列表,比如mp3的服务列表就是Music音乐这一个
		 * 第三个参数就是调用处理器,构造一个具体委托类的调用处理器
		 * 这里的显式转换,说明了这个生成的动态代理实现了第二个参数的接口,即它可以提供这些接口的服务
		 */
		Music music = (Music)Proxy.newProxyInstance(Music.class.getClassLoader(), mp3.getClass().getInterfaces(), 
				new MusicInvocationHandler(mp3));
		//会调用委托类mp3内的方法
		music.listenMusic();
		
		System.out.println("--------------------------");
		
		//这是一个只想播放电影的傲娇手机,放弃了音乐播放功能
		Movie movie = (Movie)Proxy.newProxyInstance(Movie.class.getClassLoader(), new Class[]{Movie.class}, 
				new MovieInvocationHandler(mobile));
		movie.seeMovie();
		
		System.out.println("--------------------------");
		//下面这样就会报错:ClassCastException,第二个参数中并没有Music服务,所以不能提供
		/*music = (Music)Proxy.newProxyInstance(Movie.class.getClassLoader(), new Class[]{Movie.class}, 
				new MovieInvocationHandler(mobile));
		music.listenMusic();*/
		
		/*第二个参数里加了音乐服务,这个手机就可以播放音乐了,但是调用处理器使用的是电影调用处理器,会打印出
			before see a movie...
			...now begin music with mobile
			after see a movie...
		*所以此时调用处理器也需要更改
		*/
		/*music = (Music)Proxy.newProxyInstance(Movie.class.getClassLoader(), new Class[]{Movie.class,Music.class}, 
				new MovieInvocationHandler(mobile));
		music.listenMusic();
		System.out.println("--------------------------");*/
		
		//使用索尼mp3播放音乐
		music = (Music)Proxy.newProxyInstance(Mp3.class.getClassLoader(), Mp3.class.getInterfaces(), 
				new Mp3InvocationHandler(sonyMp3));
		music.listenMusic();
		System.out.println("--------------------------");
		
		//使用iphone播放音乐
		music = (Music)Proxy.newProxyInstance(mobile.getClass().getClassLoader(), Mobile.class.getInterfaces(), 
				new MobileInvocationHandler(iphone));
		music.listenMusic();
		System.out.println("--------------------------");
		
		//使用iphone看电影
		movie = (Movie)Proxy.newProxyInstance(Mobile.class.getClassLoader(), Mobile.class.getInterfaces(),
				new MobileInvocationHandler(iphone));
		movie.seeMovie();
	}

}

/*
 * 动态代理是一个实现了一个或多个接口的具体委托类,
 * 通过代理类将方法请求发送到中介调用处理器,中介再根据具体委托类的类型进行具体方法的调用
 */




/*具体委托类和代理类要实现的模板接口--begin*/
//·听音乐的功能接口
interface Music{
	void listenMusic();
}
//·看电影的功能接口
interface Movie{
	void seeMovie();
}
/*具体委托类和代理类要实现的模板接口--end*/





/*具体的委托类:即真正提供“服务”的类--begin*/

//·可以播放音乐的mp3类
class Mp3 implements Music{

	@Override
	public void listenMusic() {
		System.out.println("...now begin music with mp3");
	}
}

//·既可以听音乐也可以看视频的手机类
class Mobile implements Music,Movie{

	@Override
	public void seeMovie() {
		System.out.println("...now begin movie with mobile");
	}

	@Override
	public void listenMusic() {
		System.out.println("...now begin music with mobile");
	}
	
}
/*具体的委托类:即真正提供“服务”的类--end*/




/*更具体的委托类,比如mp3下有索尼Sony的牌子,手机有iphone牌子--begin*/
class SonyMp3 extends Mp3{
	@Override
	public void listenMusic() {
		System.out.println("...now begin music with SonyMp3");
	}
}

class Iphone extends Mobile{
	@Override
	public void seeMovie() {
		System.out.println("...now begin movie(装逼) with Iphone");
	}

	@Override
	public void listenMusic() {
		System.out.println("...now begin music(装逼) with Iphone");
	}
}
/*更具体的委托类,比如mp3下有索尼Snoy的牌子,手机有iphone牌子--begin*/





/*调用处理器:将对代理的方法调用分发到真正的委托类,让对应的委托类提供服务--begin*/

// @可以进行音乐播放服务分发的调用处理器
class MusicInvocationHandler implements InvocationHandler{
	//委托类的实例对象
	private Music music;
	public MusicInvocationHandler(Music music){
		this.music = music;
	}
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		/*参数的含义:
		 *proxy:代理类对象
		 *method:要执行的方法
		 *args:方法的参数
		*/
		//音乐播放前可以做一些你想做的事
		System.out.println("before listen music...");
		//播放音乐
		method.invoke(music, args);
		//音乐播放后也可以做一些事情
		System.out.println("after listen music...");
		return null;
	}
	
}

//@ 可以进行视频播放服务分发的调用处理器
class MovieInvocationHandler implements InvocationHandler{
	//委托类的实例对象
	private Movie movie;
	public MovieInvocationHandler(Movie movie){
		this.movie = movie;
	}
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		/*参数的含义:
		 *proxy:代理类对象
		 *method:要执行的方法
		 *args:方法的参数
		*/
		//视频播放前可以做一些你想做的事
		System.out.println("before see a movie...");
		//播放视频
		method.invoke(movie, args);
		//视频播放后也可以做一些事情
		System.out.println("after see a movie...");
		return null;
	}
	
}

//当然也可以不面向具体的服务来写调用处理器,比如你就想面向mp3或者手机来写调用处理器,这只是继承树的层次问题,看具体需求了
//@ 面向mp3的调用处理器
class Mp3InvocationHandler implements InvocationHandler{
	//委托类的实例对象
	private Mp3 mp3;
	public Mp3InvocationHandler(Mp3 mp3){
		this.mp3 = mp3;
	}
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		/*参数的含义:
		 *proxy:代理类对象
		 *method:要执行的方法
		 *args:方法的参数
		*/
		//使用mp3前可以做一些你想做的事
		System.out.println("before use mp3...");
		//播放音乐
		method.invoke(mp3, args);
		//使用mp3后可以做一些你想做的事
		System.out.println("after use mp3...");
		return null;
	}
	
}

//@ 面向mobile的调用处理器
class MobileInvocationHandler implements InvocationHandler{
	//委托类的实例对象
	private Mobile mobile;
	public MobileInvocationHandler(Mobile mobile){
		this.mobile = mobile;
	}
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		/*参数的含义:
		 *proxy:代理类对象
		 *method:要执行的方法
		 *args:方法的参数
		*/
		//使用手机前可以做一些你想做的事
		System.out.println("before use mobile...");
		//播放视频
		method.invoke(mobile, args);
		//使用手机后也可以做一些事情
		System.out.println("after use mobile...");
		return null;
	}
	
}

/*调用处理器:将对代理的方法调用分发到真正的委托类,让对应的委托类提供服务--end*/

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值