欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

手寫Node靜態(tài)資源服務器的實現(xiàn)方法

想寫靜態(tài)資源服務器,首先我們需要知道如何創(chuàng)建一個http服務器,它的原理是什么

10年積累的成都網站建設、成都網站制作經驗,可以快速應對客戶對網站的新想法和需求。提供各種問題對應的解決方案。讓選擇我們的客戶得到更好、更有力的網絡服務。我雖然不認識你,你也不認識我。但先網站制作后付款的網站建設流程,更有三亞免費網站建設讓你可以放心的選擇與我們合作。

http服務器是繼承自tcp服務器 http協(xié)議是應用層協(xié)議,是基于TCP的

http的原理是對請求和響應進行了包裝,當客戶端連接上來之后先觸發(fā)connection事件,然后可以多次發(fā)送請求,每次請求都會觸發(fā)request事件

let server = http.createServer();
let url = require('url');
server.on('connection', function (socket) {
  console.log('客戶端連接 ');
});
server.on('request', function (req, res) {
  let { pathname, query } = url.parse(req.url, true);
  let result = [];
  req.on('data', function (data) {
    result.push(data);
  });
  req.on('end', function () {
    let r = Buffer.concat(result);
    res.end(r);
  })
});
server.on('close', function (req, res) {
  console.log('服務器關閉 ');
});
server.on('error', function (err) {
  console.log('服務器錯誤 ');
});
server.listen(8080, function () {
  console.log('server started at http://localhost:8080');
});

  1. req 代表客戶端的連接,server服務器把客戶端的請求信息進行解析,然后放在req上面
  2. res 代表響應,如果希望向客戶端回應消息,需要通過 res
  3. req和res都是從socket來的,先監(jiān)聽socket的data事件,然后等事件發(fā)生的時候,進行解析,解析出請頭對象,再創(chuàng)建請求對象,再根據(jù)請求對象創(chuàng)建響應對象
  4. req.url 獲取請求路徑
  5. req.headers 請求頭對象

接下來我們對一些核心功能進行講解

深刻理解并實現(xiàn)壓縮和解壓

為什么要壓縮呢?有什么好處?

可以使用zlib模塊進行壓縮及解壓縮處理,壓縮文件以后可以減少體積,加快傳輸速度和節(jié)約帶寬代碼

壓縮和解壓縮對象都是transform轉換流,繼承自duplex雙工流即可讀可寫流

  1. zlib.createGzip:返回Gzip流對象,使用Gzip算法對數(shù)據(jù)進行壓縮處理
  2. zlib.createGunzip:返回Gzip流對象,使用Gzip算法對壓縮的數(shù)據(jù)進行解壓縮處理
  3. zlib.createDeflate:返回Deflate流對象,使用Deflate算法對數(shù)據(jù)進行壓縮處理
  4. zlib.createInflate:返回Deflate流對象,使用Deflate算法對數(shù)據(jù)進行解壓縮處理

實現(xiàn)壓縮和解壓

因為壓縮我文件可能很大也可能很小,所以為了提高處理速度,我們用流來實現(xiàn)

let fs = require("fs");
let path = require("path");
let zlib = require("zlib");
function gzip(src) {
 fs
  .createReadStream(src)
  .pipe(zlib.createGzip())
  .pipe(fs.createWriteStream(src + ".gz"));
}
gzip(path.join(__dirname,'msg.txt'));
function gunzip(src) {
 fs
  .createReadStream(src)
  .pipe(zlib.createGunzip())
  .pipe(
   fs.createWriteStream(path.join(__dirname, path.basename(src, ".gz")))
  );
}
gunzip(path.join(__dirname, "msg.txt.gz"));
  1. gzip方法用于實現(xiàn)壓縮
  2. gunzip方法用于實現(xiàn)解壓
  3. 其中文件msg.txt是同級目錄
  4. 為什么需要這么寫:gzip(path.join(__dirname,'msg.txt'));
  5. 因為console.log(process.cwd());打印出當前工作目錄是根目錄,并不是文件所在目錄,如果這么寫gzip('msg.txt');找不到文件就會報錯
  6. basename 從一個路徑中得到文件名,包括擴展名的,可以傳一個擴展名參數(shù),去掉擴展名
  7. extname 獲取擴展名
  8. 壓縮的格式和解壓的格式需要對上,否則會報錯

