讨论Web前后端分离的意义

自然是有很大意义的。下面我可能说的比较多……方便题主能够更全面的了解为什么说是有有意义的。另外,本文是以Java的角度谈前后端分离。放心,大家一定会有种是我了,没错,的感觉。

 

一、先来明晰下概念

  前后端分离是通过Ngnix+Tomcat的方式(也可以中间加一个Node.js)有效的进行解耦,并且前后端分离会为以后的大型分布式架构、弹性计算架构、微服务架构、多端化服务(多种客户端,例如:浏览器,车载终端,安卓,iOS等等)打下坚实的基础。这个步骤是系统架构从猿进化成人的必经之路。

  它的核心思想是前端HTML页面通过Ajax调用后端的Restful API接口并使用json数据进行交互。这点题主也有提到。

 

二、其次,让我们了解下,没有前后端分离的时代(各种耦合)

  过去,Java Web项目大多数都是Java程序员又当爹又当妈,又搞前端,又搞后端。

  感觉就是,怀疑猿生……

  那时的JavaWeb项目都是使用了若干后台框架,Spring MVC/Struts + Spring + Spring JDBC/Hibernate/Mybatis 等等。

  大多数项目在Java后端都是分了三层,控制层,业务层,持久层。控制层负责接收参数,调用相关业务层,封装数据,以及路由&渲染到JSP页面。然后JSP页面上使用各种标签或者手写Java表达式将后台的数据展现出来,玩的是MVC那套思路。

  我们先看这种情况:需求定完了,代码写完了,测试测完了,然后呢?要发布了吧?你需要用maven或者eclipse等工具把你的代码打成一个war包,然后把这个war包发布到你的生产环境下的web容器里,对吧?

  发布完了之后,你要启动你的Web容器,开始提供服务,这时候你通过配置域名,DNS等等相关,你的网站就可以访问了(假设你是个网站)。那我们来看,你的前后端代码是不是全都在那个war包里?包括你的js,css,图片,各种第三方的库,对吧?

  好,下面在浏览器中输入你的网站域名(),之后发生了什么?

  浏览器在通过域名通过DNS服务器找到服务器外网ip,将http请求发送到服务器,在tcp3次握手之后(http下面是tcp/ip),通过tcp协议开始传输数据,服务器得到请求后,开始提供服务,接收参数,之后返回应答给浏览器,浏览器再通过content-type来解析返回的内容,呈现给用户。

  那么我们来看,我们先假设你的首页中有100张图片,此时,用户的看似一次http请求,其实并不是一次,用户在第一次访问的时候,浏览器中不会有缓存,100张图片,浏览器要连着请求100次http请求,服务器接收这些请求,都需要耗费内存去创建socket来玩tcp传输。

  重点来了,这样的话,服务器的压力会非常大,因为页面中的所有请求都是只请求到这台服务器上,如果1个人还好,如果10000个人并发访问呢,那服务器能扛住多少个tcp连接?带宽有多大?服务器的内存有多大?硬盘是高性能的吗?能抗住多少IO?web服务器分的内存有多大?会不会宕机?

  这就是为什么,越是大中型的web应用,他们越是要解耦。理论上可以把数据库+应用服务+消息队列+缓存+用户上传的文件+日志+等等都扔在一台服务器上,这也不用玩什么服务治理,也不用做什么性能监控,什么报警机制等等,就乱成一锅粥好了。但是这样就好像是把鸡蛋都放在一个篮子里,隐患非常大。如果因为一个子应用的内存不稳定导致整个服务器内存溢出而hung住,那整个网站就挂掉了。

  如果出意外挂掉,而恰好这时你们的业务又处于井喷式发展高峰期,业务成功被技术卡住,很可能会流失大量用户,后果不堪设想。

  此外,应用全部都耦合在一起,相当于一个巨石,当服务端负载能力不足时,一般会使用负载均衡的方式,将服务器做成集群,这样其实你是在水平扩展一块块巨石,性能加速度会越来越低,要知道,本身负载就低的功能or模块是没有必要水平扩展的,在本文中的例子就是性能瓶颈不在前端,那干嘛要水平扩展前端呢?还有发版部署上线的时候,明明只改了后端的代码,为什么要前端也跟着发布呢?

  正常的互联网架构,是都要拆开的,web服务器集群,应用服务器集群+文件服务器集群+数据库服务器集群+消息队列集群+缓存集群等等。

 

