原生JS进行前后端同构


Posted in Javascript onApril 22, 2018

什么是前后端同构

明确三个概念:「后端渲染」指传统的 ASP、Java 或 PHP 的渲染机制;「前端渲染」指使用 JS 来渲染页面大部分内容,代表是现在流行的 SPA 单页面应用;「同构渲染」指前后端共用 JS,首次渲染时使用 Node.js 来直出 HTML。一般来说同构渲染是介于前后端中的共有部分。

感觉前端的确是折腾,之前还在流行前后端分离,现在怎么又要做前后端同构了?

原因是现在流行的SPA前端单页面应用比较沉重,首次访问需要加载文件较多,第一次加载过慢,用户需要等待前端进行渲染页面。而且不利于SEO及缓存,并且有一定的开发门槛。

前后端同构通过复用模板和JS文件,让一份代码可以同时跑在服务器和浏览器,首次渲染使用nodejs渲染页面,之后使用SPA路由跳转。可以有效减少用户首次访问的等待时间,并且对SEO比较友好,也便于缓存。

项目简介

本前后端同构项目主要分为两个部分,一个是基于koa2的渲染服务器,另一个是基于原生JS和zepto的前端SPA。

项目的特点是不使用vue和react等框架,门槛低,开发速度快,便于上手,比较轻巧,核心的router部分只有一百行左右的代码。适用于页面交互较少,变化不频繁的场景下,可以有效的提升性能和加载速度。

前端部分

前端部分的核心是路由部分,具体实现可以基于history API或是hash,网上有很多实现,这次主要讲下架构
前端部分采用MVC分层结构。

router层做的主要是创建路由示例,调用路由的get方法,给特定页面绑定来自control层的函数。
形式如:

import control from '../control'

//路由的构造函数支持传入渲染函数,路由的全局名称,路由跳转前调用的钩子
router = new Router(render,'ROUTER',beforeFn)