有些時候我們拿到的字符串不是一個流,那怎么解決呢

let zlib=require('zlib');
let str='hello';
zlib.gzip(str,(err,buffer)=>{
  console.log(buffer.length);
  zlib.unzip(buffer,(err,data)=>{
    console.log(data.toString());
  })
});

有可能壓縮后的內容比原來還大,要是內容太少的話,壓縮也沒什么意義了

文本壓縮的效果會好一點,因為有規(guī)律

在http中應用壓縮和解壓

下面實現(xiàn)這樣一個功能,如圖:

手寫Node靜態(tài)資源服務器的實現(xiàn)方法

客戶端向服務器發(fā)起請求的時候,會通過accept-encoding(比如:Accept-Encoding:gzip,default)告訴服務器我支持的解壓縮的格式

  1. 服務器端需要根據(jù)Accept-Encoding顯示的格式進行壓縮,沒有的格式就不能壓縮,因為瀏覽器無法解壓
  2. 如果客戶端需要的Accept-Encoding中的格式服務端沒有,也無法實現(xiàn)壓縮
let http = require("http");
let path = require("path");
let url = require("url");
let zlib = require("zlib");
let fs = require("fs");
let { promisify } = require("util");
let mime = require("mime");
//把一個異步方法轉成一個返回promise的方法
let stat = promisify(fs.stat);
http.createServer(request).listen(8080);
async function request(req, res) {
 let { pathname } = url.parse(req.url); 
 let filepath = path.join(__dirname, pathname); 
 // fs.stat(filepath,(err,stat)=>{});現(xiàn)在不這么寫了,異步的處理起來比較麻煩
 try {
  let statObj = await stat(filepath);
  res.setHeader("Content-Type", mime.getType(pathname));
  let acceptEncoding = req.headers["accept-encoding"];
  if (acceptEncoding) {
   if (acceptEncoding.match(/\bgzip\b/)) {
    
    res.setHeader("Content-Encoding", "gzip");
    fs
     .createReadStream(filepath)
     .pipe(zlib.createGzip())
     .pipe(res);
   } else if (acceptEncoding.match(/\bdeflate\b/)) {
    res.setHeader("Content-Encoding", "deflate");
    fs
     .createReadStream(filepath)
     .pipe(zlib.createDeflate())
     .pipe(res);
   } else {
    fs.createReadStream(filepath).pipe(res);
   }
  } else {
   fs.createReadStream(filepath).pipe(res);
  }
 } catch (e) {
  res.statusCode = 404;
  res.end("Not Found");
 }
}
  1. mime:通過文件的名稱、路徑拿到一個文件的內容類型, 可以根據(jù)不同的文件內容類型返回不同的Content-Type
  2. acceptEncoding:全部寫成小寫是為了兼容不同的瀏覽器,node把所有的請求頭全轉成了小寫
  3. filepath:得到文件的絕對路徑
  4. 啟動服務后,訪問http://localhost:8080/msg.txt 可看到結果

深刻理解并實現(xiàn)緩存

為什么要緩存呢,緩存有什么好處?

  1. 減少了冗余的數(shù)據(jù)傳輸,節(jié)省了網費。
  2. 減少了服務器的負擔, 大大提高了網站的性能
  3. 加快了客戶端加載網頁的速度

緩存的分類

強制緩存:

強制緩存,在緩存數(shù)據(jù)未失效的情況下,可以直接使用緩存數(shù)據(jù)
在沒有緩存數(shù)據(jù)的時候,瀏覽器向服務器請求數(shù)據(jù)時,服務器會將數(shù)據(jù)和緩存規(guī)則一并返回,緩存規(guī)則信息包含在響應header中

手寫Node靜態(tài)資源服務器的實現(xiàn)方法

手寫Node靜態(tài)資源服務器的實現(xiàn)方法

對比緩存:

瀏覽器第一次請求數(shù)據(jù)時,服務器會將緩存標識與數(shù)據(jù)一起返回給客戶端,客戶端將二者備份至緩存數(shù)據(jù)庫中

