国产无遮挡裸体免费直播视频,久久精品国产蜜臀av,动漫在线视频一区二区,欧亚日韩一区二区三区,久艹在线 免费视频,国产精品美女网站免费,正在播放 97超级视频在线观看,斗破苍穹年番在线观看免费,51最新乱码中文字幕

剖析CocosCreator新資源管理系統(tǒng)

 更新時(shí)間:2021年04月16日 09:32:47   作者:深圳-寶爺  
這篇文章主要介紹了CocosCreator新資源管理系統(tǒng),從v2.4開始,Creator使用AssetBundle完全重構(gòu)了資源底層,提供了更加靈活強(qiáng)大的資源管理方式,也解決了之前版本資源管理的痛點(diǎn)(資源依賴與引用),本文將帶你深入了解Creator的新資源底層。

1.資源與構(gòu)建

1.1 creator資源文件基礎(chǔ)

在了解引擎如何解析、加載資源之前,我們先來了解一下這些資源文件(圖片、Prefab、動畫等)的規(guī)則,在creator項(xiàng)目目錄下有幾個(gè)與資源相關(guān)的目錄:

  • assets 所有資源的總目錄,對應(yīng)creator編輯器的資源管理器
  • library 本地資源庫,預(yù)覽項(xiàng)目時(shí)使用的目錄
  • build 構(gòu)建后的項(xiàng)目默認(rèn)目錄

在assets目錄下,creator會為每個(gè)資源文件和目錄生成一個(gè)同名的.meta文件,meta文件是一個(gè)json文件,記錄了資源的版本、uuid以及各種自定義的信息(在編輯器的屬性檢查器中設(shè)置),比如prefab的meta文件,就記錄了我們可以在編輯器修改的optimizationPolicy和asyncLoadAssets等屬性。

{
  "ver": "1.2.7",
  "uuid": "a8accd2e-6622-4c31-8a1e-4db5f2b568b5",
  "optimizationPolicy": "AUTO",     // prefab創(chuàng)建優(yōu)化策略
  "asyncLoadAssets": false,         // 是否延遲加載
  "readonly": false,
  "subMetas": {}
}

在library目錄下的imports目錄,資源文件名會被轉(zhuǎn)換成uuid,并取uuid前2個(gè)字符進(jìn)行目錄分組存放,creator會將所有資源的uuid到assets目錄的映射關(guān)系,以及資源和meta的最后更新時(shí)間戳放到一個(gè)名為uuid-to-mtime.json的文件中,如下所示。

{
  "9836134e-b892-4283-b6b2-78b5acf3ed45": {
    "asset": 1594351233259,
    "meta": 1594351616611,
    "relativePath": "effects"
  },
  "430eccbf-bf2c-4e6e-8c0c-884bbb487f32": {
    "asset": 1594351233254,
    "meta": 1594351616643,
    "relativePath": "effects\\__builtin-editor-gizmo-line.effect"
  },
  ...
}

與assets目錄下的資源相比,library目錄下的資源合并了meta文件的信息。文件目錄則只在uuid-to-mtime.json中記錄,library目錄并沒有為目錄生成任何東西。

1.2 資源構(gòu)建

在項(xiàng)目構(gòu)建之后,資源會從library目錄下移動到構(gòu)建輸出的build目錄中,基本只會導(dǎo)出參與構(gòu)建的場景和resources目錄下的資源,及其引用到的資源。腳本資源會由多個(gè)js腳本合并為一個(gè)js,各種json文件也會按照特定的規(guī)則進(jìn)行打包。我們可以在Bundle的配置界面和項(xiàng)目的構(gòu)建界面為Bundle和項(xiàng)目設(shè)置

1.2.1 圖片、圖集、自動圖集

  • https://docs.cocos.com/creator/manual/zh/asset-workflow/sprite.html
  • https://docs.cocos.com/creator/manual/zh/asset-workflow/atlas.html
  • https://docs.cocos.com/creator/manual/zh/asset-workflow/auto-atlas.html

導(dǎo)入編輯器的每張圖片都會對應(yīng)生成一個(gè)json文件,用于描述Texture的信息,如下所示,默認(rèn)情況下項(xiàng)目中所有的Texture2D的json文件會被壓縮成一個(gè),如果選擇無壓縮,則每個(gè)圖片都會生成一個(gè)Texture2D的json文件。

{
  "__type__": "cc.Texture2D",
  "content": "0,9729,9729,33071,33071,0,0,1"
}

如果將紋理的Type屬性設(shè)置為Sprite,Creator還會自動生成了SpriteFrame類型的json文件。
圖集資源除了圖片外,還對應(yīng)一個(gè)圖集json,這個(gè)json包含了cc.SpriteAtlas信息,以及每個(gè)碎圖的SpriteFrame信息
自動圖集在默認(rèn)情況下只包含了cc.SpriteAtlas信息,在勾選內(nèi)聯(lián)所有SpriteFrame的情況下,會合并所有SpriteFrame

1.2.2 Prefab與場景

  • https://docs.cocos.com/creator/manual/zh/asset-workflow/prefab.html
  • https://docs.cocos.com/creator/manual/zh/asset-workflow/scene-managing.html

場景資源與Prefab資源非常類似,都是一個(gè)描述了所有節(jié)點(diǎn)、組件等信息的json文件,在勾選內(nèi)聯(lián)所有SpriteFrame的情況下,Prefab引用到的SpriteFrame會被合并到prefab所在的json文件中,如果一個(gè)SpriteFrame被多個(gè)prefab引用,那么每個(gè)prefab的json文件都會包含該SpriteFrame的信息。而在沒有勾選內(nèi)聯(lián)所有SpriteFrame的情況下,SpriteFrame會是單獨(dú)的json文件。

1.2.3 資源文件合并規(guī)則

當(dāng)Creator將多個(gè)資源合并到一個(gè)json文件中,我們可以在config.json中的packs字段找到被打包的資源信息,一個(gè)資源有可能被重復(fù)打包到多個(gè)json中。下面舉一個(gè)例子,展示在不同的選項(xiàng)下,creator的構(gòu)建規(guī)則:

  • a.png 一個(gè)單獨(dú)的Sprite類型圖片
  • dir/b.png、c.png、AutoAtlas dir目錄下包含2張圖片,以及一個(gè)AutoAtlas
  • d.png、d.plist 普通圖集
  • e.prefab 引用了SpriteFrame a和b的prefab
  • f.prefab 引用了SpriteFrame b的prefab

下面是按不同規(guī)則構(gòu)建后的文件,可以看到,無壓縮的情況下生成的文件數(shù)量是最多的,不內(nèi)聯(lián)的文件會比內(nèi)聯(lián)多,但內(nèi)聯(lián)可能會導(dǎo)致同一個(gè)文件被重復(fù)包含,比如e和f這兩個(gè)Prefab都引用了同一個(gè)圖片,這個(gè)圖片的SpriteFrame.json會被重復(fù)包含,合并成一個(gè)json則只會生成一個(gè)文件。

資源文件 無壓縮 默認(rèn)(不內(nèi)聯(lián)) 默認(rèn)(內(nèi)聯(lián)) 合并json
a.png a.texture.json + a.spriteframe.json a.spriteframe.json
./dir/b.png b.texture.json + b.spriteframe.json b.spriteframe.json
./dir/c.png c.texture.json + c.spriteframe.json c.spriteframe.json c.spriteframe.json
./dir/AutoAtlas autoatlas.json autoatlas.json autoatlas.json
d.png d.texture.json + d.spriteframe.json d.spriteframe.json d.spriteframe.json
d.plist d.plist.json d.plist.json d.plist.json
e.prefab e.prefab.json e.prefab.json e.prefab.json(pack a+b)
f.prefab f.prefab.json f.prefab.json f.prefab.json(pack b)
g.allTexture.json g.allTexture.json all.json

默認(rèn)選項(xiàng)在絕大多數(shù)情況下都是一個(gè)不錯(cuò)的選擇,如果是web平臺,建議勾選內(nèi)聯(lián)所有SpriteFrame這可以減少網(wǎng)絡(luò)io,提高性能,而原生平臺不建議勾選,這可能會增加包體大小以及熱更時(shí)要下載的內(nèi)容。對于一些緊湊的Bundle(比如加載該Bundle就需要用到里面所有的資源),我們可以配置為合并所有的json。

2. 理解與使用 Asset Bundle

2.1 創(chuàng)建Bundle

Asset Bundle是creator 2.4之后的資源管理方案,簡單地說就是通過目錄來對資源進(jìn)行規(guī)劃,按照項(xiàng)目的需求將各種資源放到不同的目錄下,并將目錄配置成Asset Bundle。能夠起到以下作用:

  • 加快游戲啟動時(shí)間
  • 減小首包體積
  • 跨項(xiàng)目復(fù)用資源
  • 方便實(shí)現(xiàn)子游戲
  • 以Bundle為單位的熱更新

Asset Bundle的創(chuàng)建非常簡單,只要在目錄的屬性檢查器中勾選配置為bundle即可,其中的選項(xiàng)官方文檔都有比較詳細(xì)的介紹。

其中關(guān)于壓縮的理解,文檔并沒有詳細(xì)的描述,這里的壓縮指的并不是zip之類的壓縮,而是通過packAssets的方式,把多個(gè)資源的json文件合并到一個(gè),達(dá)到減少io的目的

在選項(xiàng)上打勾非常簡單,真正的關(guān)鍵在于如何規(guī)劃Bundle,規(guī)劃的原則在于減少包體、加速啟動以及資源復(fù)用。根據(jù)游戲的模塊來規(guī)劃資源是比較不錯(cuò)的選擇,比如按子游戲、關(guān)卡副本、或者系統(tǒng)功能來規(guī)劃。

Bundle會自動將文件夾下的資源,以及文件夾中引用到的其它文件夾下的資源打包(如果這些資源不是在其它Bundle中),如果我們按照模塊來規(guī)劃資源,很容易出現(xiàn)多個(gè)Bundle共用了某個(gè)資源的情況??梢詫⒐操Y源提取到一個(gè)Bundle中,或者設(shè)置某個(gè)Bundle有較高的優(yōu)先級,構(gòu)建Bundle的依賴關(guān)系,否則這些資源會同時(shí)放到多個(gè)Bundle中(如果是本地Bundle,這會導(dǎo)致包體變大)。

2.2 使用Bundle

  • 關(guān)于加載資源 https://docs.cocos.com/creator/manual/zh/scripting/load-assets.html
  • 關(guān)于釋放資源 https://docs.cocos.com/creator/manual/zh/asset-manager/release-manager.html

Bundle的使用也非常簡單,如果是resources目錄下的資源,可以直接使用cc.resources.load來加載

cc.resources.load("test assets/prefab", function (err, prefab) {
    var newNode = cc.instantiate(prefab);
    cc.director.getScene().addChild(newNode);
});

如果是其它自定義Bundle(本地Bundle或遠(yuǎn)程Bundle都可以用Bundle名加載),可以使用cc.assetManager.loadBundle來加載Bundle,然后使用加載后的Bundle對象,來加載Bundle中的資源。對于原生平臺,如果Bundle被配置為遠(yuǎn)程包,在構(gòu)建時(shí)需要在構(gòu)建發(fā)布面板中填寫資源服務(wù)器地址。

cc.assetManager.loadBundle('01_graphics', (err, bundle) => {
    bundle.load('xxx');
});

原生或小游戲平臺下,我們還可以這樣使用Bundle:

  • 如果要加載其它項(xiàng)目的遠(yuǎn)程Bundle,則需要使用url的方式加載(其它項(xiàng)目指另一個(gè)cocos工程)
  • 如果希望自己管理Bundle的下載和緩存,可以放到本地可寫路徑,并傳入路徑來加載這些Bundle
// 當(dāng)復(fù)用其他項(xiàng)目的 Asset Bundle 時(shí)
cc.assetManager.loadBundle('https://othergame.com/remote/01_graphics', (err, bundle) => {
    bundle.load('xxx');
});

// 原生平臺
cc.assetManager.loadBundle(jsb.fileUtils.getWritablePath() + '/pathToBundle/bundleName', (err, bundle) => {
    // ...
});

// 微信小游戲平臺
cc.assetManager.loadBundle(wx.env.USER_DATA_PATH + '/pathToBundle/bundleName', (err, bundle) => {
    // ...
});

其它注意項(xiàng):

  • 加載Bundle僅僅只是加載了Bundle的配置和腳本而已,Bundle中的其它資源還需要另外加載
  • 目前原生的Bundle并不支持zip打包,遠(yuǎn)程包下載方式為逐文件下載,好處是操作簡單,更新方便,壞處是io多,流量消耗大
  • 不同Bundle下的腳本文件不要重名
  • 一個(gè)Bundle A依賴另一個(gè)Bundle B,如果B沒有被加載,加載A時(shí)并不會自動加載B,而是在加載A中依賴B的那個(gè)資源時(shí)報(bào)錯(cuò)

3. 新資源框架剖析

v2.4重構(gòu)后的新框架代碼更加簡潔清晰,我們可以先從宏觀角度了解一下整個(gè)資源框架,資源管線是整個(gè)框架最核心的部分,它規(guī)范了整個(gè)資源加載的流程,并支持對管線進(jìn)行自定義。

公共文件

  • helper.js 定義了一堆公共函數(shù),如decodeUuid、getUuidFromURL、getUrlWithUuid等等
  • utilities.js 定義了一堆公共函數(shù),如getDepends、forEach、parseLoadResArgs等等
  • deserialize.js 定義了deserialize方法,將json對象反序列化為Asset對象,并設(shè)置其__depends__屬性
  • depend-util.js 控制資源的依賴列表,每個(gè)資源的所有依賴都放在_depends成員變量中
  • cache.js 通用緩存類,封裝了一個(gè)簡易的鍵值對容器
  • shared.js 定義了一些全局對象,主要是Cache和Pipeline對象,如加載好的assets、下載完的files以及bundles等

Bundle部分

  • config.js bundle的配置對象,負(fù)責(zé)解析bundle的config文件
  • bundle.js bundle類,封裝了config以及加載卸載bundle內(nèi)資源的相關(guān)接口
  • builtins.js 內(nèi)建bundle資源的封裝,可以通過 cc.assetManager.builtins 訪問

管線部分

CCAssetManager.js 管理管線,提供統(tǒng)一的加載卸載接口

管線框架

  • pipeline.js 實(shí)現(xiàn)了管線的管道組合以及流轉(zhuǎn)等基本功能
  • task.js 定義了一個(gè)任務(wù)的基本屬性,并提供了簡單的任務(wù)池功能
  • request-item.js 定義了一個(gè)資源下載項(xiàng)的基本屬性,一個(gè)任務(wù)可能會生成多個(gè)下載項(xiàng)

預(yù)處理管線

  • urlTransformer.js parse將請求參數(shù)轉(zhuǎn)換成RequestItem對象(并查詢相關(guān)的資源配置),combine負(fù)責(zé)轉(zhuǎn)換真正的url
  • preprocess.js 過濾出需要進(jìn)行url轉(zhuǎn)換的資源,并調(diào)用transformPipeline

下載管線

  • download-dom-audio.js 提供下載音效的方法,使用audio標(biāo)簽進(jìn)行下載
  • download-dom-image.js 提供下載圖片的方法,使用Image標(biāo)簽進(jìn)行下載
  • download-file.js 提供下載文件的方法,使用XMLHttpRequest進(jìn)行下載
  • download-script.js 提供下載腳本的方法,使用script標(biāo)簽進(jìn)行下載
  • downloader.js 支持下載所有格式的下載器,支持并發(fā)控制、失敗重試

解析管線

  • factory.js 創(chuàng)建Bundle、Asset、Texture2D等對象的工廠
  • fetch.js 調(diào)用packManager下載資源,并解析依賴
  • parser.js 對下載完成的文件進(jìn)行解析

其它

  • releaseManager.js 提供資源釋放接口、負(fù)責(zé)釋放依賴資源以及場景切換時(shí)的資源釋放
  • cache-manager.d.ts 在非WEB平臺上,用于管理所有從服務(wù)器上下載下來的緩存
  • pack-manager.js 處理打包資源,包括拆包,加載,緩存等等

3.1 加載管線

creator使用管線(pipeline)來處理整個(gè)資源加載的流程,這樣的好處是解耦了資源處理的流程,將每一個(gè)步驟獨(dú)立成一個(gè)單獨(dú)的管道,管道可以很方便地進(jìn)行復(fù)用和組合,并且方便了我們自定義整個(gè)加載流程,我們可以創(chuàng)建一些自己的管道,加入到管線中,比如資源加密。

AssetManager內(nèi)置了3條管線,普通的加載管線、預(yù)加載、以及資源路徑轉(zhuǎn)換管線,最后這條管線是為前面兩條管線服務(wù)的。

// 正常加載
this.pipeline = pipeline.append(preprocess).append(load);
// 預(yù)加載
this.fetchPipeline = fetchPipeline.append(preprocess).append(fetch);
// 轉(zhuǎn)換資源路徑
this.transformPipeline = transformPipeline.append(parse).append(combine);

3.1.1 啟動加載管線【加載接口】

接下來我們看一下一個(gè)普通的資源是如何加載的,比如最簡單的cc.resource.load,在bundle.load方法中,調(diào)用了cc.assetManager.loadAny,在loadAny方法中,創(chuàng)建了一個(gè)新的任務(wù),并調(diào)用正常加載管線pipeline的async方法執(zhí)行任務(wù)。

注意要加載的資源路徑,被放到了task.input中、options是一個(gè)對象,對象包含了type、bundle和__requestType__等字段

// bundle類的load方法
load (paths, type, onProgress, onComplete) {
  var { type, onProgress, onComplete } = parseLoadResArgs(type, onProgress, onComplete);
  cc.assetManager.loadAny(paths, { __requestType__: RequestType.PATH, type: type, bundle: this.name }, onProgress, onComplete);
},

// assetManager的loadAny方法
loadAny (requests, options, onProgress, onComplete) {
  var { options, onProgress, onComplete } = parseParameters(options, onProgress, onComplete);
  
  options.preset = options.preset || 'default';
  let task = new Task({input: requests, onProgress, onComplete: asyncify(onComplete), options});
  pipeline.async(task);
},

pipeline由兩部分組成 preprocess 和 load。preprocess 由以下管線組成 preprocess、transformPipeline { parse、combine },preprocess實(shí)際上只創(chuàng)建了一個(gè)子任務(wù),然后交由transformPipeline執(zhí)行。對于加載一個(gè)普通的資源,子任務(wù)的input和options與父任務(wù)相同。

let subTask = Task.create({input: task.input, options: subOptions});
task.output = task.source = transformPipeline.sync(subTask);

3.1.2 transformPipeline管線【準(zhǔn)備階段】

transformPipeline由parse和combine兩個(gè)管線組成,parse的職責(zé)是為每個(gè)要加載的資源生成RequestItem對象并初始化其資源信息(AssetInfo、uuid、config等):

先將input轉(zhuǎn)換成數(shù)組進(jìn)行遍歷,如果是批量加載資源,每個(gè)加載項(xiàng)都會生成RequestItem

如果輸入的item是object,則先將options拷貝到item身上(實(shí)際上每個(gè)item都會是object,如果是string的話,第一步就先轉(zhuǎn)換成object了)

  • 對于UUID類型的item,先檢查bundle,并從bundle中提取AssetInfo,對于redirect類型的資源,則從其依賴的bundle中獲取AssetInfo,找不到bundle就報(bào)錯(cuò)
  • PATH類型和SCENE類型與UUID類型的處理基本類似,都是要拿到資源的詳細(xì)信息
  • DIR類型會從bundle中取出指定路徑的信息,然后批量追加到input尾部(額外生成加載項(xiàng))
  • URL類型是遠(yuǎn)程資源類型,無需特殊處理
