(三)Spring注解和代理模式

一、使用注解开发

1.1、说明

在spring4之后,想要使用注解形式,必须得要引入spring-aop的包
在配置文件中引入context约束,开启注解的支持

<?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:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">
	<context:annotation-config/>
</beans>

1.2、Bean的实现

之前都是使用 bean 的标签进行bean注入,但是实际开发中,一般都会使用注解

  1. 配置扫描哪些包下的注解

    <!--指定注解扫描包-->
    <context:component-scan base-package="com.kuang.pojo"/>
    
  2. 在指定包下编写类,增加注解

    // 等价于 <bean id="user" class="com.study06.pojo.User"/>
    // @Component 组件放在类上,就说明这个类被Spring容器管理了,就是bean,名字默认为类名小写
    @Component("user")
    public class User {
        public String name = "哈哈";
    }
    
  3. 测试

    @Test
    public void test(){
    	ApplicationContext applicationContext = new ClassPathXmlApplicationContext("beans.xml");
    	User user = (User) applicationContext.getBean("user");
    	System.out.println(user.name);
    }
    

1.3、属性注入

  1. 可以不用提供set方法,直接在直接名上添加@value(“值”)

    @Component("user")
    // 相当于配置文件中 <bean id="user" class="当前注解的类"/>
    public class User {
    	@Value("秦疆")
    	// 相当于配置文件中 <property name="name" value="秦疆"/>
    	public String name;
    }
    
  2. 如果提供了set方法,在set方法上添加@value(“值”)

    @Component("user")
    public class User {
    	public String name;
    	@Value("秦疆")
    	public void setName(String name) {
    		this.name = name;
    	}
    }
    

1.4、衍生注解

@Component有几个衍生注解,在web开发中,会按照mvc三层架构分层

  • dao:@Repository
  • service:@Service
  • controller:@Controller

这四个注解功能都是一样的,都是代表将某个类注册到Spring容器中,装配bean

1.5、自动装配注解

上一章讲过了

  • @Autowried:通过类型、名字自动装配
  • @Qualifier:如果@Autowried不能唯一自动装配上属性,则需要通过@Qualifier(value=“xxx”)指定bean名字
  • @Nullable:字段标记了这个注解,说明这个字段可以为null
  • @Resource:通过名字、类型装配

1.6、作用域

@scope

  • singleton:默认的,Spring会采用单例模式创建这个对象。关闭工厂 ,所有的对象都会销毁。
  • prototype:多例模式。关闭工厂 ,所有的对象不会销毁。内部的垃圾回收机制会回收
@Controller("user")
@Scope("prototype")
public class User {
	@Value("秦疆")
	public String name;
}

1.7、小结

xml与注解:

  • XML可以适用任何场景 ,结构清晰,维护方便
  • 注解不是自己提供的类使用不了,开发简单方便

xml与注解整合开发:推荐

  • xml管理Bean
  • 注解完成属性注入
  • 在使用的过程中,只需要注意一个问题:必须让注解生效,就需要开启注解的支持

<context:annotation-config/>作用:

  • 进行注解驱动注册,从而使注解生效
  • 用于激活那些已经在spring容器里注册过的bean上面的注解,也就是显示的向Spring注册
  • 如果不扫描包,就需要手动配置bean
  • 如果不加注解驱动,则注入的值为null

二、使用Java的方式配置Bean

JavaConfig 原来是 Spring 的一个子项目,它通过 Java 类的方式提供 Bean 的定义信息,在 Spring4 的版本, JavaConfig 已正式成为 Spring4 的核心功能 。

  1. 编写一个实体类

    @Component //将这个类标注为Spring的一个组件,放到容器中!
    public class Dog {
    	public String name = "dog";
    }
    
  2. 新建一个config配置包,编写一个MyConfig配置类

    // 这个也会被Spring托管,注册到容器中。因为它本质上也是一个@component
    // @Configuration代表这是一个配置类,就和我们之前看的beans.xml一样的
    @Configuration
    @ComponentScan("com.study07")
    public class MyConfig {
        // 注册一个bean,就相当于之前写的一个bean标签
        // 这个方法的名字,就相当于bean标签中的ID属性
        // 这个方法的返回值,就相当于bean标签中的class属性
    	@Bean
    	public Dog dog(){
    		return new Dog(); // 就是返回要注入到bean的对象
    	}
    }
    
  3. 测试