再次請求數(shù)據(jù)時,客戶端將備份的緩存標識發(fā)送給服務器,服務器根據(jù)緩存標識進行判斷,判斷成功后,返回304狀態(tài)碼,通知客戶端比較成功,可以使用緩存數(shù)據(jù)

手寫Node靜態(tài)資源服務器的實現(xiàn)方法

手寫Node靜態(tài)資源服務器的實現(xiàn)方法

兩類緩存的區(qū)別和聯(lián)系

強制緩存如果生效,不需要再和服務器發(fā)生交互,而對比緩存不管是否生效,都需要與服務端發(fā)生交互
兩類緩存規(guī)則可以同時存在,強制緩存優(yōu)先級高于對比緩存,也就是說,當執(zhí)行強制緩存的規(guī)則時,如果緩存生效,直接使用緩存,不再執(zhí)行對比緩存規(guī)則

實現(xiàn)對比緩存

實現(xiàn)對比緩存一般是按照以下步驟:

第一次訪問服務器的時候,服務器返回資源和緩存的標識,客戶端則會把此資源緩存在本地的緩存數(shù)據(jù)庫中。

第二次客戶端需要此數(shù)據(jù)的時候,要取得緩存的標識,然后去問一下服務器我的資源是否是最新的。

如果是最新的則直接使用緩存數(shù)據(jù),如果不是最新的則服務器返回新的資源和緩存規(guī)則,客戶端根據(jù)緩存規(guī)則緩存新的數(shù)據(jù)

實現(xiàn)對比緩存一般有兩種方式

通過最后修改時間來判斷緩存是否可用

let http = require('http');
let url = require('url');
let path = require('path');
let fs = require('fs');
let mime = require('mime');
// http://localhost:8080/index.html
http.createServer(function (req, res) {
  let { pathname } = url.parse(req.url, true);
  //D:\vipcode\201801\20.cache\index.html
  let filepath = path.join(__dirname, pathname);
  fs.stat(filepath, (err, stat) => {
    if (err) {
      return sendError(req, res);
    } else {
      let ifModifiedSince = req.headers['if-modified-since'];
      let LastModified = stat.ctime.toGMTString();
      if (ifModifiedSince == LastModified) {
        res.writeHead(304);
        res.end('');
      } else {
        return send(req, res, filepath, stat);
      }
    }
  });
}).listen(8080);
function sendError(req, res) {
  res.end('Not Found');
}
function send(req, res, filepath, stat) {
  res.setHeader('Content-Type', mime.getType(filepath));
  //發(fā)給客戶端之后,客戶端會把此時間保存起來,下次再獲取此資源的時候會把這個時間再發(fā)回服務器
  res.setHeader('Last-Modified', stat.ctime.toGMTString());
  fs.createReadStream(filepath).pipe(res);
}

這種方式有很多缺陷

  1. 某些服務器不能精確得到文件的最后修改時間, 這樣就無法通過最后修改時間來判斷文件是否更新了
  2. 某些文件的修改非常頻繁,在秒以下的時間內進行修改.Last-Modified只能精確到秒。
  3. 一些文件的最后修改時間改變了,但是內容并未改變。 我們不希望客戶端認為這個文件修改了
  4. 如果同樣的一個文件位于多個cdn服務器上的時候內容雖然一樣,修改時間不一樣

ETag

ETag是根據(jù)實體內容生成的一段hash字符串,可以標識資源的狀態(tài)
資源發(fā)生改變時,ETag也隨之發(fā)生變化。 ETag是Web服務端產生的,然后發(fā)給瀏覽器客戶端

let http = require('http');
let url = require('url');
let path = require('path');
let fs = require('fs');
let mime = require('mime');
let crypto = require('crypto');