function parse (task) {
    // 將input轉(zhuǎn)換成數(shù)組
    var input = task.input, options = task.options;
    input = Array.isArray(input) ? input : [ input ];

    task.output = [];
    for (var i = 0; i < input.length; i ++ ) {
        var item = input[i];
        var out = RequestItem.create();
        if (typeof item === 'string') {
            // 先創(chuàng)建object
            item = Object.create(null);
            item[options.__requestType__ || RequestType.UUID] = input[i];
        }
        if (typeof item === 'object') {
            // local options will overlap glabal options
            // 將options的屬性復(fù)制到item身上,addon會復(fù)制options上有,而item沒有的屬性
            cc.js.addon(item, options);
            if (item.preset) {
                cc.js.addon(item, cc.assetManager.presets[item.preset]);
            }
            for (var key in item) {
                switch (key) {
                    // uuid類型資源,從bundle中取出該資源的詳細(xì)信息
                    case RequestType.UUID: 
                        var uuid = out.uuid = decodeUuid(item.uuid);
                        if (bundles.has(item.bundle)) {
                            var config = bundles.get(item.bundle)._config;
                            var info = config.getAssetInfo(uuid);
                            if (info && info.redirect) {
                                if (!bundles.has(info.redirect)) throw new Error(`Please load bundle ${info.redirect} first`);
                                config = bundles.get(info.redirect)._config;
                                info = config.getAssetInfo(uuid);
                            }
                            out.config = config;
                            out.info = info;
                        }
                        out.ext = item.ext || '.json';
                        break;
                    case '__requestType__':
                    case 'ext': 
                    case 'bundle':
                    case 'preset':
                    case 'type': break;
                    case RequestType.DIR: 
                        // 解包后動態(tài)添加到input列表尾部,后續(xù)的循環(huán)會自動parse這些資源
                        if (bundles.has(item.bundle)) {
                            var infos = [];
                            bundles.get(item.bundle)._config.getDirWithPath(item.dir, item.type, infos);
                            for (let i = 0, l = infos.length; i < l; i++) {
                                var info = infos[i];
                                input.push({uuid: info.uuid, __isNative__: false, ext: '.json', bundle: item.bundle});
                            }
                        }
                        out.recycle();
                        out = null;
                        break;
                    case RequestType.PATH: 
                        // PATH類型的資源根據(jù)路徑和type取出該資源的詳細(xì)信息
                        if (bundles.has(item.bundle)) {
                            var config = bundles.get(item.bundle)._config;
                            var info = config.getInfoWithPath(item.path, item.type);
                            
                            if (info && info.redirect) {
                                if (!bundles.has(info.redirect)) throw new Error(`you need to load bundle ${info.redirect} first`);
                                config = bundles.get(info.redirect)._config;
                                info = config.getAssetInfo(info.uuid);
                            }

                            if (!info) {
                                out.recycle();
                                throw new Error(`Bundle ${item.bundle} doesn't contain ${item.path}`);
                            }
                            out.config = config; 
                            out.uuid = info.uuid;
                            out.info = info;
                        }
                        out.ext = item.ext || '.json';
                        break;
                    case RequestType.SCENE:
                        // 場景類型,從bundle中的config調(diào)用getSceneInfo取出該場景的詳細(xì)信息
                        if (bundles.has(item.bundle)) {
                            var config = bundles.get(item.bundle)._config;
                            var info = config.getSceneInfo(item.scene);
                            
                            if (info && info.redirect) {
                                if (!bundles.has(info.redirect)) throw new Error(`you need to load bundle ${info.redirect} first`);
                                config = bundles.get(info.redirect)._config;
                                info = config.getAssetInfo(info.uuid);
                            }
                            if (!info) {
                                out.recycle();
                                throw new Error(`Bundle ${config.name} doesn't contain scene ${item.scene}`);
                            }
                            out.config = config; 
                            out.uuid = info.uuid;
                            out.info = info;
                        }
                        break;
                    case '__isNative__': 
                        out.isNative = item.__isNative__;
                        break;
                    case RequestType.URL: 
                        out.url = item.url;
                        out.uuid = item.uuid || item.url;
                        out.ext = item.ext || cc.path.extname(item.url);
                        out.isNative = item.__isNative__ !== undefined ? item.__isNative__ : true;
                        break;
                    default: out.options[key] = item[key];
                }
                if (!out) break;
            }
        }
        if (!out) continue;
        task.output.push(out);
        if (!out.uuid && !out.url) throw new Error('unknown input:' + item.toString());
    }
    return null;
}

RequestItem的初始信息,都是從bundle對象中查詢的,bundle的信息則是從bundle自帶的config.json文件中初始化的,在打包bundle的時(shí)候,會將bundle中的資源信息寫入config.json中。

經(jīng)過parse方法處理后,我們會得到一系列RequestItem,并且很多RequestItem都自帶了AssetInfo和uuid等信息,combine方法會為每個(gè)RequestItem構(gòu)建出真正的加載路徑,這個(gè)加載路徑最終會轉(zhuǎn)換到item.url中。

function combine (task) {
    var input = task.output = task.input;
    for (var i = 0; i < input.length; i++) {
        var item = input[i];
        // 如果item已經(jīng)包含了url,則跳過,直接使用item的url
        if (item.url) continue;

        var url = '', base = '';
        var config = item.config;
        // 決定目錄的前綴
        if (item.isNative) {
            base = (config && config.nativeBase) ? (config.base + config.nativeBase) : cc.assetManager.generalNativeBase;
        } 
        else {
            base = (config && config.importBase) ? (config.base + config.importBase) : cc.assetManager.generalImportBase;
        }

        let uuid = item.uuid;
            
        var ver = '';
        if (item.info) {
            if (item.isNative) {
                ver = item.info.nativeVer ? ('.' + item.info.nativeVer) : '';
            }
            else {
                ver = item.info.ver ? ('.' + item.info.ver) : '';
            }
        }

        // 拼接最終的url
        // ugly hack, WeChat does not support loading font likes 'myfont.dw213.ttf'. So append hash to directory
        if (item.ext === '.ttf') {
            url = `${base}/${uuid.slice(0, 2)}/${uuid}${ver}/${item.options.__nativeName__}`;
        }
        else {
            url = `${base}/${uuid.slice(0, 2)}/${uuid}${ver}${item.ext}`;
        }
        
        item.url = url;
    }
    return null;
}

3.1.3 load管線【加載流程】

load方法做的事情很簡單,基本只是創(chuàng)建了新的任務(wù),在loadOneAssetPipeline中執(zhí)行每個(gè)子任務(wù)

function load (task, done) {
    if (!task.progress) {
        task.progress = {finish: 0, total: task.input.length};
    }
    
    var options = task.options, progress = task.progress;
    options.__exclude__ = options.__exclude__ || Object.create(null);
    task.output = [];
    forEach(task.input, function (item, cb) {
        // 對每個(gè)input項(xiàng)都創(chuàng)建一個(gè)子任務(wù),并交由loadOneAssetPipeline執(zhí)行
        let subTask = Task.create({ 
            input: item, 
            onProgress: task.onProgress, 
            options, 
            progress, 
            onComplete: function (err, item) {
                if (err && !task.isFinish && !cc.assetManager.force) done(err);
                task.output.push(item);
                subTask.recycle();
                cb();
            }
        });
        // 執(zhí)行子任務(wù),loadOneAssetPipeline有fetch和parse組成
        loadOneAssetPipeline.async(subTask);
    }, function () {
        // 每個(gè)input執(zhí)行完成后,最后執(zhí)行該函數(shù)
        options.__exclude__ = null;
        if (task.isFinish) {
            clear(task, true);
            return task.dispatch('error');
        }
        gatherAsset(task);
        clear(task, true);
        done();
    });
}

loadOneAssetPipeline如其函數(shù)名所示,就是加載一個(gè)資源的管線,它分為2步,fetch和parse:

fetch方法用于下載資源文件,由packManager負(fù)責(zé)下載的實(shí)現(xiàn),fetch會將下載完的文件數(shù)據(jù)放到item.file中

parse方法用于將加載完的資源文件轉(zhuǎn)換成我們可用的資源對象

對于原生資源,調(diào)用parser.parse進(jìn)行解析,該方法會根據(jù)資源類型調(diào)用不同的解析方法

  • import資源調(diào)用parseImport方法,根據(jù)json數(shù)據(jù)反序列化出Asset對象,并放到assets中
  • 圖片資源會調(diào)用parseImage、parsePVRTex或parsePKMTex方法解析圖像格式(但不會創(chuàng)建Texture對象)
  • 音效資源調(diào)用parseAudio方法進(jìn)行解析
  • plist資源調(diào)用parsePlist方法進(jìn)行解析

對于其它資源

如果uuid在task.options.__exclude__中,則標(biāo)記為完成,并添加引用計(jì)數(shù),否則,根據(jù)一些復(fù)雜的條件來決定是否加載資源的依賴

var loadOneAssetPipeline = new Pipeline('loadOneAsset', [
    function fetch (task, done) {
        var item = task.output = task.input;
        var { options, isNative, uuid, file } = item;
        var { reload } = options;
        // 如果assets里面已經(jīng)加載了這個(gè)資源,則直接完成
        if (file || (!reload && !isNative && assets.has(uuid))) return done();
        // 下載文件,這是一個(gè)異步的過程,文件下載完會被放到item.file中,并執(zhí)行done驅(qū)動管線
        packManager.load(item, task.options, function (err, data) {
            if (err) {
                if (cc.assetManager.force) {
                    err = null;
                } else {
                    cc.error(err.message, err.stack);
                }
                data = null;
            }
            item.file = data;
            done(err);
        });
    },
    // 將資源文件轉(zhuǎn)換成資源對象的過程
    function parse (task, done) {
        var item = task.output = task.input, progress = task.progress, exclude = task.options.__exclude__;
        var { id, file, options } = item;

        if (item.isNative) {
            // 對于原生資源,調(diào)用parser.parse進(jìn)行處理,將處理完的資源放到item.content中,并結(jié)束流程
            parser.parse(id, file, item.ext, options, function (err, asset) {
                if (err) {
                    if (!cc.assetManager.force) {
                        cc.error(err.message, err.stack);
                        return done(err);
                    }
                }
                item.content = asset;
                task.dispatch('progress', ++progress.finish, progress.total, item);
                files.remove(id);
                parsed.remove(id);
                done();
            });
        } else {
            var { uuid } = item;
            // 非原生資源,如果在task.options.__exclude__中,直接結(jié)束
            if (uuid in exclude) {
                var { finish, content, err, callbacks } = exclude[uuid];
                task.dispatch('progress', ++progress.finish, progress.total, item);
    
                if (finish || checkCircleReference(uuid, uuid, exclude) ) {
                    content && content.addRef();
                    item.content = content;
                    done(err);
                } else {
                    callbacks.push({ done, item });
                }
            } else {
                // 如果不是reload,且asset中包含了該uuid
                if (!options.reload && assets.has(uuid)) {
                    var asset = assets.get(uuid);
                    // 開啟了options.__asyncLoadAssets__,或asset.__asyncLoadAssets__為false,直接結(jié)束,不加載依賴
                    if (options.__asyncLoadAssets__ || !asset.__asyncLoadAssets__) {
                        item.content = asset.addRef();
                        task.dispatch('progress', ++progress.finish, progress.total, item);
                        done();
                    }
                    else {
                        loadDepends(task, asset, done, false);
                    }
                } else {
                    // 如果是reload,或者assets中沒有,則進(jìn)行解析,并加載依賴
                    parser.parse(id, file, 'import', options, function (err, asset) {
                        if (err) {
                            if (cc.assetManager.force) {
                                err = null;
                            }
                            else {
                                cc.error(err.message, err.stack);
                            }
                            return done(err);
                        }
                        
                        asset._uuid = uuid;
                        loadDepends(task, asset, done, true);
                    });
                }
            }
        }
    }
]);

3.2 文件下載

creator使用packManager.load來完成下載的工作,當(dāng)要下載一個(gè)文件時(shí),有2個(gè)問題需要考慮:

  • 該文件是否被打包了,比如由于勾選了內(nèi)聯(lián)所有SpriteFrame,導(dǎo)致SpriteFrame的json文件被合并到prefab中
  • 當(dāng)前平臺是原生平臺還是web平臺,對于一些本地資源,原生平臺需要從磁盤讀取
// packManager.load的實(shí)現(xiàn)
load (item, options, onComplete) {
  // 如果資源沒有被打包,則直接調(diào)用downloader.download下載(download內(nèi)部也有已下載和加載中的判斷)
  if (item.isNative || !item.info || !item.info.packs) return downloader.download(item.id, item.url, item.ext, item.options, onComplete);
  // 如果文件已經(jīng)下載過了,則直接返回
  if (files.has(item.id)) return onComplete(null, files.get(item.id));

  var packs = item.info.packs;
  // 如果pack已經(jīng)在加載中,則將回調(diào)添加到_loading隊(duì)列,等加載完成后觸發(fā)回調(diào)
  var pack = packs.find(isLoading);
  if (pack) return _loading.get(pack.uuid).push({ onComplete, id: item.id });

  // 下載一個(gè)新的pack
  pack = packs[0];
  _loading.add(pack.uuid, [{ onComplete, id: item.id }]);
  let url = cc.assetManager._transform(pack.uuid, {ext: pack.ext, bundle: item.config.name});
  // 下載pack并解包,
  downloader.download(pack.uuid, url, pack.ext, item.options, function (err, data) {
      files.remove(pack.uuid);
      if (err) {
          cc.error(err.message, err.stack);
      }
      // unpack package,內(nèi)部實(shí)現(xiàn)包含2種解包,一種針對prefab、圖集等json數(shù)組的分割解包,另一種針對Texture2D的content進(jìn)行解包
      packManager.unpack(pack.packs, data, pack.ext, item.options, function (err, result) {
          if (!err) {
              for (var id in result) {
                  files.add(id, result[id]);
              }
          }
          var callbacks = _loading.remove(pack.uuid);
          for (var i = 0, l = callbacks.length; i < l; i++) {
              var cb = callbacks[i];
              if (err) {
                  cb.onComplete(err);
                  continue;
              }

              var data = result[cb.id];
              if (!data) {
                  cb.onComplete(new Error('can not retrieve data from package'));
              }
              else {
                  cb.onComplete(null, data);
              }
          }
      });
  });
}

3.2.1 Web平臺的下載

web平臺的download實(shí)現(xiàn)如下:

  • 用一個(gè)downloaders數(shù)組來管理各種資源類型對應(yīng)的下載方式
  • 使用files緩存來避免重復(fù)下載
  • 使用_downloading隊(duì)列來處理并發(fā)下載同一個(gè)資源時(shí)的回調(diào),并保證時(shí)序
  • 支持了下載的優(yōu)先級、重試等邏輯
download (id, url, type, options, onComplete) {
  // 取出downloaders中對應(yīng)類型的下載回調(diào)
  let func = downloaders[type] || downloaders['default'];
  let self = this;
  // 避免重復(fù)下載
  let file, downloadCallbacks;
  if (file = files.get(id)) {
      onComplete(null, file);
  }
  // 如果在下載中,添加到隊(duì)列
  else if (downloadCallbacks = _downloading.get(id)) {
      downloadCallbacks.push(onComplete);
      for (let i = 0, l = _queue.length; i < l; i++) {
          var item = _queue[i];
          if (item.id === id) {
              var priority = options.priority || 0;
              if (item.priority < priority) {
                  item.priority = priority;
                  _queueDirty = true;
              } 
              return;
          }
      } 
  }
  else {
      // 進(jìn)行下載,并設(shè)置好下載失敗的重試
      var maxRetryCount = options.maxRetryCount || this.maxRetryCount;
      var maxConcurrency = options.maxConcurrency || this.maxConcurrency;
      var maxRequestsPerFrame = options.maxRequestsPerFrame || this.maxRequestsPerFrame;

      function process (index, callback) {
          if (index === 0) {
              _downloading.add(id, [onComplete]);
          }
          if (!self.limited) return func(urlAppendTimestamp(url), options, callback);
          updateTime();

          function invoke () {
              func(urlAppendTimestamp(url), options, function () {
                  // when finish downloading, update _totalNum
                  _totalNum--;
                  if (!_checkNextPeriod && _queue.length > 0) {
                      callInNextTick(handleQueue, maxConcurrency, maxRequestsPerFrame);
                      _checkNextPeriod = true;
                  }
                  callback.apply(this, arguments);
              });
          }

          if (_totalNum < maxConcurrency && _totalNumThisPeriod < maxRequestsPerFrame) {
              invoke();
              _totalNum++;
              _totalNumThisPeriod++;
          }
          else {
              // when number of request up to limitation, cache the rest
              _queue.push({ id, priority: options.priority || 0, invoke });
              _queueDirty = true;

              if (!_checkNextPeriod && _totalNum < maxConcurrency) {
                  callInNextTick(handleQueue, maxConcurrency, maxRequestsPerFrame);
                  _checkNextPeriod = true;
              }
          }
      }

      // retry完成后,將文件添加到files緩存中,從_downloading隊(duì)列中移除,并執(zhí)行callbacks回調(diào)
      // when retry finished, invoke callbacks
      function finale (err, result) {
          if (!err) files.add(id, result);
          var callbacks = _downloading.remove(id);
          for (let i = 0, l = callbacks.length; i < l; i++) {
              callbacks[i](err, result);
          }
      }

      retry(process, maxRetryCount, this.retryInterval, finale);
  }
}

downloaders是一個(gè)map,映射了各種資源類型對應(yīng)的下載方法,在web平臺主要包含以下幾類下載方法:

圖片類 downloadImage

  • downloadDomImage 使用Html的Image元素,指定其src屬性來下載
  • downloadBlob 以文件下載的方式下載圖片

文件類,這里可以分為二進(jìn)制文件、json文件和文本文件

  • downloadArrayBuffer 指定arraybuffer類型調(diào)用downloadFile,用于skel、bin、pvr等文件下載
  • downloadText 指定text類型調(diào)用downloadFile,用于atlas、tmx、xml、vsh等文件下載
  • downloadJson 指定json類型調(diào)用downloadFile,并在下載完后解析json,用于plist、json等文件下載

字體類 loadFont 構(gòu)建css樣式,指定url下載

聲音類 downloadAudio

  • downloadDomAudio 創(chuàng)建Html的audio元素,指定其src屬性來下載
  • downloadBlob 以文件下載的方式下載音效

視頻類 downloadVideo web端直接返回了

腳本 downloadScript 創(chuàng)建Html的script元素,指定其src屬性來下載并執(zhí)行

Bundle downloadBundle 同時(shí)下載了Bundle的json和腳本

downloadFile使用了XMLHttpRequest來下載文件,具體實(shí)現(xiàn)如下:

function downloadFile (url, options, onProgress, onComplete) {
    var { options, onProgress, onComplete } = parseParameters(options, onProgress, onComplete);
    var xhr = new XMLHttpRequest(), errInfo = 'download failed: ' + url + ', status: ';
    xhr.open('GET', url, true);
    
    if (options.responseType !== undefined) xhr.responseType = options.responseType;
    if (options.withCredentials !== undefined) xhr.withCredentials = options.withCredentials;
    if (options.mimeType !== undefined && xhr.overrideMimeType ) xhr.overrideMimeType(options.mimeType);
    if (options.timeout !== undefined) xhr.timeout = options.timeout;

    if (options.header) {
        for (var header in options.header) {
            xhr.setRequestHeader(header, options.header[header]);
        }
    }

    xhr.onload = function () {
        if ( xhr.status === 200 || xhr.status === 0 ) {
            onComplete && onComplete(null, xhr.response);
        } else {
            onComplete && onComplete(new Error(errInfo + xhr.status + '(no response)'));
        }

    };

    if (onProgress) {
        xhr.onprogress = function (e) {
            if (e.lengthComputable) {
                onProgress(e.loaded, e.total);
            }
        };
    }

    xhr.onerror = function(){
        onComplete && onComplete(new Error(errInfo + xhr.status + '(error)'));
    };
    xhr.ontimeout = function(){
        onComplete && onComplete(new Error(errInfo + xhr.status + '(time out)'));
    };
    xhr.onabort = function(){
        onComplete && onComplete(new Error(errInfo + xhr.status + '(abort)'));
    };

    xhr.send(null);
    return xhr;
}

3.2.2 原生平臺下載

原生平臺的引擎相關(guān)文件可以在引擎目錄的resources/builtin/jsb-adapter/engine目錄下,資源加載相關(guān)的實(shí)現(xiàn)在jsb-loader.js文件中,這里的downloader重新注冊了回調(diào)函數(shù)。

downloader.register({
    // JS
    '.js' : downloadScript,
    '.jsc' : downloadScript,

    // Images
    '.png' : downloadAsset,
    '.jpg' : downloadAsset,
    ...
});

在原生平臺下,downloadAsset等方法都會調(diào)用download來進(jìn)行資源的下載,在資源下載之前會調(diào)用transformUrl對url進(jìn)行檢測,主要判斷該資源是網(wǎng)絡(luò)資源還是本地資源,如果是網(wǎng)絡(luò)資源,是否已經(jīng)下載過了。只有沒下載過的網(wǎng)絡(luò)資源,才需要進(jìn)行下載。不需要下載的在文件解析的地方會直接讀文件。

// func傳入的是下載完成之后的處理,比如腳本下載完成后需要執(zhí)行,此時(shí)會調(diào)用window.require
// 如果說要下載的是json資源之類的,傳入的func是doNothing,也就是直接調(diào)用onComplete方法
function download (url, func, options, onFileProgress, onComplete) {
    var result = transformUrl(url, options);
    // 如果是本地文件,直接指向func
    if (result.inLocal) {
        func(result.url, options, onComplete);
    }
    // 如果在緩存中,更新資源的最后使用時(shí)間(lru)
    else if (result.inCache) {
        cacheManager.updateLastTime(url)
        func(result.url, options, function (err, data) {
            if (err) {
                cacheManager.removeCache(url);
            }
            onComplete(err, data);
        });
    }
    else {
        // 未下載的網(wǎng)絡(luò)資源,調(diào)用downloadFile進(jìn)行下載
        var time = Date.now();
        var storagePath = '';
        if (options.__cacheBundleRoot__) {
            storagePath = `${cacheManager.cacheDir}/${options.__cacheBundleRoot__}/${time}${suffix++}${cc.path.extname(url)}`;
        }
        else {
            storagePath = `${cacheManager.cacheDir}/${time}${suffix++}${cc.path.extname(url)}`;
        }
        // 使用downloadFile下載并緩存
        downloadFile(url, storagePath, options.header, onFileProgress, function (err, path) {
            if (err) {
                onComplete(err, null);
                return;
            }
            func(path, options, function (err, data) {
                if (!err) {
                    cacheManager.cacheFile(url, storagePath, options.__cacheBundleRoot__);
                }
                onComplete(err, data);
            });
        });
    }
}

