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

深入理解 JSX:從零開始實現一個 JSX 解析器

開發 前端
本質上,這里使用標簽值創建一個包裝元素,為其添加屬性(如果有的話),最后,遍歷子列表(這是一個包含所有添加屬性的剩余屬性),在此過程中,將簡單地將這些值作為字符串返回(第 9 行)。

JSX 表示 JavaScript XML,它是 JavaScript 的擴展,允許開發人員在 JavaScript 代碼中使用類似 HTML 的語法。此擴展使組件的組合更易于閱讀,它隨著 React 一起出現,簡化了在 HTML 和 JavaScript 中編寫代碼的方式。

那 JSX 究竟是如何工作的呢?它背后又有怎樣的奇技淫巧?本文將介紹 JSX 的基本用法,然后從零開始編寫一個 JSX 解析器,將 JSX “組件”轉換為實際返回的有效 HTML 的JavaScript 代碼。

1、JSX 概述

基本語法

JSX 是 JavaScript XML 的縮寫,它是一種在JavaScript代碼中編寫類似于HTML結構和語法的擴展。通過使用JSX,可以更直觀地描述組件的結構,并使得代碼更易于閱讀和維護。盡管JSX看起來像HTML,但它實際上是通過編譯器轉換為純JavaScript代碼的。在編譯過程中,JSX元素會被轉換為React.createElement()函數的調用,創建相應的React元素。

JSX 允許創建自定義元素并在 React 應用中重用它們。 在下面的示例中,Main 組件包裝在 main 標簽中。 它還允許在 HTML 標簽中嵌入 JavaScript 表達式。 在下面的示例中,“Main content”文本是一個將被計算并渲染為文本的表達式。

使用 JSX,您可以構建如下組件:

function App() {
  return (
    <div>
      <h1>Hello</h1>
    </div>
  )
}

這段代碼 return 之后的就是JSX。

使用 JSX 的主要好處之一是它使代碼更具可讀性和簡潔性。來看下面的代碼塊,比較了帶有和不帶有 JSX 的簡單列表。

// 非 JSX
const fruits = ["apple", "banana", "cherry"];

// JSX
const jsxFruits = [<li>apple</li>, <li>banana</li>, <li>cherry</li>];

JSX 還具有許多使其比 HTML 使用起來更方便的功能。例如,可以在 JSX 標簽內使用 JavaScript 表達式來動態創建和填充 HTML 元素。還可以使用內置 JavaScript 函數來操作 HTML 元素并設置其樣式。

需要注意,JSX 屬性使用駝峰命名約定而不是 HTML 屬性名稱。

<button onClick = {handleClick}>Click</button>
<div className = "hello"> Div </div>
<label htmlFor="">Label</label>

JSX 表達式只能有一個父元素

JSX 表達式只能有一個父元素,那為什么不能有多個父元素呢?

function App() {
  return (
    <div>Why</div>
    <div>Can I not do this?</div>
    )
}

或者:

function App() {
  return (
    <div>
      {isOpen && (
        <div>Why again</div>
        <div>Can I not do this</div>
      )}
    </div>
  )
}

下面就來看看原因!

JSX 是 React.createElement 的語法糖,它是一個普通的 JavaScript 方法。 JSX 被編譯成瀏覽器可以理解的普通 JavaScript。

要像在沒有 JSX 的情況下創建 React 元素,可以在 React 對象上使用 createElement 方法。 該方法的語法是:

React.createElement(element, props, ...children)

例如,對于以下 JSX:

function App() {
  return (
    <div>
      <h1>Hello</h1>
    </div>
  )
}

是以下代碼的語法糖:

function App() {
  return React.createElement(
    "div",
    null,
    React.createElement("h1", null, "Hello")
  )
}

那如果在根上想要兩個父元素怎么辦? 就像上面的第一個例子一樣:

function App() {
  return (
    <div>Why</div>
    <div>Can I not do this?</div>
  )
}

這段 JSX 會編譯為:

