实验四 Spring和AOP编程

实验目的

利用Spring技术实现【实验二】中的校友信息网站。要求采用MVC框架,同时要求加入面向切面的编程。

构建一个用户记录的切面

实验内容

  • 对于所有的登录操作,记录各各次登录的时间、用户,存入UserLog表格中。
  • 对于所有的登出操作,记录各各次登录的时间、用户,存入UserLog表格中。
  • 对于用户新的输入操作,记录其表单值,存入InsertLog表格中。

注意:只记录成功登陆和成功登出的信息(比如用户账号密码错误没有成功登入,不做记录;用户还没有登入便调用登出的接口,也不做记录)。

实验过程

  • 搭建springboot+mybatis web开发框架,利用generator插件自动生成mapper和entity
  • 实现/login,/logout接口,用于:
    1.验证是否成功登陆或成功登出,将此信息保存至response参数的status里,用于aop获取;
    (利用回调,实现了代理类和被代理类的简单通信)
    2.将用户信息保存在session中,以便记录用户登录状态。
  • 实现loginAOP,logoutAOP,用于记录登录登出的信息。
  • 实现insertAOP,用于记录用户Insert的信息。
  • 实现登陆拦截器,用于拦截所有除登录的请求,如果用户未登录,则跳转到登录页面,同时将登录前访问的url保存至session中,以便登录成功后重新跳转到之前的页面。

搭建springboot+mybatis web开发框架

参考https://blog.csdn.net/weixin_42685022/article/details/82215893

实现/login,/logout接口

    @Autowired
    private AdminMapper adminMapper;

    @RequestMapping(value = "/login", method = RequestMethod.POST)
    public void login(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
        HttpSession session = httpServletRequest.getSession();
        //如果已登录则自动退出登录
        if (session.getAttribute("userDetail") != null) {
            httpServletResponse.sendRedirect("/logout");
        }
        //验证用户是否成功登录
        else {
            String username = httpServletRequest.getParameter("username");
            String password = httpServletRequest.getParameter("password");
            AdminDetail adminDetail = adminMapper.getAdminDetail(username, password);
            if (adminDetail != null) {
                System.out.println(adminDetail.getId() + ":" + "login");
                adminDetail.setLogin(true);
                session = httpServletRequest.getSession();
                session.setAttribute("userDetail", adminDetail);
                //登录成功,将response中的status设置为200,以便aop获得此信息
                httpServletResponse.setStatus(200);
            } else {
                //登录成功,将response中的status设置为403,以便aop获得此信息
                httpServletResponse.setStatus(403);
            }
        }
    }

    @RequestMapping(value = "/logout", method = RequestMethod.GET)
    public void logout(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
        HttpSession session = httpServletRequest.getSession();
        AdminDetail adminDetail = (AdminDetail) session.getAttribute("userDetail");
        //验证用户是否还未登录就登出
        if (adminDetail != null) {
            System.out.println(adminDetail.getId() + ":" + "logout");
            //登出成功,将response中的status设置为200,以便aop获得此信息
            httpServletResponse.setStatus(200);
        } else {
            //登录失败,将response中的status设置为403,以便aop获得此信息
            httpServletResponse.setStatus(403);
        }
    }