router.get('/page/a', control.pageA')

control层主要做的是加载跟后端共有的渲染模板和渲染数据,渲染出页面后运行页面函数

形式如:

let control = {
 pageA(req,res) {
  //webpack的动态加载,代码分割功能
  import(/* webpackChunkName: "pageA" */'script/pageA').then(module=> {
  // 检测该页面是否已有服务器渲染好,是的话直接运行module.default
  //否则加载模板和数据进行渲染,最后再调用页面函数
  if(this.needRender(module.default)) {
  //加载数据时访问的地址就是当前准备渲染的页面地址,只是加上了json=1的参数
   loadData('pageA').then(data => 
    res.render(xtpl,data,module.default))
  }
 }
}

// 捕捉webpack热更新,让他只进行相当于页面跳转的操作而不是刷新页面
if(module.hot) {
 module.hot.accept(['script/pageA'], () => {
  control[ROUTER.req.currentControl].call(ROUTER,null,ROUTER.res)
 })
}

view层即模板,这里使用的是xtpl模板,在服务器环境和前端环境下都支持渲染页面

页面函数的形式

页面函数要求使用es6的模块写法,配合webpack的按需加载功能

export default () => {
 window.addEventListener('scroll', fn)
//页面函数支持返回一个卸载函数,在页面离开的时候会被调用
//主要用于内存的释放,定时器的清除,事件监听的移除等等
 return function () {
  window.removeEventListener('scroll', fn)
 }
}

后端部分

使用koa2搭建的一个渲染服务器,在收到前端传来的页面请求时,会向API服务器请求数据,并识别页面请求是否带有json=1的参数,如果带有,则为前端路由跳转时的请求,直接返回数据即可,如果没有带json参数,加载跟前端共用的模板,配合数据进行渲染,发送到浏览器。

Javascript 相关文章推荐
基于jquery的direction图片渐变动画效果
May 24 Javascript
JS刷新框架外页面七种实现代码
Feb 18 Javascript
仿谷歌主页js动画效果实现代码
Jul 14 Javascript
禁用页面部分JavaScript不是全部而是部分
Sep 03 Javascript
JQuery异步获取返回值中文乱码的解决方法
Jan 29 Javascript
jquery实现跳到底部,回到顶部效果的简单实例(类似锚)
Jul 10 Javascript
Angular实现响应式表单
Aug 04 Javascript
JavaScript控制浏览器全屏显示简单示例
Jul 05 Javascript
详解vue的数据劫持以及操作数组的坑
Apr 18 Javascript
javascript跳转与返回和刷新页面的实例代码
Nov 20 Javascript
在vue中使用Echarts利用watch做动态数据渲染操作
Jul 20 Javascript
使用webpack和rollup打包组件库的方法
Feb 25 Javascript
如何以Angular的姿势打开Font-Awesome详解
Apr 22 #Javascript
vue移动端实现下拉刷新
Apr 22 #Javascript
实例讲解Vue.js中router传参
Apr 22 #Javascript
用Vue写一个分页器的示例代码
Apr 22 #Javascript
vue-cli3.0 特性解读
Apr 22 #Javascript
JS实现的哈夫曼编码示例【原始版与修改版】
Apr 22 #Javascript
使用Angular CLI快速创建Angular项目的一些基本概念和写法小结
Apr 22 #Javascript
You might like
php生成缩略图示例代码分享(使用gd库实现)
2014/01/20 PHP
php合并数组中相同元素的方法
2014/11/13 PHP
thinkPHP5.0框架模块设计详解
2017/03/18 PHP
javascript下IE与FF兼容函数收集
2008/09/17 Javascript
css3元素简单的闪烁效果实现(html5 jquery)
2013/12/28 Javascript
HTML页面弹出居中可拖拽的自定义窗口层
2014/05/07 Javascript
PHP 数组current和next用法分享
2015/03/05 Javascript
在线所见即所得HTML编辑器的实现原理浅析
2015/04/25 Javascript
javascript控制台详解
2015/06/25 Javascript
纯JS代码实现一键分享功能
2016/04/20 Javascript
JS获取填报扩展单元格控件的值的解决办法
2017/07/14 Javascript
解决vue打包之后静态资源图片失效的问题
2018/02/21 Javascript
Vue在页面右上角实现可悬浮/隐藏的系统菜单
2018/05/04 Javascript
Vue中使用sass实现换肤功能
2018/09/07 Javascript
JS监听滚动和id自动定位滚动
2018/12/18 Javascript
javascript删除数组元素的七个方法示例
2019/09/09 Javascript
js实现鼠标拖曳效果
2020/12/30 Javascript
[01:04:49]KG vs LGD 2019国际邀请赛小组赛 BO2 第二场 8.15
2019/08/16 DOTA
Python中类型检查的详细介绍
2017/02/13 Python
查看django版本的方法分享
2018/05/14 Python
python画一个玫瑰和一个爱心
2020/08/18 Python
Python XlsxWriter模块Chart类用法实例分析
2019/03/11 Python
Django保护敏感信息的方法示例
2019/05/09 Python
Pyqt5 关于流式布局和滚动条的综合使用示例代码
2020/03/24 Python
如何开发一款堪比APP的微信小程序(腾讯内部团队分享)
2016/12/22 HTML / CSS
HTML5 Canvas玩转酷炫大波浪进度图效果实例(附demo)
2016/12/14 HTML / CSS
猫途鹰英国网站:TripAdvisor英国(旅游社区和旅游评论)
2016/08/30 全球购物
环境工程大学生个人的自我评价
2013/10/08 职场文书
单位委托书范本
2014/04/04 职场文书
人力资源职位说明书
2014/07/29 职场文书
查摆问题自我剖析材料
2014/08/18 职场文书
2014年教师党员自我评议
2014/09/19 职场文书
新教师教学工作总结
2015/08/14 职场文书
小学五年级班主任工作经验交流材料
2015/11/02 职场文书
七年级写作指导之游记作文
2019/10/07 职场文书
Vue鼠标滚轮滚动切换路由效果的实现方法
2021/08/04 Vue.js