function App() {
  return React.createElement("div", null, "Why")
  React.createElement("div", null, "Can I not do this?")
}

這里嘗試一次返回兩個內容,但這并不是一段有效的 JavaScript。因此,只能返回一個父元素,而該父元素可以有任意數量的子元素。要返回多個子元素,可以將它們作為參數傳遞給 createElement,如下所示:

return React.createElement(
  "h1",
  null,
  "Hello",
  "Hi",
  React.createElement("span", null, "Hello")
  // 其他子元素
)

其 JSX 表示為:

return (
  <h1>
    Hello Hi
    <span>Hello</span>
  </h1>
)

接下來,檢查一下之前的代碼塊:

function App() {
  return (
    <div>
      {isOpen && (
        <div>Why again</div>
        <div>Can I not do this</div>
      )}
    </div>
  )
}

這個有一個根父級 div,但仍然會報錯: isOpen 表達式中有多個父級。 為什么?

如果只使用一個 div 標簽:

function App() {
  return <div>{isOpen && <div>Why again</div>}</div>
}

這會編譯為:

function App() {
  return React.createElement(
    "div",
    null,
    isOpen && React.createElement("div", null, "Why again")
  )
}

isOpen 表達式是第一個 createElement 中的子級,該表達式使用邏輯 && 運算符將第二個 createElement 父級作為子級添加到第一個 createElement 中。

這意味著這段代碼有兩個父級:

function App() {
  return (
    <div>
      {isOpen && (
        <div>Why again</div>
        <div>Can I not do this</div>
      )}
    </div>
  )
}

這會編譯為:

function App() {
  return React.createElement(
    "div",
    null,
    isOpen
    && React.createElement("div", null, "Why again")
    React.createElement("div", null, "Can I not do this")
  )
}

這段代碼是錯誤的語法,因為在 && 運算符之后,嘗試返回兩個內容,而 JavaScript 只允許一次返回一個表達式。 返回的表達式應該有一個父表達式和多個的子表達式。

這就是為什么 JSX 表達式只能有一個父元素。

2、實現 JSX 解析器

先來看看最終要解析的 JSX 文件:

import * as MyLib from './MyLib.js'

export function Component() {
    let myRef = null
    let name = "Fernando"
    let myClass = "open"
    
  	return (
        <div className={myClass} ref={myRef}>
            <h1>Hello {name}!</h1>
        </div>
    )
}

console.log(Component())

如果在 React 中編寫這段代碼,會得到這樣的東西:

import * as React from 'react'

export function Component() {
    let myRef = null
    let name = "Fernando"
    let myClass = "open"
    
  	return (
        <div className={myClass} ref={myRef}>
            <h1>Hello {name}!</h1>
        </div>
    )
}

console.log(Component())

這里唯一改變的就是初始導入,接下來編寫 JSX 時,你就會明白為什么需要導入 React。

雖然解析本身需要一些工作,但其背后的邏輯實際上非常簡單。 React 官方文檔就展示了解析 JSX 的輸出。

圖片圖片

這里實際上是將每個 JSX 元素轉換為對React.createElement的調用。因此,需要導入React,即使并沒有直接使用它,一旦解析完成,生成的JavaScript代碼將使用到它。

React.createElement 方法的第一個屬性是要創建的元素的標簽名。第二個屬性是一個包含與正在創建的元素相關的所有屬性的對象,其余的屬性(可以有一個或多個)將成為此元素的直接子級(它們可以是純文本或其他元素)。

因此,實現 JSX 解析器的大致步驟總結如下:

  1. 捕獲 JavaScript 中的 JSX。
  2. 將其解析為可以遍歷和查詢的樹狀結構。
  3. 將該結構轉換為將代替 JSX 編寫的 JavaScript 代碼(文本)。
  4. 將步驟 3 的輸出保存到磁盤中,并保存為擴展名為 .js 的文件。

(1)從組件中提取并解析 JSX