实现loginAOP,logoutAOP

 @Autowired
    private UserlogMapper userlogMapper;

    @Pointcut("execution(public * com.bao.schoolfellow.controller.Login.login(..))")
    public void LoginPoint() {
    }

    @Pointcut("execution(public * com.bao.schoolfellow.controller.Login.logout(..))")
    public void LogoutPoint() {
    }


    @Around("LoginPoint()")
    public Object doLoginAdvice(ProceedingJoinPoint proceedingJoinPoint) throws java.lang.Throwable {
        //先执行/login接口的方法
        proceedingJoinPoint.proceed();
        //通过反射机制获取/login方法response参数的status信息
        int status = (Integer) HttpServletResponse.class.getMethod("getStatus").invoke(proceedingJoinPoint.getArgs()[1]);

        //登录成功,记录信息并跳转到主页或登录前的页面
        if (status == 200) {
            //通过反射机制获取/login方法resquest参数的session信息
            HttpSession session = (HttpSession) HttpServletRequest.class.getMethod("getSession").invoke(proceedingJoinPoint.getArgs()[0]);
            //通过session获取用户信息,并记录登录信息至数据库
            Userlog userlog = new Userlog();
            userlog.setTime(new Date());
            userlog.setType("login");
            AdminDetail adminDetail = (AdminDetail) (session.getAttribute("userDetail"));
            userlog.setUserId(adminDetail.getId());
            userlogMapper.insertSelective(userlog);
            System.out.println("login success:" + "record");

            //获取用户登录前访问的url信息,如果有,则跳转;如果没有,跳转至主页
            String preURL=(String) session.getAttribute("preURL");
            if(preURL!=null){
                System.out.println("preURL"+preURL);
                HttpServletResponse.class.getMethod("sendRedirect", String.class).invoke(proceedingJoinPoint.getArgs()[1], session.getAttribute("preURL"));
            }
            else {
                HttpServletResponse.class.getMethod("sendRedirect", String.class).invoke(proceedingJoinPoint.getArgs()[1], "/index.html");
            }

        }
        //登录失败,跳转到错误页面
        else if (status == 403) {
            System.out.println("login fail:" + "not record!");
            HttpServletResponse.class.getMethod("sendRedirect", String.class).invoke(proceedingJoinPoint.getArgs()[1], "/login_error.html");
        }
        return null;
    }

    //logoutAOP原理同上
    @Around("LogoutPoint()")
    public Object doLogoutAdvice(ProceedingJoinPoint proceedingJoinPoint) throws java.lang.Throwable {
        proceedingJoinPoint.proceed();
        int status = (Integer) HttpServletResponse.class.getMethod("getStatus").invoke(proceedingJoinPoint.getArgs()[1]);
        if (status == 200) {
            HttpSession session = (HttpSession) HttpServletRequest.class.getMethod("getSession").invoke(proceedingJoinPoint.getArgs()[0]);
            Userlog userlog = new Userlog();
            userlog.setTime(new Date());
            userlog.setType("logout");
            AdminDetail adminDetail = (AdminDetail) (session.getAttribute("userDetail"));
            userlog.setUserId(adminDetail.getId());
            userlogMapper.insertSelective(userlog);
            session.removeAttribute("userDetail");
            System.out.println("logout success:" + "record");
        } else if (status == 403) {
            System.out.println("illegal operate:" + "not login but logout!");
        }
        HttpServletResponse.class.getMethod("sendRedirect", String.class).invoke(proceedingJoinPoint.getArgs()[1], "/login.html");
        return null;
    }

实现insertAOP

这里我想用Around通知同时实现两个切面:
1.对于所有的除GET方法的请求,操作成功后让returnAOP返回操作成功的状态码。
2.对于/insert接口,操作成功后记录用户本次操作的信息。

先贴下代码:
InsertController

    @Autowired
    private SchoolfellowMapper schoolfellowMapper;

    @RequestMapping(method = RequestMethod.POST)
    //注意:这里返回值不能写void,这样即使AOP返回了返回值,服务器也不会返回给客户端
    public Object add(HttpServletRequest httpServletRequest,HttpServletResponse httpServletResponse, @RequestBody Schoolfellow schoolfellow) {
        System.out.println("insert!!!");
        schoolfellowMapper.insertSelective(schoolfellow);
        return null;
    }

AOP

    @Autowired
    private InsertlogMapper insertlogMapper;

    @Pointcut("execution(public * com.bao.schoolfellow.controller.Operate.*(..))")
    public void ReturnStatusPoint() {
    }

    @Pointcut("execution(public * com.bao.schoolfellow.controller.Operate.add(..))")
    public void InsertPoint() {
    }

    @Around("ReturnStatusPoint()")
    public Object returnStatus(ProceedingJoinPoint proceedingJoinPoint) throws java.lang.Throwable{
        System.out.println("changeReturnStart!");
        //先调用原方法
        Object object=proceedingJoinPoint.proceed(proceedingJoinPoint.getArgs());
        Method invokeMethod=null;
        //通过反射获取被调用的接口的方法信息
        for(Method method:proceedingJoinPoint.getTarget().getClass().getMethods()){
            if(method.getName().equals(proceedingJoinPoint.getSignature().getName())){
                invokeMethod=method;
                break;
            }
        }
        //获取此方法的注解信息
        for(Annotation annotation:invokeMethod.getDeclaredAnnotations()){
            //如果它不是GET的方法,则返回操作成功的状态status
            if(annotation.annotationType().equals(RequestMapping.class)){
                if(((RequestMapping)annotation).method()[0]!=RequestMethod.GET){
                    Map<String,String> status=new HashMap<>();
                    status.put("status",String.valueOf(HttpServletResponse.class.getMethod("getStatus").invoke(proceedingJoinPoint.getArgs()[1])));
                    System.out.println("changeReturnOver!");
                    return status;
                }
            }
        }
        //如果不是,返回原方法的返回值
        return object;
    }

    @Around("InsertPoint()")
    public Object saveInsert(ProceedingJoinPoint proceedingJoinPoint) throws java.lang.Throwable {
        System.out.println("saveInsertStart!!");
        //先调用原方法
        Object object=proceedingJoinPoint.proceed(proceedingJoinPoint.getArgs());
        //记录用户的操作信息至数据库
        HttpSession session = (HttpSession) HttpServletRequest.class.getMethod("getSession").invoke(proceedingJoinPoint.getArgs()[0]);
        AdminDetail adminDetail=(AdminDetail) session.getAttribute("userDetail");
        Insertlog insertlog=new Insertlog();
        insertlog.setContent(proceedingJoinPoint.getArgs()[2].toString());
        insertlog.setUserId(adminDetail.getId());
        insertlog.setTime(new Date());
        insertlogMapper.insertSelective(insertlog);
        System.out.println("saveInsertOver!!");
        return object;
    }

