精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

手把手帶你用 82 行代碼實現一個簡易版 Express 框架

開發 前端
connect 的 repo 描述是:“Connect is a middleware layer for Node.js”,也就是一個 Node.js 的中間件層。中間件層是一個非常有用的機制,它類似一個插件系統,讓我們可以通過插拔的方式組合不同功能來處理請求。

本文將帶大家實現輕量級 web 框架 connect[1] 的主要功能,只要 82 行代碼就能搞定。

我并沒有標題黨,因為 Express 在 v3 版本之前就是基于 connect 進行封裝的,不過在 v4 版本就將 connect 依賴移除了[2],代碼被搬到 Express 倉庫里,并做了一些細微調整。因此某種程度上,學習 connect 就是在學習 Express。

connect 的 repo 描述是:“Connect is a middleware layer for Node.js”,也就是一個 Node.js 的中間件層。中間件層是一個非常有用的機制,它類似一個插件系統,讓我們可以通過插拔的方式組合不同功能來處理請求。

基本使用

先來看看 connect 的使用。

const connect = require('connect')

const app = connect()

// respond to all requests
app.use(function(req, res){
  res.end('Hello from Connect!\n')
})

// create node.js http server and listen on port
http.createServer(app).listen(3000)

跟 Express 一樣。

另外,app 上還提供了 .listen() 方法,用于替代 http.createServer(app).listen(3000) 的冗長寫法。

app.listen(3000) // 等價于 http.createServer(app).listen(3000)

再看看中間件的使用。

app.use(function middleware1(req, res, next) {
  // middleware 1
  next()
});
app.use(function middleware2(req, res, next) {
  // middleware 2
  next()
});

我們通過 app.use() 方法收集并使用中間件。

中間件就是一個函數,包含 3 個參數:req、res 還有 next()。在一個中間件內調用 next(),就進入到下一個中間件的執行。

同時,我們還可以為中間件指定路由,這樣中間件只在特定路徑下起作用。

app.use('/foo', function fooMiddleware(req, res, next) {
  // req.url starts with "/foo"
  next()
})
app.use('/bar', function barMiddleware(req, res, next) {
  // req.url starts with "/bar"
  next()
})

本質上,純中間件的寫法就是在設置根路由('/'),所以會對所有請求有效。

app.use(function middleware1(req, res, next) {
  // middleware 1
  next()
})
// 等同于
app.use('/', function middleware1(req, res, next) {
  // middleware 1
  next()
})

不過還有一類特殊中間件——異常中間件,專門用于處理前面流程里的異常錯誤。

// regular middleware
app.use(function (req, res, next) {
  // i had an error
  next(new Error('boom!'));
});

// error middleware for errors that occurred in middleware
// declared before this
app.use(function onerror(err, req, res, next) {
  // an error occurred!
});

異常中間件必須是 4 個參數,第一個參數就是 error,對應前面流程中傳遞給 next() 的 Error 對象。

以上,我們就講完了 connect 庫的基本使用。接下來,就著手實現。

代碼實現

基于 connect v3.7.0 版本[3]。

剛學 Node.js 的時候,我們學到第一個例子,可能就是啟動一個會說“Hello World”的服務器了。

const http = require('node:http')

const hostname = '127.0.0.1'
const port = 3000

const server = http.createServer((req, res) => {
  res.statusCode = 200
  res.setHeader('Content-Type', 'text/plain')
  res.end('Hello World\n')
})

server.listen(port, hostname, () => {
  console.log(`Server running at http://${hostname}:${port}/`)
})

回顧 connect 的使用。

const connect = require('connect')

const app = connect()

// respond to all requests
app.use(function(req, res){
  res.end('Hello from Connect!\n')
})

// create node.js http server and listen on port
app.listen(3000)

實現 app.listen()

我們已經知道 app.listen(3000) 內部實現就是 http.createServer(app).listen(3000)。

因此,我們先實現 .listen() 方法。

module.exports = function createApplication() {
  const app = {}

  app.listen = function listen(...args) {
    const server = require('node:http').createServer(/* ? */)
    return server.listen(...args);
  }

  return app
}

假設 app 是一個對象。不過,http.createServer(/* ? */) 中的 ? 內容該如何實現呢?

實現 app.use()

前一步,我們做了 app.use() 的調用。