@Test
public void test2(){
	// 如果完全使用了配置类方式去做,就只能通过AnnotationConfig上下文来获取容器,通过配置类的class对象来加载
	ApplicationContext applicationContext = new AnnotationConfigApplicationContext(MyConfig.class);
	// 要取方法名
	Dog dog = (Dog) applicationContext.getBean("dog");
	System.out.println(dog.name);
}

此外,还可以导入其他配置类:

  1. 新建一个配置类

    @Configuration
    public class MyConfig2 {
    }
    
  2. 导入这个新的配置类

    @Configuration
    @Import(MyConfig2.class) //导入合并其他配置类,类似于配置文件中的 inculde 标签
    public class MyConfig {
    	@Bean
    	public Dog dog(){
    		return new Dog();
    	}
    }
    

三、代理模式

AOP的低层机制就是动态代理
代理模式的分类:

  • 静态代理
  • 动态代理

3.1、静态代理

静态代理角色分析:

  • 抽象角色:一般使用接口或者抽象类来实现
  • 真实角色:被代理的角色
  • 代理角色:代理真实角色,代理真实角色后,一般会做一些附属的操作
  • 客户:使用代理角色来做一些操作

代码实现:

  1. 接口

    /**
     * 租房的接口
     */
    public interface Rent {
        void rent();
    }
    
  2. 真实角色

    /**
     * 房东
     */
    public class Host implements Rent {
        public void rent() {
            System.out.println("房东要出租房子");
        }
    }
    
  3. 代理角色

    /**
     * 中介(代理)
     */
    public class Proxy implements Rent {
        private Host host;
        public Proxy() {
        }
        public Proxy(Host host) {
            this.host = host;
        }
        @Override
        public void rent() {
            seeHouse();
            host.rent();
            hetong();
            fare();
        }
        // 看房
        public void seeHouse() {
            System.out.println("中介带你看房");
        }
        // 收中介费
        public void fare() {
            System.out.println("收中介费");
        }
        // 签合同
        public void hetong() {
            System.out.println("签租赁合同");
        }
    
    }
    
  4. 客户端访问代理角色

    /**
     * 客户端,即租房的人
     */
    public class Client {
        public static void main(String[] args) {
            // 房东要租房子
            Host host = new Host();
            // 代理,中介帮房东租房子。但是代理角色一般会有一些附属操作
            Proxy proxy = new Proxy(host);
            // 不用面对房东,直接找中介租房即可
            proxy.rent();
        }
    }
    

3.2、代理模式的优缺点

优点:

  • 可以使真实角色的操作更加纯粹,不用去关注一些公共的业务
  • 公共业务就交给了代理角色,实现了业务的分工
  • 公共业务发生扩展的时候,方便集中管理

缺点:

  • 一个真实角色就会产生一个代理角色,代码量会翻倍,开发效率会变低

3.3、静态代理再理解

  1. 创建一个抽象角色

    public interface UserService {
        void add();
        void delete();
        void update();
        void query();
    }
    
  2. 真实角色

    // 真实对象
    public class UserServiceImpl implements UserService {
        @Override
        public void add() {
            System.out.println("增加了一个用户");
        }
        @Override
        public void delete() {
            System.out.println("删除了一个用户");
        }
        @Override
        public void update() {
            System.out.println("修改了一个用户");
        }
        @Override
        public void query() {
            System.out.println("查询了一个用户");
        }
        // 改动原有的业务代码,在公司是大忌
    }
    