第一步就是通過某種方式從組件中提取 JSX 并將其解析為樹狀結構。

我們需要做的第一件事是讀取 JSX 文件,然后使用正則表達式來捕獲 JSX 代碼。最后,就可以使用 HTML 解析器來解析它。

此時,我們關心的是結構,而不是 JSX 的實際功能。 因此,可以使用 Node 中的 fs 模塊和 node-html-parser 包來讀取文件。

該函數如下所示:

const JSX_STRING = /\(\s*(<.*)>\s*\)/gs

async function parseJSXFile(fname) {
    let content = await fs.promises.readFile(fname)
    let str = content.toString()

    let matches = JSX_STRING.exec(str)
    if(matches) {
        let HTML = matches[1] + ">"
        const root = parse(HTML)
        let translated = (translate(root.firstChild))
        str = str.replace(matches[1] + ">", translated)
        await fs.promises.writeFile("output.js", str)
    }
}

parseJSXFile 函數使用 RegExp 來查找函數中第一個組件的開始標簽。在第 10 行調用了解析函數,該函數返回一個根元素,其中 firstChild 是 JSX 中的根元素(在開始的例子中是 div 元素)。

現在有了樹狀結構,就可以將其轉換為代碼了。 為此,將調用 translate 函數。

(2)將 HTML 轉譯為 JS 代碼

由于處理的樹狀結構的深度有限,因此可以安全地使用遞歸來遍歷這棵樹。

該函數如下所示:

function translate(root) {
    if(Array.isArray(root) && root.length == 0) return

    let children = []
    if(root.childNodes.length > 0) {
        children = root.childNodes.map( child => translate(child) ).filter( c => c != null)
    }
    // 文本節點
    if(root.nodeType == 3) {
        if(root._rawText.trim() === "") return null
        return parseText(root._rawText)
        
    }
    let tagName = root.rawTagName

    let opts = getAttrs(root.rawAttrs)
   
    return `MyLib.createElement("${tagName}", ${replaceInterpolations(JSON.stringify(opts, replacer), true)}, ${children})`
    
}

首先,遍歷所有子項,并對它們調用 translate 函數。 如果子級為空,則該調用將返回 null,將在第 7 行過濾這些結果。

處理完子節點后,接下來看一下第 9 行,在其中對節點類型進行快速健全性檢查。如果類型為 3,則意味著這是一個文本節點,將返回解析后的文本。

為什么要調用 parseText 函數呢? 因為即使在文本節點內部,我們也需要在 {…} 中查找 JSX 表達式。 因此,如果需要,此函數將負責檢查并正確更改返回的字符串。

接下來,獲取標簽名稱(第 14 行),然后解析屬性(第 16 行)。 解析屬性意味著將獲取原始字符串并將其轉換為正確的 JSON。

最后,返回想要生成的代碼行(即使用正確的參數調用 createElement)。

注意,生成的代碼會從 MyLib 模塊調用 createElement 方法。這就是為什么在 JSX 文件內有 import * as MyLib from './MyLib.js' 的原因。

接下來就需要處理字符串來替換 JSX 表達式,無論是在文本節點還是每個元素的屬性對象內。

(3)解析表達式

在此實現中支持的 JSX 表達式類型是最簡單的一種。正如示例中看到的,可以在這些表達式中添加 JS 變量,它們將在最終輸出中保留為變量。

以下是執行此操作的函數:

const JSX_INTERPOLATION = /\{([a-zA-Z0-9]+)\}/gs

function parseText(txt) {
    let interpolations = txt.match(JSX_INTERPOLATION)
    if(!interpolations) {
        return txt
    } else {
        txt = replaceInterpolations(txt)
        return `"${txt}"`
    }
}

function replaceInterpolations(txt, isOnJSON = false) {
    let interpolations = null;

    while(interpolations = JSX_INTERPOLATION.exec(txt)) {
        if(isOnJSON) {
            txt = txt.replace(`"{${interpolations[1]}}"`, interpolations[1])
        } else {
            txt = txt.replace(`{${interpolations[1]}}`, `"+ ${interpolations[1]} +"`)
        }
    } 
    return txt
}