// respond to all requests
app.use(function(req, res){
  res.end('Hello from Connect!\n')
})

所以,當服務啟動后,訪問 localhost:3000 時,應該返回 "Hello from Connect!" 的文本。

同時,app.use() 又支持重復調用。

// respond to all requests
app.use(function(req, res, next) {
  console.log('log req.url', req.url)
  next()
})

// respond to all requests
app.use(function(req, res) {
  res.end('Hello from Connect!\n')
})

那我們就考慮先用個數組,把通過 app.use() 調用傳入進來的回調函數存起來。

module.exports = function createApplication() {
  const app = {}
 app.stack = []
  
  app.use = function use(route, fn) {
   let path = route
   let handle = fn

        // default route to '/'
   if (typeof route !== 'string') {
      path = '/'
      handle = route
    }
    
    this.stack.push({ route: path, handle })
    return this
  }
  
  app.listen = function listen(...args) {
    const server = require('node:http').createServer(/* ? */)
    return server.listen(...args)
  }

  return app
}

我們把調用 app.use() 傳入的中間件都存到了 app.stack 里。

根據定義可知,http.createServer(/* ? */) 中的 ? 內容應該是一個函數。針對當前場景,它是用來處理 stack 中的這些中間件的。

實現 app.handle()

我們把這些邏輯寫在 app.handle() 內。

module.exports = function createApplication() {
  const app = {}
  app.stack = []

  // ...

  app.listen = function listen(...args) {
    const server = require('node:http').createServer(app.handle.bind(app))
    return server.listen(...args)
  }

  app.handle = function handle(res, res) {
    // TODO
  }

  return app
}

每當請求來臨,都由 app.handle 負責處理。

app.handle 的主要邏輯主要是處理 3 件事情。

  1. 獲取當前要處理的路由,沒有的話就交由最終處理函數 done
  2. 路由不匹配就跳過
  3. 路由匹配就執行當前中間件
app.handle = function handle(req, res) {
  let index = 0

  const done = function (err) { /* ... */ }

  function next(err) {
    // next callback
    const layer = app.stack[index++]

    // 1) all done
    if (!layer) {
      setImmdiate(done, err)
      return
    }

    // route data
    const path = require('node:url').parse(req.url).pathname
    const route = layer.route

    // 2) skip this layer if the route doesn't match
    if (!path.toLowerCase().startsWith(route.toLowerCase())) {
      return next(err)
    }

    // 3) call the layer handle
    const arity = handle.length
    const hasError = !!err
    let error = err

    try {
      if (hasError && arity === 4) {
        // error-handling middleware
        layer.handle(err, req, res, next)
        return
      } else if (!hasError && arity < 4) {
        // request-handling middleware
        layer.handle(req, res, next)
        return
      }
    } catch (e) {
      error = e
    }

    next(error)
  }

  next()
}

以上的關鍵處理就封裝在 next() 函數中。而 next() 函數就是傳遞給 connect 中間件的 next 參數。

這樣,每次請求進來,我們都會從 app.stack 的第一個中間件(stack[0])開始處理,就實現了以 next 參數為連接橋梁的中間件機制。

值得注意的是調用當前中間件的邏輯,當我們調用 layer.handle(err, req, res, next)/layer.handle(req, res, next) 時,處理流程會流入中間件內部,當內部調用 next() 函數后,控制權會重新回到 app.handle,繼續處理隊列中的下一個中間件。

當請求最終沒有任何中間件可以處理時,就會流入到 done,這是最終處理器。處理器內部,會根據是否存在錯誤,分別返回 404 或 5xx 響應。

const done = function (err) {
  if (err) {
    res.statusCode = err.status ?? err.statusCode ?? 500
    res.statusMessage = require('node:http').STATUS_CODES[404]
  } else {
    res.statusCode = 404
    res.statusMessage = `Cannot ${req.method} ${require('node:url').parse(req.url).pathname}`
  }
  res.end(`${res.statusCode} ${res.statusMessage}`)
}

至此,我們基本寫完了所有的邏輯。

當然,有一個地方,可以做一個小小的優化。將 http.createServer(app.handle.bind(app)) 簡化成 http.createServer(this),不過此時 app 就不能是對象,而是函數了。

module.exports = function createApplication() {
 function app(req, res) { app.handle(req, res) }

  // ...
  
  app.listen = function listen(...args) {
    const server = require('node:http').createServer(app)
    return server.listen(...args)
  }

  // ...
  
 return app
}

