自从抛弃Express的Jade模板,加上静态资源服务器,花满楼的小站已经全部静态化了(参见:Nodejs基于Express4的动态页面静态化),速度确实提上来了,目前还剩服务器端的优化加速了;为什么要抛弃Jade(包括Express的其他模板)?其实我也还没能力自己造一套模板,待会说说放弃后如何重新选择,不好意思Jade,可能我要将这条路走到底了;
我是个偏外行的小前端,在业余初学Nodejs这一路来,断断续续,胸无大志的只限于捯饬我这个其貌不扬的博客;讲起前端结构,结合各种工具(什么Grunt、Gulp、Webpack、FIS等),打造一套前端工程化的体系,哈哈,那肯定是我,肯定是我没有发言权啊;不过这确实非常诱人和令人向往的,加油!
一开始,前端我是用的Seajs,加上自己造的和网上copy的,弄了一堆自认为的模块和组件作为基础模块,全站由main.js来代理,作为全站的入口,并由他内部分发各个页面的主模块,各个主模块内require基础模块;后来,逐步优化前端资源,可能基础模块划分的有点过细,你知道的,http请求次数感觉有些超标,不愠不火的spm,有点坑,懒得去用了;然后我就换个思路:用Gulp和browserify,将Javascript引入到Nodejs端;首先是将那堆模块改造为符合Nodejs模块化规范的模块,但是没有改造像jQuery这样的偏大的基础类库和公共模块,防止单个文件体积过大;然后,写好Gulp browserify将各个页面的主模块一键编译,这次不用担心太过细分模块导致http请求次数太多,因为browserify会帮你处理依赖并且合并到一起,good job;最后,你就可以像写Nodejs程序一样去require模块,来写你的前端Javascript了;并且由browserify,NPM模块、内部核心模块、你造的模块之间是可以相互共存、相互调用的;当然,即便这样做了,那个作为全站入口的main.js还是可以用的,作用还是差不多;(参见:关于前端结构调整的一次实践)
妈呀,好像跑偏了;想必“伪静态”大多数人都听说过或者实践过吧,我仅仅知道.net的URLRewrite,还有短连接什么的,因为确实挺复杂,而且不好维护和管理;呵呵,初入前端,我只对Nodejs情有独钟;我不是要“伪静态”,我就是要静态;Express有这么好用的Router,咱也不必URLRewrite了;是的,我要真静态,同时做到“伪动态”;那么,问题也就来了,全站是你的静态页面,你又抛弃了内置的模板,这一堆静态页面怎么统一管理呢?还有静态页面的缓存也比较严重,总不能老是提醒访客强刷吧,你以为都像你,老按F5啊;
不曾拥有,谈何抛弃;既然已与Jade擦肩而过,全当是一次美丽的邂逅吧,志在远方,活在当下嘛!我得从新做出选择了,按照大众的做法,提取最基础的Header和Bottom;为了保留熟悉的感觉,页面上仅用<Header></Header>和<Bottom></Bottom>来占位吧;Header、Bottom模块大概这样,原谅我弄的比较粗糙,先实现吧:
1 module.exports=function(host) { 2 return [ 3 '<header>', 4 ...... 5 '</header>' 6 ].join(''); 7 };
这样一来,可能会有三个目录;一个是开发时的静态页面,一个是本地重造的静态页面,另一个是准备上线的静态页面;后两者都是重新生成的,别怕麻烦,重点是将提取的公共部分填充到开发时的静态页面里,再分别生成到后两个目录;这样做并不适合所有情况,只适合静态架子的页面,等页面加载后,ajax请求新数据填充的情况,如果要考虑SEO,就不能单独的生成页面了,最好在请求到达时就拿数据填充,同时生成对应的静态页面,response到客户端访问;先完成独立生成的静态页面吧,这种情况也挺常见的,而且这种情况按照这种做法,还有一大好处就是,每次更新不需要重启服务器,直接本地生成好,直接丢服务器上覆盖就是;呵呵,意外的收获;看看怎么重新生成页面到本地测试和备上线目录吧:
1 var online=false; 2 var host="localhost:4000"; 3 var header=require('../templs/header.js')(host); 4 var bottom=require('../templs/bottom.js')(host); 5 function createStaticPages(name,html,fn) { 6 var path='./path/'+name+'.html'; 7 if (online) { 8 path='./path_online/'+name+'.html'; 9 }; 10 var ws=fs.createWriteStream(path); 11 ws.write(html,function(err) { 12 console.log('writePage:'+path); 13 fn&&fn(); 14 }); 15 ws.on('drain',function() { 16 ws.end(); 17 }); 18 }; 19 function renderStaticPage(name,online) { 20 var rs=fs.createReadStream('./pages/'+name+'.html'); 21 var body=[],size=0; 22 rs.on('data',function(data) { 23 body.push(data); 24 size+=data.length; 25 }); 26 rs.on('error',function(err) { 27 console.log(err); 28 }); 29 rs.on('end',function() { 30 var buffer=Buffer.concat(body,size); 31 var html=buffer.toString(); 32 var $=cheerio.load(html); 33 if ($('header').html()=='') { 34 $('header').replaceWith(header); 35 }; 36 if ($('bottom').html()=='') { 37 $('bottom').replaceWith(bottom); 38 }; 39 var _html='<!DOCTYPE html><html>'+$('html').html()+'</html>'; 40 if (online) { 41 var _html='<!DOCTYPE html><html>'+$('html').html().replace(/(localhost:4000)/ig,'cdn.famanoder.com')+'</html>'; 42 }; 43 createStaticPages(name,reconvert(_html)); 44 }); 45 };
不多解释了,一般没什么高大上的写法;
然后执行以下代码就能生成好了,提前设置好最上面的变量即可:
1 (function(files) { 2 for(var i in files){ 3 (function(name) { 4 renderStaticPage(name,online); 5 })(files[i]); 6 } 7 })([ 8 'h5lab', 9 'minips', 10 'sprite' 11 ]);
初步尝试,刚刚上线,可能解释的还不怎么详细,还有就是访问及时生成的情况,以后再讨论吧,其实差不多少;关于页面的静态化涉及到的点挺多的,而且有些复杂,还包括服务器端的一些配置和实现,网上大侠们有好多长篇大论,每次看到都长见识了;我这只是实现了简单的静态资源服务器,为了前端上依然潇洒的zencoding,非逼自己保留了html文件,像上面提到的,只在里面做了占位,然后抛弃了Jade,自己弄堆乱七八糟的,来实现统一的管理;不过那个意外的收获多少还是有些令人欣慰的;至于静态页面的缓存,我想还是在src和href里,每次更新文件后手动加版本号什么的吧?或者在本地加好版本号,再生成一次,然后直接丢服务器吧,本地操作总不能也老是嫌麻烦吧,怎么说服务器不用重启了啊;
折腾来折腾去,现在Express项目骨架里的views目录已被我置空了;替换他的是那堆html页面和一些粗制乱造的templs,原谅我这颗时而躁动时而不安分的心,想安静的做个美男子好难的!想来想去,说的还是不够详细,他日走过来时路,再续!