三、不分离,JSP的痛点,前端工程师的痛点,扎心……

  以前的Java Web项目大多数使用JSP作为页面层展示数据给用户,因为流量不高,因此也没有那么苛刻的性能要求。但现在是大数据时代,对于互联网项目的性能要求是越来越高,因此原始的前后端耦合在一起的架构模式已经逐渐不能满足我们,因此我们需要需找一种解耦的方式,来大幅度提升我们的负载能力。

  动态资源和静态资源全部耦合在一起,服务器压力大,因为服务器会收到各种http请求,例如css的http请求,js的,图片的等等。一旦服务器出现状况,前后台一起玩完,用户体验极差。 UI出好设计图后,前端工程师只负责将设计图切成HTML,需要由Java工程师来将HTML套成JSP页面,出错率较高(因为页面中经常会出现大量的js代码),修改问题时需要双方协同开发,效率低下。

  JSP必须要在支持Java的Web服务器里运行(例如Tomcat,Jetty,resin等),无法使用Ngnix等(Ngnix据说单实例http并发高达5w,这个优势要用上),性能提不上来。 第一次请求JSP,必须要在Web服务器中编译成Servlet,第一次运行会较慢。

  每次请求JSP都是访问servlet再用输出流输出的html页面,效率没有直接使用HTML高(是每次哟,亲~)。 jsp内有较多标签和表达式,前端工程师在修改页面时会捉襟见肘,遇到很多痛点。 如果jsp中的内容很多,页面响应会很慢,因为是同步加载。 需要前端工程师使用Java的IDE(例如eclipse),以及需要配置各种后端的开发环境,你们有考虑过前端工程师的感受吗?

 

四、So!前后端分离后的意义?它可解决以上的(及未列的)大部分“矛盾”

  前后端分离可以实现真正的前后端解耦,前端服务器使用Ngnix。

  前端/Web服务器放的是css,js,图片等等一系列静态资源(甚至你还可以css,js,图片等资源放到特定的文件服务器,例如阿里云的OSS,并使用CDN加速),前端服务器负责控制页面引用&跳转&路由,前端页面异步调用后端的接口,后端/应用服务器使用Tomcat(把Tomcat想象成一个数据提供者),加快整体响应速度。(这里需要使用一些前端工程化的框架比如Node.js,React,router,redux,webpack) 发现bug,可以快速定位是谁的问题,不会出现互相踢皮球的现象。页面逻辑,跳转错误,浏览器兼容性问题,脚本错误,页面样式等问题,全部由前端工程师来负责。接口数据出错,数据没有提交成功,应答超时等问题,全部由后端工程师来解决。

  双方互不干扰,前端与后端是相亲相爱的一家人。

  另外,在大并发情况下,我可以同时水平扩展前后端服务器,比如某宝的一个首页就需要2000+台前端服务器做集群来抗住日均多少亿+的日均pv。(去参加阿里的技术峰会,听他们说他们的web容器都是自己写的,就算他单实例抗10万http并发,2000台是2亿http并发,并且他们还可以根据预知洪峰来无限拓展,很恐怖,就一个首页。。。) 减少后端服务器的并发/负载压力。

  除了接口以外的其他所有http请求全部转移到前端Ngnix上,接口的请求调用Tomcat,参考Ngnix反向代理Tomcat。且除了第一次页面请求外,浏览器会大量调用本地缓存。 即使后端服务暂时超时或者宕机了,前端页面也会正常访问,只不过数据刷不出来而已。

  也许你也需要有微信相关的轻应用,那样你的接口完全可以共用,如果也有app相关的服务,那么只要通过一些代码重构,也可以大量复用接口,提升效率。(多端应用) 页面显示的东西再多也不怕,因为是异步加载。 nginx支持页面热部署,不用重启服务器,前端升级更无缝。 增加代码的维护性&易读性(前后端耦在一起的代码读起来相当费劲)。

  提升开发效率。因为可以前后端并行开发,而不是像以前的强依赖。 在Ngnix中部署证书,外网使用https访问,并且只开放443和80端口,其他端口一律关闭(防止×××端口扫描),内网使用http,性能和安全都有保障。 前端大量的组件代码得以复用,组件化,提升开发效率,抽出来。

 

