点击上方“方志朋”,选择“设为星标”

回复”666“获取新整理的面试资料

来源:http://sina.lt/gauR

本文总结自实习中对项目的重构。原先项目采用Springboot+freemarker模版,开发过程中觉得前端逻辑写的实在恶心,后端Controller层还必须返回Freemarker模版的ModelAndView,逐渐有了前后端分离的想法,由于之前,没有接触过,主要参考的还是网上的一些博客教程等,初步完成了前后端分离,在此记录以备查阅。

一、前后端分离思想

前端从后端剥离,形成一个前端工程,前端只利用Json来和后端进行交互,后端不返回页面,只返回Json数据。前后端之间完全通过public API约定。

二、后端 Springboot

Springboot就不再赘述了,Controller层返回Json数据。

    @RequestMapping(value = "/add", method = RequestMethod.POST)@ResponseBodypublic JSONResult addClient(@RequestBody String param) {JSONObject jsonObject = JSON.parseObject(param);String task = jsonObject.getString("task");List<Object> list = jsonObject.getJSONArray("attributes");List<String> attrList = new LinkedList(list);Client client = JSON.parseObject(jsonObject.getJSONObject("client").toJSONString(),new TypeReference<Client>(){});clientService.addClient(client, task, attrList);return JSONResult.ok();}

Post请求使用@RequestBody参数接收。

三、前端 Vue + ElementUI + Vue router + Vuex + axios + webpack

主要参考:

https://cn.vuejs.org/v2/guide/

https://github.com/PanJiaChen/vue-admin-template/blob/master/README-zh.md

https://github.com/PanJiaChen/vue-element-admin

这里主要说一下开发工程中遇到的问题:

1.跨域

由于开发中前端工程使用webpack启了一个服务,所以前后端并不在一个端口下,必然涉及到跨域:

XMLHttpRequest会遵守同源策略(same-origin policy). 也即脚本只能访问相同协议/相同主机名/相同端口的资源, 如果要突破这个限制, 那就是所谓的跨域, 此时需要遵守CORS(Cross-Origin Resource Sharing)机制。

解决跨域分两种:

1、server端是自己开发的,这样可以在在后端增加一个拦截器

@Component
public class CommonIntercepter implements HandlerInterceptor {private final Logger logger = LoggerFactory.getLogger(this.getClass());@Overridepublic boolean preHandle(HttpServletRequest request,HttpServletResponse response, Object handler) throws Exception {//允许跨域,不能放在postHandle内response.setHeader("Access-Control-Allow-Origin", "*");if (request.getMethod().equals("OPTIONS")) {response.addHeader("Access-Control-Allow-Methods", "GET,HEAD,POST,PUT,DELETE,TRACE,OPTIONS,PATCH");response.addHeader("Access-Control-Allow-Headers", "Content-Type, Accept, Authorization");}return true;}
}

response.setHeader("Access-Control-Allow-Origin", "*");

主要就是在Response Header中增加 "Access-Control-Allow-Origin: *"