http.createServer(function (req, res) {
  let { pathname } = url.parse(req.url, true);
  
  let filepath = path.join(__dirname, pathname);
  fs.stat(filepath, (err, stat) => {
    if (err) {
      return sendError(req, res);
    } else {
      let ifNoneMatch = req.headers['if-none-match'];
      let out = fs.createReadStream(filepath);
      let md5 = crypto.createHash('md5');

      out.on('data', function (data) {
        md5.update(data);
      });
      out.on('end', function () {
      
        let etag = md5.digest('hex');
        let etag = `${stat.size}`;
        if (ifNoneMatch == etag) {
          res.writeHead(304);
          res.end('');
        } else {
          return send(req, res, filepath, etag);
        }
      });

    }
  });
}).listen(8080);
function sendError(req, res) {
  res.end('Not Found');
}
function send(req, res, filepath, etag) {
  res.setHeader('Content-Type', mime.getType(filepath));
  
  res.setHeader('ETag', etag);
  fs.createReadStream(filepath).pipe(res);
}

客戶端想判斷緩存是否可用可以先獲取緩存中文檔的ETag,然后通過If-None-Match發(fā)送請求給Web服務器詢問此緩存是否可用。

服務器收到請求,將服務器的中此文件的ETag,跟請求頭中的If-None-Match相比較,如果值是一樣的,說明緩存還是最新的,Web服務器將發(fā)送304 Not Modified響應碼給客戶端表示緩存未修改過,可以使用。

如果不一樣則Web服務器將發(fā)送該文檔的最新版本給瀏覽器客戶端

實現(xiàn)強制緩存

把資源緩存在客戶端,如果客戶端再次需要此資源的時候,先獲取到緩存中的數(shù)據(jù),看是否過期,如果過期了。再請求服務器

如果沒過期,則根本不需要向服務器確認,直接使用本地緩存即可

let http = require('http');
let url = require('url');
let path = require('path');
let fs = require('fs');
let mime = require('mime');
let crypto = require('crypto');
http.createServer(function (req, res) {
  let { pathname } = url.parse(req.url, true);
  let filepath = path.join(__dirname, pathname);
  console.log(filepath);
  fs.stat(filepath, (err, stat) => {
    if (err) {
      return sendError(req, res);
    } else {
      send(req, res, filepath);
    }
  });
}).listen(8080);
function sendError(req, res) {
  res.end('Not Found');
}
function send(req, res, filepath) {
  res.setHeader('Content-Type', mime.getType(filepath));
  res.setHeader('Expires', new Date(Date.now() + 30 * 1000).toUTCString());
  res.setHeader('Cache-Control', 'max-age=30');
  fs.createReadStream(filepath).pipe(res);
}

瀏覽器會將文件緩存到Cache目錄,第二次請求時瀏覽器會先檢查Cache目錄下是否含有該文件,如果有,并且還沒到Expires設置的時間,即文件還沒有過期,那么此時瀏覽器將直接從Cache目錄中讀取文件,而不再發(fā)送請求

Expires是服務器響應消息頭字段,在響應http請求時告訴瀏覽器在過期時間前瀏覽器可以直接從瀏覽器緩存取數(shù)據(jù)

Cache-Control與Expires的作用一致,都是指明當前資源的有效期,控制瀏覽器是否直接從瀏覽器緩存取數(shù)據(jù)還是重新發(fā)請求到服務器取數(shù)據(jù),如果同時設置的話,其優(yōu)先級高于Expires

下面開始寫靜態(tài)服務器

首先創(chuàng)建一個http服務,配置監(jiān)聽端口

 let http = require('http');
 let server = http.createServer();
    server.on('request', this.request.bind(this));
    server.listen(this.config.port, () => {
      let url = `http://${this.config.host}:${this.config.port}`;
      debug(`server started at ${chalk.green(url)}`);
    });

下面寫個靜態(tài)文件服務器