思考:这里使用两个Around调用了两次proceedingJoinPoint.proceed,会不会原方法也调用了两次呢?两次调用都修改了原方法的返回值,最终取谁的返回值呢?

如果搞清了AOP的原理——动态代理,这个问题就解决了:
这就是代理模式的神奇之处,可以嵌套代理,但是被代理类只被调用一次。
假设原方法的类为class,切面1的代理类proxy1,切面2的代理类为proxy2。事实上,SpringAOP使proxy2代理了class,proxy1代理了proxy2。调用的入口在proxy2,返回值的决定权也在proxy2,即最外层的代理类。(代理类的代理顺序可以进行配置)
打个比方:
spring aop就是一个同心圆,要执行的方法为圆心,proxy2相应的方法为圆AOP2,proxy1相应的方法为圆AOP1


image.png

理解并读懂了代码,会发现成功执行一次insert操作会输出如下结果:


image.png

实现登陆拦截器

拦截器和AOP一样采用了动态代理的方式,可以设置指定的url被拦截或不被拦截,配置方便,很好用。

@Component
public class LoginInterceptor implements HandlerInterceptor {
    public String getURL(HttpServletRequest httpServletRequest){
        StringBuffer url=new StringBuffer(httpServletRequest.getRequestURL());
        Map map=httpServletRequest.getParameterMap();
        if(!map.isEmpty()){
            url.append("?");
            for(Object key:map.keySet()){
                url.append(key+"="+map.get(key)+"&");
            }
            url.substring(0,url.length()-1);
        }
        return url.toString();
    }

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        //获取request的URL(包括request参数),保存至session中
        HttpSession session=request.getSession();
        AdminDetail adminDetail=(AdminDetail) session.getAttribute("userDetail");
        session.setAttribute("preURL",getURL(request));

        //验证用户是否登录,如果没有则跳转到登录页面
        if(adminDetail==null){
            response.sendRedirect("/login.html");
            return false;
        }
        else {
            return true;
        }
    }
}

构建一个用户记录的切面

实验内容

  • 对于所有的Alumni表的查询操作,验证用户已经登录;如果用户没有登录,先导航到登录页面;
  • 对于所有的Alumni表的更新(更新和删除)操作,在Read权限的基础上验证用户具有Update的权限。如果没有,该操作取消,并导航到错误页面。
  • 对于所有的Alumni表的汇总和下载操作,验证用户具有Aggregate权限;如果没有,该操作取消,并导航到错误页面。

实验过程(在原项目的基础上)

  • 实现登陆拦截器,用于拦截所有除登录的请求,如果用户未登录,则跳转到登录页面,同时将登录前访问的url保存至session中,以便登录成功后重新跳转到之前的页面。
  • 实现PermissionMetadata组件,用于加载所有需要验证的URL的信息。
  • 实现权限验证拦截器,用于拦截除登录登出以及获取当前用户信息外的所有请求,先查看此url是否在PermissionMetadata中,如果不在,则放行;如果在,根据sesssion中保存的用户的信息查看用户是否具有此权限,如果有,则放行。

实现PermissionMetadata组件

@Component
public class PermissionMetadata {
    @Autowired
    private AdminMapper adminMapper;

    private List<String> allPermissions;

    public List<String> getAllPermissions() {
        if(allPermissions==null){
            allPermissions=adminMapper.selectAllPermission();
        }
        return allPermissions;
    }
}

实现权限验证拦截器

@Component
public class PermissionInerceptor implements HandlerInterceptor {
    @Autowired
    private AdminMapper adminMapper;

    @Autowired
    private PermissionMetadata permissionMetadata;

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        AdminDetail adminDetail=(AdminDetail) request.getSession().getAttribute("userDetail");

        //将当前访问的URL处理成method/URI的形式
        String requestURI=request.getRequestURI();
        String url=request.getMethod();
        int count=0;
        for(String subPath:requestURI.split("/")){
            if(count%2!=0){
                url+="/"+subPath;
            }
            count++;
        }
        System.out.println("url:"+url);

        //查看此权限是否在permissionMetada中,如果不在则放行
        boolean isExist=false;
        for(String permission:permissionMetadata.getAllPermissions()){
            if(url.equals(permission)){
                isExist=true;
            }
        }
        if(!isExist){
            System.out.println("Permission not exist so pass");
            return true;
        }

        //查看用户是否具有此权限,如果有则放行
        for(String permission:adminDetail.getPermissions()){
            if(permission.equals(url)){
                System.out.println("You have permission:"+url);
                return true;
            }
        }
        System.out.println("You have no permission:"+url);
        response.sendRedirect("/login_error.html");
        return false;

    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容