      if (request.getMethod().equals("OPTIONS")) {response.addHeader("Access-Control-Allow-Methods", "GET,HEAD,POST,PUT,DELETE,TRACE,OPTIONS,PATCH");response.addHeader("Access-Control-Allow-Headers", "Content-Type, Accept, Authorization");}

由于我们在前后端分离中集成了shiro,因此需要在headers中自定义一个'Authorization'字段,此时普通的GET、POST等请求会变成preflighted request,即在GET、POST请求之前会预先发一个OPTIONS请求,这个后面再说。推荐一篇博客介绍 preflighted request。

https://blog.csdn.net/cc1314_/article/details/78272329

2、server端不是自己开发的,可以在前端加proxyTable。

不过这个只能在开发的时候用,后续部署,可以把前端项目作为静态资源放到后端,这样就不存在跨域(由于项目需要,我现在是这么做的,根据网上博客介绍,可以使用nginx,具体怎么做可以在网上搜一下)。

遇到了网上很多人说的,proxyTable无论如何修改,都没效果的现象。

1、(非常重要)确保proxyTable配置的地址能访问,因为如果不能访问,在浏览器F12调试的时候看到的依然会是提示404。

并且注意,在F12看到的js提示错误的域名,是js写的那个域名,并不是代理后的域名。(l楼主就遇到这个问题,后端地址缺少了查询参数,代理设置为后端地址,然而F12看到的错误依然还是本地的域名,并不是代理后的域名)

2、就是要手动再执行一次npm run dev

四、前后端分离项目中集成shiro

可以参考:

blog.csdn.net/u013615903/article/details/78781166

这里说一下实际开发集成过程中遇到的问题:

1、OPTIONS请求不带'Authorization'请求头字段:

前后端分离项目中,由于跨域,会导致复杂请求,即会发送preflighted request,这样会导致在GET/POST等请求之前会先发一个OPTIONS请求,但OPTIONS请求并不带shiro的'Authorization'字段(shiro的Session),即OPTIONS请求不能通过shiro验证,会返回未认证的信息。

解决方法:给shiro增加一个过滤器,过滤OPTIONS请求

public class CORSAuthenticationFilter extends FormAuthenticationFilter {private static final Logger logger = LoggerFactory.getLogger(CORSAuthenticationFilter.class);public CORSAuthenticationFilter() {super();}@Overridepublic boolean isAccessAllowed(ServletRequest request, ServletResponse response, Object mappedValue) {//Always return true if the request's method is OPTIONSif (request instanceof HttpServletRequest) {if (((HttpServletRequest) request).getMethod().toUpperCase().equals("OPTIONS")) {return true;}}
return super.isAccessAllowed(request, response, mappedValue);}@Overrideprotected boolean onAccessDenied(ServletRequest request, ServletResponse response) throws Exception {HttpServletResponse res = (HttpServletResponse)response;res.setHeader("Access-Control-Allow-Origin", "*");res.setStatus(HttpServletResponse.SC_OK);res.setCharacterEncoding("UTF-8");PrintWriter writer = res.getWriter();Map<String, Object> map= new HashMap<>();map.put("code", 702);map.put("msg", "未登录");writer.write(JSON.toJSONString(map));writer.close();return false;}
}

贴一下我的config文件:

@Configuration
public class ShiroConfig {@Beanpublic Realm realm() {return new DDRealm();}@Beanpublic CacheManager cacheManager() {return new MemoryConstrainedCacheManager();}/*** cookie对象;* rememberMeCookie()方法是设置Cookie的生成模版,比如cookie的name,cookie的有效时间等等。* @return*/@Beanpublic SimpleCookie rememberMeCookie(){//System.out.println("ShiroConfiguration.rememberMeCookie()");//这个参数是cookie的名称,对应前端的checkbox的name = rememberMeSimpleCookie simpleCookie = new SimpleCookie("rememberMe");//<!-- 记住我cookie生效时间30天 ,单位秒;-->simpleCookie.setMaxAge(259200);return simpleCookie;}/*** cookie管理对象;* rememberMeManager()方法是生成rememberMe管理器,而且要将这个rememberMe管理器设置到securityManager中* @return*/@Beanpublic CookieRememberMeManager rememberMeManager(){//System.out.println("ShiroConfiguration.rememberMeManager()");CookieRememberMeManager cookieRememberMeManager = new CookieRememberMeManager();cookieRememberMeManager.setCookie(rememberMeCookie());//rememberMe cookie加密的密钥 建议每个项目都不一样 默认AES算法 密钥长度(128 256 512 位)cookieRememberMeManager.setCipherKey(Base64.decode("2AvVhdsgUs0FSA3SDFAdag=="));return cookieRememberMeManager;}@Beanpublic SecurityManager securityManager() {DefaultWebSecurityManager sm = new DefaultWebSecurityManager();sm.setRealm(realm());sm.setCacheManager(cacheManager());//注入记住我管理器sm.setRememberMeManager(rememberMeManager());//注入自定义sessionManagersm.setSessionManager(sessionManager());return sm;}//自定义sessionManager@Beanpublic SessionManager sessionManager() {return new CustomSessionManager();}public CORSAuthenticationFilter corsAuthenticationFilter(){return new CORSAuthenticationFilter();}@Bean(name = "shiroFilter")public ShiroFilterFactoryBean getShiroFilterFactoryBean(SecurityManager securityManager) {ShiroFilterFactoryBean shiroFilter = new ShiroFilterFactoryBean();shiroFilter.setSecurityManager(securityManager);//SecurityUtils.setSecurityManager(securityManager);Map<String, String> filterChainDefinitionMap = new LinkedHashMap<>();//配置不会被拦截的链接,顺序判断filterChainDefinitionMap.put("/", "anon");filterChainDefinitionMap.put("/static/js/**", "anon");filterChainDefinitionMap.put("/static/css/**", "anon");filterChainDefinitionMap.put("/static/fonts/**", "anon");filterChainDefinitionMap.put("/login/**", "anon");filterChainDefinitionMap.put("/corp/call_back/receive", "anon");//authc:所有url必须通过认证才能访问,anon:所有url都可以匿名访问filterChainDefinitionMap.put("/**", "corsAuthenticationFilter");shiroFilter.setFilterChainDefinitionMap(filterChainDefinitionMap);//自定义过滤器Map<String, Filter> filterMap = new LinkedHashMap<>();filterMap.put("corsAuthenticationFilter", corsAuthenticationFilter());shiroFilter.setFilters(filterMap);return shiroFilter;}/*** Shiro生命周期处理器 * @return*/@Beanpublic LifecycleBeanPostProcessor lifecycleBeanPostProcessor() {return new LifecycleBeanPostProcessor();}/*** 开启Shiro的注解(如@RequiresRoles,@RequiresPermissions),需借助SpringAOP扫描使用Shiro注解的类,并在必要时进行安全逻辑验证 * 配置以下两个bean(DefaultAdvisorAutoProxyCreator(可选)和AuthorizationAttributeSourceAdvisor)即可实现此功能 * @return*/@Bean@DependsOn({"lifecycleBeanPostProcessor"})public DefaultAdvisorAutoProxyCreator advisorAutoProxyCreator() {DefaultAdvisorAutoProxyCreator advisorAutoProxyCreator = new DefaultAdvisorAutoProxyCreator();advisorAutoProxyCreator.setProxyTargetClass(true);return advisorAutoProxyCreator;}@Beanpublic AuthorizationAttributeSourceAdvisor authorizationAttributeSourceAdvisor(SecurityManager securityManager) {AuthorizationAttributeSourceAdvisor authorizationAttributeSourceAdvisor = new AuthorizationAttributeSourceAdvisor();authorizationAttributeSourceAdvisor.setSecurityManager(securityManager);return authorizationAttributeSourceAdvisor;}
}

2、设置session失效时间

shiro session默认失效时间是30min,我们在自定义的sessionManager的构造函数中设置失效时间为其他值

public class CustomSessionManager extends DefaultWebSessionManager {private static final Logger logger = LoggerFactory.getLogger(CustomSessionManager.class);private static final String AUTHORIZATION = "Authorization";private static final String REFERENCED_SESSION_ID_SOURCE = "Stateless request";public CustomSessionManager() {super();setGlobalSessionTimeout(DEFAULT_GLOBAL_SESSION_TIMEOUT * 48);}@Overrideprotected Serializable getSessionId(ServletRequest request, ServletResponse response) {String sessionId = WebUtils.toHttp(request).getHeader(AUTHORIZATION);//如果请求头中有 Authorization 则其值为sessionIdif (!StringUtils.isEmpty(sessionId)) {request.setAttribute(ShiroHttpServletRequest.REFERENCED_SESSION_ID_SOURCE, REFERENCED_SESSION_ID_SOURCE);request.setAttribute(ShiroHttpServletRequest.REFERENCED_SESSION_ID, sessionId);request.setAttribute(ShiroHttpServletRequest.REFERENCED_SESSION_ID_IS_VALID, Boolean.TRUE);return sessionId;} else {//否则按默认规则从cookie取sessionIdreturn super.getSessionId(request, response);}}
}

五、部署项目

前端项目部署主要分两种方法:

1.将前端项目打包(npm run build)成静态资源文件,放入后端,一起打包。后端写一个Controller返回前端界面(我使用Vue开发的是单页面应用),但是这样其实又将前后端耦合在一起了,不过起码做到前后端分离开发,方便开发的目的已经达成,也初步达成了要求,由于项目的需要,我是这样做的,并且免去了跨域问题。

@RequestMapping(value = {"/", "/index"}, method = RequestMethod.GET)public String index() {return "/index";}

2.将前端工程另启一个服务(tomcat,nginx,nodejs),这样有跨域的问题。

说一下我遇到的问题:

1、nginx反向代理,导致当访问无权限的页面时,shiro 302到unauth的controller,访问的地址是https,重定向地址是http,导致了无法访问。

不使用shiro的 shiroFilter.setLoginUrl("/unauth");

当页面无权限访问时,我们在过滤器里直接返回错误信息,不利用shiro自带的跳转。看过滤器中的onAccessDenied函数

public class CORSAuthenticationFilter extends FormAuthenticationFilter {private static final Logger logger = LoggerFactory.getLogger(CORSAuthenticationFilter.class);public CORSAuthenticationFilter() {super();}@Overridepublic boolean isAccessAllowed(ServletRequest request, ServletResponse response, Object mappedValue) {//Always return true if the request's method is OPTIONSif (request instanceof HttpServletRequest) {if (((HttpServletRequest) request).getMethod().toUpperCase().equals("OPTIONS")) {return true;}}return super.isAccessAllowed(request, response, mappedValue);}@Overrideprotected boolean onAccessDenied(ServletRequest request, ServletResponse response) throws Exception {HttpServletResponse res = (HttpServletResponse)response;res.setHeader("Access-Control-Allow-Origin", "*");res.setStatus(HttpServletResponse.SC_OK);res.setCharacterEncoding("UTF-8");PrintWriter writer = res.getWriter();Map<String, Object> map= new HashMap<>();map.put("code", 702);map.put("msg", "未登录");writer.write(JSON.toJSONString(map));writer.close();return false;}
}

先记录这么多,有不对的地方,欢迎指出!

热门内容:   

  