五、再说说分离前后的对比吧,比如,开发模式、请求方式

  开发模式

  以前老的方式是:

  产品经历/领导/客户提出需求
  UI做出设计图
  前端工程师做HTML页面
  后端工程师将HTML页面套成JSP页面(前后端强依赖,后端必须要等前端的HTML做好才能套JSP。如果HTML发生变更,就更痛了,开发效率低)
  集成出现问题
  前端返工
  后端返工
  二次集成
  集成成功
  交付


  新的方式是:

  产品经历/领导/客户提出需求
  UI做出设计图
  前后端约定接口&数据&参数
  前后端并行开发(无强依赖,可前后端并行开发,如果需求变更,只要接口&参数不变,就不用两边都修改代码,开发效率高)
  前后端集成
  前端页面调整
  集成成功
  交付

  请求方式

  以前老的方式是:

  客户端请求
  服务端的Servlet或Controller接收请求(后端控制路由与渲染页面,整个项目开发的权重大部分在后端)
  调用Service,Dao代码完成业务逻辑
  返回JSP
  JSP展现一些动态的代码
  

  新的方式是

  浏览器发送请求
  直接到达HTML页面(前端控制路由与渲染页面,整个项目开发的权重前移)
  HTML页面负责调用服务端接口产生数据(通过Ajax等等,后台返回json格式数据,JSON数据格式因为简洁高效而取代XML)
  填充HTML,展现动态效果,在页面上进行解析并操作DOM。
  总结一下新的方式的请求步骤:

  大量并发浏览器请求--->Web服务器集群(Ngnix)--->应用服务器集群(Tomcat)--->文件/数据库/缓存/消息队列服务器集群

  同时又可以玩分模块,还可以按业务拆成一个个的小集群,为后面的架构升级做准备。

 

六、总之

  前后端分离并非仅仅只是一种开发模式,而是一种架构模式(前后端分离架构)。千万不要以为只有在撸代码的时候把前端和后端分开就是前后端分离了,需要区分前后端项目。前端项目与后端项目是两个项目,放在两个不同的服务器,需要独立部署,两个不同的工程,两个不同的代码库,不同的开发人员。

  前后端工程师需要约定交互接口,实现并行开发,开发结束后需要进行独立部署,前端通过ajax来调用http请求调用后端的Restful API。前端只需要关注页面的样式与动态数据的解析&渲染,而后端专注于具体业务逻辑。

 

七、最后,这里是拓展环节,一点建议(现在很多公司的前后端界限越来越明确,前端工程师只管前端的事情,后端工程师只管后端的事情,所以我觉得“术业有专攻”这句话,还是没错的)

  对于后端Java工程师:

    把精力放在Java基础,设计模式,jvm原理,spring+springmvc原理及源码,linux,mysql事务隔离与锁机制,mongodb,http/tcp,多线程,分布式架构,弹性计算架构,微服务架构,Java性能优化,以及相关的项目管理等等。

  后端追求的是:三高(高并发,高可用,高性能),安全,存储,业务等等。

  对于前端工程师:

    把精力放在HTML5,CSS3,jQuery,AngularJS,Bootstrap,React.js,Vue.js,webpack,less/sass,gulp,Node.js,Google V8引擎,Javascript多线程,模块化,面向切面编程,设计模式,浏览器兼容性,性能优化等等。

  前端追求的是:页面表现,速度流畅,兼容性,用户体验等等。

 

  祝大家阅读愉快~

  作者:慕课网
  链接:https://www.zhihu.com/question/28207685/answer/575863155
  来源:知乎
  著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

猜你喜欢

转载自www.cnblogs.com/yijiahao/p/11716844.html