简述JSP为什么被“抛弃”

简述JSP为什么被“抛弃”


一、前言

以前的项目大多数都是Java程序猿又当爹又当妈,既搞前端,又搞后端。

随着时代的发展,渐渐的许多大中小公司开始把前后端的界限分的越来越明确,前端工程师只负责前端的事情,后端工程师只管后端的事情。正所谓术业有专攻,一个人如果什么都会,那么每一样都很难达到精通。大中型公司需要专业人才,小公司需要全才。

二、前后端一体

曾几何时, 我们的Java Web项目都是使用了若干后台框架进行开发,Spring、Spring MVC、MyBatis、Hibernate等等。而且大多数项目在Java后端都是分了三层,控制层、业务层、持久层。控制层负责接收参数,调用相关业务层,封装数据,以及路由到JSP页面。然后Jsp页面上使用各种标签(jstl/el)表达式将后台的数据展现出来。

上述这种情况,需求定完了,代码写完了,测试测完了,然后发布。需要用maven或者eclipse等工具把代码打成一个war包,然后把这个war包发布到Web容器里,发布完了之后,启动Web容器,开始提供服务。然后通过配置域名,DNS等等相关,网站就可以访问了。这时前后端代码全都在那个war包里,包括js,css,图片,各种第三方的库。

用户在浏览器中输入网站域名:www.xxx.com,之后浏览器再通过ip路由到Web服务器,在tcp3次握手之后,通过tcp协议开始访问Web服务器,Web服务器得到请求后,开始提供服务,接收请求,之后通过response返回应答给浏览器。

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

这样的话,Web服务器的压力会非常大,因为页面中的所有请求都是只请求到这台服务器上,如果1个人还好,如果10000个人并发访问呢(先不聊web服务器集群,这里就说是单实例Web服务器),那Web服务器能扛住多少个tcp链接?服务器的内存有多大?能抗住多少IO?web服务器分的内存有多大?会不会宕机?

所以,越是大中型的Web应用,他们越是要解耦。理论上可以把数据库+应用服务+消息队列+缓存+用户上传的文件+日志+等等都扔在一台主机上,但是这样就好像是把鸡蛋都放在一个篮子里,隐患非常大。

正常的分布式架构,是都要拆开的,应用服务器集群(前,后)+文件服务器集群+数据库服务器集群+消息队列集群+缓存集群等等。

Java web项目都尽量要避免使用JSP,要搞前后台解耦,玩分布式架构,这样我们的应用架构才更强。

使用 JSP 的痛点

  1. 动态资源和静态资源全部耦合在一起,无法做到真正的动静分离。服务器压力大,因为服务器会收到各种http请求,例如css的http请求、js的、图片的、动态代码的等等。一旦服务器出现状况,前后台一起玩完,用户体验极差。
  2. 前端工程师做好html后,需要由Java工程师来将html修改成jsp页面,出错率较高(因为页面中经常会出现大量的js代码),修改问题时需要双方协同开发,效率低下。
  3. JSP 必须要在支持Java的Web服务器里运行(例如tomcat等),无法使用nginx等(nginx单实例http并发高达5w),性能提不上来。
  4. 第一次请JSP,必须要在web服务器中编译成servlet,第一次运行会较慢。
  5. 每次请求JSP都是访问Servlet再用输出流输出的html页面,效率没有直接使用html高。
  6. JSP 内有较多标签和表达式,前端工程师在修改页面时会捉襟见肘,遇到很多痛点。
  7. 如果JSP中的内容很多,页面响应会很慢,因为是同步加载。

基于上述的一些痛点,我们应该把整个项目的开发权重往前移,实现前后端真正的解耦!

三、前后端解耦

老的方式:

  1. 客户端请求
  2. 服务端的servlet或controller接收请求(路由规则由后端制定,整个项目开发的权重大部分在后端)
  3. 调用service,dao代码完成业务逻辑
  4. 返回JSP
  5. jsp展现一些动态的代码

新的方式:

  1. 浏览器发送请求
  2. 直接到达html页面(路由规则由前端制定,整个项目开发的权重前移)
  3. html页面负责调用服务端接口产生数据(通过ajax等等)
  4. 填充html,展现动态效果。

我们访问阿里巴巴等大型网站时,刷新一次页面,http请求大多数都是单独请求后台数据,使用 json传输数据,而不是一个大而全的http请求把整个页面包括动+静全部返回过来

这样做的好处是:

1.可以实现真正的前后端解耦,前端服务器使用nginx

前端服务器放的是css,js,图片等等一系列静态资源。甚至还可以css,js,图片等资源放到特定的文件服务器,例如阿里云的oss,并使用cdn加速,前端服务器负责控制页面引用,跳转,调用后端的接口,后端服务器使用tomcat。

这里需要使用一些前端工程化的框架比如nodejs,react,router,react,redux,webpack等。

2.发现bug,可以快速定位是谁的问题,不会出现互相踢皮球的现象

页面逻辑,跳转错误,浏览器兼容性问题,脚本错误,页面样式等问题,全部由前端工程师来负责。

接口数据出错,数据没有提交成功,应答超时等问题,全部由后端工程师来解决。

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

3.在大并发情况下,可以同时水平扩展前后端服务器

比如淘宝的一个首页就需要2000台前端服务器做集群来抗住日均多少亿+的日均pv。他们的web容器都是自己写的,就算单实例抗10万http并发,2000台是2亿http并发,并且他们还可以根据预知洪峰来无限拓展,很恐怖,就一个首页。

4.减少后端服务器的并发压力,除了接口以外的其他所有http请求全部转移到前端nginx上。

5.即使后端服务暂时超时或者宕机了,前端页面也会正常访问,只不过数据刷不出来而已。

6.接口完全可以共用,Web,APP可以复用接口,提升效率。

7.页面显示的东西再多也不怕,因为是异步加载。

但是也有很多需要注意的地方,比如:

  1. 在开需求会议的时候,前后端工程师必须全部参加,并且需要制定好接口文档,后端工程师要写好测试用例,不要让前端工程师充当后端开发的专职测试,推荐使用chrome的插件postman,service层的测试用例拿junit写。
  2. 加重了前端团队的工作量,减轻了后端团队的工作量,提高了性能和可扩展性。
  3. 使用前端的框架来解决类似于页面嵌套,分页,页面跳转控制等功能。
  4. 以前还有人在使用类似于velocity/freemarker等模板框架来生成静态页面,现在这种做法也被淘汰掉了。

虽然JSP在大型外网Java web项目中被淘汰掉,但是对于一些学生朋友来说,servlet等相关的Java web基础还是要掌握牢的,Spring MVC这些框架是基于它来写的。

©️2020 CSDN 皮肤主题: Age of Ai 设计师:meimeiellie 返回首页