12、面向切面的Spring(补)(spring笔记)

之前书中的例子代码不全,这里我们通过几个例子来演示说明spring aop的功能。

一、基本的 AOP 使用

工程结构

1

首先定义切面 SecurityHandler.java

package win.iot4yj.spring.aspect;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;

/**
 * 
 * @author yj
 * 这个类就是一个切面
 */

@Aspect
@Component
public class SecurityHandler {
    
    //定义一个切点,只是相当于一个标识。即匹配所有add开头的方法,参数和返回值任意。而allAddMethod方法不能有返回值
    @Pointcut("execution(* add*(..))")
    private void allAddMethod(){}
    
    //这个方法就相当于一个通知,标识在哪个切入点织入
    @Before("allAddMethod()")
    private void checkSevurity(){
        System.out.println("-------安全性检查-------");
    }
}

说明:切点的标识方法可以有参数,但不能有返回值。

业务类 UserManager.java UserManagerImpl.java

package win.iot4yj.spring.manager;
public interface UserManager {
    
    public void addUser(String username, String password);
}
package win.iot4yj.spring.manager;
import org.springframework.stereotype.Component;

@Component("userManager")
public class UserManagerImpl implements UserManager {

    @Override
    public void addUser(String username, String password) {
        System.out.println("-------添加用户-------");
    }
}

配置类 Config.java

package win.iot4yj.spring.config;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import win.iot4yj.spring.aspect.SecurityHandler;
import win.iot4yj.spring.manager.UserManager;

@Configuration
@EnableAspectJAutoProxy
@ComponentScan(basePackageClasses={UserManager.class, SecurityHandler.class})
public class Config {
}

测试 IoCTest.java

package win.iot4yj.spring.test;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import win.iot4yj.spring.manager.UserManager;

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes=win.iot4yj.spring.config.Config.class)
public class IoCTest {
    
    @Autowired
    private UserManager userManager;

    @Test
    public void test01() {
        userManager.addUser("张三", "123");
    }
}

说明:这里我们使用的是前置通知。会在添加用户之前进行安全性检查。

二、环绕通知

下面看环绕通知,没有给出的代码和之前一样。

SecurityHandler.java

package win.iot4yj.spring.aspect;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;

/**
 * 
 * @author yj
 * 这个类就是一个切面
 */

@Aspect
@Component
public class SecurityHandler {
    
    //定义一个切点,只是相当于一个标识。即匹配所有add开头的方法,参数和返回值任意。而allAddMethod方法不能有返回值
    @Pointcut("execution(* add*(..))")
    private void allAddMethod(){}
    
    
    //这个方法就相当于一个通知,标识在哪个切入点织入
    @Around("allAddMethod()")
    private void checkSevurity(ProceedingJoinPoint jp){
        System.out.println("-------安全性检查之前-------");
        try {
            jp.proceed();//调用被通知的方法
        } catch (Throwable e) {
            e.printStackTrace();
        }
        System.out.println("-------安全性检查之后-------");
    }
}

三、传递参数

可以从之前的测试代码中看到,添加用户肯定是传递了用户名和密码的,这里我们将其传递到通知中去。

SecurityHandler.java

package win.iot4yj.spring.aspect;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
/**
 * 
 * @author yj
 * 这个类就是一个切面
 */

@Aspect
@Component
public class SecurityHandler {
    
    //定义一个切点,只是相当于一个标识。即匹配所有add开头的方法,参数和返回值任意。而allAddMethod方法不能有返回值
    @Pointcut("execution(* add*(..)) && args(username, password)")
    private void allAddMethod(String username, String password){}
    
    //这个方法就相当于一个通知,标识在哪个切入点织入
    @Around("allAddMethod(username, password)")
    private void checkSevurity(ProceedingJoinPoint jp, String username, String password){
        System.out.println("-------安全性检查之前-------");
        try {
            jp.proceed();
        } catch (Throwable e) {
            e.printStackTrace();
        }
        System.out.println("用户名: " + username + ", 密码: " + password);
        System.out.println("-------安全性检查之后-------");
    }
}

说明:这里要注意的是一定要使用args(username, password)进行标识,这标明相关的参数也会传递到通知中去。

四、引入新功能

这里首先定义要引入的新功能的类和接口

package win.iot4yj.spring.manager;
public interface NewFun {
    public void newFunction();
}
package win.iot4yj.spring.manager;
import org.springframework.stereotype.Component;

@Component
public class NewFunImpl implements NewFun {

    @Override
    public void newFunction() {
        System.out.println("-----引入的新功能-----");
    }
}

切面 SecurityHandler.java

package win.iot4yj.spring.aspect;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.DeclareParents;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
import win.iot4yj.spring.manager.NewFun;
import win.iot4yj.spring.manager.NewFunImpl;
/**
 * @author yj
 * 这个类就是一个切面
 */
@Aspect
@Component
public class SecurityHandler {
    
    @DeclareParents(value = "win.iot4yj.spring.manager.UserManager+",
                    defaultImpl = NewFunImpl.class)
    public static NewFun n;
    
    //定义一个切点,只是相当于一个标识。即匹配所有add开头的方法,参数和返回值任意。而allAddMethod方法不能有返回值
    @Pointcut("execution(* add*(..)) && args(username, password)")
    private void allAddMethod(String username, String password){}
    
    //这个方法就相当于一个通知,标识在哪个切入点织入
    @Around("allAddMethod(username, password)")
    private void checkSevurity(ProceedingJoinPoint jp, String username, String password){
        System.out.println("-------安全性检查之前-------");
        try {
            jp.proceed();
        } catch (Throwable e) {
            e.printStackTrace();
        }
        System.out.println("用户名: " + username + ", 密码: " + password);
        System.out.println("-------安全性检查之后-------");
    }
}

说明:相对于之前的代码,这里添加了引入新功能的配置。具体说明参考前一小节文章。

测试 IoCTest.java

package win.iot4yj.spring.test;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import win.iot4yj.spring.manager.NewFun;
import win.iot4yj.spring.manager.UserManager;

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes=win.iot4yj.spring.config.Config.class)
public class IoCTest {
    
    @Autowired
    private UserManager userManager;
    
    @Autowired
    ApplicationContext applicationContext;

    @Test
    public void test01() {
        userManager.addUser("张三", "123");
        NewFun newFun = (NewFun) applicationContext.getBean("userManager");
        newFun.newFunction();
    }
}

说明:这里首先自动注入了应用上下文类ApplicationContextspring中很多类,如上下文类、环境类Environment类都可以这样获得。而其实通过ApplicationContext就可以获得应用中已存在的bean,所以这里的UserManager bean自动注入可以省略。此时就可以使用新引入的功能了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,099评论 19 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 173,972评论 25 709
  • 我的中专同学群里又多了一位老同学,她叫王礼花,是我的张舍老乡,是我师傅把她拉进群里的,我师傅接着发了欢迎...
    夏俊智爸爸阅读 129评论 0 1
  • 月初确定了办公室,很喜欢它的“中国味”:大红色的主色调,中式的家具,方正的四合院,玻璃阳光房,露天咖啡厅,健身房,...
    景景相依阅读 185评论 0 0
  • 那时候她正不堪,父亲去世,事业受挫,婚姻破裂,人生的不幸赶着趟儿地朝她砸过来,她疲于应付,心力交瘁,整个世界在她...
    聪明可爱小郎君阅读 319评论 0 0