  • 学 Redis ,至少要看看这篇!7000 字小结

  • 系统运行缓慢,CPU 100%,以及Full GC次数过多问题的排查思路

  • 恕我直言,HttpClient 你不一定会用

  • 除了不要 SELECT * ,数据库还有哪些技巧

  • Redis为什么这么快?一文深入了解Redis内存模型!

  • 面试环节:在浏览器输入 URL 回车之后发生了什么?(超详细版)

  • 让 Spring Boot 启动更快一点

  • 一次非常有意思的 SQL 优化经历:从 30248.271s 到 0.001s

  • 进程与线程的一个简单解释

最近面试BAT,整理一份面试资料《Java面试BAT通关手册》,覆盖了Java核心技术、JVM、Java并发、SSM、微服务、数据库、数据结构等等。

获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。

明天见(。・ω・。)ノ♡

Spring Boot + Vue + Shiro 实现前后端分离、权限控制相关推荐

  1. Spring Boot + Vue + Shiro 实现前后端分离、权限控制 (附源码)

    点击上方[全栈开发者社区]→右上角[...]→[设为星标⭐] 原项目采用Springboot+freemarker模版,开发过程中觉得前端逻辑写的实在恶心,后端Controller层还必须返回Free ...

  2. Spring Boot + Vue.js 实现前后端分离(附源码)