在操作上增加一个日志功能

  1. 代理角色

    public class UserServiceProxy implements UserService {
        UserServiceImpl userService;
        public void setUserService(UserServiceImpl userService) {
            this.userService = userService;
        }
        @Override
        public void add() {
            log("add");
            userService.add();
        }
        @Override
        public void delete() {
            log("delete");
            userService.delete();
        }
        @Override
        public void update() {
            log("update");
            userService.update();
        }
        @Override
        public void query() {
            userService.query();
        }
        // 日志方法
        public void log(String msg) {
            System.out.println("使用了" + msg + "方法");
        }
    }
    
  2. 客户端访问代理角色

    public static void main(String[] args) {
        UserServiceImpl userService = new UserServiceImpl();
    
        UserServiceProxy proxy = new UserServiceProxy();
        proxy.setUserService(userService);
    
        proxy.add();
    }
    

3.4、总结:

在不改变原来的代码的情况下,实现了对原有功能的增强,这是AOP中最核心的思想

3.5、动态代理

  • 动态代理和静态代理角色一样
  • 动态代理的代理类是动态生成的,不是直接写好的
  • 动态代理分为:基于接口的动态代理,基于类的动态代理两大类
    • 基于接口的动态代理------JDK动态代理【在这里使用】
    • 基于类的动态代理------cglib
    • 现在用的比较多的是:Java字节码实现的Javassist

需要了解两个类:Proxy:代理,InvocationHandler:调用处理程序

  1. 接口

    /**
     * 租房的接口
     */
    public interface Rent {
        void rent();
    }
    
  2. 真实角色

    /**
     * 房东
     */
    public class Host implements Rent {
        public void rent() {
            System.out.println("房东要出租房子");
        }
    }
    
  3. ProxyInvocationHandler. java 即代理角色

    // 用这个类,自动生成代理类
    public class ProxyInvocationHandler implements InvocationHandler {
    
        // 被代理的接口
        private Rent rent;
    
        public void setRent(Rent rent) {
            this.rent = rent;
        }
    
        // 生成代理类
        public Object getProxy() {
            return Proxy.newProxyInstance(this.getClass().getClassLoader(), rent.getClass().getInterfaces(), this);
        }
    
        // 处理代理实例,并返回结果
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            // 动态代理的本质,就是使用反射机制实现的
            seeHouse();
            Object result = method.invoke(rent, args);
            fare();
            return result;
        }
    
        public void seeHouse() {
            System.out.println("中介带看房");
        }
        public void fare() {
            System.out.println("收中介费");
        }
    }
    
  4. Client

    public class Client {
        public static void main(String[] args) {
            // 真实角色
            Host host = new Host();
    
            // 代理角色:现在没有
            ProxyInvocationHandler pih = new ProxyInvocationHandler();
            // 通过调用程序处理角色来处理要调用的接口
            pih.setRent(host);
            Rent proxy = (Rent)pih.getProxy(); // 这里的Proxy就是动态生成的
    
            proxy.rent();
        }
    }
    

3.6、动态代理再理解

使用动态代理实现代理后面写的UserService

// 用这个类,自动生成代理类
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);
    }

    // 处理代理实例,并返回结果
    @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 msg) {
        System.out.println("执行了"+ msg +"方法");
    }
}

测试:

public class Client {
    public static void main(String[] args) {
        // 真实角色
        UserServiceImpl userService = new UserServiceImpl();
        // 代理角色
        ProxyInvocationHandler pih = new ProxyInvocationHandler();
        pih.setTarget(userService);  // 设置要代理的对象
        // 动态生成代理类
        UserService proxy = (UserService)pih.getProxy();
        proxy.add();
    }
}

动态代理的好处:

  • 可以使真实角色的操作更加纯粹,不用去关注一些公共的业务
  • 公共业务就交给了代理角色,实现了业务的分工
  • 公共业务发生扩展的时候,方便集中管理
  • 一个动态代理类代理的是一个接口,一般就是对应的一类业务
  • 一个动态代理类可以代理多个类,只要是实现了同一个接口即可
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值