最后,我們整體來回顧一下。

module.exports = function createApplication() {
  function app(req, res) { app.handle(req, res) }
  app.stack = []

  app.use = function use(route, fn) {
    let path = route
    let handle = fn
    
    // default route to '/'
    if (typeof route !== 'string') {
      path = '/'
      handle = route
    }

    this.stack.push({ route: path, handle })
    return this
  }

  app.listen = function listen(...args) {
    const server = require('node:http').createServer(app)
    return server.listen(...args)
  }

  app.handle = function handle(req, res) {
    let index = 0

    const done = function (err) {
      if (err) {
        res.statusCode = err.status ?? err.statusCode ?? 500
        res.statusMessage = require('node:http').STATUS_CODES[404]
      } else {
        res.statusCode = 404
        res.statusMessage = `Cannot ${req.method} ${require('node:url').parse(req.url).pathname}`
      }
      res.end(`${res.statusCode} ${res.statusMessage}`)
    }

    function next(err) {
      // next callback
      const layer = app.stack[index++]

      // 1) all done
      if (!layer) {
        setImmediate(done, err)
        return
      }

      const path = require('node:url').parse(req.url).pathname
      const route = layer.route
      
      // 2) skip this layer if the route doesn't match
      if (!path.toLowerCase().startsWith(route.toLowerCase())) {
        return next(err)
      }

      // 3) call the layer handle
      const arity = handle.length
      const hasError = !!err
      let error = err

      try {
        // error-handling middleware
        if (hasError && arity === 4) {
          layer.handle(err, req, res, next)
          return
        // request-handling middleware
        } else if (!hasError && arity < 4) { 
          layer.handle(req, res, next)
          return
        }
      } catch (e) {
        error = e
      }

      next(error)
    }

    next()
  }
  
  return app
}

連上注釋,我們只用了 82 行代碼,就實現了 connect 的主要功能。

總結

本文帶大家實現了輕量級 Web 框架 connect 的主要功能,同樣這也是一個簡易版本  Express!

實現核心是 2 個函數。

  • app.use(route, fn):用于收集中間件
  • app.handle(res, req):用于消費中間件。主要邏輯位于 next() 函數,這是傳遞給中間件的 next 參數。每一次接收請求來臨時,都由 app.handle 負責處理

而這兩個函數之間的橋梁就是 app.stack。

行文最后,給大家留一個思考題。

connect() 實例的真實實現,是支持作為子應用,掛載到父應用之上的,也就是下面的用法。

const connect = require('connect')
const app = connect()
const blogApp = connect()

app.use('/blog', blogApp)
app.listen(3000)

甚至 http.Server 實例也支持掛載。

const connect = require('connect')
const app = connect()

const blog = http.createServer(function(req, res){
  res.end('blog')
})

app.use('/blog', blog)

那是如何實現呢?

大家可以參照 app.use()[4] 函數的源碼進行學習。

感謝的你的閱讀,再見~

參考資料

[1]connect: https://github.com/senchalabs/connect

[2]在 v4 版本就將 connect 依賴移除了: https://github.com/expressjs/express/compare/3.21.2...4.0.0#diff-7ae45ad102eab3b6d7e7896acd08c427a9b25b346470d7bc6507b6481575d519

[3]connect v3.7.0 版本: https://github.com/senchalabs/connect/blob/3.7.0/index.js

[4]app.use(): https://github.com/senchalabs/connect/blob/3.7.0/index.js#L76

責任編輯:武曉燕 來源: 寫代碼的寶哥
相關推薦

2022-11-07 18:36:03

組件RPC框架

2017-10-29 21:43:25

人臉識別

2022-10-20 11:49:49

JS動畫幀,CSS

2020-12-15 08:58:07

Vue編輯器vue-cli

2017-11-27 15:17:24

PythonMonkey腳本

2017-07-19 13:27:44

前端Javascript模板引擎

2022-02-28 12:07:56

RxJS函數式

2014-01-22 09:19:57

JavaScript引擎

2022-06-28 15:29:56

Python編程語言計時器

2024-07-16 11:26:35

微前端代碼JS

2017-10-27 10:29:35

人臉識別UbuntuPython

2017-09-20 16:39:31

云主機網站MySQL

