从手写一个静态服务器了解http

2018-03-07  本文已影响0人  JOKER_HAN

如果我们想在终端输入命令行启动一个服务,这是我们需要下载一个工具

npm  i yargs --save

yags 可以帮助我们解析命令行参数,把参数解析成对象

如何发布自己的npm包

  1. 注册npm
  2. 登陆之后验证你的邮箱
  3. 代码提交github
  4. 进入项目跟目录,命令行 npm -y生成package.json文件
  5. npm adduser 输入完用户名,密码,邮箱后没有错误信息就完成了
  6. 发布包 npm publish

这是我的静态服务器,安装方法:

npm install hj-static-server

项目地址:Github

怎样处理缓存

1.对比缓存

Last-Modified

1 .但是如果文件修改比较频繁,如果一秒内修改多次那么,Last-Modified是没法准确判断了,因为他只能精确到秒。
2 . 如果文件的修改的时间变了,但是内容未改变,我们也不希望客户端认为文件修改了 。
3 .如果同样的一个文件位于多个CDN服务器上的时候内容虽然一样,修改时间不一样。

ETag

2.强制缓存

Cache-Control 和 Expires
强制缓存的好处是浏览器不需要发送HTTP请求,一般不常更改的页面都会设置一个较长的强制缓存。
Cache-Control与Expires的作用一致,都是指明当前资源的有效期,控制浏览器是否直接从浏览器缓存取数据还是重新发请求到服务器取数据,如果同时设置的话,其优先级高于Expires。但Expires是HTTP1.0的内容,现在浏览器均默认使用HTTP1.1,所以基本可以忽略 。
Cache-Control 有几个参数

下面用代码说话,是如何处理缓存的。

 handleCache(req,res,filepath,statObj){
        let isNoneMatch = req.headers['is-none-match'];
        res.setHeader('Expires',new Date(Date.now() + 30 *1000).toGMTString());
        let etag = statObj.size;
        let ifModifiedSince = req.headers['if-modified-since'];
        let lastModified = statObj.ctime.toGMTString();
        res.setHeader('ETag',etag);
        res.setHeader('Last-Modified',lastModified);
        //如果任何一个对比缓存头不匹配,则不走缓存
        if(isNoneMatch && isNoneMatch !== etag){
            return false
        }
        if(lastModified && lastModified !==ifModifiedSince){
           return false
        }
        if (isNoneMatch || ifModifiedSince) {
            res.writeHead(304);
            res.end();
            return true;
        } else {
            return false;
        }

    }

怎样处理压缩

浏览器请求头中,都会携带自己的压缩类型,最常用的两种是gzip和deflate,服务端可以根据Accept-Ecoding头来返回响应的压缩资源
具体实现代码:

    getEncoding(req,res){
       let acceptEncoding = req.headers['accept-encoding'];
       if(/\bgzip\b/.test(acceptEncoding)){
           res.setHeaders('Content-Encoding','gzip');
           return zlib.createGzip();
       }else if(/\bdeflate\b/.test(acceptEncoding)){
           res.setHeader('Content-Encoding', 'deflate');
           return zlib.createDeflate();
       }else{
           return null;
       }
    }

什么是断点续传

对一个文件根据我们请求头中range字段中对应的bytes的值,下载文件内容。

getStream(req, res, filepath, statObj) {
        let start = 0;
        let end = statObj.size - 1;
        let range = req.headers['range'];
        if (range) {
            res.setHeader('Accept-Range', 'bytes');
            res.statusCode = 206;//返回整个内容的一块
            let result = range.match(/bytes=(\d*)-(\d*)/);
            if (result) {
                start = isNaN(result[1]) ? start : parseInt(result[1]);
                end = isNaN(result[2]) ? end : parseInt(result[2]) - 1;
            }
        }
        return fs.createReadStream(filepath, {
            start, end
        });
    }

源代码 Github地址

上一篇 下一篇

猜你喜欢

热点阅读