    点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试文章 作者: 梁小生0101 链接:juejin.im/post/5c6 ...

  3. 基于Spring boot + Mybatis +Netty 实现前后端分离的聊天App,部署到阿里云线上服务器...

    前后端分离Spring boot 项目部署 了解前后端分离项目 配置云服务器 java maven tomcat nginx mysql 部署后端项目 部署前端项目 部署Java环境 1.下载JDK软 ...

  4. Java Spring boot element ui activiti前后端分离,流程审批,权限管理框架

    基于react ant design pro typescript 技术框架已经重磅推出 预览地址 系统介绍 是什么? 使用springboot,activiti,mybatis,vue elemen ...

  5. springboot接收json参数_Springboot + Vue + shiro 实现前后端分离、权限控制

    小Hub领读: 嘿嘿,之前我也发了一篇类似的项目,SpringBoot+Vue的项目,还有视频讲解,如果这篇文章看完不懂,不妨去看看我的视频讲解哈,超级详细! 太赞了,SpringBoot+Vue前后 ...

  6. VUE+Spring Boot整合MyBatis实现前后端分离项目壁纸网站

    目录 前言 一.项目运行 二.环境需要 三.技术栈 四.项目说明 五.后端代码 前言 每次换桌面,壁纸总是不好找,搜索图片得不到好的索引与反馈,很难找到自己喜欢的壁纸,而壁纸网站可以免去我们去寻找壁纸 ...