function transformUrl (url, options) {
    var inLocal = false;
    var inCache = false;
    // 通過正則匹配是不是URL
    if (REGEX.test(url)) {
        if (options.reload) {
            return { url };
        }
        else {
            // 檢查是否在緩存中(本地磁盤緩存)
            var cache = cacheManager.cachedFiles.get(url);
            if (cache) {
                inCache = true;
                url = cache.url;
            }
        }
    }
    else {
        inLocal = true;
    }
    return { url, inLocal, inCache };
}

downloadFile會調(diào)用原生平臺的jsb_downloader來下載資源,并保存到本地磁盤中

downloadFile (remoteUrl, filePath, header, onProgress, onComplete) {
  downloading.add(remoteUrl, { onProgress, onComplete });
  var storagePath = filePath;
  if (!storagePath) storagePath = tempDir + '/' + performance.now() + cc.path.extname(remoteUrl);
  jsb_downloader.createDownloadFileTask(remoteUrl, storagePath, header);
},

3.3 文件解析

在loadOneAssetPipeline中,資源會經(jīng)過fetch和parse兩個(gè)管線進(jìn)行處理,fetch負(fù)責(zé)下載而parse負(fù)責(zé)解析資源,并實(shí)例化資源對象。在parse方法中調(diào)用了parser.parse將文件內(nèi)容傳入,解析成對應(yīng)的Asset對象,并返回。

3.3.1 Web平臺解析

Web平臺下的parser.parse主要做的是對解析中的文件的管理,為解析中、解析完的文件維護(hù)一個(gè)列表,避免重復(fù)解析。同時(shí)維護(hù)了解析完成后的回調(diào)列表,而真正的解析方法在parsers數(shù)組中。

parse (id, file, type, options, onComplete) {
  let parsedAsset, parsing, parseHandler;
  if (parsedAsset = parsed.get(id)) {
      onComplete(null, parsedAsset);
  }
  else if (parsing = _parsing.get(id)){
      parsing.push(onComplete);
  }
  else if (parseHandler = parsers[type]){
      _parsing.add(id, [onComplete]);
      parseHandler(file, options, function (err, data) {
          if (err) {
              files.remove(id);
          } 
          else if (!isScene(data)){
              parsed.add(id, data);
          }
          let callbacks = _parsing.remove(id);
          for (let i = 0, l = callbacks.length; i < l; i++) {
              callbacks[i](err, data);
          }
      });
  }
  else {
      onComplete(null, file);
  }
}

parsers映射了各種類型文件的解析方法,下面以圖片和普通的asset資源為例:

注意:在parseImport方法中,反序列化方法會將資源的依賴放到asset.__depends__中,結(jié)構(gòu)為數(shù)組,數(shù)組中每個(gè)對象包含3個(gè)字段,資源id uuid、owner 對象、prop 屬性。比如一個(gè)Prefab資源,下面有2個(gè)節(jié)點(diǎn),都引用了同一個(gè)資源,depends列表需要為這兩個(gè)節(jié)點(diǎn)對象分別記錄一條依賴信息 [{uuid:xxx, owner:1, prop:tex}, {uuid:xxx, owner:2, prop:tex}]

// 映射圖片格式到解析方法
var parsers = {
  '.png' : parser.parseImage,
  '.jpg' : parser.parseImage,
  '.bmp' : parser.parseImage,
  '.jpeg' : parser.parseImage,
  '.gif' : parser.parseImage,
  '.ico' : parser.parseImage,
  '.tiff' : parser.parseImage,
  '.webp' : parser.parseImage,
  '.image' : parser.parseImage,
  '.pvr' : parser.parsePVRTex,
  '.pkm' : parser.parsePKMTex,
  // Audio
  '.mp3' : parser.parseAudio,
  '.ogg' : parser.parseAudio,
  '.wav' : parser.parseAudio,
  '.m4a' : parser.parseAudio,

  // plist
  '.plist' : parser.parsePlist,
  'import' : parser.parseImport
};

// 圖片并不會解析成Asset對象,而是解析成對應(yīng)的圖片對象
parseImage (file, options, onComplete) {
  if (capabilities.imageBitmap && file instanceof Blob) {
      let imageOptions = {};
      imageOptions.imageOrientation = options.__flipY__ ? 'flipY' : 'none';
      imageOptions.premultiplyAlpha = options.__premultiplyAlpha__ ? 'premultiply' : 'none';
      createImageBitmap(file, imageOptions).then(function (result) {
          result.flipY = !!options.__flipY__;
          result.premultiplyAlpha = !!options.__premultiplyAlpha__;
          onComplete && onComplete(null, result);
      }, function (err) {
          onComplete && onComplete(err, null);
      });
  }
  else {
      onComplete && onComplete(null, file);
  }
},

// Asset對象的解析,通過deserialize實(shí)現(xiàn),大致流程是解析json然后找到對應(yīng)的class,并調(diào)用對應(yīng)class的_deserialize方法拷貝數(shù)據(jù)、初始化變量,并將依賴資源放到asset.__depends
parseImport (file, options, onComplete) {
  if (!file) return onComplete && onComplete(new Error('Json is empty'));
  var result, err = null;
  try {
      result = deserialize(file, options);
  }
  catch (e) {
      err = e;
  }
  onComplete && onComplete(err, result);
},

3.3.2 原生平臺解析

在原生平臺下,jsb-loader.js中重新注冊了各種資源的解析方法:

parser.register({
    '.png' : downloader.downloadDomImage,
    '.binary' : parseArrayBuffer,
    '.txt' : parseText,
    '.plist' : parsePlist,
    '.font' : loadFont,
    '.ExportJson' : parseJson,
    ...
});

圖片的解析方法竟然是downloader.downloadDomImage?跟蹤原生平臺調(diào)試了一下,確實(shí)是調(diào)用的這個(gè)方法,創(chuàng)建了Image對象并指定src來加載圖片,這種方式加載本地磁盤的圖片也是可以的,但紋理對象又是如何創(chuàng)建的呢?通過Texture2D對應(yīng)的json文件,creator在加載真正的原生紋理之前,就已經(jīng)創(chuàng)建好了Texture2D這個(gè)Asset對象,而在加載完原生圖片資源后,會將Image對象設(shè)置為Texture2D對象的_nativeAsset,在這個(gè)屬性的set方法中,會調(diào)用initWithData或initWithElement,這里才真正使用紋理數(shù)據(jù)創(chuàng)建了用于渲染的紋理對象。

