Spring-AOP

一、代理模式1.创建项目-1.创建项目springdemo8_proxy-2.完成入门代码测试-3.编辑Service接口package com.jt.service;public interface UserService { void addUser();}-4.编辑ServiceImpl实现类package com.jt.service;import org.springframework.stereotype.Service;@Se..
摘要由CSDN通过智能技术生成

目录

一、代理模式

1.创建项目

-1.创建项目springdemo8_proxy

 -2.完成入门代码测试

 -3.编辑Service接口

-4.编辑ServiceImpl实现类

-5业务层如何控制事务 

 -6.业务代码-问题说明

2.代理机制

-1.代理模式特点

-2.代理特点

 3.动态代理-JDK模式

-1.JDK代理的说明

 -2.编辑代理类

 -3.编辑测试类

4.动态代理-JDK模式案例

-1.编辑代理对象

-2.编辑测试方法

5.动态代理-CGLIB代理

-1.CGLIB说明

-2.CGLIB代理模式

-3.CGLIB代理测试类

6. 关于JDK代理和CGlib代理总结(高程/架构)!!!

二、Spring AOP介绍

1.创建项目

-1.创建项目springdemo9_aop

-2. 层级代码结构

2.AOP

-1.AOP介绍

 3.AOP 入门案例

-1.编辑UserService接口

-2.编辑UserServiceImpl实现类 

-3.编辑配置类

-4.编辑切面类

-5.编辑测试类

-6.常见通知类型

4.切入点表达式

-1.bean表达式

-2.within表达式

-3.execution表达式

-4. @annotation表达式

5.通知方法:

-1.关于AOP 通知的用法

-2.前置通知用法

-3.后置通知用法

-4.异常通知用法

 -5.切面排序

-6.环绕通知

6.关于代理对象生成策略说明

三、关于Spring总结


一、代理模式

1.创建项目

-1.创建项目springdemo8_proxy

 -2.完成入门代码测试

 -3.编辑Service接口

package com.jt.service;

public interface UserService {

    void addUser();
}

-4.编辑ServiceImpl实现类

package com.jt.service;

import org.springframework.stereotype.Service;

@Service
public class UserServiceImpl implements UserService{

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

-5业务层如何控制事务 

事务: 可以保证数据的/原子性/一致性/持久性/隔离性.
说明: 业务层操作时,需要考虑数据库的事务.代码结构如下

package com.jt.service;

import org.springframework.stereotype.Service;

@Service
public class UserServiceImpl implements UserService{

    @Override
    public void addUser() {
        try {
            System.out.println("开启数据库事务");
            System.out.println("新增用户");
            int a = 1/0;
            System.out.println("提交数据库事务");
        }catch (Exception e){
            System.out.println("事务回滚");
        }
    }
}

 -6.业务代码-问题说明

  1. 如果有多个方法,则每个方法都需要控制事务. 代码重复率高.
  2. 业务层service,应该只处理业务,不要和事务代码耦合在一起.否则扩展性不好,耦合性高.

如何解决: 采用代理机制解决.

2.代理机制

-1.代理模式特点

说明: 一般采用代理模式,主要的目的就是为了解耦.将公共的通用的方法(功能/业务)放到代理对象中. 由业务层专注于业务执行即可

-2.代理特点

  1. 为什么使用代理? 因为自己不方便(没有资源)
  2. 代理的作用? 代理要解决(扩展)某些实际问题.
  3. 用户最终执行目标方法!!!

 

 3.动态代理-JDK模式

-1.JDK代理的说明

  1. JDK代理模式是java原生提供的API,无需导包
  2. JDK代理要求: 被代理者必须 要么是接口,要么实现接口
  3. 灵活: 代理对象应该看起来和被代理者 一模一样!!! (方法相同)

 -2.编辑代理类

package com.jt.proxy;

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

public class JDKProxy {

    //传入target目标对象获取代理对象的
    //利用代理对象 实现方法的扩展
    public static Object getProxy(Object target){
        //1.获取类加载器
        ClassLoader classLoader = target.getClass().getClassLoader();
        //2.获取接口数组类型
        Class[] interfaces = target.getClass().getInterfaces();
        //3.代理对象执行方法时的回调方法(代理对象调用方法时,执行InvocationHandler)
        return Proxy.newProxyInstance(classLoader,interfaces,invocationHandler(target));
    }

    //要求必须传递目标对象
    public static InvocationHandler invocationHandler(Object target){
       return new InvocationHandler() {
           @Override
           public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
               System.out.println("事务开启");
               //获取目标方法的返回值
               Object result = method.invoke(target,args);
               System.out.println("事务提交");
               return result;
           }
       };
    }
}

 -3.编辑测试类

package com.jt;

import com.jt.config.SpringConfig;
import com.jt.proxy.JDKProxy;
import com.jt.service.UserService;
import org.junit.jupiter.api.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;

public class TestSpring {

    @Test
    public void demo1(){
        ApplicationContext context =
                new AnnotationConfigApplicationContext(SpringConfig.class);
        //获取目标对象
        UserService userService = context.getBean(UserService.class);
        //userService.addUser();
        //获取代理对象
        UserService proxy = (UserService) JDKProxy.getProxy(userService);
        //代理对象执行方法, 调用invoke方法
        proxy.addUser();
    }
}

4.动态代理-JDK模式案例

需求: 要求大家计算addUser()方法的运行时间? 要求采用动态代理的方式完成.

-1.编辑代理对象

 public static Object getTimePorxy(Object target){
        ClassLoader classLoader = target.getClass().getClassLoader();
        Class[] interfaces = target.getClass().getInterfaces();
        return Proxy.newProxyInstance(classLoader,interfaces,invocationHandlerTime(target));
    }

    //要求必须传递目标对象
    public static InvocationHandler invocationHandlerTime(Object target){
        return new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                //开始时间
                long startTime = System.currentTimeMillis();
                //让目标方法执行 结果
                Object result = method.invoke(target,args);
                long endTime = System.currentTimeMillis();
                System.out.println("耗时:"+(endTime - startTime));
                return result;
            }
        };
    }

-2.编辑测试方法

package com.jt;

import com.jt.config.SpringConfig;
import com.jt.proxy.JDKProxy;
import com.jt.service.UserService;
import org.junit.jupiter.api.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.Ann
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值