2017-09-20 12:36:50

云主機網站MySQL

2022-09-22 12:38:46

antd form組件代碼

2023-03-29 10:02:36

2023-01-30 09:27:57

開發自動化配置

2020-05-09 09:59:52

Python數據土星

2021-06-22 10:43:03

Webpack loader plugin

2022-08-11 07:32:51

Starter自動裝配

2021-10-09 13:08:58

C++STLSet
點贊
收藏

51CTO技術棧公眾號

无码精品在线观看| 九九热最新地址| 成人免费av电影| 国产精品美女久久久久久久久 | 午夜欧美性电影| 国产美女免费看| 在线视频亚洲| 久久精品成人一区二区三区 | 中文字幕亚洲在线观看| 欧美日韩视频免费播放| 亚洲精品白虎| 日本高清视频免费看| 老司机精品视频网站| www.国产精品一二区| 日韩aaaaa| 先锋影音网一区二区| 黄色成人在线播放| 在线观看视频黄色| 欧洲毛片在线| 成人网在线播放| 国产精品最新在线观看| 久久夜色精品亚洲| 国产精品麻豆久久| 亚洲欧洲在线免费| 图片区偷拍区小说区| 久久麻豆视频| 日本精品一区二区三区四区的功能| 中文字幕乱码免费| www在线播放| 91小视频在线观看| www.久久艹| 国产女人18毛片18精品| 日韩综合在线视频| 欧美一级片在线播放| 久草国产在线视频| 亚洲最大av| 日韩中文在线不卡| 免费福利视频网站| 天海翼亚洲一区二区三区| 欧美一级一级性生活免费录像| 熟妇人妻va精品中文字幕| 超碰97免费在线| 一区二区三区精品| 日本xxx免费| 色大18成网站www在线观看| www激情久久| 精品综合在线| 五月婷婷狠狠干| yourporn久久国产精品| 俄罗斯精品一区二区三区| 国产又大又黑又粗| 国产综合久久久久影院| 国产欧美精品va在线观看| 国产天堂第一区| 久久精品理论片| 国产精品视频久久久| 成人黄色免费网| 奇米色777欧美一区二区| 国产成人啪精品视频免费网| 国产性生活视频| 三级久久三级久久| 日韩av黄色在线观看| 青青青国产在线 | 欧洲精品毛片网站| 在线能看的av| 日韩在线一区二区三区| 国产精品久久久久久婷婷天堂 | 欧美久久久久久蜜桃| 波多野结衣天堂| 成人黄色免费观看| 欧美日韩国产一级二级| 亚洲五月激情网| 亚洲国产视频二区| 精品对白一区国产伦| 欲求不满的岳中文字幕| 亚洲三级性片| 色阁综合伊人av| 成年人av电影| 91久久午夜| 国产精彩精品视频| 国产又黄又大又爽| 国产999精品久久| 麻豆视频成人| 在线免费av网站| 亚洲特黄一级片| 国产无限制自拍| 精品亚洲美女网站| 3d动漫精品啪啪1区2区免费| 特级特黄刘亦菲aaa级| 在线亚洲a色| xxav国产精品美女主播| 久久精品99国产精| 久久久久久夜| 91丨九色丨国产在线| 你懂的网站在线| 国产日产欧美一区| 日韩精品久久一区二区| 欧美最新精品| 欧美不卡在线视频| 91在线无精精品白丝| 女生裸体视频一区二区三区| 奇米一区二区三区四区久久| 国产一区二区自拍视频| 91蝌蚪porny九色| gogogo免费高清日本写真| av在线视屏| 欧美日韩黄视频| 97人妻精品一区二区三区免| 久久久五月天| 日本午夜在线亚洲.国产| 精品国产va久久久久久久| 久久久久国产免费免费| 国产小视频免费| 日日夜夜一区| 亚洲欧美日韩中文在线制服| 我家有个日本女人| 麻豆国产精品777777在线| 精品乱码一区| 黑人玩欧美人三根一起进| 欧美色图免费看| 日韩av一二区| 国内在线观看一区二区三区| 国产欧美va欧美va香蕉在| 青青草观看免费视频在线| 亚洲一区二区三区美女| 亚洲精品第三页| 激情婷婷综合| 91av在线播放| 国产91免费在线观看| 中文字幕在线视频一区| 成人亚洲视频在线观看| 西野翔中文久久精品国产| 国内精品视频一区| 国产成人久久精品77777综合| 欧美国产亚洲另类动漫| 91精品91久久久中77777老牛| 99久久香蕉| 欧美成人中文字幕| 91av久久久| 中文字幕中文在线不卡住| 欧美亚洲日本在线观看| 国内精品久久久久久久影视简单 | 在线视频一区观看| 素人啪啪色综合| 亚洲人成网站999久久久综合| 可以免费看的av毛片| www.av亚洲| 黄页网站大全在线观看| 国产亚洲精品美女久久| 欧美大片在线看| 午夜精品久久久久久久99老熟妇| 亚洲老司机在线| 国产xxxxhd| 亚洲视频福利| 国产一区二区免费电影| 免费毛片b在线观看| 亚洲国产精品网站| 日本在线观看视频网站| 99久久伊人网影院| 日韩欧美视频网站| 欧美女王vk| 国产精品视频1区| 免费观看久久久久| 日韩一区二区三区四区五区六区| 性生交大片免费全黄| 国内精品第一页| 日本男女交配视频| 极品一区美女高清| 1769国产精品| 99免在线观看免费视频高清| 欧美日韩色综合| 精品国产欧美日韩不卡在线观看| 国产一区二区免费看| av在线播放天堂| 一本色道久久综合亚洲精品酒店 | av手机天堂网| 国产精品久久久久一区二区三区 | 一本色道久久综合熟妇| 亚洲欧美激情在线| 亚洲一级Av无码毛片久久精品| 亚洲日本视频| 日本午夜精品一区二区三区| 色综合视频一区二区三区日韩| 日韩一区二区三区xxxx| 丰满人妻一区二区三区无码av | 亚洲色成人一区二区三区小说| 国产精品三级| 亚洲a级在线播放观看| bl在线肉h视频大尺度| 亚洲视频第一页| 国产露脸无套对白在线播放| 黄色一区二区三区| 亚洲欧洲综合网| 99精品国产视频| 亚洲国产日韩欧美在线观看| 黄色亚洲在线| 亚洲啪啪av| 久久中文资源| 成人av番号网| 中文字幕高清在线播放| 按摩亚洲人久久| 美州a亚洲一视本频v色道| 欧美一区二区三区喷汁尤物| 欧美a视频在线观看| 中文字幕日韩欧美一区二区三区| 国产美女视频免费观看下载软件| 久久国产综合精品| 国产淫片免费看| 欧美日韩一区自拍| 亚洲成人第一| 天堂俺去俺来也www久久婷婷| 国产一区二区在线免费视频| 咪咪网在线视频| 欧美成人全部免费| 国产二区视频在线观看| 亚洲丁香婷深爱综合| 国产又粗又大又爽视频| 91黄色激情网站| 国产成人自拍视频在线| 一区二区三区自拍| 国产黄色片在线| 久久精品一区八戒影视| 稀缺小u女呦精品呦| 国产一区二区三区观看| 中文字幕永久视频| 久久不射2019中文字幕| 欧美一区二区激情| 伊人久久大香线| 一区二区三区在线视频111| 视频一区中文| 久久精品国产第一区二区三区最新章节 | 大地资源网3页在线观看| 国产一区二区日韩| 日韩一区二区三区中文字幕| 欧美精品一区二区三区视频| 性欧美8khd高清极品| 正在播放亚洲一区| 在线播放成人av| 欧美无人高清视频在线观看| 欧美h在线观看| 欧美日韩亚洲成人| 国产成人无码精品亚洲| 精品久久久久久久久久久久久久| 久久国产精品波多野结衣| 亚洲三级在线免费| 国产老头老太做爰视频| 亚洲欧洲精品一区二区三区不卡| 欧美色图17p| 国产精品不卡一区二区三区| 成人一级片免费看| 亚洲国产高清在线观看视频| 国产精品情侣呻吟对白视频| 国产精品美女一区二区| 日日噜噜夜夜狠狠久久波多野| 亚洲欧美激情小说另类| 久草免费在线观看视频| 亚洲超碰精品一区二区| 久久狠狠高潮亚洲精品| 欧美日韩一区二区在线| 少妇久久久久久久| 欧美视频一区在线观看| 国产一区二区在线不卡| 欧美一级二级三级蜜桃| 国产激情无套内精对白视频| 亚洲成人久久久| 色哟哟在线观看| 国产亚洲成精品久久| 在线观看美女网站大全免费| 久久九九国产精品怡红院 | 好男人在线视频www| 日韩av在线免播放器| jizz在线观看| 美女999久久久精品视频| h片视频在线观看| 国产va免费精品高清在线观看| 澳门av一区二区三区| 成人福利视频在线观看| 1769国产精品视频| 欧美成人免费在线| 91日韩视频| 日本xxxxxxxxxx75| 日日夜夜免费精品| av地址在线观看| 久久影院视频免费| 情侣偷拍对白清晰饥渴难耐| 亚洲国产精品尤物yw在线观看| 亚洲第一在线播放| 欧美日韩亚洲综合| 成人午夜视频一区二区播放| 亚洲人成绝费网站色www| 国产激情在线| 欧美一区二三区| 国产一区二区在线观| 欧美精品与人动性物交免费看| 9999国产精品| 久久精品免费一区二区| 看国产成人h片视频| av鲁丝一区鲁丝二区鲁丝三区| 欧美激情一区二区在线| 青青草在线观看视频| 91黄色在线观看| 免费观看黄色一级视频| 日韩在线视频免费观看| 卡通欧美亚洲| 国产精品久久一区二区三区| 日韩精品dvd| 免费av观看网址| 国产高清精品在线| 韩国女同性做爰三级| 亚洲国产成人精品视频| 97精品人妻一区二区三区在线| 日韩精品在线播放| 日本资源在线| 国产精品一区二区三区在线播放| 啪啪国产精品| 日韩a级黄色片| 国精产品一区一区三区mba视频| 亚洲最大的黄色网| 又紧又大又爽精品一区二区| 中文字字幕在线观看| 亚洲美女激情视频| zzzwww在线看片免费| 91黄色精品| 外国成人免费视频| 69久久久久久| 国产欧美精品国产国产专区 | 欧美日韩一区成人| 欧美挠脚心网站| 91精品国产91久久久久久吃药| 精品视频在线播放一区二区三区 | 日本乱人伦一区| 神马精品久久| 午夜精品视频在线| 成人爽a毛片| 日本aa在线观看| 国产精品综合二区| 亚洲不卡在线播放| 欧美日韩第一区日日骚| 91网在线播放| 国产一区二区丝袜| 99久久九九| www.五月天色| 亚洲精品高清视频在线观看| 91精品人妻一区二区三区果冻| 在线视频欧美性高潮| 欧美色网在线| 视频一区视频二区视频| 蜜桃免费网站一区二区三区| 99精品欧美一区二区| 欧美亚洲丝袜传媒另类| av在线播放网站| 国产欧洲精品视频| 亚洲a一区二区三区| 久久精品一卡二卡| 亚洲综合自拍偷拍| 黄色一级a毛片| 97视频在线观看视频免费视频| 欧美日韩一区二区三区在线电影| 男人添女人下部高潮视频在观看| 91在线视频免费观看| 日韩在线 中文字幕| 国产一区二区av| 欧美黄色网络| 国产人妻人伦精品| 99久久久国产精品| 黄色片视频免费| 日韩在线视频免费观看高清中文| 精品一区二区三区中文字幕视频 | 日韩一级片在线观看| 最新av在线播放| 精品欧美日韩| 日韩精品乱码av一区二区| 国产福利在线导航| 日韩欧美久久一区| 国产理论在线| 性欧美videosex高清少妇| 激情五月婷婷综合网| 动漫精品一区一码二码三码四码| 亚洲精品aⅴ中文字幕乱码| 免费成人直播| 国产免费一区二区三区四在线播放 | 1024国产精品| 色香蕉在线视频| 国产精品久久国产精品99gif| 99视频精品全部免费在线视频| 色综合久久久无码中文字幕波多| 欧美日韩视频在线| 日韩三级影院| 国产一区免费在线观看| 美女免费视频一区| 久久精品国产亚洲av香蕉| 亚洲网站在线播放| 免费一区二区三区在线视频| 日本欧美黄色片| 日韩毛片精品高清免费| 无码国精品一区二区免费蜜桃| 国产精品美乳在线观看| 亚洲经典在线看| 手机在线免费看毛片| 亚洲午夜性刺激影院|