先取到客戶端想說的文件或文件夾路徑,如果是目錄的話,應該顯示目錄下面的文件列表

 async request(req, res) {
    let { pathname } = url.parse(req.url);
    if (pathname == '/favicon.ico') {
      return this.sendError('not found', req, res);
    }
    let filepath = path.join(this.config.root, pathname);
    try {
      let statObj = await stat(filepath);
      if (statObj.isDirectory()) {
        let files = await readdir(filepath);
        files = files.map(file => ({
          name: file,
          url: path.join(pathname, file)
        }));
        let html = this.list({
          title: pathname,
          files
        });
        res.setHeader('Content-Type', 'text/html');
        res.end(html);
      } else {
        this.sendFile(req, res, filepath, statObj);
      }
    } catch (e) {
      debug(inspect(e));
      this.sendError(e, req, res);
    }
  }
  
  sendFile(req, res, filepath, statObj) {
    if (this.handleCache(req, res, filepath, statObj)) return;
    res.setHeader('Content-Type', mime.getType(filepath) + ';charset=utf-8');
    let encoding = this.getEncoding(req, res);
    let rs = this.getStream(req, res, filepath, statObj);

    if (encoding) {
      rs.pipe(encoding).pipe(res);
    } else {
      rs.pipe(res);
    }
  }

支持斷點續(xù)傳

 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
    });
  }

支持對比緩存,通過etag的方式

handleCache(req, res, filepath, statObj) {
    let ifModifiedSince = req.headers['if-modified-since'];
    let isNoneMatch = req.headers['is-none-match'];
    res.setHeader('Cache-Control', 'private,max-age=30');
    res.setHeader('Expires', new Date(Date.now() + 30 * 1000).toGMTString());
    let etag = statObj.size;
    let lastModified = statObj.ctime.toGMTString();
    res.setHeader('ETag', etag);
    res.setHeader('Last-Modified', lastModified);
    if (isNoneMatch && isNoneMatch != etag) {
      return fasle;
    }
    if (ifModifiedSince && ifModifiedSince != lastModified) {
      return fasle;
    }
    if (isNoneMatch || ifModifiedSince) {
      res.writeHead(304);
      res.end();
      return true;
    } else {
      return false;
    }
  }

支持文件壓縮

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

編譯模板,得到一個渲染的方法,然后傳入實際數(shù)據(jù)數(shù)據(jù)就可以得到渲染后的HTML了

function list() {
  let tmpl = fs.readFileSync(path.resolve(__dirname, 'template', 'list.html'), 'utf8');
  return handlebars.compile(tmpl);
}

這樣一個簡單的靜態(tài)服務器就完成了,其中包含了靜態(tài)文件服務,實現(xiàn)緩存,實現(xiàn)斷點續(xù)傳,分塊獲取,實現(xiàn)壓縮的功能

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。

分享題目:手寫Node靜態(tài)資源服務器的實現(xiàn)方法
文章起源:http://aaarwkj.com/article22/jegpjc.html

成都網站建設公司_創(chuàng)新互聯(lián),為您提供虛擬主機、Google軟件開發(fā)、域名注冊、靜態(tài)網站、網站設計

廣告

聲明:本網站發(fā)布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創(chuàng)新互聯(lián)

成都定制網站建設
国产日韩欧在线视频| 亚洲va在线va天堂va在线| 中文字幕国产成人在线视频| 日韩高清av一区二区三区| 日韩欧美国产午夜精品| 欧美精品激情在线不卡| av 一区二区三区av| 亚洲国产日韩精品欧美| 欧美日韩另类综合一区| 色婷婷精品综合久久狠狠| 日本东京热免费一二三区| 亚洲一区二区日本乱码| 免费看欧美粗又大爽老| 成人黄色av大片在线观看| 深夜十八禁在线免费观看| 欧美日韩精品人妻一区| 少妇毛片一区二区三区| 久久精人妻一区二区三区| 国产剧情av在线资源| 亚洲精品国产精品成人| 亚洲欧美天堂一区二区| 亚洲色图综合在线观看| 视频一区二区三区不卡| 婷婷久久五月综合激情| 午夜未满十八禁止观看| 亚洲日本欧美激情综合| 日本欧美二区在线看| 日本啪啪精品一区二区三区| 韩国黄色理论片一区二区麻豆| 亚州精品少妇久久久久久| 日韩人妻高清精品专区| 久久久久久成人综合色| 国产日韩欧美另类专区| 91女厕偷拍女厕偷拍| 国产日韩精品专区一区| 给我免费在线观看视频| 亚洲新大香蕉视频在线播放| 成人色视频免费在线观看| 久久久久亚洲av成人网人| 六月丁香花五月婷婷| 午夜剧场福利在线观看|