一、缓存的作用

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中实现http服务器与浏览器之间的内容缓存的更多相关文章

  1. 浏览器中的 JS 和 Node.js 中的 JS

    一个是前端技术,一个是后端技术 浏览器中的 JavaScript ECMAScript  语言基础,如语法.数据类型结构.一些内置对象 BOM(Browser Object Model)  一些操作页 ...

  2. node.js中express模块创建服务器和http模块客户端发请求

    首先下载express模块,命令行输入 npm install express 1.node.js中express模块创建服务端 在js代码同文件位置新建一个文件夹(www_root),里面存放网页文 ...

  3. node.js中使用http模块创建服务器和客户端

    node.js中的 http 模块提供了创建服务器和客户端的方法,http 全称是超文本传输协议,基于 tcp 之上,属于应用层协议. 一.创建http服务器 const http = require ...

  4. node.js中通过dgram数据报模块创建UDP服务器和客户端

    node.js中 dgram 模块提供了udp数据包的socket实现,可以方便的创建udp服务器和客户端. 一.创建UDP服务器和客户端 服务端: const dgram = require('dg ...

  5. node.js中net模块创建服务器和客户端(TCP)

    node.js中net模块创建服务器和客户端 1.node.js中net模块创建服务器(net.createServer) // 将net模块 引入进来 var net = require(" ...

  6. js中ajax连接服务器open函数的另外两个默认参数get请求和默认异步(open的post方式send函数带参数)(post请求和get请求区别:get:快、简单 post:安全,量大,不缓存)(服务器同步和异步区别:同步:等待服务器响应当中浏览器不能做别的事情)(ajax和jquery一起用的)

    js中ajax连接服务器open函数的另外两个默认参数get请求和默认异步(open的post方式send函数带参数)(post请求和get请求区别:get:快.简单 post:安全,量大,不缓存)( ...

  7. Node.js中的Session,不要觉得简单哦。

    本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,博客地址为http://www.cnblogs.com/jasonnode/ .学习网站上有对应 ...

  8. 在node.js中使用COOKIE

    node.js中如何向客户端发送COOKIE呢?有如下两个方案: 一.使用response.writeHead,代码示例: //设置过期时间为一分钟 var today = new Date(); v ...

  9. 使用Node.js搭建静态资源服务器

    对于Node.js新手,搭建一个静态资源服务器是个不错的锻炼,从最简单的返回文件或错误开始,渐进增强,还可以逐步加深对http的理解.那就开始吧,让我们的双手沾满网络请求! Note: 当然在项目中如 ...

随机推荐

  1. 1.2.4 Excel快速建立n个文件夹

    1.准备员工信息表,选中名字 2.[设置单元格格式]>[数字]>[自定义]>右侧的[类型]>输入”md ”@>单击[确定] 3.确定后在姓名前会出现md,新建文本文档,将 ...

  2. 1、Sql-oracle-日期问题

    1.月份差 --MONTHS_BETWEEN(date2,date1) select months_between('19-12月-1999','19-3月-1999') from dual; sel ...

  3. tomcat接口调用时延开关

    项目中有些页面时延不稳定,需要看每次接口调用时延,怎么看,有两种方法:一种是直接去catalina.out日志中看,一种是直接去localhost_access_log日志中看,第一种需要在代码中实现 ...

  4. 域名到站点的负载均衡技术一览(主要是探讨一台Nginx抵御大并发的解决方案)(转)https://www.cnblogs.com/EasonJim/p/7823410.html

    一.问题域 Nginx.LVS.Keepalived.F5.DNS轮询,往往讨论的是接入层的这样几个问题: 1)可用性:任何一台机器挂了,服务受不受影响 2)扩展性:能否通过增加机器,扩充系统的性能 ...

  5. Python变量以及类型

    变量的定义 在程序中,有时我们需要对2个数据进行求和,那么该怎样做呢? 大家类比一下现实生活中,比如去超市买东西,往往咱们需要一个菜篮子,用来进行存储物品,等到所有的物品都购买完成后,在收银台进行结账 ...

  6. 一键精简Windows不常用的字体.cmd

    一键精简Windows不常用的字体.cmd @Echo offset path=%path%;%~dp0setlocal EnableDelayedExpansionset /P dv=请输入要精简W ...

  7. 高性能mysql 第六章查询性能优化 总结(上)查询的执行过程

    6  查询性能优化 6.1为什么查询会变慢 这里说明了的查询执行周期,从客户端到服务器端,服务器端解析,优化器生成执行计划,执行(可以细分,大体过程可以通过show profile查看),从服务器端返 ...

  8. 使用Mechanize实现自动化表单处理

    使用Mechanize实现自动化表单处理   mechanize是对urllib2的部分功能的替换,能够更好的模拟浏览器行为,在web访问控制方面做得更全面 mechanize的特点: 1 http, ...

  9. PHP运行出现Notice : Use of undefined constant 的解决办法

    这些是 PHP 的提示而非报错,PHP 本身不需要事先声明变量即可直接使用,但是对未声明变量会有提示.一般作为正式的网站会把提示关掉的,甚至连错误信息也被关掉 关闭 PHP 提示的方法 搜索php.i ...

  10. 自然语言处理(NLP)入门学习资源清单

    Melanie Tosik目前就职于旅游搜索公司WayBlazer,她的工作内容是通过自然语言请求来生产个性化旅游推荐路线.回顾她的学习历程,她为期望入门自然语言处理的初学者列出了一份学习资源清单. ...