原生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 相关文章推荐
JS与C#编码解码
Dec 03 Javascript
javascripit实现密码强度检测代码分享
Dec 12 Javascript
JavaScript函数的4种调用方法详解
Apr 22 Javascript
详解JavaScript中this关键字的用法
May 26 Javascript
zepto与jquery的区别及zepto的不同使用8条小结
Jul 28 Javascript
深入理解Commonjs规范及Node模块实现
May 17 Javascript
VUE2.0+Element-UI+Echarts封装的组件实例
Mar 02 Javascript
JavaScript多态与封装实例分析
Jul 27 Javascript
JS实现字符串翻转的方法分析
Aug 31 Javascript
JS/jQuery实现简单的开关灯效果【案例】
Feb 19 jQuery
webpack DllPlugin xxx is not defined解决办法
Dec 13 Javascript
JS Thunk 函数的含义和用法实例总结
Apr 08 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
海河写的 Discuz论坛帖子调用js的php代码
2007/08/23 PHP
php cache类代码(php数据缓存类)
2010/04/15 PHP
setcookie中Cannot modify header information-headers already sent by错误的解决方法详解
2013/05/08 PHP
JS 用6N±1法求素数 实例教程
2009/10/20 Javascript
JavaScript 上万关键字瞬间匹配实现代码
2013/07/07 Javascript
js鼠标滑轮滚动事件绑定的简单实例(兼容主流浏览器)
2014/01/14 Javascript
js函数模拟显示桌面.scf程序示例
2014/04/20 Javascript
详解JS函数重载
2014/12/04 Javascript
D3.js中data(), enter() 和 exit()的问题详解
2015/08/17 Javascript
JavaScript程序开发之JS代码放置的位置
2016/01/15 Javascript
jQuery实现滚动鼠标放大缩小图片的方法(附demo源码下载)
2016/03/05 Javascript
浅析$(function) ready和onload 的区别
2016/09/03 Javascript
微信小程序 input输入框控件详解及实例(多种示例)
2016/12/14 Javascript
一个例子轻松学会Vue.js
2017/01/02 Javascript
Vue2.0组件间数据传递示例
2017/03/07 Javascript
Vuex之理解state的用法实例
2017/04/19 Javascript
Vue2.x中利用@font-size引入字体图标报错的解决方法
2018/09/28 Javascript
node.js express框架简介与实现
2019/07/23 Javascript
jQuery操作选中select下拉框的值代码实例
2020/02/07 jQuery
vue中利用three.js实现全景图的完整示例
2020/12/07 Vue.js
Python中的连接符(+、+=)示例详解
2017/01/13 Python
解决python3中自定义wsgi函数,make_server函数报错的问题
2017/11/21 Python
python编程通过蒙特卡洛法计算定积分详解
2017/12/13 Python
python从子线程中获得返回值的方法
2019/01/30 Python
NumPy 数组使用大全
2019/04/25 Python
python3+PyQt5 使用三种不同的简便项窗口部件显示数据的方法
2019/06/17 Python
windows下Python安装、使用教程和Notepad++的使用教程
2019/10/06 Python
python中通过pip安装库文件时出现“EnvironmentError: [WinError 5] 拒绝访问”的问题及解决方案
2020/08/11 Python
澳大利亚药房在线:ThePharmacy
2017/10/04 全球购物
学校食堂采购员岗位职责
2013/12/05 职场文书
车队司机自我鉴定
2014/03/02 职场文书
教师作风建设剖析材料
2014/10/11 职场文书
工程技术员岗位职责
2015/04/11 职场文书
2016年秋季运动会通讯稿
2015/11/25 职场文书
Spring Bean的实例化之属性注入源码剖析过程
2021/06/13 Java/Android
详细介绍MySQL中limit和offset的用法
2022/05/06 MySQL