博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
springboot+vue的前后端分离与合并方案
阅读量:7045 次
发布时间:2019-06-28

本文共 5540 字,大约阅读时间需要 18 分钟。

hot3.png

springboot和vue结合的方案网络上的主要有以下两种:

1. 【不推荐】在html中直接使用script标签引入vue和一些常用的组件,这种方式和以前传统的开发是一样的,只是可以很爽的使用vue的双向数据绑定,这种方式只适合于普通的全栈开发。

2.【推荐】使用vue官方的脚手架创建单独的前端工程项目,做到和后端完全独立开发和部署,后端单独部署一个纯restful的服务,而前端直接采用nginx来部署,这种称为完全的前后端分离架构开发模式,但是在分离中有很多api权限的问题需要解决,包括部署后的vue router路由需要在nginx中配置rewrite规则。这种前后端完全分离的架构也是目前互联网公司所采用的,后端服务器不再需要处理静态资源,也能减少后端服务器一些压力。

一、为什么做前后端分离开发合并

    在传统行业中很多是以项目思想来主导的,而不是产品,一个项目会卖给很多的客户,并且部署到客户本地的机房里。在一些传统行业里面,部署实施人员的技术无法和互联网公司的运维团队相比,由于各种不定的环境也无法做到自动构建,容器化部署等。因此在这种情况下尽量减少部署时的服务软件需求,打出的包数量也尽量少。针对这种情况这里采用的在开发中做到前后端独立开发,整个开发方式和上面提到的第二种方式是相同的,但是在后端springboot打包发布时将前端的构建输出一起打入,最后只需部署springboot的项目即可,无需再安装nginx服务器。

二、springboot和vue整合的关键操作

    实际上本文中这种前后端分离的开发,前端开发好后将build构建好的dist下static中的文件拷贝到springboot的resource的static下,index.html则直接拷贝到springboot的resource的static下。下面是示例图:

vue前端项目

234113_QeL2_1760791.png

springboot项目:

234232_LoW8_1760791.png

重点:上面这是最简单的合并方式,但是如果作为工程级的项目开发,并不推荐使用手工合并,也不推荐将前端代码构建后提交到springboot的resouce下,好的方式应该是保持前后端完全独立开发代码,项目代码互不影响,借助jenkins这样的构建工具在构建springboot时触发前端构建并编写自动化脚本将前端webpack构建好的资源拷贝到springboot下再进行jar的打包,最后就得到了一个完全包含前后端的springboot项目了。

三、整合的核心问题处理

    通过上面的整合后会出现两个比较大的问题:

    1. 无法正常访问静态资源 。

    2. vue router路由的路径无法正常解析 。

   解决第一个问题,我们必须重新指定springboot的静态资源处理前缀,代码:

@Configurationpublic class SpringWebMvcConfig extends WebMvcConfigurerAdapter {    @Override    public void addResourceHandlers(ResourceHandlerRegistry registry) {        registry.addResourceHandler("/static/**").addResourceLocations("classpath:/static/");        super.addResourceHandlers(registry);    }}

  解决第二个问题的方式是对vue的路由的路径做rewrite,交给router来处理,而不是springboot自己处理,rewrite时可以考虑路由的路径统一增加后缀,然后在springboot中编写过滤拦截特定后缀来做请求转发交给vue的路由处理。如:

const router = new VueRouter({  mode: 'history',  base: __dirname,  routes: [    {      path: '/ui/first.vhtml',      component: First    },    {      path: '/ui/second.vhtml',      component: secondcomponent    }  ]})

后端拦截到带有vhtml的都交给router来处理,这种方式在后端写过滤器拦截后打包是完全可行的,但是前端开发的直接访问带后缀的路径会有问题。

另外一种方式是给前端的路由path统一加个前缀比如/ui,当然就可以把之前的后缀删除了,这时后端写过滤器匹配该前缀,也不会影响前端单独开发是的路由解析问题。过滤器参考如下:

/** * be used to rewrite vue router * * @author yu on 2017-11-22 19:47:23. */public class RewriteFilter implements Filter {    /**     * 需要rewrite到的目的地址     */    public static final String REWRITE_TO = "rewriteUrl";    /**     * 拦截的url,url通配符之前用英文分号隔开     */    public static final String REWRITE_PATTERNS = "urlPatterns";    private Set
urlPatterns = null;//配置url通配符 private String rewriteTo = null; @Override public void init(FilterConfig cfg) throws ServletException { //初始化拦截配置 rewriteTo = cfg.getInitParameter(REWRITE_TO); String exceptUrlString = cfg.getInitParameter(REWRITE_PATTERNS); if (StringUtil.isNotEmpty(exceptUrlString)) { urlPatterns = Collections.unmodifiableSet( new HashSet<>(Arrays.asList(exceptUrlString.split(";", 0)))); } else { urlPatterns = Collections.emptySet(); } } @Override public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws IOException, ServletException { HttpServletRequest request = (HttpServletRequest) req; String servletPath = request.getServletPath(); String context = request.getContextPath(); //匹配的路径重写 if (isMatches(urlPatterns, servletPath)) { req.getRequestDispatcher(context+"/"+rewriteTo).forward(req, resp); }else{ chain.doFilter(req, resp); } } @Override public void destroy() { } /** * 匹配返回true,不匹配返回false * @param patterns 正则表达式或通配符 * @param url 请求的url * @return */ private boolean isMatches(Set
patterns, String url) { if(null == patterns){ return false; } for (String str : patterns) { if (str.endsWith("/*")) { String name = str.substring(0, str.length() - 2); if (url.contains(name)) { return true; } } else { Pattern pattern = Pattern.compile(str); if (pattern.matcher(url).matches()) { return true; } } } return false; }}

过滤器的注册:

@SpringBootApplicationpublic class SpringBootMainApplication {    public static void main(String[] args) {        SpringApplication.run(SpringBootMainApplication.class, args);    }    @Bean    public EmbeddedServletContainerCustomizer containerCustomizer() {        return (container -> {                ErrorPage error401Page = new ErrorPage(HttpStatus.UNAUTHORIZED, "/errors/401.html");                ErrorPage error404Page = new ErrorPage(HttpStatus.NOT_FOUND, "/errors/404.html");                ErrorPage error500Page = new ErrorPage(HttpStatus.INTERNAL_SERVER_ERROR, "/errors/500.html");                container.addErrorPages(error401Page, error404Page, error500Page);        });    }    @Bean    public FilterRegistrationBean testFilterRegistration() {        FilterRegistrationBean registration = new FilterRegistrationBean();        registration.setFilter(new RewriteFilter());//注册rewrite过滤器        registration.addUrlPatterns("/*");        registration.addInitParameter(RewriteFilter.REWRITE_TO,"/index.html");        registration.addInitParameter(RewriteFilter.REWRITE_PATTERNS, "/ui/*");        registration.setName("rewriteFilter");        registration.setOrder(1);        return registration;    }}

这时springboot就可以将前端的路由资源交给路由来处理了。至此整个完整前后端分离开发合并方案就完成了。这种方式在后期有条件情况下也可以很容易做到前后端的完全分离开发部署。

ps:本方案只是在特定场景下的选择,如果一切条件允许,强力推荐做完全的前后端分离

版权申明:转载请注明出处,否则后果自负

转载于:https://my.oschina.net/u/1760791/blog/1577662

你可能感兴趣的文章
echarts x y轴设置
查看>>
abap优化工具事务代码: ST05
查看>>
浅谈进程地址空间与虚拟存储空间
查看>>
CodeForces 506D Mr. Kitayuta's Colorful Graph
查看>>
【实验吧】Reverse400
查看>>
LINQ Enumerable
查看>>
项目二业务逻辑整理 非常有用
查看>>
略少面试题 项目中用到的技术详解 有用
查看>>
实验三+116+陈洁
查看>>
Centos 7.4 源码 Nginx 安装
查看>>
iOS中retain和copy的区别
查看>>
mysql 安装记录
查看>>
[转载] 七龙珠第一部——第020话 修行的威力
查看>>
Python Application
查看>>
mybatis简介
查看>>
Here With You
查看>>
前端基础之CSS_1
查看>>
Uva 11600 期望DP
查看>>
HDU 5527 贪心
查看>>
Junit单元测试对线程测试没反应
查看>>