  7. Docker中Spring boot+VueJS+MongoDB的前后端分离哲学摔跤

    一图胜千言 目标 想将VueJs,Spring boot,MongoDB全部都放到Docker中运行,并且做到VueJs和Spring boot在不同都Docker容器中. Docker带来的变化 开 ...

  8. 企业电子招标采购系统源码Spring Cloud + Spring Boot + MybatisPlus + Redis + Layui + 前后端分离 + 二次开发

    项目说明 随着公司的快速发展,企业人员和经营规模不断壮大,公司对内部招采管理的提升提出了更高的要求.在企业里建立一个公平.公开.公正的采购环境,最大限度控制采购成本至关重要.符合国家电子招投标法律法规 ...

  9. 视频教程-springboot+Vue整合前后端分离权限后台管理系统-Java

    springboot+Vue整合前后端分离权限后台管理系统 拥有八年的Java项目开发经验,擅长Java.vue.SpringBoot.springCloud.spring.springmvc.myb ...

最新文章

  1. python比java简单好学-21、PHP和python/JAVA比,哪个更好学?
  2. POJ 3177 Redundant Paths (边双连通+缩点)
  3. 实录分享 | 计算未来轻沙龙:深度学习工具专场(PPT下载)
  4. Invalid character found in the request target. The valid characters are defined in RFC 7230 and RFC
  5. javascript提取标签之间的信息
  6. 4月27日微软云训练营活动-现场图集
  7. 致家长:疫情期间教育好自己的孩子,就是你最重要的事业!
  8. 如何看云服务器性能,从存储速度看云服务器性能测试
  9. RMI(Remote Method Invocation,远程方法调用)
  10. LinuxQuestions.org庆祝16岁生日
  11. Linux | 如何通过Xshell上传文件到Linux
  12. 计算机基本操作怎么保存,win10重置此电脑保留我的文件怎么操作
  13. Cannot create PoolableConnectionFactory (Access denied for user 'root'@'localhos
  14. 华为第十届 关灯计划
  15. 21年年后离职找工作的随笔杂谈
  16. APICloud开发app学习(一)
  17. Boboniu Plays Chess
  18. iOS开发者账户密码修改流程
  19. Macadam自然语言处理(NLP)工具包(TextClassification, SequenceLabeling, RelationExtraction)
  20. 学习《JavaScript高级程序设计》----day06

热门文章

  1. 前端基础之JQuery
  2. 微酷WeiKuCMS现赠送高速开发系统软件。公司、程序猿的福音呀!
  3. basePath = request.getScheme()+://+request.getServerName()+:+r
  4. 一 梳理 从 HDFS 到 MR。
  5. Java5中的线程池实例讲解
  6. ASP.NET Cookie
  7. 参加海峡两岸城市地理信息系统论坛2010 年会(一张图、规划信息化和空间句法的碎碎念)...
  8. KDE社区:首个KDialogue正式开放
  9. Numpy入门教程:09. 输入和输出
  10. 【怎样写代码】工厂三兄弟之工厂方法模式(一):问题案例