node.js实现http服务器与浏览器之间的内容缓存操作示例
本文实例讲述了node.js实现http服务器与浏览器之间的内容缓存操作。分享给大家供大家参考,具体如下:
一、缓存的作用
1、减少了数据传输,节约流量。
2、减少服务器压力,提高服务器性能。
3、加快客户端加载页面的速度。
二、缓存的分类
1、强制缓存,如果缓存有效,则不需要与服务器发生交互,直接使用缓存。
2、对比缓存,每次都需要与服务器发生交互,对缓存进行比较判断是否可以使用缓存。
三、通过使用 last-modified / if-modified-since 来进行缓存判断
1、last-modified 是服务器向客户端发送的头信息,用于告诉客户端资源的 最后修改时间,该信息浏览器会保存起来。
2、if-modified-since 是客户端向服务器发送的头信息,当客户端再次请求资源时,浏览器会带上该信息发送给服务器,服务器通过该信息来判断资源是否过期。
3、如果没有过期,则响应 304 表示 未更新,告诉浏览器使用保存的缓存。
4、如果过期了,则响应 200,返回最新的资源。
const http = require('http'); const url = require('url'); const path = require('path'); const fs = require('fs'); const util = require('util'); const mime = require('mime'); //创建http服务器并监听端口 let server = http.createserver(); server.listen(1234, '0.0.0.0', function () { console.log('开始监听'); }); function sendfile(req, res, filepath, stats) { //设置文件内容类型 res.setheader('content-type', mime.gettype(filepath)); //设置资源最后修改时间头信息 res.setheader('last-modified', stats.ctime.togmtstring()); //通过管道将文件数据发送给客户端 fs.createreadstream(filepath).pipe(res); } server.on('request', function (req, res) { let {pathname} = url.parse(req.url, true); //获取文件真实路径 let filepath = path.join(__dirname, pathname); //判断文件是否存在 fs.stat(filepath, function (err, stats) { if (err) { return res.end(util.inspect(err)); } if (!stats.isfile()) { return res.end('is not file'); } //获取客户端请求的if-modified-since头信息 let ifmodifiedsince = req.headers['if-modified-since']; if (ifmodifiedsince) { //如果最后修改时间相同,说明该资源并未修改,直接响应 304,让浏览器从缓存中获取数据。 if (ifmodifiedsince == stats.ctime.togmtstring()) { res.statuscode = 304; res.end(); } else { sendfile(req, res, filepath, stats); } } else { sendfile(req, res, filepath, stats); } }); });
通过最后修改时间判断缓存是否可用,并不是很精确,有如下几个问题:
1、last-modified 只精确到秒,秒以下的时间修改,将无法准确判断。
2、文件最后修改时间变了,但 内容并没有发生改变。
3、文件存在于多个 cdn 上,那该文件的最后修改时间是不一样的。
四、通过 etag / if-none-match 进行判断
etag 表示 实体标签,将内容通过 hash 算法生成一段字符串,用以标识资源,如果资源发生变化,则 etag 也会变化。
etag 是服务器生成的,发送给客户端的。
1、客户端请求资源,服务器根据资源生成etag,发送给客户端。浏览器会保存该信息。
2、当客户端再次请求时,浏览器会发送 if-none-match 给服务器,值为第1步保存的信息,服务器通过该信息进行判断,资源是否修改过。
3、如果没有修改过,则响应 304 未更新,告诉浏览器使用保存的缓存。
4、如果修改过,则响应 200,返回最新资源。
const http = require('http'); const url = require('url'); const path = require('path'); const fs = require('fs'); const util = require('util'); const crypto = require('crypto'); const mime = require('mime'); //创建http服务器并监听端口 let server = http.createserver(); server.listen(1234, '0.0.0.0', function () { console.log('开始监听'); }); function sendfile(req, res, filepath, etag) { //设置文件内容类型 res.setheader('content-type', mime.gettype(filepath)); //设置etag头信息 res.setheader('etag', etag); //通过管道将文件数据发送给客户端 fs.createreadstream(filepath).pipe(res); } server.on('request', function (req, res) { let {pathname} = url.parse(req.url, true); //获取文件真实路径 let filepath = path.join(__dirname, pathname); //判断文件是否存在 fs.stat(filepath, function (err, stats) { if (err) { return res.end(util.inspect(err)); } if (!stats.isfile()) { return res.end('is not file'); } //获取客户端请求的if-none-match头信息 let ifnonematch = req.headers['if-none-match']; //创建可读流 let rs = fs.createreadstream(filepath); //创建md5算法 let md5 = crypto.createhash('md5'); rs.on('data', function (data) { md5.update(data); }); rs.on('end', function () { let etag = md5.digest('hex'); if (ifnonematch) { //判断etag与客户端发送过来的if-none-match是否相等 if (ifnonematch == etag) { res.statuscode = 304; res.end(); } else { sendfile(req, res, filepath, etag); } } else { sendfile(req, res, filepath, etag); } }); }); });
五、让浏览器在缓存有效期内不用发请求
expires 是http1.0的内容,用于设置缓存的有效期,在有效期内浏览器直接从浏览器缓存中获取数据。
cache-control 与expires作用一样,是http1.1的内容,用于指明当前资源的有效期,优先级高于expires。
cache-control可以设置的值 :
1、private 客户端可以缓存
2、public 客户端和代理服务器都可以缓存
3、max-age=10 缓存内容在10秒后失效
4、no-cache 使用对比缓存验证,强制向服务器验证
5、no-store 内容都不缓存,强制缓存和对比缓存都不会触发
const http = require('http'); const url = require('url'); const path = require('path'); const fs = require('fs'); const util = require('util'); const mime = require('mime'); //创建http服务器并监听端口 let server = http.createserver(); server.listen(1234, '0.0.0.0', function () { console.log('开始监听'); }); function sendfile(req, res, filepath, stats) { //设置文件内容类型 res.setheader('content-type', mime.gettype(filepath)); //设置缓存失效时间60秒 res.setheader('expires', new date(date.now() + 60 * 1000).toutcstring()); //设置缓存失效时间60秒 res.setheader('cache-control', 'max-age=60'); //通过管道将文件数据发送给客户端 fs.createreadstream(filepath).pipe(res); } server.on('request', function (req, res) { let {pathname} = url.parse(req.url, true); //获取文件真实路径 let filepath = path.join(__dirname, pathname); //判断文件是否存在 fs.stat(filepath, function (err, stats) { if (err) { return res.end(util.inspect(err)); } if (!stats.isfile()) { return res.end('is not file'); } sendfile(req, res, filepath, stats) }); });
希望本文所述对大家node.js程序设计有所帮助。
上一篇: go语言中linkname的用法