如果有插值(即大括號內的變量),就會調用replaceInterpolation函數,該函數會遍歷所有匹配的插值,并將它們替換為正確格式的字符串(本質上以在寫入JS文件時生成JS變量的方式保留變量名稱)。

我們也將這些函數與屬性對象一起使用。 由于在返回 JS 代碼時使用 JSON.stringify 方法,因此該函數會將所有值轉換為字符串。 因此,將解析 stringify 方法返回的字符串,并確保正確替換插值變量。

getAttrs 函數的實現如下:

function getAttrs(attrsStr) {
    if(attrsStr.trim().length == 0) return {}
    let objAttrs = {}
    let parts = attrsStr.split(" ")
    parts.forEach( p => {
        const [name, value] = p.split("=")
        console.log(name)
        console.log(value)
        objAttrs[name] = (value)
    })
    return objAttrs
}

(4)JavaScript 代碼

接下來看一下解析 JSX 文件所輸出的代碼:

import * as MyLib from './MyLib.js'

export function Component() {
    let myRef = null
    let name = "Fernando"
    let myClass = "open"
  
    return (
        MyLib.createElement("div", 
           {"className":myClass,"ref":myRef}, 
           MyLib.createElement( 
              "h1", 
              {}, 
              "Hello "+ name +"!"))
    )
}

console.log(Component())

這段代碼真正有趣的地方是生成的對 createElement 的調用。 可以看到它們是如何嵌套的,以及它們引用了在 JSX 文件中插回的變量。

如果執行這段代碼,輸出如下:

<div class="open" ref="null">
  <h1 >
  Hello Fernando!
  </h1>
</div>

那 createElement 方法是如何實現的呢?這里有一個簡化的版本:

function mapAttrName(name) {
    if(name == "className") return "class"
    return name
}

export function createElement(tag, opts, ...children) {
    return `<${tag} ${Object.keys(opts).map(oname => `${mapAttrName(oname)}="${opts[oname]}"`).join(" ")}>
     ${children.map( c => c)}
     </${tag}>
    `
}

本質上,這里使用標簽值創建一個包裝元素,為其添加屬性(如果有的話),最后,遍歷子列表(這是一個包含所有添加屬性的剩余屬性),在此過程中,將簡單地將這些值作為字符串返回(第 9 行)。

這樣,一個簡易的 JSX 解析器就完成了,下面是完整的代碼:

import * as fs from 'fs'
import { parse } from 'node-html-parser';


