Spring学习笔记--代理

静态代理

1.静态代理的角色分析(UML画图推荐使用StarUML软件)

 抽象角色---一般使用接口或者抽象类来实现。

 真是角色---被代理的角色。

代理角色---代理真实角色—代理真是角色后一般会做一些附属操作。

客户-------使用代理角色来进行一些操作.

2.代码的实现(房东-中介-客户)

 Rent.java--抽象角色

package cn.sxt.staticproxy;

public interface Rent {
	public void rent();
}

Host.java--真实角色

package cn.sxt.staticproxy;

public class Host implements Rent{
	@Override
	public void rent(){
		System.out.println("房屋出租");
	}
}

Proxy.java--代理角色

package cn.sxt.staticproxy;

public class Proxy implements Rent{
	private Host host;
	public Proxy(){

	}
	public Proxy(Host host) {
		super();
		this.host = host;
	}
	public void setHost(Host host) {
		this.host = host;
	}
	//租房
	public void rent(){
		seeHouse();
		host.rent();
		fare();
	}
	//看房
	private void seeHouse(){
		System.out.println("带房客看房");
	}
	//收中介费
	private void fare(){
		System.out.println("收取中介费");
	}
}

Client.java--客户

package cn.sxt.staticproxy;

public class Client {
	public static void main(String[] args) {

		Host host=new Host();
		Proxy proxy=new Proxy(host);
		proxy.rent();
	}
}

3.使用静态代理的好处:

使得真实角色处理的业务更加的纯粹,不再去关注一些公共的事情;

      公共的业务由代理来完成---实现了业务的分工;

      公共业务发生扩展时,变得更加集中和方便;

举例说明:

UserService.java:

public interface UserService {
	public void add();
	public void update();
	public void delete();
	public void search();
}

UserServiceImpl.java:关注纯粹的业务逻辑

public class UserServiceImpl implements UserService {

	@Override
	public void add() {
		System.out.println("增加用户");
	}

	@Override
	public void update() {

		System.out.println("修改用户");
	}

	@Override
	public void delete() {

		System.out.println("删除用户");
	}

	@Override
	public void search() {

		System.out.println("查询用户");
	}
}

UserServiceProxy.java:处理一些公共的事务,比如日志

public class UserServiceProxy implements UserService {

	private UserService userService;

	@Override
	public void add() {
		log("add");
		userService.add();
	}

	@Override
	public void update() {
		log("update");
		userService.update();
	}

	@Override
	public void delete() {
		log("delete");
		userService.delete();
	}

	@Override
	public void search() {
		log("search");
		userService.delete();
	}

	public void log(String methodName) {
		System.out.println("执行" + methodName + "执行");
	}

}

静态代理的缺点:

类变的多了---多了代理类,工作量变大了,开发效率降低了

于是乎----动态代理出现了:有静态代理的好处,抛弃静态代理的缺点

动态代理

1.动态代理和静态代理的角色是一样的。

2.动态代理的代理类是动态生成的。

3.动态代理分为两类

a)基于接口的动态代理---jdk动态代理

b)基于类的动态代理---cglib

现在用javasist来生成动态代理

4.jdk的动态代理---Proxy类和InvocationHandler接口

InvocationHandler 是代理实例的调用处理程序 实现的接口。

每个代理实例都具有一个关联的调用处理程序。对代理实例调用方法时,将对方法调用进行编码并将其指派到它的调用处理程序的 invoke 方法。

invoke方法:

Object invoke(Object proxy, Method method, Object[] args)
          在代理实例上处理方法调用并返回结果。

在代理实例上处理方法调用并返回结果。在与方法关联的代理实例上调用方法时,将在调用处理程序上调用此方法。

参数:
proxy - 在其上调用方法的代理实例
method - 对应于在代理实例上调用的接口方法的 Method 实例。Method 对象的声明类将是在其中声明方法的接口,该接口可以是代理类赖以继承方法的代理接口的超接口。
args - 包含传入代理实例上方法调用的参数值的对象数组,如果接口方法不使用参数,则为 null。基本类型的参数被包装在适当基本包装器类(如 java.lang.Integerjava.lang.Boolean)的实例中。
返回:
从代理实例的方法调用返回的值。如果接口方法的声明返回类型是基本类型,则此方法返回的值一定是相应基本包装对象类的实例;否则,它一定是可分配到声明返回类型的类型。如果此方法返回的值为 null 并且接口方法的返回类型是基本类型,则代理实例上的方法调用将抛出 NullPointerException。否则,如果此方法返回的值与上述接口方法的声明返回类型不兼容,则代理实例上的方法调用将抛出 ClassCastException

Proxy 提供用于创建动态代理类和实例的静态方法,它还是由这些方法创建的所有动态代理类的超类。

static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
          返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。

返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。此方法相当于:

     Proxy.getProxyClass(loader, interfaces).
         getConstructor(new Class[] { InvocationHandler.class }).
         newInstance(new Object[] { handler }); 

Proxy.newProxyInstance 抛出 IllegalArgumentException,原因与 Proxy.getProxyClass 相同。

参数:
loader - 定义代理类的类加载器
interfaces - 代理类要实现的接口列表
h - 指派方法调用的调用处理程序

实现:

public class ProxyInvocationHandler implements InvocationHandler{
	//目标对象--真实对象
	private Object target;
	public void setTarget(Object target) {
		this.target = target;
	}
	/**
	 * 生成代理类:
	 */
	public Object getProxy(){
		return Proxy.newProxyInstance(this.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
	}

	/**
	 * proxy--是代理类
	 * method--代理类的实例(proxy)调用的处理程序(add)的方法对象
	 * 比如调用proxy.add():此处method.getName就是add
	 */
	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		log(method.getName());
		Object result=method.invoke(target, args);
		return result;
	}
	public void log(String methodName){
		System.out.println("执行"+methodName+"方法");
	}
	//看房
	private void seeHouse(){
		System.out.println("带房客看房");
	}
	//收中介费
	private void fare(){
		System.out.println("收取中介费");
	}

}

Client:

import cn.sxt.service.UserService;
import cn.sxt.service.UserServiceImpl;

public class Client {
	public static void main(String[] args) {

		UserService userService=new UserServiceImpl();
		ProxyInvocationHandler pih=new ProxyInvocationHandler();
		pih.setTarget(userService);
		UserService proxy=(UserService)pih.getProxy();
		proxy.add();
	}
}

一个动态代理一般代理某一类业务,一个动态代理可以代理多各类

时间: 2024-10-31 13:09:39

Spring学习笔记--代理的相关文章

不错的Spring学习笔记(转)

Spring学习笔记(1)----简单的实例 ---------------------------------   首先需要准备Spring包,可从官方网站上下载.   下载解压后,必须的两个包是spring.jar和commons-logging.jar.此外为了便于测试加入了JUnit包.   在Myeclipse中创建Java项目.   编写一个接口类,为了简单,只加入了一个方法.   Java代码   1.package com.szy.spring.interfacebean;  

Spring学习笔记(一)

Spring学习笔记(一) Spring核心思想: IOC:  Inversion Of Control (控制反转) / DI: Dependency Injection (依赖注入) AOP: Aspect Oriented Programming (面向切面编程) IOC 1. 简单的应用 Model package com.wangj.spring.model; public class User { private String username; private String pas

《Spring学习笔记》:Spring、Hibernate、struts2的整合(以例子来慢慢讲解,篇幅较长)

<Spring学习笔记>:Spring.Hibernate.struts2的整合(以例子来慢慢讲解,篇幅较长) 最近在看马士兵老师的关于Spring方面的视频,讲解的挺好的,到了Spring.Hibernate.struts2整合这里,由于是以例子的形式来对Spring+Hibernate+struts2这3大框架进行整合,因此,自己还跟着写代码的过程中,发现还是遇到了很多问题,因此,就记录下. 特此说明:本篇博文完全参考于马士兵老师的<Spring视频教程>. 本篇博文均以如下这

spring学习笔记(19)mysql读写分离后端AOP控制实例

在这里,我们接上一篇文章,利用JNDI访问应用服务器配置的两个数据源来模拟同时操作不同的数据库如同时操作mysql和oracle等.实际上,上个例子可能用来模拟mysql数据库主从配置读写分离更贴切些.既然如此,在本例中,我们就完成读写分离的模拟在web端的配置实例. 续上次的例子,关于JNDI数据源的配置和spring datasource的配置这里不再重复.下面着重加入AOP实现DAO层动态分库调用.可先看上篇文章<spring学习笔记(18)使用JNDI模拟访问应用服务器多数据源实例 >

Spring学习笔记(三)

Spring学习笔记(三) AOP 一.使用Annotation方式实现AOP.步骤: xml里加入配置:<aop:aspectj-autoproxy /> <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org

Spring学习笔记--initmethod和构造函数、setter方法的加载顺序

今天学习了一下spring中bean的初始化和销毁,突然想了解一下初始化方法跟构造函数及setter方法注入的执行顺序,记录在此,仅作为学习笔记. 当实例化一个bean时,可能需要执行一些初始化操作来确保该bean处于可用状态.同样地,当不再需要bean时,将其从容器中移除是,我们可以还需要按顺序 执行一些清除工作. package com.zp.chapter2; public class Auditorium { private String name; public void doBefo

【Spring学习笔记-MVC-3.1】SpringMVC返回Json数据-方式1-扩展

<Spring学习笔记-MVC>系列文章,讲解返回json数据的文章共有3篇,分别为: [Spring学习笔记-MVC-3]SpringMVC返回Json数据-方式1:http://www.cnblogs.com/ssslinppp/p/4528892.html [Spring学习笔记-MVC-4]返回Json数据-方式2:http://www.cnblogs.com/ssslinppp/p/4530002.html [Spring学习笔记-MVC-3.1]SpringMVC返回Json数据-

Spring学习笔记 2014-7-9

Spring需要applicationContext.xml来管理各个Bean,其基本格式: <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:t

Spring学习笔记一(Spring核心思想)

通过学习<Spring in action (Third edition)>的第一章,我大概了解了Spring的基本思想: 1,依赖注入(Dependnecy Injection): 在不使用Spring框架的情况下,一个类要跟另一个类建立联系,可能会使用如下的模式: class A{...} class B{ private A a; ...       } 这样的话,每次实例化一个B的对象,如b1,必定实例化一个A的对象,如a1,并且b1和a1是紧耦合的,b1牢牢地和a1绑定在一起了.他们