var Texture2D = cc.Class({
    name: 'cc.Texture2D',
    extends: require('../assets/CCAsset'),
    mixins: [EventTarget],

    properties: {
        _nativeAsset: {
            get () {
                // maybe returned to pool in webgl
                return this._image;
            },
            set (data) {
                if (data._data) {
                    this.initWithData(data._data, this._format, data.width, data.height);
                }
                else {
                    this.initWithElement(data);
                }
            },
            override: true
        },

而對于parseJson、parseText、parseArrayBuffer等實(shí)現(xiàn),這里只是簡單地調(diào)用了文件系統(tǒng)讀取文件而已。像一些拿到文件內(nèi)容之后,需要進(jìn)一步解析才能使用的資源呢?比如模型、骨骼等資源依賴二進(jìn)制的模型數(shù)據(jù),這些數(shù)據(jù)的解析在哪里呢?沒錯(cuò),跟上面的Texture2D一樣,都是放在對應(yīng)的Asset資源本身,有些在_nativeAsset字段的setter回調(diào)中初始化,而有些會在真正使用這個(gè)資源時(shí)才惰性地進(jìn)行初始化。

// 在jsb-loader.js文件中
function parseText (url, options, onComplete) {
    readText(url, onComplete);
}

function parseArrayBuffer (url, options, onComplete) {
    readArrayBuffer(url, onComplete);
}

function parseJson (url, options, onComplete) {
    readJson(url, onComplete);
}

// 在jsb-fs-utils.js文件中
    readText (filePath, onComplete) {
        fsUtils.readFile(filePath, 'utf8', onComplete);
    },

    readArrayBuffer (filePath, onComplete) {
        fsUtils.readFile(filePath, '', onComplete);
    },

    readJson (filePath, onComplete) {
        fsUtils.readFile(filePath, 'utf8', function (err, text) {
            var out = null;
            if (!err) {
                try {
                    out = JSON.parse(text);
                }
                catch (e) {
                    cc.warn('Read json failed: ' + e.message);
                    err = new Error(e.message);
                }
            }
            onComplete && onComplete(err, out);
        });
    },

像圖集、Prefab這些資源又是怎么初始化的呢?Creator還是使用parseImport方法進(jìn)行解析,因?yàn)檫@些資源對應(yīng)的類型是import,原生平臺下并沒有覆蓋這種類型對應(yīng)的parse函數(shù),而這些資源會直接反序列化成可用的Asset對象。

3.4 依賴加載

creator將資源分為兩大類,普通資源和原生資源,普通資源包括cc.Asset及其子類,如cc.SpriteFrame、cc.Texture2D、cc.Prefab等等。原生資源包括各種格式的紋理、音樂、字體等文件,在游戲中我們無法直接使用這些原生資源,而是需要讓creator將他們轉(zhuǎn)換成對應(yīng)的cc.Asset對象之后才能使用。

在creator中,一個(gè)Prefab可能會依賴很多資源,這些依賴也可以分為普通依賴和原生資源依賴,creator的cc.Asset提供了_parseDepsFromJson_parseNativeDepFromJson方法來檢查資源的依賴。loadDepends通過getDepends方法搜集了資源的依賴。

loadDepends創(chuàng)建了一個(gè)子任務(wù)來負(fù)責(zé)依賴資源的加載,并調(diào)用pipeline執(zhí)行加載,實(shí)際上無論有無依賴需要加載,都會執(zhí)行這段邏輯,加載完成后執(zhí)行以下重要邏輯:

  • 初始化assset:在依賴加載完成后,將依賴的資源賦值到asset對應(yīng)的屬性后調(diào)用asset.onLoad
  • 將資源對應(yīng)的files和parsed緩存移除,并緩存資源到assets中(如果是場景的話,不會緩存)
  • 執(zhí)行repeatItem.callbacks列表中的回調(diào)(在loadDepends的開頭構(gòu)造,默認(rèn)記錄傳入的done方法)
// 加載指定asset的依賴項(xiàng)
function loadDepends (task, asset, done, init) {

    var item = task.input, progress = task.progress;
    var { uuid, id, options, config } = item;
    var { __asyncLoadAssets__, cacheAsset } = options;

    var depends = [];
    // 增加引用計(jì)數(shù)來避免加載依賴的過程中資源被釋放,調(diào)用getDepends獲取依賴資源
    asset.addRef && asset.addRef();
    getDepends(uuid, asset, Object.create(null), depends, false, __asyncLoadAssets__, config);
    task.dispatch('progress', ++progress.finish, progress.total += depends.length, item);

    var repeatItem = task.options.__exclude__[uuid] = { content: asset, finish: false, callbacks: [{ done, item }] };

    let subTask = Task.create({ 
        input: depends, 
        options: task.options, 
        onProgress: task.onProgress, 
        onError: Task.prototype.recycle, 
        progress, 
        onComplete: function (err) {
            // 在所有依賴項(xiàng)加載完成之后回調(diào)
            asset.decRef && asset.decRef(false);
            asset.__asyncLoadAssets__ = __asyncLoadAssets__;
            repeatItem.finish = true;
            repeatItem.err = err;

            if (!err) {
                var assets = Array.isArray(subTask.output) ? subTask.output : [subTask.output];
                // 構(gòu)造一個(gè)map,記錄uuid到asset的映射
                var map = Object.create(null);
                for (let i = 0, l = assets.length; i < l; i++) {
                    var dependAsset = assets[i];
                    dependAsset && (map[dependAsset instanceof cc.Asset ? dependAsset._uuid + '@import' : uuid + '@native'] = dependAsset);
                }

                // 調(diào)用setProperties將對應(yīng)的依賴資源設(shè)置到asset的成員變量中
                if (!init) {
                    if (asset.__nativeDepend__ && !asset._nativeAsset) {
                        var missingAsset = setProperties(uuid, asset, map);
                        if (!missingAsset) {
                            try {
                                asset.onLoad && asset.onLoad();
                            }
                            catch (e) {
                                cc.error(e.message, e.stack);
                            }
                        }
                    }
                }
                else {
                    var missingAsset = setProperties(uuid, asset, map);
                    if (!missingAsset) {
                        try {
                            asset.onLoad && asset.onLoad();
                        }
                        catch (e) {
                            cc.error(e.message, e.stack);
                        }
                    }
                    files.remove(id);
                    parsed.remove(id);
                    cache(uuid, asset, cacheAsset !== undefined ? cacheAsset : cc.assetManager.cacheAsset); 
                }
                subTask.recycle();
            }
            
            // 這個(gè)repeatItem可能有很多個(gè)地方都加載了它,要通知所有回調(diào)加載完成
            var callbacks = repeatItem.callbacks;
            for (var i = 0, l = callbacks.length; i < l; i++) {
                var cb = callbacks[i];
                asset.addRef && asset.addRef();
                cb.item.content = asset;
                cb.done(err);
            }
            callbacks.length = 0;
        }
    });

    pipeline.async(subTask);
}

3.4.1 依賴解析

getDepends (uuid, data, exclude, depends, preload, asyncLoadAssets, config) {
  var err = null;
  try {
      var info = dependUtil.parse(uuid, data);
      var includeNative = true;
      if (data instanceof cc.Asset && (!data.__nativeDepend__ || data._nativeAsset)) includeNative = false; 
      if (!preload) {
          asyncLoadAssets = !CC_EDITOR && (!!data.asyncLoadAssets || (asyncLoadAssets && !info.preventDeferredLoadDependents));
          for (let i = 0, l = info.deps.length; i < l; i++) {
              let dep = info.deps[i];
              if (!(dep in exclude)) {
                  exclude[dep] = true;
                  depends.push({uuid: dep, __asyncLoadAssets__: asyncLoadAssets, bundle: config && config.name});
              }
          }

          if (includeNative && !asyncLoadAssets && !info.preventPreloadNativeObject && info.nativeDep) {
              config && (info.nativeDep.bundle = config.name);
              depends.push(info.nativeDep);
          }
          
      } else {
          for (let i = 0, l = info.deps.length; i < l; i++) {
              let dep = info.deps[i];
              if (!(dep in exclude)) {
                  exclude[dep] = true;
                  depends.push({uuid: dep, bundle: config && config.name});
              }
          }
          if (includeNative && info.nativeDep) {
              config && (info.nativeDep.bundle = config.name);
              depends.push(info.nativeDep);
          }
      }
  }
  catch (e) {
      err = e;
  }
  return err;
},

dependUtil是一個(gè)控制依賴列表的單例,通過傳入uuid和asset對象來解析該對象的依賴資源列表,返回的依賴資源列表可能包含以下4個(gè)字段:

  • deps 依賴的Asset資源
  • nativeDep 依賴的原生資源
  • preventPreloadNativeObject 禁止預(yù)加載原生對象,這個(gè)值默認(rèn)是false
  • preventDeferredLoadDependents 禁止延遲加載依賴,默認(rèn)為false,對于骨骼動畫、TiledMap等資源為true
  • parsedFromExistAsset 是否直接從asset.__depends__中取出

dependUtil還維護(hù)了_depends緩存來避免依賴的重復(fù)查詢,這個(gè)緩存會在首次查詢某資源依賴時(shí)添加,當(dāng)該資源被釋放時(shí)移除

// 根據(jù)json信息獲取其資源依賴列表,實(shí)際上json信息就是asset對象
parse (uuid, json) {
  var out = null;
  // 如果是場景或者Prefab,data會是一個(gè)數(shù)組,scene or prefab
  if (Array.isArray(json)) {
      // 如果已經(jīng)解析過了,在_depends中有依賴列表,則直接返回
      if (this._depends.has(uuid)) return this._depends.get(uuid)
      out = {
          // 對于Prefab或場景,直接使用_parseDepsFromJson方法返回
          deps: cc.Asset._parseDepsFromJson(json),
          asyncLoadAssets: json[0].asyncLoadAssets
      };
  }
  // 如果包含__type__,獲取其構(gòu)造函數(shù),并從json中查找依賴資源 get deps from json
  // 實(shí)際測試,預(yù)加載的資源會走下面這個(gè)分支,預(yù)加載的資源并沒有把json反序列化成Asset對象
  else if (json.__type__) {
      if (this._depends.has(uuid)) return this._depends.get(uuid);
      var ctor = js._getClassById(json.__type__);
      // 部分資源重寫了_parseDepsFromJson和_parseNativeDepFromJson方法
      // 比如cc.Texture2D
      out = {
          preventPreloadNativeObject: ctor.preventPreloadNativeObject,
          preventDeferredLoadDependents: ctor.preventDeferredLoadDependents,
          deps: ctor._parseDepsFromJson(json),
          nativeDep: ctor._parseNativeDepFromJson(json)
      };
      out.nativeDep && (out.nativeDep.uuid = uuid);
  }
  // get deps from an existing asset 
  // 如果沒有__type__字段,則無法找到它對應(yīng)的ctor,從asset的__depends__字段中取出依賴
  else {
      if (!CC_EDITOR && (out = this._depends.get(uuid)) && out.parsedFromExistAsset) return out;
      var asset = json;
      out = {
          deps: [],
          parsedFromExistAsset: true,
          preventPreloadNativeObject: asset.constructor.preventPreloadNativeObject,
          preventDeferredLoadDependents: asset.constructor.preventDeferredLoadDependents
      };
      let deps = asset.__depends__;
      for (var i = 0, l = deps.length; i < l; i++) {
          var dep = deps[i].uuid;
          out.deps.push(dep);
      }
  
      if (asset.__nativeDepend__) {
          // asset._nativeDep會返回類似這樣的對象 {__isNative__: true, uuid: this._uuid, ext: this._native}
          out.nativeDep = asset._nativeDep;
      }
  }
  // 第一次找到依賴,直接放到_depends列表中,cache dependency list
  this._depends.add(uuid, out);
  return out;
}

CCAsset默認(rèn)的_parseDepsFromJson_parseNativeDepFromJson實(shí)現(xiàn)如下,_parseDepsFromJson通過調(diào)用parseDependRecursively遞歸json,將json對象及其子對象的所有__uuid__全部找到放到depends數(shù)組中。Texture2D、TTFFont、AudioClip的實(shí)現(xiàn)為直接返回空數(shù)組,而SpriteFrame的實(shí)現(xiàn)為返回cc.assetManager.utils.decodeUuid(json.content.texture),這個(gè)字段記錄了SpriteFrame對應(yīng)紋理的uuid。

_parseNativeDepFromJson在改asset的_native有值的情況下,會返回{ __isNative__: true, ext: json._native}。實(shí)際上大部分的native資源走的是_nativeDep,這個(gè)屬性的get方法會返回一個(gè)包含類似這樣的對象{__isNative__: true, uuid: this._uuid, ext: this._native}。

_parseDepsFromJson (json) {
      var depends = [];
      parseDependRecursively(json, depends);
      return depends;
},

_parseNativeDepFromJson (json) {
if (json._native) return { __isNative__: true, ext: json._native};
      return null;
}

3.5 資源釋放

這一小節(jié)重點(diǎn)介紹在Creator中釋放資源的三種方式以及其背后的實(shí)現(xiàn),最后介紹在項(xiàng)目中如何排查資源泄露的情況。

3.5.1 Creator的資源釋放

Creator支持以下3種資源釋放的方式:

釋放方式 釋放效果
勾選:場景->屬性檢查器->自動釋放資源 在場景切換后,自動釋放新場景不使用的資源
引用計(jì)數(shù)釋放res.decRef 使用addRef和decRef維護(hù)引用計(jì)數(shù),在decRef后引用計(jì)數(shù)為0時(shí)自動釋放
手動釋放cc.assetManager.releaseAsset(texture); 手動釋放資源,強(qiáng)制釋放

3.5.2 場景自動釋放

當(dāng)一個(gè)新場景運(yùn)行的時(shí)候會執(zhí)行Director.runSceneImmediate方法,這里調(diào)用了_autoRelease來實(shí)現(xiàn)老場景資源的自動釋放(如果老場景勾選了自動釋放資源)。

runSceneImmediate: function (scene, onBeforeLoadScene, onLaunched) {
  // 省略代碼...
  var oldScene = this._scene;
  if (!CC_EDITOR) {
      // 自動釋放資源
      CC_BUILD && CC_DEBUG && console.time('AutoRelease');
      cc.assetManager._releaseManager._autoRelease(oldScene, scene, persistNodeList);
      CC_BUILD && CC_DEBUG && console.timeEnd('AutoRelease');
  }

  // unload scene
  CC_BUILD && CC_DEBUG && console.time('Destroy');
  if (cc.isValid(oldScene)) {
      oldScene.destroy();
  }
  // 省略代碼...
},

最新版本的_autoRelease的實(shí)現(xiàn)非常簡潔干脆,將持久節(jié)點(diǎn)的引用從老場景遷移到新場景,然后直接調(diào)用資源的decRef減少引用計(jì)數(shù),而是否釋放老場景引用的資源,則取決于老場景是否設(shè)置了autoReleaseAssets。

// do auto release
_autoRelease (oldScene, newScene, persistNodes) { 
  // 所有持久節(jié)點(diǎn)依賴的資源自動addRef、并記錄到sceneDeps.persistDeps中
  for (let i = 0, l = persistNodes.length; i < l; i++) {
      var node = persistNodes[i];
      var sceneDeps = dependUtil._depends.get(newScene._id);
      var deps = _persistNodeDeps.get(node.uuid);
      for (let i = 0, l = deps.length; i < l; i++) {
          var dependAsset = assets.get(deps[i]);
          if (dependAsset) {
              dependAsset.addRef();
          }
      }
      if (sceneDeps) {
          !sceneDeps.persistDeps && (sceneDeps.persistDeps = []);
          sceneDeps.persistDeps.push.apply(sceneDeps.persistDeps, deps);
      }
  }

  // 釋放老場景的依賴
  if (oldScene) {
      var childs = dependUtil.getDeps(oldScene._id);
      for (let i = 0, l = childs.length; i < l; i++) {
          let asset = assets.get(childs[i]);
          asset && asset.decRef(CC_TEST || oldScene.autoReleaseAssets);
      }
      var dependencies = dependUtil._depends.get(oldScene._id);
      if (dependencies && dependencies.persistDeps) {
          var persistDeps = dependencies.persistDeps;
          for (let i = 0, l = persistDeps.length; i < l; i++) {
              let asset = assets.get(persistDeps[i]);
              asset && asset.decRef(CC_TEST || oldScene.autoReleaseAssets);
          }
      }
      dependUtil.remove(oldScene._id);
  }
},

3.5.3 引用計(jì)數(shù)和手動釋放資源

剩下兩種釋放資源的方式,本質(zhì)上都是調(diào)用releaseManager.tryRelease來實(shí)現(xiàn)資源釋放,區(qū)別在于decRef是根據(jù)引用計(jì)數(shù)和autoRelease來決定是否調(diào)用tryRelease,而releaseAsset是強(qiáng)制釋放。資源釋放的完整流程大致如下圖所示:

// CCAsset.js 減少引用
decRef (autoRelease) {
  this._ref--;
  autoRelease !== false && cc.assetManager._releaseManager.tryRelease(this);
  return this;
}

// CCAssetManager.js 手動釋放資源
releaseAsset (asset) {
  releaseManager.tryRelease(asset, true);
},

tryRelease支持延遲釋放和強(qiáng)制釋放2種模式,當(dāng)傳入force參數(shù)為true時(shí)直接進(jìn)入釋放流程,否則creator會將資源放入待釋放的列表中,并在EVENT_AFTER_DRAW事件中執(zhí)行freeAssets方法真正清理資源。不論何種方式,資源會傳入到_free方法處理,這個(gè)方法做了以下幾件事情。

  • 從_toDelete中移除
  • 在非force釋放時(shí),需要檢查是否還有其它引用,如果是則返回
  • 從assets緩存中移除
  • 自動釋放依賴資源
  • 調(diào)用資源的destroy方法銷毀資源
  • 從dependUtil中移除資源的依賴記錄

checkCircularReference返回值如果大于0,表示資源還有被其它地方引用,其它地方指所有我們addRef的地方,該方法會先記錄asset當(dāng)前的refCount,然后消除掉資源和依賴資源中對asset的引用,這相當(dāng)于資源A內(nèi)部掛載了組件B和C,它們都引用了資源A,此時(shí)資源A的引用計(jì)數(shù)為2,而組件B和C其實(shí)是要跟著A釋放的,而A被B和C引用著,計(jì)數(shù)就不為0無法釋放,所以checkCircularReference先排除了內(nèi)部的引用。如果資源的refCount減去了內(nèi)部的引用次數(shù)還大于1,說明有其它地方還引用著它,不能釋放。

tryRelease (asset, force) {
  if (!(asset instanceof cc.Asset)) return;
  if (force) {
      releaseManager._free(asset, force);
  }
  else {
      _toDelete.add(asset._uuid, asset);
      // 在下次Director繪制完成之后,執(zhí)行freeAssets
      if (!eventListener) {
          eventListener = true;
          cc.director.once(cc.Director.EVENT_AFTER_DRAW, freeAssets);
      }
  }
}

// 釋放資源
_free (asset, force) {
  _toDelete.remove(asset._uuid);

  if (!cc.isValid(asset, true)) return;

  if (!force) {
      if (asset.refCount > 0) {
          // 檢查資源內(nèi)部的循環(huán)引用
          if (checkCircularReference(asset) > 0) return; 
      }
  }

  // 從緩存中移除
  assets.remove(asset._uuid);
  var depends = dependUtil.getDeps(asset._uuid);
  for (let i = 0, l = depends.length; i < l; i++) {
      var dependAsset = assets.get(depends[i]);
      if (dependAsset) {
          dependAsset.decRef(false);
          releaseManager._free(dependAsset, false);
      }
  }
  asset.destroy();
  dependUtil.remove(asset._uuid);
},

// 釋放_toDelete中的資源并清空
function freeAssets () {
  eventListener = false;
  _toDelete.forEach(function (asset) {
      releaseManager._free(asset);
  });
  _toDelete.clear();
}

asset.destroy做了什么?資源對象是如何被釋放掉的?像紋理、聲音這樣的資源又是如何被釋放掉的呢?Asset對象本身并沒有destroy方法,而是Asset對象所繼承的CCObject對象實(shí)現(xiàn)了destroy,這里的實(shí)現(xiàn)只是將對象放到了一個(gè)待釋放的數(shù)組中,并打上ToDestroy的標(biāo)記。Director每一幀都會調(diào)用deferredDestroy來執(zhí)行_destroyImmediate進(jìn)行資源釋放,這個(gè)方法會對對象的Destroyed標(biāo)記進(jìn)行判斷和操作、調(diào)用_onPreDestroy方法執(zhí)行回調(diào)、以及_destruct方法進(jìn)行析構(gòu)。

prototype.destroy = function () {
    if (this._objFlags & Destroyed) {
        cc.warnID(5000);
        return false;
    }
    if (this._objFlags & ToDestroy) {
        return false;
    }
    this._objFlags |= ToDestroy;
    objectsToDestroy.push(this);

    if (CC_EDITOR && deferredDestroyTimer === null && cc.engine && ! cc.engine._isUpdating) {
        // 在編輯器模式下可以立即銷毀
        deferredDestroyTimer = setImmediate(deferredDestroy);
    }
    return true;
};

// Director每一幀都會調(diào)用這個(gè)方法
function deferredDestroy () {
    var deleteCount = objectsToDestroy.length;
    for (var i = 0; i < deleteCount; ++i) {
        var obj = objectsToDestroy[i];
        if (!(obj._objFlags & Destroyed)) {
            obj._destroyImmediate();
        }
    }
    // 當(dāng)我們在a.onDestroy中調(diào)用b.destroy,objectsToDestroy數(shù)組的大小會變化,我們只銷毀在這次deferredDestroy之前objectsToDestroy中的元素
    if (deleteCount === objectsToDestroy.length) {
        objectsToDestroy.length = 0;
    }
    else {
        objectsToDestroy.splice(0, deleteCount);
    }

    if (CC_EDITOR) {
        deferredDestroyTimer = null;
    }
}

// 真正的資源釋放
prototype._destroyImmediate = function () {
    if (this._objFlags & Destroyed) {
        cc.errorID(5000);
        return;
    }
    // 執(zhí)行回調(diào)
    if (this._onPreDestroy) {
        this._onPreDestroy();
    }

    if ((CC_TEST ? (/* make CC_EDITOR mockable*/ Function('return !CC_EDITOR'))() : !CC_EDITOR) || cc.engine._isPlaying) {
        this._destruct();
    }

    this._objFlags |= Destroyed;
};

在這里_destruct做的事情就是將對象的屬性清空,比如將object類型的屬性置為null,將string類型的屬性置為'',compileDestruct方法會返回一個(gè)該類的析構(gòu)函數(shù),compileDestruct先收集了普通object和cc.Class這兩種類型下的所有屬性,并根據(jù)類型構(gòu)建了一個(gè)propsToReset用來清空屬性,支持JIT的情況下會根據(jù)要清空的屬性生成一個(gè)類似這樣的函數(shù)返回function(o) {o.a='';o.b=null;o.['c']=undefined...},而非JIT情況下會返回一個(gè)根據(jù)propsToReset遍歷處理的函數(shù),前者占用更多內(nèi)存,但效率更高。

prototype._destruct = function () {
    var ctor = this.constructor;
    var destruct = ctor.__destruct__;
    if (!destruct) {
        destruct = compileDestruct(this, ctor);
        js.value(ctor, '__destruct__', destruct, true);
    }
    destruct(this);
};

function compileDestruct (obj, ctor) {
    var shouldSkipId = obj instanceof cc._BaseNode || obj instanceof cc.Component;
    var idToSkip = shouldSkipId ? '_id' : null;

    var key, propsToReset = {};
    for (key in obj) {
        if (obj.hasOwnProperty(key)) {
            if (key === idToSkip) {
                continue;
            }
            switch (typeof obj[key]) {
                case 'string':
                    propsToReset[key] = '';
                    break;
                case 'object':
                case 'function':
                    propsToReset[key] = null;
                    break;
            }
        }
    }
    // Overwrite propsToReset according to Class
    if (cc.Class._isCCClass(ctor)) {
        var attrs = cc.Class.Attr.getClassAttrs(ctor);
        var propList = ctor.__props__;
        for (var i = 0; i < propList.length; i++) {
            key = propList[i];
            var attrKey = key + cc.Class.Attr.DELIMETER + 'default';
            if (attrKey in attrs) {
                if (shouldSkipId && key === '_id') {
                    continue;
                }
                switch (typeof attrs[attrKey]) {
                    case 'string':
                        propsToReset[key] = '';
                        break;
                    case 'object':
                    case 'function':
                        propsToReset[key] = null;
                        break;
                    case 'undefined':
                        propsToReset[key] = undefined;
                        break;
                }
            }
        }
    }

    if (CC_SUPPORT_JIT) {
        // compile code
        var func = '';
        for (key in propsToReset) {
            var statement;
            if (CCClass.IDENTIFIER_RE.test(key)) {
                statement = 'o.' + key + '=';
            }
            else {
                statement = 'o[' + CCClass.escapeForJS(key) + ']=';
            }
            var val = propsToReset[key];
            if (val === '') {
                val = '""';
            }
            func += (statement + val + ';\n');
        }
        return Function('o', func);
    }
    else {
        return function (o) {
            for (var key in propsToReset) {
                o[key] = propsToReset[key];
            }
        };
    }
}

那么_onPreDestroy又做了什么呢?主要是將各種事件、定時(shí)器進(jìn)行注銷,對子節(jié)點(diǎn)、組件等進(jìn)行刪除,詳情可以看下面這段代碼。

// Node的_onPreDestroy
_onPreDestroy () {
  // 調(diào)用_onPreDestroyBase方法,實(shí)際是調(diào)用BaseNode.prototype._onPreDestroy,這個(gè)方法下面介紹
  var destroyByParent = this._onPreDestroyBase();

  // 注銷Actions
  if (ActionManagerExist) {
      cc.director.getActionManager().removeAllActionsFromTarget(this);
  }

  // 移除_currentHovered
  if (_currentHovered === this) {
      _currentHovered = null;
  }

  this._bubblingListeners && this._bubblingListeners.clear();
  this._capturingListeners && this._capturingListeners.clear();

  // 移除所有觸摸和鼠標(biāo)事件監(jiān)聽
  if (this._touchListener || this._mouseListener) {
      eventManager.removeListeners(this);
      if (this._touchListener) {
          this._touchListener.owner = null;
          this._touchListener.mask = null;
          this._touchListener = null;
      }
      if (this._mouseListener) {
          this._mouseListener.owner = null;
          this._mouseListener.mask = null;
          this._mouseListener = null;
      }
  }

  if (CC_JSB && CC_NATIVERENDERER) {
      this._proxy.destroy();
      this._proxy = null;
  }

  // 回收到對象池中
  this._backDataIntoPool();

  if (this._reorderChildDirty) {
      cc.director.__fastOff(cc.Director.EVENT_AFTER_UPDATE, this.sortAllChildren, this);
  }

  if (!destroyByParent) {
      if (CC_EDITOR) {
          // 確保編輯模式下的,節(jié)點(diǎn)的被刪除后可以通過ctrl+z撤銷(重新添加到原來的父節(jié)點(diǎn))
          this._parent = null;
      }
  }
},

// BaseNode的_onPreDestroy
_onPreDestroy () {
  var i, len;

  // 加上Destroying標(biāo)記
  this._objFlags |= Destroying;
  var parent = this._parent;
  
  // 根據(jù)檢測父節(jié)點(diǎn)的標(biāo)記判斷是不是由父節(jié)點(diǎn)的destroy發(fā)起的釋放
  var destroyByParent = parent && (parent._objFlags & Destroying);
  if (!destroyByParent && (CC_EDITOR || CC_TEST)) {
      // 從編輯器中移除
      this._registerIfAttached(false);
  }

  // 把所有子節(jié)點(diǎn)進(jìn)行釋放,它們的_onPreDestroy也會被執(zhí)行
  var children = this._children;
  for (i = 0, len = children.length; i < len; ++i) {
      children[i]._destroyImmediate();
  }

  // 把所有的組件進(jìn)行釋放,它們的_onPreDestroy也會被執(zhí)行
  for (i = 0, len = this._components.length; i < len; ++i) {
      var component = this._components[i];
      component._destroyImmediate();
  }

  // 注銷事件監(jiān)聽,比如otherNode.on(type, callback, thisNode) 注冊了事件
  // thisNode被釋放時(shí),需要注銷otherNode身上的監(jiān)聽,避免事件回調(diào)到已銷毀的對象上
  var eventTargets = this.__eventTargets;
  for (i = 0, len = eventTargets.length; i < len; ++i) {
      var target = eventTargets[i];
      target && target.targetOff(this);
  }
  eventTargets.length = 0;

  // 如果自己是常駐節(jié)點(diǎn),則從常駐節(jié)點(diǎn)列表中移除
  if (this._persistNode) {
      cc.game.removePersistRootNode(this);
  }

  // 如果是自己釋放的自己,而不是從父節(jié)點(diǎn)釋放的,要通知父節(jié)點(diǎn),把這個(gè)失效的子節(jié)點(diǎn)移除掉
  if (!destroyByParent) {
      if (parent) {
          var childIndex = parent._children.indexOf(this);
          parent._children.splice(childIndex, 1);
          parent.emit && parent.emit('child-removed', this);
      }
  }

  return destroyByParent;
},

// Component的_onPreDestroy
_onPreDestroy () {
  // 移除ActionManagerExist和schedule
  if (ActionManagerExist) {
      cc.director.getActionManager().removeAllActionsFromTarget(this);
  }
  this.unscheduleAllCallbacks();

  // 移除所有的監(jiān)聽
  var eventTargets = this.__eventTargets;
  for (var i = eventTargets.length - 1; i >= 0; --i) {
      var target = eventTargets[i];
      target && target.targetOff(this);
  }
  eventTargets.length = 0;

  // 編輯器模式下停止監(jiān)控
  if (CC_EDITOR && !CC_TEST) {
      _Scene.AssetsWatcher.stop(this);
  }

  // destroyComp的實(shí)現(xiàn)為調(diào)用組件的onDestroy回調(diào),各個(gè)組件會在回調(diào)中銷毀自身的資源
  // 比如RigidBody3D組件會調(diào)用body的destroy方法,而Animation組件會調(diào)用stop方法
  cc.director._nodeActivator.destroyComp(this);

  // 將組件從節(jié)點(diǎn)身上移除
  this.node._removeComponent(this);
},    

3.5.4 資源釋放的問題

最后我們來聊一聊資源釋放的問題與定位,在加入引用計(jì)數(shù)后,最常見的問題還是沒有正確增減引用計(jì)數(shù)導(dǎo)致的內(nèi)存泄露(循環(huán)引用、少調(diào)用了decRef或多調(diào)用了addRef),以及正在使用的資源被釋放的問題(和內(nèi)存泄露相反,資源被提前釋放了)。

從目前的代碼來看,如果正確使用了引用計(jì)數(shù),新的資源底層是可以避免內(nèi)存泄露等問題的

這種問題怎么解決呢?首先是定位出哪些資源出了問題,如果是被提前釋放,我們可以直接定位到這個(gè)資源,如果是內(nèi)存泄露,當(dāng)我們發(fā)現(xiàn)問題時(shí)程序往往已經(jīng)占用了大量的內(nèi)存,這種情況下可以切換到一個(gè)空場景,并清理資源,把資源清理完后,可以檢查assets中殘留的資源是否有未被釋放的資源。

要了解資源為什么會泄露,可以通過跟蹤addRef和decRef的調(diào)用得到,下面提供了一個(gè)示例方法,用于跟蹤某資源的addRef和decRef調(diào)用,然后調(diào)用資源的dump方法打印出所有調(diào)用的堆棧:

public static traceObject(obj : cc.Asset) {
  let addRefFunc = obj.addRef;
  let decRefFunc = obj.decRef;
  let traceMap = new Map();

  obj.addRef = function() : cc.Asset {
      let stack = ResUtil.getCallStack(1);
      let cnt = traceMap.has(stack) ? traceMap.get(stack) + 1 : 1;
      traceMap.set(stack, cnt);
      return addRefFunc.apply(obj, arguments);
  }

  obj.decRef = function() : cc.Asset {
      let stack = ResUtil.getCallStack(1);
      let cnt = traceMap.has(stack) ? traceMap.get(stack) + 1 : 1;
      traceMap.set(stack, cnt);
      return decRefFunc.apply(obj, arguments);
  }

  obj['dump'] = function() {
      console.log(traceMap);
  }
}

以上就是剖析CocosCreator新資源管理系統(tǒng)的詳細(xì)內(nèi)容,更多關(guān)于CococCreator的資料,請關(guān)注腳本之家其他相關(guān)文章!

相關(guān)文章

最新評論

免费手机黄页网址大全| 日本欧美视频在线观看三区| av日韩在线免费播放| 久草电影免费在线观看| 偷拍自拍 中文字幕| 久久麻豆亚洲精品av| av高潮迭起在线观看| 亚洲特黄aaaa片| 亚洲推理片免费看网站| 91九色国产熟女一区二区| 亚洲精品福利网站图片| 国产精品久久久久久久女人18| 亚洲国产中文字幕啊啊啊不行了 | 日本男女操逼视频免费看| 精品视频中文字幕在线播放| 国产成人精品一区在线观看| sw137 中文字幕 在线| av高潮迭起在线观看| 欧美成人综合视频一区二区| 亚洲精品国产在线电影| 中出中文字幕在线观看| 国产精品国产三级国产精东| 欧美80老妇人性视频| 午夜频道成人在线91| 亚洲一区久久免费视频| 1区2区3区不卡视频| 激情图片日韩欧美人妻| 爆乳骚货内射骚货内射在线| 天码人妻一区二区三区在线看 | 欧美国品一二三产区区别| 男人靠女人的逼视频| 一级黄色av在线观看| 91九色porny国产在线| 性感美女高潮视频久久久| 亚洲欧美一区二区三区爱爱动图| 国产在线拍揄自揄视频网站| 成人sm视频在线观看| 中文字幕一区二区人妻电影冢本| 麻豆精品成人免费视频| 韩国女主播精品视频网站| 又色又爽又黄又刺激av网站| 亚洲国产欧美一区二区三区…| 人人妻人人澡人人爽人人dvl| 大鸡巴后入爆操大屁股美女| 久久久制服丝袜中文字幕| 天天操天天爽天天干| 动漫精品视频在线观看| 激情伦理欧美日韩中文字幕| av手机在线观播放网站| 日韩在线中文字幕色| 丰满熟女午夜福利视频| 久久精品在线观看一区二区| gogo国模私拍视频| 久久丁香花五月天色婷婷| 天天想要天天操天天干| 精彩视频99免费在线| 一区二区视频在线观看视频在线| 亚洲第一黄色在线观看| 大白屁股精品视频国产| 国产精品一二三不卡带免费视频 | 久久久人妻一区二区| 性感美女诱惑福利视频| 熟妇一区二区三区高清版| 中文字幕av一区在线观看| 日本www中文字幕| 97人妻夜夜爽二区欧美极品| 大鸡巴后入爆操大屁股美女| 97瑟瑟超碰在线香蕉| 男人的天堂在线黄色| 天天射,天天操,天天说| 天天日天天日天天擦| 日本阿v视频在线免费观看| 在线观看免费av网址大全| 天天日天天干天天舔天天射| 超pen在线观看视频公开97| 久久这里只有精品热视频| 高潮视频在线快速观看国家快速| 国产精品视频男人的天堂| 18禁免费av网站| 亚洲图库另类图片区| 一区二区三区麻豆福利视频| 精品91高清在线观看| 青青青青青青青青青青草青青| 国产熟妇人妻ⅹxxxx麻豆| 亚洲国产精品美女在线观看| 国产精品黄大片在线播放| 红杏久久av人妻一区| av在线免费观看亚洲天堂| 欧美日韩在线精品一区二区三| 亚洲最大黄了色网站| 亚洲精品一区二区三区老狼| a v欧美一区=区三区| jiuse91九色视频| 中文字幕欧美日韩射射一| av大全在线播放免费| 黄网十四区丁香社区激情五月天| 久久久久久久99精品| 极品粉嫩小泬白浆20p主播| 亚洲精品欧美日韩在线播放| 亚洲国产在人线放午夜| 黄色在线观看免费观看在线| 日韩美女精品视频在线观看网站| 晚上一个人看操B片| 在线视频这里只有精品自拍| 在线制服丝袜中文字幕| 日本五十路熟新垣里子| 91精品国产综合久久久蜜| 美女少妇亚洲精选av| av破解版在线观看| 色在线观看视频免费的| 老鸭窝日韩精品视频观看| 色爱av一区二区三区| 欧美成人综合视频一区二区| 国产成人午夜精品福利| 黄片大全在线观看观看| 老司机深夜免费福利视频在线观看| 日本三极片中文字幕| 淫秽激情视频免费观看| 国产精品亚洲在线观看| 特一级特级黄色网片| 综合国产成人在线观看| 制服丝袜在线人妻中文字幕| 不卡一区一区三区在线| 久久机热/这里只有| 中文字幕高清免费在线人妻| 欧美综合婷婷欧美综合| 人妻3p真实偷拍一二区| 啊用力插好舒服视频| 日韩近亲视频在线观看| 4个黑人操素人视频网站精品91| 黄色视频在线观看高清无码| 91久久国产成人免费网站| 精品一区二区三四区| 青青青青青青青青青青草青青| 在线视频这里只有精品自拍| 美女福利写真在线观看视频| okirakuhuhu在线观看| 国产日韩一区二区在线看| 青青青青爽手机在线| 中文字母永久播放1区2区3区| 日本真人性生活视频免费看| 天天插天天色天天日| 欧美日韩国产一区二区三区三州| 啊啊啊想要被插进去视频| 国产在线观看黄色视频| 日本少妇人妻xxxxxhd| 男人在床上插女人视频| 日本人妻少妇18—xx| 欧美日韩熟女一区二区三区| 在线观看成人国产电影| 91www一区二区三区| 亚洲视频乱码在线观看| 538精品在线观看视频| 日本三极片中文字幕| 男人的天堂av日韩亚洲| 五十路丰满人妻熟妇| 日本熟妇喷水xxx| 亚洲自拍偷拍综合色| 福利视频广场一区二区| 欧美精品黑人性xxxx| 亚洲熟女久久久36d| 风流唐伯虎电视剧在线观看| 特大黑人巨大xxxx| 狠狠的往里顶撞h百合| 91快播视频在线观看| 国产麻豆剧果冻传媒app| av在线观看网址av| 国产精品手机在线看片| 熟女人妻一区二区精品视频| 大鸡吧插逼逼视频免费看| 成年人该看的视频黄免费| 换爱交换乱高清大片| 国产精品久久久久网| 97人人模人人爽人人喊| 岛国毛片视频免费在线观看| 99精品国产免费久久| 天天草天天色天天干| 女生被男生插的视频网站| av久久精品北条麻妃av观看| 国产精品黄页网站视频| 亚洲精品欧美日韩在线播放| 大陆胖女人与丈夫操b国语高清| 精品区一区二区三区四区人妻| 玩弄人妻熟妇性色av少妇| 天天操,天天干,天天射| 日曰摸日日碰夜夜爽歪歪| 亚洲精品国产综合久久久久久久久| 国产在线自在拍91国语自产精品| 91福利视频免费在线观看| 欧美专区第八页一区在线播放| 国产黄色片在线收看| 日韩近亲视频在线观看| 欧美亚洲国产成人免费在线| 国产精品久久久久国产三级试频| 婷婷久久久久深爱网| 99精品免费久久久久久久久a| 2012中文字幕在线高清| 亚洲日本一区二区三区| 黑人解禁人妻叶爱071| 免费在线福利小视频| 亚洲色偷偷综合亚洲AV伊人| 国产精品黄页网站视频| 精品国产乱码一区二区三区乱| 日本18禁久久久久久| 色在线观看视频免费的| 欧美日本国产自视大全| 亚洲中文字幕人妻一区| 最近中文字幕国产在线| 青草久久视频在线观看| 93精品视频在线观看| 成年人的在线免费视频| 欧美视频中文一区二区三区| 男人操女人的逼免费视频| 绝色少妇高潮3在线观看| 99精品国产aⅴ在线观看| 93精品视频在线观看| 2020av天堂网在线观看| 91一区精品在线观看| 亚洲1区2区3区精华液| 国产麻豆剧果冻传媒app| 黄色的网站在线免费看| 青青青青草手机在线视频免费看| 熟女俱乐部一二三区| 丝袜国产专区在线观看| 欧美亚洲偷拍自拍色图| 日本中文字幕一二区视频| 日本一道二三区视频久久 | 日本男女操逼视频免费看| 国产一区自拍黄视频免费观看| 日韩精品电影亚洲一区| 午夜在线观看一区视频| 欧美精品一区二区三区xxxx| 99国内小视频在现欢看| 中文字幕亚洲久久久| 老师啊太大了啊啊啊尻视频| 亚洲午夜福利中文乱码字幕| 欧美一区二区三区乱码在线播放 | 日韩一区二区三区三州| 人妻av无码专区久久绿巨人| 又色又爽又黄又刺激av网站| 日韩成人性色生活片| 女警官打开双腿沦为性奴| 天堂v男人视频在线观看| 大鸡吧插入女阴道黄色片| 日韩美女搞黄视频免费| 经典国语激情内射视频| 在线成人日韩av电影| 欧美性受xx黑人性猛交| 2021天天色天天干| 国产实拍勾搭女技师av在线| 阴茎插到阴道里面的视频| 91精品国产综合久久久蜜| 日韩二区视频一线天婷婷五| 亚洲国产成人在线一区| 在线不卡成人黄色精品| 久久h视频在线观看| 色婷婷精品大在线观看| 国产综合高清在线观看| 国产刺激激情美女网站| 水蜜桃国产一区二区三区| 黄色男人的天堂视频| 中文字幕人妻一区二区视频| 成年人免费看在线视频| 亚洲av第国产精品| av视屏免费在线播放| 亚洲一区二区三区uij| 人人妻人人人操人人人爽| 男人天堂色男人av| 夜色福利视频在线观看| 日本18禁久久久久久| 免费无毒热热热热热热久| 91天堂精品一区二区| 欧美精产国品一二三产品价格| 中文字幕+中文字幕| 夜色撩人久久7777| 日韩精品激情在线观看| 欧美日本在线视频一区| 又粗又长 明星操逼小视频| 亚洲美女自偷自拍11页| 丁香花免费在线观看中文字幕| 超碰97人人澡人人| asmr福利视频在线观看| 含骚鸡巴玩逼逼视频| 欧美偷拍亚洲一区二区| 97精品视频在线观看| 亚洲一区二区三区偷拍女厕91 | 日韩欧美高清免费在线| 欧美日韩熟女一区二区三区| 国产日韩一区二区在线看 | 五色婷婷综合狠狠爱| 日韩av免费观看一区| av黄色成人在线观看| 天天日天天干天天要| 日本高清在线不卡一区二区| 欧美成人综合视频一区二区 | 91 亚洲视频在线观看| 99热色原网这里只有精品| 亚洲精品久久视频婷婷| 99热这里只有国产精品6| 日本裸体熟妇区二区欧美| 91精品综合久久久久3d动漫| 国产成人精品久久二区91| 我想看操逼黄色大片| 日韩成人免费电影二区| 操日韩美女视频在线免费看| 中文字幕中文字幕人妻| 欧美国品一二三产区区别| 91极品大一女神正在播放| 粉嫩欧美美人妻小视频| 中文字幕,亚洲人妻| 绯色av蜜臀vs少妇| 精品国产乱码一区二区三区乱| 人妻少妇性色欲欧美日韩| 亚洲福利天堂久久久久久| 丁香花免费在线观看中文字幕| 亚国产成人精品久久久| 亚洲福利天堂久久久久久| 黑人大几巴狂插日本少妇| 人妻另类专区欧美制服| 在线观看免费av网址大全| 日韩美女综合中文字幕pp| 激情色图一区二区三区| 亚洲国产免费av一区二区三区 | 99热久久这里只有精品8| 一区二区三区蜜臀在线| 亚洲综合另类精品小说| 视频 国产 精品 熟女 | 亚洲国产最大av综合| 亚洲精品乱码久久久本| 3344免费偷拍视频| 日本少妇在线视频大香蕉在线观看| 免费一级特黄特色大片在线观看| 精品久久久久久久久久久久人妻| 男人的天堂av日韩亚洲| 天天日天天天天天天天天天天| 93视频一区二区三区| 狍和女人的王色毛片| 熟女人妻一区二区精品视频| 欧美老妇精品另类不卡片| 中文字幕在线视频一区二区三区| 国产精品免费不卡av| 天堂v男人视频在线观看| 亚洲欧美成人综合在线观看| 国产福利在线视频一区| 亚洲精品乱码久久久本| 人妻久久无码中文成人| 国产精品大陆在线2019不卡 | 午夜福利资源综合激情午夜福利资| 3337p日本欧洲大胆色噜噜| 日韩欧美中文国产在线 | 亚洲午夜精品小视频| 色97视频在线播放| 成人国产小视频在线观看| 韩国爱爱视频中文字幕| 欧美国品一二三产区区别| 人妻在线精品录音叫床| 天堂v男人视频在线观看| 女同互舔一区二区三区| 人妻素人精油按摩中出| 欧美麻豆av在线播放| 欧美成人小视频在线免费看| 国产一区成人在线观看视频 | 红桃av成人在线观看| 日本韩国在线观看一区二区| 亚洲va国产va欧美精品88| 超碰在线观看免费在线观看| 在线免费观看99视频| 男女啪啪视频免费在线观看| 在线免费观看国产精品黄色| 2022中文字幕在线| 女同性ⅹxx女同hd| 偷拍自拍亚洲视频在线观看| 国产又粗又猛又爽又黄的视频在线| 韩国三级aaaaa高清视频| 亚洲av第国产精品| 中国熟女@视频91| 国产九色91在线视频| 午夜免费观看精品视频| 被大鸡吧操的好舒服视频免费 | 婷婷色国产黑丝少妇勾搭AV | 欧美精品资源在线观看| 制丝袜业一区二区三区| 1区2区3区4区视频在线观看| 人妻熟女在线一区二区| 啪啪啪啪啪啪啪啪啪啪黄色| 91www一区二区三区| 国产av欧美精品高潮网站| 成年午夜免费无码区| 久精品人妻一区二区三区| 成人色综合中文字幕| 2020久久躁狠狠躁夜夜躁| 天天干夜夜操天天舔| 最新97国产在线视频| 精品美女在线观看视频在线观看| 密臀av一区在线观看| 青青擦在线视频国产在线| 中国视频一区二区三区| 中文字幕第一页国产在线| 在线免费观看av日韩| 欧美一区二区三区高清不卡tv| 美洲精品一二三产区区别| 天天夜天天日天天日| 午夜av一区二区三区| 国产va精品免费观看| 欧美精品免费aaaaaa| 国产卡一卡二卡三乱码手机| 男女第一次视频在线观看| 国产视频一区在线观看| 青青草在观免费国产精品| 欧美日本在线观看一区二区| 又粗又硬又猛又爽又黄的| 岛国av高清在线成人在线| 男女啪啪视频免费在线观看 | 九九热99视频在线观看97| 在线观看国产网站资源| 亚洲在线免费h观看网站| 国产女人露脸高潮对白视频| 天天躁日日躁狠狠躁av麻豆| 久久人人做人人妻人人玩精品vr| 亚洲av在线观看尤物| 亚洲国产精品美女在线观看| 少妇一区二区三区久久久| 91色网站免费在线观看| 国产精品久久久黄网站| 国产精品一区二区三区蜜臀av| 一区二区三区精品日本| 这里只有精品双飞在线播放| 亚洲欧美综合在线探花| 国产一级精品综合av| 97国产在线观看高清| 在线观看一区二区三级| 99视频精品全部15| 欧美特级特黄a大片免费| 很黄很污很色的午夜网站在线观看 | 日噜噜噜夜夜噜噜噜天天噜噜噜| 一区二区免费高清黄色视频| 91国产在线视频免费观看| 日本人妻欲求不满中文字幕| 55夜色66夜色国产精品站| 区一区二区三国产中文字幕| 99精品久久久久久久91蜜桃| 中文字幕在线第一页成人| 亚洲欧美清纯唯美另类 | 久久久久久久精品成人热| 成人动漫大肉棒插进去视频| 特大黑人巨大xxxx| 中国视频一区二区三区| 国产日本精品久久久久久久| 国产精品大陆在线2019不卡| 懂色av之国产精品| 国产密臀av一区二区三| 国产黄色片在线收看| 久久麻豆亚洲精品av| 在线视频自拍第三页| 天天爽夜夜爽人人爽QC| 91一区精品在线观看| 91色网站免费在线观看| 在线观看免费视频色97| 超碰97人人澡人人| 日本后入视频在线观看| 亚洲一区二区三区久久午夜| 亚洲成人av在线一区二区| 一色桃子久久精品亚洲| 亚洲一级美女啪啪啪| 国产使劲操在线播放| 亚洲精品 日韩电影| 天天想要天天操天天干| 天天干天天操天天爽天天摸| 欧美va亚洲va天堂va| 午夜精品一区二区三区更新| 中文字幕最新久久久| 快插进小逼里大鸡吧视频| 性色蜜臀av一区二区三区| 天天躁夜夜躁日日躁a麻豆| 欧美成人精品欧美一级黄色| 国产极品精品免费视频| 亚洲一区制服丝袜美腿| 新婚人妻聚会被中出| 色偷偷伊人大杳蕉综合网| 综合精品久久久久97| 日本高清撒尿pissing| 日韩人妻在线视频免费| 丝袜肉丝一区二区三区四区在线看| 日本av熟女在线视频| 97黄网站在线观看| 日韩精品二区一区久久| 亚洲图库另类图片区| 色哟哟国产精品入口| 2018在线福利视频| 日本丰满熟妇大屁股久久| 天天色天天操天天舔| 精品国产污污免费网站入口自 | 日韩不卡中文在线视频网站| 国产成人精品av网站| 乱亲女秽乱长久久久| 天堂av在线官网中文| 唐人色亚洲av嫩草| 国产黄色片蝌蚪九色91| 91av中文视频在线| 国产精品久久久黄网站| 欧美怡红院视频在线观看| AV无码一区二区三区不卡| 国产又粗又猛又爽又黄的视频美国| 夜女神免费福利视频| 亚洲国产精品久久久久蜜桃| 91福利视频免费在线观看| 绯色av蜜臀vs少妇| 97超碰国语国产97超碰| xxx日本hd高清| 天天做天天干天天舔| 日本av高清免费网站| 亚洲国产精品中文字幕网站| 日曰摸日日碰夜夜爽歪歪| 日本黄在免费看视频| 夜色福利视频在线观看| 国产91久久精品一区二区字幕| 日韩av大胆在线观看| 精品av久久久久久久| 大香蕉伊人国产在线| 91av中文视频在线| 人妻丝袜诱惑我操她视频| 最新中文字幕免费视频| 欧美日韩熟女一区二区三区| 国产精品久久久久国产三级试频| 老熟妇xxxhd老熟女| 黄色成年网站午夜在线观看| 亚洲国际青青操综合网站| 青青草亚洲国产精品视频| 国产大鸡巴大鸡巴操小骚逼小骚逼| 亚洲特黄aaaa片| 国产美女一区在线观看| 欧美性受xx黑人性猛交| 亚洲精品av在线观看| 91九色porny蝌蚪国产成人| 国产精品一区二区三区蜜臀av| 国产高清在线在线视频| 黄色男人的天堂视频| 午夜精品久久久久久99热| 天天操夜夜骑日日摸| 国产女人露脸高潮对白视频| 天天做天天干天天操天天射| 97欧洲一区二区精品免费| 黄片色呦呦视频免费看| 热思思国产99re| 老司机深夜免费福利视频在线观看| 91精品资源免费观看| 爱有来生高清在线中文字幕| 国产精品自拍视频大全| 国产普通话插插视频| 欧美精品黑人性xxxx| 99精品国自产在线人| 中文字幕 亚洲av| 欧美偷拍亚洲一区二区| 人妻凌辱欧美丰满熟妇| 精品成人啪啪18免费蜜臀| 精品国产成人亚洲午夜| 91色九色porny| 亚洲免费福利一区二区三区| 午夜激情高清在线观看| 可以在线观看的av中文字幕| 成人乱码一区二区三区av| 免费成人va在线观看| 成人18禁网站在线播放| 欧美在线精品一区二区三区视频| 天天操天天爽天天干| 五月天色婷婷在线观看视频免费| 成人高潮aa毛片免费| 人妻无码色噜噜狠狠狠狠色| 偷拍美女一区二区三区| 日本高清撒尿pissing| 中文字幕网站你懂的| 哥哥姐姐综合激情小说| 抽查舔水白紧大视频| 北条麻妃高跟丝袜啪啪| 日本一道二三区视频久久 | 不卡一不卡二不卡三| 欧美爆乳肉感大码在线观看| 美女福利写真在线观看视频| av在线观看网址av| 国产精品伦理片一区二区| 大鸡巴插入美女黑黑的阴毛| 男人插女人视频网站| 中文字幕在线观看国产片| 91超碰青青中文字幕| 91色老99久久九九爱精品| 快插进小逼里大鸡吧视频| 视频一区 二区 三区 综合| 免费在线福利小视频| 亚洲精品 欧美日韩| 九一传媒制片厂视频在线免费观看 | 国产成人精品亚洲男人的天堂| 成年午夜影片国产片| 天天日天天干天天插舔舔| 日本高清成人一区二区三区| 亚洲av日韩高清hd| 黑人性生活视频免费看| 精品区一区二区三区四区人妻| 色av色婷婷人妻久久久精品高清| 成人蜜臀午夜久久一区| av一本二本在线观看| 国产中文精品在线观看| 青青青青青青青青青国产精品视频| 三级黄色亚洲成人av| 大屁股熟女一区二区三区| 偷拍美女一区二区三区| 自拍 日韩 欧美激情| 高潮喷水在线视频观看| 东游记中文字幕版哪里可以看到| 又粗又硬又猛又爽又黄的| 亚洲黄色av网站免费播放| 美女骚逼日出水来了| 美女少妇亚洲精选av| 1000部国产精品成人观看视频| 亚洲中文字幕国产日韩| 色噜噜噜噜18禁止观看| 亚洲av色香蕉一区二区三区| 亚洲老熟妇日本老妇| 亚洲码av无色中文| 日本少妇精品免费视频| 青青伊人一精品视频| 久久久精品欧洲亚洲av| 亚洲一区二区久久久人妻| 亚洲色偷偷综合亚洲AV伊人| 美女操逼免费短视频下载链接| 欧美日韩一级黄片免费观看| 亚洲av日韩精品久久久久久hd| 欧美亚洲偷拍自拍色图| av在线免费观看亚洲天堂| 春色激情网欧美成人| 黑人进入丰满少妇视频| 真实国模和老外性视频| 久草电影免费在线观看| 国产一区av澳门在线观看| 青青青青青操视频在线观看| 色秀欧美视频第一页| 国产午夜福利av导航| 久久久久只精品国产三级| 区一区二区三国产中文字幕| 亚洲1069综合男同| 欧美激情电影免费在线| 亚洲av日韩精品久久久久久hd| 国产麻豆剧果冻传媒app| AV无码一区二区三区不卡| 亚洲美女美妇久久字幕组| h国产小视频福利在线观看| 顶级尤物粉嫩小尤物网站| 亚洲av成人网在线观看| 中国黄色av一级片| 啊啊好大好爽啊啊操我啊啊视频| 桃色视频在线观看一区二区| 天天干天天操天天摸天天射| 亚洲精品av在线观看| 中文字幕视频一区二区在线观看 | 蜜桃精品久久久一区二区| 阴茎插到阴道里面的视频| 不卡一区一区三区在线| 日本最新一二三区不卡在线 | 亚洲成人熟妇一区二区三区| 91www一区二区三区| 无码精品一区二区三区人| 91色老99久久九九爱精品| 九九视频在线精品播放| 久久精品亚洲国产av香蕉| 中文字幕之无码色多多| 青青青青青青草国产| 手机看片福利盒子日韩在线播放| 无码中文字幕波多野不卡| 五月婷婷在线观看视频免费| 欧美在线偷拍视频免费看| 好太好爽好想要免费| 国产亚洲国产av网站在线| 国产高清女主播在线| 国产中文字幕四区在线观看| 老司机免费视频网站在线看| 99精品国自产在线人| 国产黄色高清资源在线免费观看| 在线观看911精品国产| 亚洲丝袜老师诱惑在线观看| 青青尤物在线观看视频网站| 可以在线观看的av中文字幕| 5528327男人天堂| 又粗又硬又猛又爽又黄的| 99热久久极品热亚洲| 精品美女久久久久久| 国产第一美女一区二区三区四区 | 亚洲图库另类图片区| 亚洲超碰97人人做人人爱| 护士小嫩嫩又紧又爽20p| 成年午夜影片国产片| 亚洲精品在线资源站| 韩国一级特黄大片做受| 欧美日韩情色在线观看| 亚洲成人av一区久久| 国产女人被做到高潮免费视频| 国产视频一区二区午夜| 亚洲 中文 自拍 另类 欧美| 91国内视频在线观看| 日韩美av高清在线| 一区二区三区av高清免费| 瑟瑟视频在线观看免费视频| 蜜桃视频17c在线一区二区| 午夜在线一区二区免费| 福利视频网久久91| yy96视频在线观看| 久久精品亚洲国产av香蕉| 黄色中文字幕在线播放| 亚洲欧美在线视频第一页| 日韩成人性色生活片| 日韩a级黄色小视频| 国产在线自在拍91国语自产精品| 中文字幕人妻av在线观看| 国产乱子伦精品视频潮优女| 天码人妻一区二区三区在线看 | 人妻少妇一区二区三区蜜桃| 中文字幕乱码av资源| 天天干夜夜操天天舔| 美女在线观看日本亚洲一区| 亚洲 图片 欧美 图片| 成人激情文学网人妻| 大香蕉福利在线观看| 国产精品自偷自拍啪啪啪| 一二三区在线观看视频| 久久99久久99精品影院| aⅴ五十路av熟女中出| 亚洲青青操骚货在线视频| 岛国一区二区三区视频在线| 肏插流水妹子在线乐播下载| okirakuhuhu在线观看| 中文字幕欧美日韩射射一| 精品人妻每日一部精品| 亚洲图片偷拍自拍区| 亚洲丝袜老师诱惑在线观看| 日韩亚国产欧美三级涩爱| japanese五十路熟女熟妇| 香港一级特黄大片在线播放| 又色又爽又黄的美女裸体| 国产视频网站国产视频| 免费黄页网站4188| 人人妻人人人操人人人爽| 成年人的在线免费视频| 把腿张开让我插进去视频| 午夜精品一区二区三区福利视频| 都市家庭人妻激情自拍视频| 水蜜桃国产一区二区三区| 一区二区三区另类在线| 成年人该看的视频黄免费| av中文在线天堂精品| 精品黑人巨大在线一区| 最近中文字幕国产在线| 日本三极片视频网站观看| 亚洲国产欧美一区二区丝袜黑人| 国产+亚洲+欧美+另类| 国产亚洲成人免费在线观看| 在线观看日韩激情视频| 99久久激情婷婷综合五月天| 黑人解禁人妻叶爱071| 男人的天堂一区二区在线观看| 日韩黄色片在线观看网站| 最新国产亚洲精品中文在线| 国产真实灌醉下药美女av福利| 97超碰最新免费在线观看| 人妻久久久精品69系列| 欧美特级特黄a大片免费| 美女 午夜 在线视频| 国产成人精品一区在线观看| 久久久久久cao我的性感人妻 | 人妻自拍视频中国大陆| 午夜频道成人在线91| 日本一本午夜在线播放| 91极品大一女神正在播放| 91久久综合男人天堂| 欧美精品国产综合久久| 国产在线观看黄色视频| 五月精品丁香久久久久福利社| 老司机深夜免费福利视频在线观看| 亚洲成人黄色一区二区三区| 综合国产成人在线观看| 国产成人小视频在线观看无遮挡| 区一区二区三国产中文字幕| 成人午夜电影在线观看 久久| 天天色天天操天天透| 日本韩国在线观看一区二区| 硬鸡巴动态操女人逼视频| 99一区二区在线观看| 国产一线二线三线的区别在哪| 日韩一个色综合导航| av中文字幕国产在线观看| 国产又大又黄免费观看| 四虎永久在线精品免费区二区| 亚洲av极品精品在线观看| 亚洲福利精品视频在线免费观看| 中国无遮挡白丝袜二区精品| 国产一线二线三线的区别在哪 | 欧美一级片免费在线成人观看| 青青青青在线视频免费观看| 久久免费看少妇高潮完整版| 亚洲一区二区三区在线高清| 国产又粗又硬又猛的毛片视频| 亚洲成人三级在线播放| 黄色三级网站免费下载| 天天爽夜夜爽人人爽QC| 亚洲人成精品久久久久久久| 久久丁香婷婷六月天| 天天爽夜夜爽人人爽QC| 国产精品视频资源在线播放 | 亚洲欧美一卡二卡三卡| 成熟丰满熟妇高潮xx×xx| 天天日天天日天天擦| 天天干天天搞天天摸| 欧美天堂av无线av欧美| 成人性爱在线看四区| 中文字幕1卡1区2区3区| 2020久久躁狠狠躁夜夜躁| 中字幕人妻熟女人妻a62v网| 在线国产日韩欧美视频| 被大鸡吧操的好舒服视频免费| 天天操天天弄天天射| 亚洲成人午夜电影在线观看| 国产精品久久久久久美女校花| 中国黄片视频一区91| 97国产在线av精品| 888亚洲欧美国产va在线播放| 新婚人妻聚会被中出| 日日夜夜大香蕉伊人| gay gay男男瑟瑟在线网站| 免费观看国产综合视频| 精品一区二区三区三区88| 免费在线福利小视频| 亚洲自拍偷拍综合色| 日本少妇精品免费视频| 啊啊啊视频试看人妻| yy96视频在线观看| 久久久久久久亚洲午夜综合福利 | 免费成人va在线观看| 成人免费公开视频无毒| 亚洲欧美自拍另类图片| 国产女人被做到高潮免费视频| 自拍 日韩 欧美激情| 日本阿v视频在线免费观看| 国产亚洲欧美45p| 99国内小视频在现欢看| 99久久久无码国产精品性出奶水| 在线免费观看亚洲精品电影| 色婷婷精品大在线观看| 中文字幕亚洲久久久| 日本一二三区不卡无| 成人国产影院在线观看| 亚洲一区二区三区五区 | 538精品在线观看视频| 97a片免费在线观看| 久久农村老妇乱69系列| 午夜在线观看岛国av,com| 神马午夜在线观看视频| 美女大bxxxx内射| 国产精品sm调教视频| 经典亚洲伊人第一页| 人妻凌辱欧美丰满熟妇| 国产va在线观看精品| 久精品人妻一区二区三区| 国产精品久久久久久久久福交| 玩弄人妻熟妇性色av少妇| 大学生A级毛片免费视频| 3D动漫精品啪啪一区二区下载| 久久久极品久久蜜桃| 亚洲精品国产综合久久久久久久久| 午夜精品九一唐人麻豆嫩草成人| 中文字幕av男人天堂| av男人天堂狠狠干| 夜色撩人久久7777| 搞黄色在线免费观看| 78色精品一区二区三区| 天天色天天爱天天爽| 欧美天堂av无线av欧美| 欧美精品伦理三区四区| 亚洲中文精品人人免费| 欧美激情精品在线观看| 精品成人午夜免费看| 色综合久久五月色婷婷综合| 骚逼被大屌狂草视频免费看| 老师啊太大了啊啊啊尻视频| 国产丰满熟女成人视频| 国产视频网站国产视频| 午夜极品美女福利视频| 亚洲av无码成人精品区辽| 欧美精产国品一二三产品价格| 成年人啪啪视频在线观看| 欧美中国日韩久久精品| 又色又爽又黄又刺激av网站| 精品美女久久久久久| xxx日本hd高清| 美女视频福利免费看| 黑人大几巴狂插日本少妇| 插小穴高清无码中文字幕| av天堂资源最新版在线看| 欧美精品激情在线最新观看视频| 黑人性生活视频免费看| tube69日本少妇| 99精品免费观看视频| 青青青青青青青在线播放视频| 激情色图一区二区三区| 成人亚洲精品国产精品| 欧美日韩v中文在线| 一区二区三区av高清免费| 高潮视频在线快速观看国家快速| 美女操逼免费短视频下载链接| 日日摸夜夜添夜夜添毛片性色av| 天天操天天干天天日狠狠插| 九一传媒制片厂视频在线免费观看 | 久久精品在线观看一区二区| ka0ri在线视频| 自拍偷区二区三区麻豆| 最近中文2019年在线看| 色婷婷六月亚洲综合香蕉| 日本少妇高清视频xxxxx| 亚洲天天干 夜夜操| 99精品国产aⅴ在线观看| 偷拍自拍亚洲美腿丝袜| 一色桃子久久精品亚洲| 一级黄片大鸡巴插入美女| 91九色porny蝌蚪国产成人| 亚洲成av人无码不卡影片一| 蜜桃臀av蜜桃臀av| 天天摸天天干天天操科普| 成人av久久精品一区二区| 大屁股熟女一区二区三区| 亚洲成av人无码不卡影片一| 亚洲国产免费av一区二区三区 | 超污视频在线观看污污污| 中文字幕—97超碰网| 日韩亚洲高清在线观看| av在线观看网址av| 青青青青草手机在线视频免费看| 搡老妇人老女人老熟女| 大屁股熟女一区二区三区| 色综合久久久久久久久中文| 色花堂在线av中文字幕九九 | 都市家庭人妻激情自拍视频| 在线观看日韩激情视频| 中文字幕在线观看国产片| 亚洲综合在线观看免费| 被大鸡吧操的好舒服视频免费| 国产精品自拍视频大全| 人人超碰国字幕观看97| 特大黑人巨大xxxx| 亚洲人妻国产精品综合| 婷婷色国产黑丝少妇勾搭AV| 日本高清在线不卡一区二区| 亚洲午夜电影在线观看| 色吉吉影音天天干天天操| av视网站在线观看| 人妻久久无码中文成人| 人人妻人人爽人人添夜| 自拍偷拍亚洲欧美在线视频| 97人妻色免费视频| 国产精品国产精品一区二区| 亚洲一区二区激情在线| 欲满人妻中文字幕在线| 中文字幕一区的人妻欧美日韩| 999九九久久久精品| 激情小视频国产在线| 亚洲av极品精品在线观看| 亚洲综合色在线免费观看| 亚洲av在线观看尤物| 精品视频国产在线观看| 日韩三级电影华丽的外出 | 国产精品日韩欧美一区二区| 亚洲高清国产一区二区三区| av中文字幕福利网| 护士小嫩嫩又紧又爽20p| 又粗又硬又猛又爽又黄的| 九色porny九色9l自拍视频| 天天色天天舔天天射天天爽| 97国产福利小视频合集| avjpm亚洲伊人久久| 久草视频福利在线首页| 中文字幕日韩人妻在线三区| 9l人妻人人爽人人爽| 亚洲福利精品视频在线免费观看| 日韩美女搞黄视频免费| 中文字日产幕乱六区蜜桃| 中文字幕日韩精品日本| 国产美女一区在线观看| 91快播视频在线观看| 亚洲国产最大av综合| 日韩午夜福利精品试看| 动漫精品视频在线观看| 日本韩国免费福利精品| 亚洲中文字幕人妻一区| 久久精品国产23696| 熟女视频一区,二区,三区| 天天干天天啪天天舔| 黄色片黄色片wyaa| 干逼又爽又黄又免费的视频| 亚洲av极品精品在线观看| 日韩av有码一区二区三区4| 韩国AV无码不卡在线播放| 亚洲熟妇无码一区二区三区| 欧美亚洲国产成人免费在线| 动漫av网站18禁| 精品一区二区三区欧美| 亚洲免费成人a v| 熟女人妻一区二区精品视频| 国产精品国产三级国产精东| 福利国产视频在线观看| 黄工厂精品视频在线观看| 91麻豆精品久久久久| av中文字幕国产在线观看| 操日韩美女视频在线免费看| 天天日天天添天天爽| 国产综合精品久久久久蜜臀| 日韩特级黄片高清在线看| 国产精品sm调教视频| 欧美老妇精品另类不卡片| 亚洲精品三级av在线免费观看| 福利视频网久久91| 91www一区二区三区| 91精品高清一区二区三区| 国产亚洲四十路五十路| 国产真实灌醉下药美女av福利| 超碰97免费人妻麻豆| 天天摸天天亲天天舔天天操天天爽| 久碰精品少妇中文字幕av| a v欧美一区=区三区| av在线shipin| 日本一道二三区视频久久| 91社福利《在线观看| japanese五十路熟女熟妇| 97人妻无码AV碰碰视频| av在线免费中文字幕| 55夜色66夜色国产精品站| 色婷婷综合激情五月免费观看 | 亚洲 欧美 自拍 偷拍 在线| 精品人妻每日一部精品| 日韩午夜福利精品试看| 欧美精产国品一二三产品价格 | 在线观看视频网站麻豆| 亚洲成人三级在线播放 | 插小穴高清无码中文字幕| 天天日天天干天天舔天天射| 999九九久久久精品| 欧美专区第八页一区在线播放| 中文字母永久播放1区2区3区 | 日本高清撒尿pissing| 亚洲人妻国产精品综合| 亚洲精品在线资源站| 男人天堂最新地址av| 亚洲第一黄色在线观看| 欧美乱妇无乱码一区二区| 亚洲人一区二区中文字幕| av森泽佳奈在线观看| 精品久久久久久久久久久a√国产| 少妇被强干到高潮视频在线观看| 伊拉克及约旦宣布关闭领空| 中文字幕最新久久久| 青娱乐极品视频青青草| 久草视频在线看免费| 99精品免费久久久久久久久a| 大香蕉伊人国产在线| 人人人妻人人澡人人| 国产精品黄大片在线播放| 玖玖一区二区在线观看| 又大又湿又爽又紧A视频| 巨乳人妻日下部加奈被邻居中出| 欧美日本在线观看一区二区| 国产三级精品三级在线不卡| 国产精品久久久久久久久福交| av在线免费中文字幕| 成人av电影免费版| 91‖亚洲‖国产熟女| 欧美在线精品一区二区三区视频 | 天天射夜夜操狠狠干| 亚洲va欧美va人人爽3p| 久久热久久视频在线观看| 人妻最新视频在线免费观看| 无码精品一区二区三区人| 91大屁股国产一区二区| 新97超碰在线观看| 中文字幕在线第一页成人| 精品视频一区二区三区四区五区| 国产精品三级三级三级| 亚洲国产欧美一区二区三区…| 男人的天堂在线黄色| 欧美老妇精品另类不卡片| 亚洲高清自偷揄拍自拍| 中文字幕1卡1区2区3区| 欧美日韩国产一区二区三区三州 | 午夜精品久久久久久99热| 国产大学生援交正在播放| 午夜蜜桃一区二区三区| 亚洲蜜臀av一区二区三区九色| 早川濑里奈av黑人番号| 成年人午夜黄片视频资源| 天天插天天色天天日| 视频一区二区在线免费播放| 国产a级毛久久久久精品| 只有精品亚洲视频在线观看| 午夜激情久久不卡一区二区| 顶级尤物粉嫩小尤物网站| 老熟妇xxxhd老熟女| 日韩中文字幕福利av| 激情啪啪啪啪一区二区三区| 亚洲成人线上免费视频观看| 欧美一级片免费在线成人观看| 国产精品人妻66p| 中文字幕一区二区三区人妻大片| 日韩美在线观看视频黄| 五月色婷婷综合开心网4438| 久久农村老妇乱69系列| 另类av十亚洲av| 欧美一区二区三区久久久aaa| 欧美一区二区三区在线资源| 亚欧在线视频你懂的| 久久一区二区三区人妻欧美| 91国内视频在线观看| 91麻豆精品传媒国产黄色片| 4个黑人操素人视频网站精品91| 日韩中文字幕福利av| 亚洲成人午夜电影在线观看| 91九色porny蝌蚪国产成人| 亚洲一区二区三区精品视频在线 | 91中文字幕免费在线观看| 亚洲av香蕉一区区二区三区犇| 大胸性感美女羞爽操逼毛片| 欧美视频不卡一区四区| 99人妻视频免费在线| japanese日本熟妇另类| 天天做天天爽夜夜做少妇| 亚洲最大黄了色网站| 亚洲综合在线观看免费| 亚洲1卡2卡三卡4卡在线观看| 国产精品中文av在线播放 | 亚洲成av人无码不卡影片一| 国产真实灌醉下药美女av福利| 成年午夜影片国产片| 五十路熟女人妻一区二区9933| 2018最新中文字幕在线观看 | 99婷婷在线观看视频| 国产又粗又硬又大视频| 天天躁夜夜躁日日躁a麻豆| 美女av色播在线播放| 亚洲精品无码色午夜福利理论片| www,久久久,com| 无码精品一区二区三区人 | 护士小嫩嫩又紧又爽20p| 91国内视频在线观看| 好吊视频—区二区三区| 密臀av一区在线观看| 国产va在线观看精品| 无忧传媒在线观看视频| 少妇被强干到高潮视频在线观看| 亚洲在线一区二区欧美| 中文字幕1卡1区2区3区| 操日韩美女视频在线免费看| 日本黄在免费看视频| 天天日天天玩天天摸| 欧洲欧美日韩国产在线| 亚洲va欧美va人人爽3p| 亚洲一区自拍高清免费视频| 欧亚乱色一区二区三区| 欧美日本在线观看一区二区| av破解版在线观看| 中文字幕在线乱码一区二区| 成人影片高清在线观看| 91在线视频在线精品3| 人妻丝袜av在线播放网址| 色呦呦视频在线观看视频| 国产麻豆剧传媒精品国产av蜜桃| 99精品久久久久久久91蜜桃| 无套猛戳丰满少妇人妻| 国产午夜男女爽爽爽爽爽视频 | 57pao国产一区二区| 98精产国品一二三产区区别| 神马午夜在线观看视频| 成人av在线资源网站| 亚洲午夜精品小视频| 国产亚洲视频在线二区| 99热这里只有精品中文| 日本一本午夜在线播放| 91精品啪在线免费| 自拍偷拍vs一区二区三区| 日本女大学生的黄色小视频| 任你操任你干精品在线视频| 18禁网站一区二区三区四区| 精品黑人巨大在线一区| 久久久91蜜桃精品ad| 久久机热/这里只有| 欧美精品免费aaaaaa| 91九色国产熟女一区二区| 一二三中文乱码亚洲乱码one | 亚洲天堂成人在线观看视频网站| 国产性感美女福利视频| 亚洲国产成人无码麻豆艾秋| 五十路av熟女松本翔子| 蜜桃色婷婷久久久福利在线| 操的小逼流水的文章| 亚洲国产香蕉视频在线播放| 国产精品一区二区久久久av| 欧洲国产成人精品91铁牛tv| 99国产精品窥熟女精品| 亚洲熟女综合色一区二区三区四区| 亚洲免费福利一区二区三区| 亚洲在线一区二区欧美| 适合午夜一个人看的视频| av手机在线免费观看日韩av| 高清一区二区欧美系列| 蜜桃臀av蜜桃臀av| 午夜频道成人在线91| heyzo蜜桃熟女人妻| 非洲黑人一级特黄片| 欧美国品一二三产区区别| av在线资源中文字幕| 福利午夜视频在线合集| 亚洲国产中文字幕啊啊啊不行了| 亚洲成人av一区久久| 欧美乱妇无乱码一区二区| 日本一道二三区视频久久| 色哟哟在线网站入口| 婷婷六月天中文字幕| 99亚洲美女一区二区三区| 夜夜骑夜夜操夜夜奸| 2012中文字幕在线高清| 亚洲免费va在线播放| 在线观看成人国产电影| 含骚鸡巴玩逼逼视频| 东游记中文字幕版哪里可以看到| 揄拍成人国产精品免费看视频 | 夜女神免费福利视频| 国产精品大陆在线2019不卡| 青青青青青青青青青青草青青| 日本女人一级免费片| 中文字幕日韩无敌亚洲精品| 亚洲激情偷拍一区二区| 一区二区在线视频中文字幕| 国产精品亚洲在线观看| 亚洲精品国产综合久久久久久久久| 国产亚洲精品品视频在线| 99re国产在线精品| 老司机福利精品免费视频一区二区| 青青色国产视频在线| 日本人妻少妇18—xx| 一区二区视频在线观看免费观看 | 亚洲欧美激情人妻偷拍| 成人免费公开视频无毒| 亚洲精品乱码久久久久久密桃明| 国产精品日韩欧美一区二区| 亚洲免费在线视频网站| 中文字幕高清资源站| 欧美成人一二三在线网| 专门看国产熟妇的网站| 国产精彩对白一区二区三区| 9l人妻人人爽人人爽| 欧美视频中文一区二区三区| 欧美一区二区三区激情啪啪啪| 99热久久极品热亚洲| 欧美日韩高清午夜蜜桃大香蕉| 日本精品视频不卡一二三| 9国产精品久久久久老师| 日韩精品中文字幕福利| 最后99天全集在线观看| 国产精品国产精品一区二区| 欧美精品国产综合久久| 中文字幕综合一区二区| 天天日天天添天天爽| 欧美一区二区三区啪啪同性| 亚洲美女美妇久久字幕组| 亚洲成人三级在线播放| 人妻自拍视频中国大陆| 在线 中文字幕 一区| 久久久精品欧洲亚洲av| 骚逼被大屌狂草视频免费看| 丝袜长腿第一页在线| 91精品国产黑色丝袜| 2021年国产精品自拍| 成人24小时免费视频| 日韩人妻xxxxx| 亚洲高清视频在线不卡| 天天干天天日天天谢综合156| 又粗又硬又猛又黄免费30| 午夜在线观看岛国av,com| 丰满的子国产在线观看| 女同性ⅹxx女同h偷拍| 日本女人一级免费片| 日日爽天天干夜夜操| 亚洲av色香蕉一区二区三区| 99av国产精品欲麻豆| 美女福利视频网址导航| 4个黑人操素人视频网站精品91| 人妻自拍视频中国大陆| 91中文字幕免费在线观看| 日本成人一区二区不卡免费在线| av老司机亚洲一区二区| 狠狠的往里顶撞h百合| 亚洲av男人天堂久久| 国产精品自拍在线视频| 天天综合天天综合天天网| 国产久久久精品毛片| 男人的天堂av日韩亚洲| 真实国模和老外性视频| 欧洲日韩亚洲一区二区三区| 天天插天天色天天日| av大全在线播放免费| 国产精品三级三级三级| 瑟瑟视频在线观看免费视频| 黄色av网站免费在线| 岛国av高清在线成人在线| 亚洲欧美人精品高清| 国产精品国产三级国产午| 精彩视频99免费在线| 天天日天天爽天天干| 精品国产成人亚洲午夜| 国产午夜福利av导航| av在线免费资源站| 亚洲精品国产久久久久久| 最新国产亚洲精品中文在线| 久久h视频在线观看| 久草视频 久草视频2| 岛国av高清在线成人在线| 蜜臀成人av在线播放| sspd152中文字幕在线| 动漫黑丝美女的鸡巴| 亚洲视频在线观看高清| 国产熟妇一区二区三区av| 中文字幕日韩精品就在这里| 视频在线亚洲一区二区| 午夜av一区二区三区| 日本三极片中文字幕| 亚洲国产在人线放午夜| 国产精品一区二区久久久av| 影音先锋女人av噜噜色| 成人动漫大肉棒插进去视频| 亚洲一区二区激情在线| 日韩av免费观看一区| v888av在线观看视频| 久草视频福利在线首页| 国产成人一区二区三区电影网站| 91免费福利网91麻豆国产精品 | 日本女大学生的黄色小视频| 亚洲免费成人a v| 国产一区成人在线观看视频| 大鸡巴操娇小玲珑的女孩逼| 亚洲老熟妇日本老妇| 偷拍自拍 中文字幕| 中文字幕人妻一区二区视频| av高潮迭起在线观看| 2020久久躁狠狠躁夜夜躁| 国产在线免费观看成人| 亚洲Av无码国产综合色区| 天美传媒mv视频在线观看| 欧美亚洲免费视频观看| 夜夜嗨av蜜臀av| 人妻少妇亚洲精品中文字幕| 精品少妇一二三视频在线| 日日夜夜精品一二三| 国产高清精品一区二区三区| 天天夜天天日天天日| 福利午夜视频在线合集| 男人在床上插女人视频| 人妻丝袜诱惑我操她视频| 91色秘乱一区二区三区| 阿v天堂2014 一区亚洲| 日本18禁久久久久久| 国产精选一区在线播放| 亚洲1区2区3区精华液| 精品91自产拍在线观看一区| 99精品国产免费久久| 视频一区二区综合精品| 亚洲激情唯美亚洲激情图片| 亚洲综合自拍视频一区| 抽查舔水白紧大视频| 国产精品欧美日韩区二区| 美女小视频网站在线| 在线成人日韩av电影| 日本熟妇色熟妇在线观看| 精内国产乱码久久久久久| 超碰中文字幕免费观看| 男人的网址你懂的亚洲欧洲av| 人妻丝袜精品中文字幕| 四川五十路熟女av| 日韩精品激情在线观看| 日本欧美视频在线观看三区| 国产美女精品福利在线| 欧美日韩精品永久免费网址| 亚洲免费国产在线日韩| 88成人免费av网站| 中文字幕一区二 区二三区四区| 色97视频在线播放| 偷拍自拍国产在线视频| 天堂av中文在线最新版| 粉嫩av蜜乳av蜜臀| 鸡巴操逼一级黄色气| 亚洲区美熟妇久久久久| 玖玖一区二区在线观看| 黄片三级三级三级在线观看| 亚洲久久午夜av一区二区| 18禁美女羞羞免费网站| 亚洲图片欧美校园春色| 国产亚州色婷婷久久99精品| 亚洲精品在线资源站| 国产一线二线三线的区别在哪| 视频在线免费观看你懂得| 成人区人妻精品一区二视频| 久久这里只有精彩视频免费| 91麻豆精品久久久久| 免费国产性生活视频| 一级黄片大鸡巴插入美女| 亚洲 中文 自拍 另类 欧美 | 中文字幕亚洲中文字幕| 91快播视频在线观看| 成人蜜臀午夜久久一区| 老有所依在线观看完整版| 欧美在线偷拍视频免费看| 少妇一区二区三区久久久| 99精品国产免费久久| 99人妻视频免费在线| 狠狠躁狠狠爱网站视频| 日日夜夜大香蕉伊人| 最后99天全集在线观看| 亚洲男人的天堂a在线| 久久久制服丝袜中文字幕| 久久一区二区三区人妻欧美| 一区二区三区激情在线| 精品美女久久久久久| 日本熟妇喷水xxx| 欧美美女人体视频一区| 国产欧美日韩在线观看不卡| 国产又粗又硬又大视频| 亚洲中文字幕人妻一区| 中文字幕人妻三级在线观看| 亚洲高清国产自产av| 日韩二区视频一线天婷婷五| 亚洲无线观看国产高清在线| 亚洲人一区二区中文字幕| 国产黄色大片在线免费播放 | 黄工厂精品视频在线观看| 久久久久久性虐视频| 综合页自拍视频在线播放| 天天干狠狠干天天操| 国产精品久久久久久久女人18| 一二三中文乱码亚洲乱码one| 久久精品久久精品亚洲人| 婷婷六月天中文字幕| 黄色资源视频网站日韩| 国产日韩欧美美利坚蜜臀懂色| 大陆精品一区二区三区久久| 男生舔女生逼逼的视频| 精品一区二区三区欧美| 亚洲精品欧美日韩在线播放| 亚洲熟妇无码一区二区三区| 夜夜嗨av一区二区三区中文字幕| 国产精品三级三级三级| 国产精品日韩欧美一区二区| 亚洲熟女女同志女同| 成年人黄色片免费网站| 午夜精品一区二区三区4| 欧美3p在线观看一区二区三区| 午夜精品福利一区二区三区p| 亚洲精品福利网站图片| 日韩一区二区电国产精品| 亚洲高清自偷揄拍自拍| 蜜臀av久久久久久久| 亚洲日产av一区二区在线| 男人的网址你懂的亚洲欧洲av| 可以在线观看的av中文字幕| av视屏免费在线播放| 91精品综合久久久久3d动漫| 青青青青操在线观看免费| 日韩精品中文字幕福利| 丝袜亚洲另类欧美变态| 日本熟妇喷水xxx| 自拍偷拍 国产资源| 亚洲欧美另类自拍偷拍色图| 玩弄人妻熟妇性色av少妇| 成人亚洲国产综合精品| 快插进小逼里大鸡吧视频| 亚洲欧美综合在线探花| 国产乱子伦精品视频潮优女| 日本在线一区二区不卡视频| 影音先锋女人av噜噜色| 亚洲国产中文字幕啊啊啊不行了| 国产麻豆乱子伦午夜视频观看| 欧美男同性恋69视频| 老司机你懂得福利视频| 97少妇精品在线观看| 影音先锋女人av噜噜色| 日日操夜夜撸天天干| 黄色大片免费观看网站| 亚洲自拍偷拍综合色| 国产不卡av在线免费| 日韩熟女系列一区二区三区| 岛国免费大片在线观看| 中文字幕网站你懂的| 91一区精品在线观看| 在线视频国产欧美日韩| 性欧美激情久久久久久久| 人妻爱爱 中文字幕| av欧美网站在线观看| av久久精品北条麻妃av观看| 丁香花免费在线观看中文字幕| av在线播放国产不卡| 视频在线亚洲一区二区| 国产精品国色综合久久| japanese五十路熟女熟妇| 日韩影片一区二区三区不卡免费| 欧美va亚洲va天堂va| 91精品国产91久久自产久强| 精品人妻一二三区久久| 夜夜嗨av一区二区三区中文字幕| 色97视频在线播放| 91片黄在线观看喷潮| 红杏久久av人妻一区| 91色九色porny| 综合页自拍视频在线播放| 一区二区视频视频视频| 国产亚洲欧美45p| 操操网操操伊剧情片中文字幕网| 男女啪啪视频免费在线观看| 亚洲的电影一区二区三区| 又粗又硬又猛又爽又黄的| 国产自拍黄片在线观看| 国产真实乱子伦a视频| 亚洲午夜高清在线观看| 快插进小逼里大鸡吧视频| 欧美亚洲中文字幕一区二区三区| 青青尤物在线观看视频网站| 亚洲国产在线精品国偷产拍| 国产精品国色综合久久| 日韩中文字幕福利av| 在线不卡成人黄色精品| 日本男女操逼视频免费看| 国产麻豆精品人妻av| 婷婷久久一区二区字幕网址你懂得| 18禁美女羞羞免费网站| 新97超碰在线观看| 青青青aaaa免费| 婷婷激情四射在线观看视频| 亚洲av自拍偷拍综合| 99热这里只有精品中文| 黄色资源视频网站日韩| 99精品国产免费久久| 国产伦精品一区二区三区竹菊| 蜜臀av久久久久久久| 人妻在线精品录音叫床| 在线观看一区二区三级| 午夜精品福利一区二区三区p| 少妇人妻100系列| 黄工厂精品视频在线观看| av在线观看网址av| 亚洲一区二区激情在线| 水蜜桃国产一区二区三区| 免费人成黄页网站在线观看国产| 国产精品自拍在线视频| 婷婷六月天中文字幕| 亚洲av男人天堂久久| 在线免费视频 自拍| 热99re69精品8在线播放| 任我爽精品视频在线播放| 91av精品视频在线| av日韩在线观看大全| av中文字幕电影在线看| 欧美亚洲少妇福利视频| 黄页网视频在线免费观看| 国产美女一区在线观看| 熟女少妇激情五十路| 中文字幕人妻熟女在线电影| 亚洲公开视频在线观看| 国产精品手机在线看片| jiuse91九色视频| 亚洲福利精品视频在线免费观看| caoporn蜜桃视频| 日本少妇的秘密免费视频| 在线视频国产欧美日韩| 欧美第一页在线免费观看视频| 沈阳熟妇28厘米大战黑人| 国产在线观看免费人成短视频| 91精品国产黑色丝袜| 老师啊太大了啊啊啊尻视频| 综合国产成人在线观看| 97年大学生大白天操逼| 一个人免费在线观看ww视频| 视频一区二区在线免费播放| 国产va在线观看精品| 偷拍自拍福利视频在线观看| 综合激情网激情五月五月婷婷| 黑人解禁人妻叶爱071| 日本xx片在线观看| 亚洲另类图片蜜臀av| 国产日韩精品一二三区久久久| 大肉大捧一进一出好爽在线视频| 日本18禁久久久久久| 国产亚洲视频在线观看| 精品亚洲国产中文自在线| 黄色片黄色片wyaa| 99久久中文字幕一本人| 亚洲激情,偷拍视频| 91国内精品自线在拍白富美| 亚洲超碰97人人做人人爱| 亚洲一级美女啪啪啪| 十八禁在线观看地址免费| 91免费观看在线网站| 亚洲粉嫩av一区二区三区| 美女大bxxxx内射| 啪啪啪18禁一区二区三区| 国产成人午夜精品福利| 一区二区三区国产精选在线播放| 人妻少妇亚洲一区二区| 久久久久五月天丁香社区| 班长撕开乳罩揉我胸好爽| 国产日韩av一区二区在线| 中文字幕在线永久免费播放| 自拍偷拍一区二区三区图片| 丝袜肉丝一区二区三区四区在线| 午夜激情精品福利视频| 人妻另类专区欧美制服| 日本韩国免费一区二区三区视频| 亚洲一区av中文字幕在线观看| 蜜桃久久久久久久人妻| 亚洲公开视频在线观看| 99热这里只有国产精品6| 成人影片高清在线观看| 欧美激情精品在线观看| 天天艹天天干天天操| 日曰摸日日碰夜夜爽歪歪| 91久久国产成人免费网站| 国产九色91在线视频| 成人av亚洲一区二区| 中国熟女一区二区性xx| 日韩在线视频观看有码在线| 国产黄色大片在线免费播放| 日本美女成人在线视频| 欧美精品国产综合久久| 国产精品国产精品一区二区| 中文字幕1卡1区2区3区| 日本一区二区三区免费小视频| av一本二本在线观看| 国产清纯美女al在线| 国产揄拍高清国内精品对白| 在线观看av亚洲情色| 国产综合视频在线看片| 一区二区三区四区中文| 国产精选一区在线播放| 亚洲精品成人网久久久久久小说 | gay gay男男瑟瑟在线网站| 伊人综合免费在线视频| 激情人妻校园春色亚洲欧美| 大香蕉伊人国产在线| 中文字幕一区二 区二三区四区| 国产中文字幕四区在线观看| 99国内小视频在现欢看| 日本性感美女写真视频| 在线新三级黄伊人网| 国产午夜亚洲精品麻豆| 青青青aaaa免费| 美女在线观看日本亚洲一区| 欧美美女人体视频一区| 激情伦理欧美日韩中文字幕| 伊人情人综合成人久久网小说 | 91精品免费久久久久久| 高清成人av一区三区| 国产高清在线观看1区2区| 91在线免费观看成人| 青青色国产视频在线| 人妻熟女在线一区二区| 国产精品国产精品一区二区| 中文字幕日韩91人妻在线| 天天日天天干天天插舔舔| 欧美第一页在线免费观看视频| 天堂av中文在线最新版| 偷拍3456eee| 国产午夜无码福利在线看| 久久久久久久精品成人热| 懂色av蜜桃a v| 亚洲美女自偷自拍11页| 亚洲国产欧美一区二区丝袜黑人| 亚洲日本一区二区三区 | 粗大的内捧猛烈进出爽大牛汉子| 欧美日韩不卡一区不区二区| 午夜dv内射一区区| 伊人精品福利综合导航| 国产视频网站国产视频| 欧美成人黄片一区二区三区| 9l人妻人人爽人人爽| 久久久精品国产亚洲AV一| 91久久精品色伊人6882| 亚洲av日韩av网站| 亚洲国产中文字幕啊啊啊不行了 | 女同久久精品秋霞网| 91精品激情五月婷婷在线| 激情色图一区二区三区| 亚洲综合另类精品小说| 精品一区二区三区三区88| 99热99re在线播放| 欧美伊人久久大香线蕉综合| 女生自摸在线观看一区二区三区 | 日本精品视频不卡一二三| 日美女屁股黄邑视频| 亚洲1区2区3区精华液| 欧美专区第八页一区在线播放| 国产内射中出在线观看| 日本熟妇色熟妇在线观看| 好吊操视频这里只有精品| 国产一区二区欧美三区| av网址国产在线观看| 欧美一级片免费在线成人观看| 国产亚洲国产av网站在线| 亚洲av无硬久久精品蜜桃| 护士小嫩嫩又紧又爽20p| 农村胖女人操逼视频| 亚洲国产免费av一区二区三区| 人人在线视频一区二区| 亚洲福利精品视频在线免费观看| 人妻少妇一区二区三区蜜桃| 精品国产污污免费网站入口自| 亚洲1069综合男同| 久久丁香花五月天色婷婷| 亚洲国产在人线放午夜| 国产麻豆乱子伦午夜视频观看| 这里只有精品双飞在线播放| 欧美性受xx黑人性猛交| 97年大学生大白天操逼| 人妻无码中文字幕专区| 国产91久久精品一区二区字幕| 一级黄片久久久久久久久| 人妻丝袜av在线播放网址| 国产一区二区在线欧美| 精品老妇女久久9g国产| 2020国产在线不卡视频| 久久精品国产23696| 亚洲图片偷拍自拍区| 涩爱综合久久五月蜜臀| 91老熟女连续高潮对白| 中文字幕中文字幕 亚洲国产| 日韩无码国产精品强奸乱伦| 社区自拍揄拍尻屁你懂的 | 硬鸡巴动态操女人逼视频| 五十路人妻熟女av一区二区| 中文字幕高清资源站| 亚洲最大黄了色网站| 国产视频一区在线观看| 国产性感美女福利视频| 国产一区av澳门在线观看| 在线观看免费岛国av| 日韩黄色片在线观看网站| 91麻豆精品传媒国产黄色片| 久久永久免费精品人妻专区| 偷拍自拍视频图片免费| 93人妻人人揉人人澡人人| 中文字幕第三十八页久久| 亚洲欧美另类手机在线| 97瑟瑟超碰在线香蕉| 操的小逼流水的文章| 91久久综合男人天堂| 99热99这里精品6国产| 亚洲免费成人a v| 99热久久极品热亚洲| 亚洲成人激情av在线| gogo国模私拍视频| 2022国产综合在线干| 大学生A级毛片免费视频| 日韩av免费观看一区| 亚洲国产40页第21页| 天天想要天天操天天干| 成人激情文学网人妻| 曰本无码人妻丰满熟妇啪啪| 国产普通话插插视频| 年轻的人妻被夫上司侵犯| 18禁污污污app下载| 天天做天天爽夜夜做少妇| 成人av在线资源网站| 国产91嫩草久久成人在线视频| 一区二区三区综合视频| 五十路在线观看完整版| 国产极品精品免费视频| 在线视频精品你懂的| 黄色三级网站免费下载| 老司机欧美视频在线看| 成人网18免费视频版国产| 日韩美女福利视频网| 懂色av之国产精品| 老司机你懂得福利视频| 免费在线看的黄片视频| 久久久久久久99精品| 一区二区三区欧美日韩高清播放| 超碰在线中文字幕一区二区| 国产chinesehd精品麻豆| 中出中文字幕在线观看| 午夜精品亚洲精品五月色| 在线国产精品一区二区三区| 欧美美女人体视频一区| 日韩av免费观看一区| 亚洲精品乱码久久久本| 色综合天天综合网国产成人| 色噜噜噜噜18禁止观看| 2020韩国午夜女主播在线| 欧美另类一区二区视频| 白嫩白嫩美女极品国产在线观看| 色秀欧美视频第一页| 亚洲av天堂在线播放| 年轻的人妻被夫上司侵犯| 成人免费做爰高潮视频| 久久这里只有精品热视频| 2021久久免费视频| 婷婷综合蜜桃av在线| 久久久久久性虐视频| 天天躁夜夜躁日日躁a麻豆| 成人伊人精品色xxxx视频| 久久综合老鸭窝色综合久久 | 欧美老妇精品另类不卡片| 五月婷婷在线观看视频免费| www日韩a级s片av| 男人的网址你懂的亚洲欧洲av| 韩国爱爱视频中文字幕| 粉嫩av蜜乳av蜜臀| 在线免费观看亚洲精品电影| 欧美日韩国产一区二区三区三州| 久久久久久久久久一区二区三区 | 精品一区二区三区午夜| 国产精品国产三级国产午| 欧美专区日韩专区国产专区| 深田咏美亚洲一区二区| 欲满人妻中文字幕在线| 亚洲va欧美va人人爽3p| www,久久久,com| 99久久超碰人妻国产| 日韩人妻在线视频免费| 99热久久极品热亚洲| 快插进小逼里大鸡吧视频| 亚洲av无乱一区二区三区性色| 国产精品女邻居小骚货| 亚洲卡1卡2卡三卡四老狼| 亚洲欧美成人综合在线观看| 婷婷色中文亚洲网68| 女生自摸在线观看一区二区三区 | 天天日天天舔天天射进去| 天天干天天操天天扣| 美女张开两腿让男人桶av| 中文字幕人妻熟女在线电影| 新婚人妻聚会被中出| 精品少妇一二三视频在线| 18禁无翼鸟成人在线| 亚洲男人让女人爽的视频| 色天天天天射天天舔| 国产高清女主播在线| 亚洲国产美女一区二区三区软件| 日韩a级黄色小视频| 中文字幕人妻三级在线观看| 国产在线91观看免费观看| 自拍偷拍日韩欧美亚洲| 水蜜桃国产一区二区三区| av亚洲中文天堂字幕网| 中文字幕高清在线免费播放| 欧美aa一级一区三区四区 | 清纯美女在线观看国产| 国产一级麻豆精品免费| 国产精品久久久久久美女校花| 在线观看视频一区麻豆| 男人的天堂av日韩亚洲| 成年人免费看在线视频| 黑人解禁人妻叶爱071| 日本高清成人一区二区三区| 女生被男生插的视频网站| 午夜久久久久久久精品熟女| 国产精品国色综合久久| 2022国产综合在线干| 91国语爽死我了不卡| 亚洲护士一区二区三区| 中文字幕人妻熟女在线电影| 欧美麻豆av在线播放| 韩国三级aaaaa高清视频| 女生自摸在线观看一区二区三区 | 91人妻精品一区二区久久| 国产精品伦理片一区二区| 国产又大又黄免费观看| 最新日韩av传媒在线| 一级黄片大鸡巴插入美女| 91国偷自产一区二区三区精品| 精品高跟鞋丝袜一区二区| 中文字幕亚洲久久久| 亚洲激情偷拍一区二区| 免费一级特黄特色大片在线观看| 天天操天天干天天插| 日曰摸日日碰夜夜爽歪歪| 激情五月婷婷免费视频| 超碰公开大香蕉97| 日本免费视频午夜福利视频| 在线国产中文字幕视频| 果冻传媒av一区二区三区| 国产亚洲视频在线观看| 精品人妻一二三区久久| 天天干夜夜操天天舔| 亚洲国产香蕉视频在线播放| 亚洲国产成人无码麻豆艾秋| 青青社区2国产视频| 日韩a级黄色小视频| 天天色天天舔天天射天天爽| 欧美另类z0z变态| 一区二区视频在线观看免费观看| 欧美另类一区二区视频| 91she九色精品国产| 黑人3p华裔熟女普通话| 一区二区三区麻豆福利视频| 小穴多水久久精品免费看| 第一福利视频在线观看| 大陆av手机在线观看| 中文字幕一区二区人妻电影冢本| 国产男女视频在线播放| 在线观看黄色成年人网站 | okirakuhuhu在线观看| 亚洲国产欧美一区二区三区久久 | 欧美黄片精彩在线免费观看| 热思思国产99re| 2022天天干天天操| 亚洲成人午夜电影在线观看| 人妻爱爱 中文字幕| 动色av一区二区三区| 国产欧美精品一区二区高清| chinese国产盗摄一区二区 | 2020国产在线不卡视频| 绝顶痉挛大潮喷高潮无码 | 久久尻中国美女视频| 直接能看的国产av| 欧美交性又色又爽又黄麻豆| 久久机热/这里只有| 97资源人妻免费在线视频| 国产日韩精品一二三区久久久| 国产 在线 免费 精品| 和邻居少妇愉情中文字幕| 精品国产污污免费网站入口自| av在线免费中文字幕| 日本人妻精品久久久久久| 成人久久精品一区二区三区| 日本福利午夜电影在线观看| 中文字幕高清在线免费播放| 黄工厂精品视频在线观看| 日本后入视频在线观看 | 91大神福利视频网| 亚洲综合在线观看免费| 任你操视频免费在线观看| 爱爱免费在线观看视频| 国产精品午夜国产小视频| 国产午夜亚洲精品麻豆| 沙月文乃人妻侵犯中文字幕在线| 国产精品免费不卡av| av手机在线免费观看日韩av| 无忧传媒在线观看视频| 超碰97人人澡人人| 91快播视频在线观看| 国产麻豆乱子伦午夜视频观看| 欧美亚洲牲夜夜综合久久| 特大黑人巨大xxxx| 国产亚洲成人免费在线观看| 秋霞午夜av福利经典影视| 亚洲最大免费在线观看| 欧美成人精品欧美一级黄色| 亚洲美女高潮喷浆视频| 91亚洲精品干熟女蜜桃频道 | 免费av岛国天堂网站| 97精品综合久久在线| 日本美女成人在线视频| 少妇人妻真实精品视频| 综合色区亚洲熟妇shxstz| 亚洲欧美成人综合在线观看| 啊慢点鸡巴太大了啊舒服视频| 天天操天天干天天艹| 亚洲伊人色一综合网| 秋霞午夜av福利经典影视| 999九九久久久精品| v888av在线观看视频| 国产不卡av在线免费| 哥哥姐姐综合激情小说| 男人的天堂在线黄色| av森泽佳奈在线观看| 任我爽精品视频在线播放| 把腿张开让我插进去视频| 人妻久久无码中文成人| 国产1区,2区,3区| 黄色男人的天堂视频| 欧美日韩情色在线观看| 国产高清在线观看1区2区| 熟妇一区二区三区高清版| 激情小视频国产在线| 一区二区视频在线观看免费观看| 日韩不卡中文在线视频网站| 天天日天天干天天插舔舔| 婷婷久久一区二区字幕网址你懂得| 98视频精品在线观看| 9色在线视频免费观看| 国产一区二区视频观看| 亚洲成人av在线一区二区| 亚洲成人线上免费视频观看| av乱码一区二区三区| 91精品国产91青青碰| 日本高清撒尿pissing| 亚洲在线观看中文字幕av| 5528327男人天堂| 亚洲在线免费h观看网站| 国产熟妇人妻ⅹxxxx麻豆| 精品一区二区三区三区88| 亚欧在线视频你懂的| 亚洲av人人澡人人爽人人爱| ka0ri在线视频| 国产大鸡巴大鸡巴操小骚逼小骚逼| 天天干夜夜操天天舔| 亚洲免费福利一区二区三区| 北条麻妃av在线免费观看| 日韩伦理短片在线观看| 超污视频在线观看污污污| 国产性生活中老年人视频网站| 操日韩美女视频在线免费看| 美女 午夜 在线视频| 成人动漫大肉棒插进去视频| 人妻3p真实偷拍一二区| 老鸭窝在线观看一区| 2022天天干天天操| 91亚洲手机在线视频播放| 91久久人澡人人添人人爽乱| 亚洲精品国偷自产在线观看蜜桃| jiujiure精品视频在线| 中文字幕—97超碰网| 在线免费91激情四射| 97黄网站在线观看| 香蕉aⅴ一区二区三区| 国产亚洲欧美视频网站| 大鸡八强奸视频在线观看| 日韩加勒比东京热二区| 日本成人不卡一区二区| a v欧美一区=区三区| 日本丰满熟妇BBXBBXHD| 午夜成午夜成年片在线观看| 把腿张开让我插进去视频| 国产精品手机在线看片| 好了av中文字幕在线| 黄色黄色黄片78在线| 91大屁股国产一区二区| 播放日本一区二区三区电影| 精品国产高潮中文字幕| 新97超碰在线观看| 国产成人精品久久二区91| 青青青青操在线观看免费| 一级黄色片夫妻性生活| 日本熟妇丰满厨房55| 日本性感美女视频网站| 3344免费偷拍视频| 最新中文字幕乱码在线| av森泽佳奈在线观看| av视网站在线观看| 天天日天天添天天爽| 88成人免费av网站| 日本乱人一区二区三区| 国产又粗又猛又爽又黄的视频在线 | 婷婷六月天中文字幕| 大尺度激情四射网站| 99热这里只有国产精品6| 国产精品久久综合久久| 成人综合亚洲欧美一区| 欧美麻豆av在线播放| 夜夜嗨av蜜臀av| 在线免费观看日本片| 美女视频福利免费看| 日本精品一区二区三区在线视频。 | 啪啪啪啪啪啪啪啪av| 午夜在线一区二区免费| 亚洲 欧美 自拍 偷拍 在线| 日韩av有码一区二区三区4| av无限看熟女人妻另类av| 女同性ⅹxx女同h偷拍| 亚洲男人让女人爽的视频| caoporm超碰国产| 女同互舔一区二区三区| 亚洲精品国产久久久久久| 中文字幕在线第一页成人| 韩国一级特黄大片做受| 91精品激情五月婷婷在线| 91精品激情五月婷婷在线| 日韩欧美一级黄片亚洲| 欧美精品国产综合久久| 2021天天色天天干| 热久久只有这里有精品| 38av一区二区三区| 大香蕉伊人国产在线| 色秀欧美视频第一页| 夜夜嗨av一区二区三区中文字幕| 非洲黑人一级特黄片| 日韩欧美一级精品在线观看| 成人午夜电影在线观看 久久| 青青青青青操视频在线观看| 久久这里只有精品热视频| 成人av电影免费版| 亚洲一区二区三区在线高清| 中文字日产幕乱六区蜜桃| 欧美精品一二三视频| 丝袜美腿视频诱惑亚洲无| 日视频免费在线观看| 亚洲天堂第一页中文字幕| 中文字幕免费福利视频6| 亚洲欧美国产麻豆综合| 欧洲黄页网免费观看| 丰满少妇人妻xxxxx| 阴茎插到阴道里面的视频| 天天干天天操天天扣| 在线 中文字幕 一区| 日美女屁股黄邑视频| 四虎永久在线精品免费区二区| lutube在线成人免费看| 水蜜桃国产一区二区三区| 亚洲av无乱一区二区三区性色| 欧美日韩熟女一区二区三区| 国产午夜亚洲精品麻豆| 精品美女久久久久久| 91一区精品在线观看| 丝袜美腿视频诱惑亚洲无 | 久久久精品欧洲亚洲av| 91精品激情五月婷婷在线| 久久精品美女免费视频| 亚洲精品精品国产综合| 日本熟妇色熟妇在线观看| 午夜91一区二区三区| 午夜av一区二区三区| 少妇ww搡性bbb91| 亚洲伊人av天堂有码在线| 青娱乐蜜桃臀av色| 一区二区三区日韩久久| 欧洲亚洲欧美日韩综合| 色97视频在线播放| 好男人视频在线免费观看网站| 好吊操视频这里只有精品| 日本美女性生活一级片| 麻豆性色视频在线观看| 人妻爱爱 中文字幕| 98视频精品在线观看| 在线观看国产网站资源| 北条麻妃高跟丝袜啪啪| 岳太深了紧紧的中文字幕| 初美沙希中文字幕在线| 亚洲另类伦春色综合小| 在线观看操大逼视频| 国产九色91在线视频| 人人妻人人澡欧美91精品| 精品美女福利在线观看| 2020久久躁狠狠躁夜夜躁| 成人久久精品一区二区三区| 在线国产日韩欧美视频| 久久久久久国产精品| 最新日韩av传媒在线| 任你操任你干精品在线视频| 亚洲精品色在线观看视频| 国产精品福利小视频a| 亚洲欧美自拍另类图片| 含骚鸡巴玩逼逼视频| 欧美精品亚洲精品日韩在线| 丰满熟女午夜福利视频| 不卡日韩av在线观看| 1区2区3区不卡视频| 国产在线观看黄色视频| 亚洲av无硬久久精品蜜桃| 免费成人av中文字幕| 青青青青青免费视频| 久久精品亚洲成在人线a| 中文字幕之无码色多多| 97国产在线av精品| 成人高潮aa毛片免费| 在线观看av亚洲情色| 九色porny九色9l自拍视频| 人妻少妇性色欲欧美日韩| 久久这里只有精品热视频| av天堂资源最新版在线看| 粗大的内捧猛烈进出爽大牛汉子| 黄色片一级美女黄色片| 国产黄色大片在线免费播放| 亚洲国产精品黑丝美女| 婷婷色中文亚洲网68| 真实国模和老外性视频| 北条麻妃av在线免费观看| 女人精品内射国产99| 亚洲卡1卡2卡三卡四老狼| 亚洲熟女女同志女同| 欧洲黄页网免费观看| 熟女少妇激情五十路| 三级黄色亚洲成人av| 天堂女人av一区二区| 精品一区二区三区三区色爱| 日韩欧美亚洲熟女人妻| 中文字幕第1页av一天堂网 | 午夜频道成人在线91| 99av国产精品欲麻豆| 国产又粗又猛又爽又黄的视频在线| 日韩欧美亚洲熟女人妻| 久久机热/这里只有| 最新欧美一二三视频| 夜鲁夜鲁狠鲁天天在线| 日韩伦理短片在线观看| 国产又粗又猛又爽又黄的视频在线| 国产亚洲欧美另类在线观看| 动漫av网站18禁| 国产chinesehd精品麻豆| 大香蕉大香蕉在线看| 熟女人妻一区二区精品视频| 成人av免费不卡在线观看| 日韩av熟妇在线观看| 精品首页在线观看视频| 黑人借宿ntr人妻的沦陷2| 神马午夜在线观看视频| 沈阳熟妇28厘米大战黑人| 91天堂精品一区二区| 亚洲一区制服丝袜美腿| 福利片区一区二体验区| 免费黄高清无码国产| 97人妻人人澡爽人人精品| 亚洲成人av在线一区二区| 内射久久久久综合网| 中文字幕乱码人妻电影| 亚洲成人精品女人久久久| 中出中文字幕在线观看| 日本熟女50视频免费| 日本熟妇一区二区x x| 国产极品精品免费视频| av无限看熟女人妻另类av| 插逼视频双插洞国产操逼插洞| 久久国产精品精品美女| 亚洲av色图18p| 91人妻精品一区二区久久| 涩爱综合久久五月蜜臀| av完全免费在线观看av| 精品一区二区三区午夜| 自拍偷拍 国产资源| 一区二区三区激情在线| 天天艹天天干天天操| 又大又湿又爽又紧A视频| 黑人巨大精品欧美视频| 中文字幕一区二区三区人妻大片| 亚洲欧美精品综合图片小说| 粉嫩小穴流水视频在线观看| 在线不卡日韩视频播放| 亚洲精品 欧美日韩| 亚洲嫩模一区二区三区| 2020国产在线不卡视频| 天天色天天操天天舔| 亚洲区欧美区另类最新章节| 日韩熟女系列一区二区三区| 亚洲区美熟妇久久久久| 丝袜美腿视频诱惑亚洲无| 蜜桃久久久久久久人妻| 欧美视频不卡一区四区| 日本女人一级免费片| 一本一本久久a久久精品综合不卡 亚洲另类综合一区小说 | 日本18禁久久久久久| 美女张开两腿让男人桶av| 中文字幕在线一区精品| 做爰视频毛片下载蜜桃视频1| 91成人在线观看免费视频| 免费黄页网站4188| aⅴ五十路av熟女中出| 国产亚洲欧美另类在线观看| 成人性爱在线看四区| 天天操天天弄天天射| 成年人该看的视频黄免费| 青青青爽视频在线播放| 亚洲中文精品字幕在线观看 | 男人操女人的逼免费视频| 青青草人人妻人人妻| 国产va精品免费观看| 在线不卡日韩视频播放| 最新91精品视频在线| 欧美偷拍自拍色图片| 国产精品入口麻豆啊啊啊| 91国产资源在线视频| 操的小逼流水的文章| 亚洲日产av一区二区在线| 红杏久久av人妻一区| 亚洲在线一区二区欧美| 欧美精产国品一二三产品价格| 青青草视频手机免费在线观看| 成年人午夜黄片视频资源| 中国无遮挡白丝袜二区精品| 综合国产成人在线观看| 男生用鸡操女生视频动漫| 青青尤物在线观看视频网站| 一级a看免费观看网站| 亚洲男人的天堂a在线| 午夜精品久久久久久99热| 在线免费观看亚洲精品电影| av乱码一区二区三区| 中文字幕人妻一区二区视频| 热久久只有这里有精品| 日本少妇高清视频xxxxx| 日曰摸日日碰夜夜爽歪歪| av老司机亚洲一区二区| 丰满的子国产在线观看| 成人亚洲国产综合精品| 五十路丰满人妻熟妇| 在线观看国产网站资源| 成人免费毛片aaaa| 欧美日韩精品永久免费网址 | 国产高清女主播在线| 水蜜桃一区二区三区在线观看视频| 99热碰碰热精品a中文| 亚洲一级av大片免费观看| 大陆胖女人与丈夫操b国语高清| 黄色资源视频网站日韩| 免费黄页网站4188| 久久机热/这里只有| 亚洲第17页国产精品| 青青青青青青青青青青草青青| 红桃av成人在线观看| 国产麻豆乱子伦午夜视频观看| 亚洲美女自偷自拍11页| 国产亚洲欧美另类在线观看| 天天操天天爽天天干| 韩国爱爱视频中文字幕| 91精品国产综合久久久蜜| 2022精品久久久久久中文字幕| 岛国免费大片在线观看| 国产使劲操在线播放| 成年人黄色片免费网站| 亚洲中文字幕校园春色| 亚洲无线观看国产高清在线| 91中文字幕免费在线观看| 久久香蕉国产免费天天| av在线播放国产不卡| 男女啪啪视频免费在线观看| 亚洲偷自拍高清视频| 国产内射中出在线观看| 91中文字幕免费在线观看| 国产亚洲精品品视频在线| 国产在线观看免费人成短视频| mm131美女午夜爽爽爽| 日本少妇在线视频大香蕉在线观看| 欧美成人综合色在线噜噜| 日本免费午夜视频网站| 免费在线看的黄片视频| 日本美女性生活一级片| 天天摸天天日天天操| 亚洲国产40页第21页| 在线视频精品你懂的| 亚洲av天堂在线播放| 婷婷午夜国产精品久久久| 亚洲一级美女啪啪啪| 天天干天天爱天天色| 97年大学生大白天操逼| 搡老熟女一区二区在线观看| 女人精品内射国产99| 亚洲久久午夜av一区二区| 91亚洲手机在线视频播放| 男人操女人逼逼视频网站| 水蜜桃一区二区三区在线观看视频 | 成人国产影院在线观看| 免费手机黄页网址大全| 免费黄色成人午夜在线网站| 亚洲av可乐操首页| 岛国毛片视频免费在线观看| 免费看国产av网站| 婷婷五月亚洲综合在线| 亚洲综合图片20p| 区一区二区三国产中文字幕| 免费在线福利小视频| 亚洲欧美自拍另类图片| gav成人免费播放| 2020国产在线不卡视频| 特一级特级黄色网片| 欧美 亚洲 另类综合| 久久久麻豆精亚洲av麻花| 天天日天天摸天天爱| sw137 中文字幕 在线| av亚洲中文天堂字幕网| 亚洲乱码中文字幕在线| 中文字幕日韩精品日本| 成人av在线资源网站| 亚洲国产免费av一区二区三区| 偷青青国产精品青青在线观看 | 噜噜色噜噜噜久色超碰| 青草久久视频在线观看| 老司机午夜精品视频资源| 传媒在线播放国产精品一区| 91九色porny蝌蚪国产成人| 欧美viboss性丰满| 免费高清自慰一区二区三区网站| 中文字幕日韩无敌亚洲精品| 国际av大片在线免费观看| 久久尻中国美女视频| 一区二区三区另类在线| 日视频免费在线观看| v888av在线观看视频| 久久精品美女免费视频| 色伦色伦777国产精品| 摧残蹂躏av一二三区| 97精品综合久久在线| 人妻激情图片视频小说| av资源中文字幕在线观看| 国产精彩对白一区二区三区| 国产女人被做到高潮免费视频| 精品国产污污免费网站入口自| 十八禁在线观看地址免费| 亚洲一级av无码一级久久精品| 91av中文视频在线| 欧美特色aaa大片| 亚洲av男人的天堂你懂的| 九色精品视频在线播放| 国产一级精品综合av| 福利午夜视频在线观看| caoporn蜜桃视频| 特级欧美插插插插插bbbbb| 免费十精品十国产网站| 18禁美女黄网站色大片下载| 男人操女人的逼免费视频| 久草极品美女视频在线观看| 人人妻人人爱人人草| AV无码一区二区三区不卡| 国产夫妻视频在线观看免费| 亚洲在线免费h观看网站| 中文字幕国产专区欧美激情| 97国产在线观看高清| 久久精品美女免费视频| 大香蕉福利在线观看| 日本一道二三区视频久久| 中文字幕无码日韩专区免费| 欧美少妇性一区二区三区| 大胆亚洲av日韩av| 国产高清97在线观看视频| 亚洲天堂av最新网址| 国产麻豆国语对白露脸剧情| 美女av色播在线播放| 天天日天天天天天天天天天天| 日韩人妻丝袜中文字幕| 国产精品久久久久国产三级试频| 亚洲高清免费在线观看视频| 1000小视频在线| 2020国产在线不卡视频| 无码日韩人妻精品久久| 欧美成人一二三在线网| 97人人模人人爽人人喊|