const JSX_STRING = /\(\s*(<.*)>\s*\)/gs
const JSX_INTERPOLATION = /\{([a-zA-Z0-9]+)\}/gs
const QUOTED_STRING = /["|'](.*)["|']/gs

function getAttrs(attrsStr) {
    if(attrsStr.trim().length == 0) return {}
    let objAttrs = {}
    let parts = attrsStr.split(" ")
    parts.forEach( p => {
        const [name, value] = p.split("=")
        console.log(name)
        console.log(value)
        objAttrs[name] = (value)
    })
    return objAttrs
}

function parseText(txt) {
    let interpolations = txt.match(JSX_INTERPOLATION)
    if(!interpolations) {
        console.log("no inerpolation found: ", txt)
        return txt
    } else {
        console.log("inerpolation found!", txt)
        txt = replaceInterpolations(txt)
        // interpolations.shift()
        // interpolations.forEach( v => {
        //     txt = txt.replace(`{${v}}`, `" + (${v}) + "`)
        // })
        return `"${txt}"`
    }
}

function replacer(k, v) {
    if(k) {
        let quoted = QUOTED_STRING.exec(v)
        if(quoted) {
            return parseText(quoted[1])
        }
        return (v)
    } else {
        return v
    }
}

function replaceInterpolations(txt, isOnJSON = false) {
    let interpolations = null;

    while(interpolations = JSX_INTERPOLATION.exec(txt)) {
        console.log("fixing interpolation for ", txt)
        console.log(interpolations)
        if(isOnJSON) {
            txt = txt.replace(`"{${interpolations[1]}}"`, interpolations[1])
        } else {
            txt = txt.replace(`{${interpolations[1]}}`, `"+ ${interpolations[1]} +"`)
        }
    } 
    return txt
}

function translate(root) {
    if(Array.isArray(root) && root.length == 0) return
    console.log("Current root: ")
    console.log(root)
    let children = []
    if(root.childNodes.length > 0) {
        children = root.childNodes.map( child => translate(child) ).filter( c => c != null)
    }
    if(root.nodeType == 3) { //Textnodes
        if(root._rawText.trim() === "") return null
        return parseText(root._rawText)
        
    }
    let tagName = root.rawTagName

    let opts = getAttrs(root.rawAttrs)
    console.log("Opts: ")
    console.log(opts)
    console.log(JSON.stringify(opts))

    return `MyLib.createElement("${tagName}", ${replaceInterpolations(JSON.stringify(opts, replacer), true)}, ${children})`
    
}

async function parseJSXFile(fname) {
    let content = await fs.promises.readFile(fname)
    let str = content.toString()

    let matches = JSX_STRING.exec(str)
    if(matches) {
        let HTML = matches[1] + ">"
console.log("parsed html")
console.log(HTML)
        const root = parse(HTML)
        //console.log(root.firstChild)
        let translated = (translate(root.firstChild))
        str = str.replace(matches[1] + ">", translated)
        await fs.promises.writeFile("output.js", str)
    }

}

(async () => {
    await parseJSXFile("./file.jsx")
})()

責任編輯:武曉燕 來源: 前端充電寶
相關推薦

2019-01-18 12:39:45

云計算PaaS公有云

2022-11-08 15:14:17

MyBatis插件

2022-10-20 11:00:52

SQL解析器

2023-03-20 09:48:23

ReactJSX

2022-06-02 09:09:27

前端React低代碼編輯器

2018-09-14 17:16:22

云計算軟件計算機網絡

2017-02-14 10:20:43

Java Class解析器

2022-06-28 08:17:10

JSON性能反射

2023-12-30 13:33:36

Python解析器JSON

2014-07-22 13:09:21

android

2023-10-24 16:44:24

RubyDNS

2019-07-05 08:39:39

GoSQL解析器

2022-09-01 10:46:02

前端組件庫

2024-11-27 16:25:54

JVMJIT編譯機制

2023-11-23 15:06:36

PythonHTTP服務器

2025-09-18 09:54:55

ReactNodeJSX.ElemenReactEleme

2024-09-18 08:10:06

2024-11-18 17:31:27

2024-10-05 00:00:06

HTTP請求處理容器

2016-12-08 15:36:59

HashMap數據結構hash函數
點贊
收藏

51CTO技術棧公眾號

在线观看一区二区三区视频| 小说区图片区图片区另类灬| 国产精品suv一区二区69| 粉嫩一区二区三区四区公司1| 婷婷综合五月天| 亚洲成人网上| 六月丁香色婷婷| 免费xxxx性欧美18vr| 欧美人成在线视频| 91精彩刺激对白露脸偷拍| 91久久青草| 一本大道久久a久久精二百| 在线免费一区| 天堂资源中文在线| 国产在线一区二区| 青青青国产精品一区二区| 天天看天天摸天天操| 琪琪久久久久日韩精品| 欧美一区二区黄| 丁香婷婷激情网| 美女网站视频在线| 国产精品国产馆在线真实露脸 | 成人在线精品| 色综合久久中文综合久久牛| 久久亚洲国产成人精品无码区| 国产区在线视频| 国产又大又黄又爽| 国产麻豆精品久久| 欧美精品一区二区三区高清aⅴ| 亚洲天堂av线| 女海盗2成人h版中文字幕| 综合久久久久久| 欧美二级三级| 刘亦菲久久免费一区二区| 国产一区二区在线视频| 国产女精品视频网站免费| 天堂а√在线中文在线新版 | 国产又粗又猛又黄视频| 影音先锋亚洲一区| 欧美床上激情在线观看| 欧美日韩生活片| 免费国产自久久久久三四区久久| 欧美精品一区二区不卡| 夜夜爽久久精品91| 亚洲伊人精品酒店| 欧美日韩视频第一区| 人妻丰满熟妇av无码区app| 日本在线高清| 午夜私人影院久久久久| 国产成人一区二区三区别| 在线播放免费av| 亚洲手机成人高清视频| 亚洲在线不卡| 免费在线观看av网站| 国产女人aaa级久久久级| 欧美日韩一区二区三区在线视频 | 成人性视频免费网站| 97人人澡人人爽| www.成人在线观看| 国产激情视频一区二区在线观看 | 免费av一区| 亚洲一区av在线播放| 欧美老女人性生活视频| 91在线播放观看| 国产精品久久久久av蜜臀| 精品国产一区二区三区久久久蜜月| 欧美性猛交乱大交| 亚洲精品在线播放| 亚洲精品福利免费在线观看| 熟女丰满老熟女熟妇| 国产999精品在线观看| 日韩欧美中文字幕制服| 国产日韩视频一区| 亚州综合一区| 中文字幕欧美日韩va免费视频| 九九热久久免费视频| 97在线精品| 久久99亚洲精品| 免费观看一区二区三区毛片| 久久久亚洲人| 成人精品一区二区三区| 亚洲国产精品二区| 久久视频一区二区| 中文字幕一区二区三区最新 | 视频一区欧美| 日韩日本欧美亚洲| 九九九国产视频| 久久久亚洲人| 5g国产欧美日韩视频| 亚洲 国产 欧美 日韩| 亚洲国产成人在线| 国产激情片在线观看| 国偷自产一区二区免费视频| 欧美日韩一区精品| 午夜不卡久久精品无码免费| 国产欧美一区| 欧美激情在线播放| 波多野结衣在线观看视频| 国模无码大尺度一区二区三区| 国产精品国产三级欧美二区| 日韩有码电影| 亚洲色图制服丝袜| 精品视频无码一区二区三区| 国产精品一区三区在线观看| 日韩高清不卡av| 国产精品 欧美激情| 性感少妇一区| 99c视频在线| www.中文字幕久久久| 日本综合在线| 中文字幕第一页久久| 日韩精品综合在线| 国产精品第一| 亚洲激情自拍图| 情侣偷拍对白清晰饥渴难耐| 一区二区黄色| 91夜夜未满十八勿入爽爽影院| 日韩av资源站| 亚洲午夜在线视频| 色噜噜狠狠一区二区三区狼国成人| 六月丁香久久丫| 不卡av在线播放| 在线播放成人av| 久久这里只有精品首页| 人妻激情另类乱人伦人妻| 国模私拍国内精品国内av| 精品爽片免费看久久| 免费又黄又爽又色的视频| 久久超碰97中文字幕| 日本高清久久一区二区三区| xxx.xxx欧美| 欧美一级免费大片| 五月天色婷婷丁香| 麻豆国产91在线播放| 日本免费一区二区三区| 欧亚在线中文字幕免费| 亚洲精品在线免费观看视频| 中文字幕在线观看成人| 九一久久久久久| 一区二区不卡在线观看| 欧美综合社区国产| 在线观看亚洲视频| 免费又黄又爽又猛大片午夜| 久久影院视频免费| 动漫av网站免费观看| 网友自拍区视频精品| 2018中文字幕一区二区三区| 色哟哟中文字幕| 亚洲国产精品天堂| 99久久久无码国产精品性波多| 欧美在线免费一级片| 亚洲一区二区三区成人在线视频精品| 乱人伦中文视频在线| 欧美日韩在线免费视频| 蜜桃av免费观看| 精品一区二区在线看| 中文字幕欧美日韩一区二区三区| 伊人久久精品| 天涯成人国产亚洲精品一区av| 日本免费久久高清视频| 国产99久一区二区三区a片| 国产精品成人网| 黄色片子免费看| 欧美激情亚洲| 国产一区二区精品在线| 极品在线视频| 亚洲片国产一区一级在线观看| 黄色污污网站在线观看| 国产精品每日更新在线播放网址| 色戒在线免费观看| 亚洲天天综合| 国产精品二区三区| 惠美惠精品网| 尤物精品国产第一福利三区| 亚洲一区二区三区高清视频| 亚洲欧美国产毛片在线| 中文字幕天堂av| 久久福利精品| 在线国产精品网| 成人午夜网址| 国产精品电影在线观看| 黄色网页在线看| 精品成人在线观看| 波多野结衣激情视频| 亚洲欧美日韩电影| 久久久久国产精品区片区无码| 日韩经典一区二区| 欧美 日韩 国产精品| 九色成人国产蝌蚪91| 成人免费大片黄在线播放| 91超碰在线播放| 伊人伊成久久人综合网小说| 国产精品久久久久久久一区二区| 亚洲高清免费视频| 夜夜春很很躁夜夜躁| 丁香亚洲综合激情啪啪综合| 黄色高清无遮挡| 国产一区二区三区四区老人| 日本免费高清一区二区| 亚洲视频国产| 国产精品自产拍在线观| 性xxxxfreexxxxx欧美丶| www.日韩系列| 久香视频在线观看| 精品国产3级a| 国产又黄又爽视频| 一本高清dvd不卡在线观看| 外国一级黄色片| 国产欧美日韩另类一区| 日韩成人av一区二区| 精品亚洲国内自在自线福利| 人妻熟女一二三区夜夜爱| 超碰一区二区| 久久在线免费观看| 国产av人人夜夜澡人人爽| 国产精品v日韩精品v欧美精品网站 | 久久久久久毛片免费看| 成人www视频在线观看| 在线天堂新版最新版在线8| 久久6精品影院| 免费黄色网址在线观看| 一区二区三区视频观看| 少妇又色又爽又黄的视频| 欧美久久久久久蜜桃| 人人爽人人爽人人片av| 婷婷成人激情在线网| 国产在线视频二区| 亚洲欧美日韩久久| 三级黄色片在线观看| 久久久激情视频| 亚洲最大成人网站| 91色|porny| 亚洲AV无码国产精品| 不卡一区二区三区四区| 性一交一黄一片| 国产在线不卡一卡二卡三卡四卡| 国产一区二区在线免费播放| 久久青草久久| 国产淫片av片久久久久久| 国产亚洲永久域名| 妞干网在线视频观看| 激情视频一区| 免费av手机在线观看| 亚洲精品少妇| 国产精品无码人妻一区二区在线| 欧美视频网站| 18黄暴禁片在线观看| 亚洲第一伊人| 久久综合色视频| 国产亚洲激情| 久久久久久久久久久久久久国产| 亚洲影音一区| 成年人视频在线免费| 日韩电影在线一区二区三区| 国产又黄又猛又粗| 美国十次了思思久久精品导航| 亚洲免费999| 国产高清无密码一区二区三区| 免费看的av网站| 风间由美一区二区三区在线观看| 特级特黄刘亦菲aaa级| 9人人澡人人爽人人精品| 国产又黄又粗又猛又爽的视频| 91婷婷韩国欧美一区二区| 性猛交娇小69hd| www国产精品| 国产精品亚洲网站| 成人亚洲精品| 国产欧美在线一区二区| 亚洲成a人片77777在线播放| 日韩视频在线播放| 国产精品99视频| 97久久国产亚洲精品超碰热| 亚洲片区在线| www.色偷偷.com| 国内精品第一页| 污污内射在线观看一区二区少妇 | 日本aⅴ亚洲精品中文乱码| 高清av免费看| 福利一区二区在线观看| 扒开jk护士狂揉免费| 国产视频一区二区在线| 亚洲伦理一区二区三区| 亚洲国产精品一区二区久久| 国产一区二区视频免费| 欧美一区二区私人影院日本| 五月婷婷六月激情| 中文字幕一区二区精品| 欧美6一10sex性hd| 国产不卡精品视男人的天堂| 精品国产三区在线| 欧美不卡福利| 一级欧洲+日本+国产 | 在线中文字幕播放| 成人激情视频在线观看| 老汉色老汉首页av亚洲| 一区二区三区av在线| 99热免费精品在线观看| 在线播放av中文字幕| 91日韩一区二区三区| 日本午夜在线观看| 色综合天天性综合| 亚洲成人久久精品| 中文字幕日韩精品有码视频| av老司机免费在线| 成人久久久久久| 最新国产一区| 黄色一级片黄色| 理论片日本一区| 精品人妻少妇嫩草av无码| 一区二区在线观看视频| 伊人精品在线视频| 亚洲激情在线观看视频免费| 成人高清免费在线| 国产精品极品美女在线观看免费| 精品资源在线| 4444亚洲人成无码网在线观看| 男人操女人的视频在线观看欧美| 久久人人妻人人人人妻性色av| 亚洲免费在线电影| 中文字幕自拍偷拍| 亚洲欧洲日产国产网站| 污污网站在线看| 91免费欧美精品| 香蕉污视频在线观看| 欧美tickling挠脚心丨vk| 欧美激情免费| 国产精品久久久一区| 婷婷成人在线| 国产av天堂无码一区二区三区| 国产麻豆精品久久一二三| 狂野欧美性猛交| 欧美性色欧美a在线播放| 色猫av在线| 97精品久久久| 精品无人区一区二区| 男人天堂手机在线视频| 国产成人av电影在线播放| 最新av电影网站| 欧美日韩在线播| 亚洲s色大片| 国产精品成人免费视频| sdde在线播放一区二区| www.国产区| 日本一区二区三区dvd视频在线| 无码人妻av免费一区二区三区 | 精品1卡二卡三卡四卡老狼| 亚洲色图清纯唯美| 国产情侣激情自拍| 久久成人国产精品| 精品视频一二| 男女裸体影院高潮| 国产麻豆精品视频| 久久久久久欧美精品se一二三四| 日韩欧美综合在线| 伦理av在线| 好看的日韩精品| 欧美专区在线| 丁香激情五月少妇| 欧美日本国产视频| av网址在线播放| 91视频免费进入| 在线视频观看日韩| 在线观看日韩精品视频| 狠狠躁夜夜躁人人爽超碰91| 欧美色18zzzzxxxxx| 国产精品久久久久久久久久尿| 不卡一区2区| 亚洲自拍第三页| 亚洲一区二区三区国产| 青梅竹马是消防员在线| 国产精品三级久久久久久电影| 欧美激情欧美| 蜜桃视频无码区在线观看| 欧美日韩国产一区在线| 九色蝌蚪在线| 亚洲在线观看视频网站| 中日韩男男gay无套| 国产成人免费观看网站| 欧美一区二区私人影院日本| 日韩伦理在线一区| 天堂精品视频| 国产一区二区在线免费观看| 日韩特黄一级片| 日韩在线观看你懂的| 99国产精品久久一区二区三区| 91黄色小网站| 亚洲色图都市小说| 少妇激情av一区二区| 国产日韩欧美在线观看| 亚洲美洲欧洲综合国产一区| 中文字幕人妻一区二区三区在线视频| 欧美精品丝袜久久久中文字幕| 天天干天天干天天操| 久久久久久久久一| 日本韩国欧美中文字幕| 久久精品视频在线播放| 欧美尿孔扩张虐视频| 天堂av8在线| 一本久久a久久免费精品不卡|