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

源碼進(jìn)階:騰訊開源輕量級緩存 Mmkv 源碼解析

存儲
MMKV 是基于 mmap 內(nèi)存映射的 key-value 組件,底層序列化/反序列化使用 protobuf 實(shí)現(xiàn),性能高,穩(wěn)定性強(qiáng)。

[[429938]]

本文轉(zhuǎn)載自微信公眾號「Android開發(fā)編程」,作者Android開發(fā)編程。轉(zhuǎn)載本文請聯(lián)系A(chǔ)ndroid開發(fā)編程公眾號。

前言

MMKV本質(zhì)上的定位和sp有點(diǎn)相似,經(jīng)常用于持久化小數(shù)據(jù)的鍵值對;

其速度可以說是當(dāng)前所有同類型中速度最快,性能最優(yōu)的庫;

今天我們就來聊聊;

一、MMKV介紹和簡單使用

1、什么是mmkv

MMKV 是基于 mmap 內(nèi)存映射的 key-value 組件,底層序列化/反序列化使用 protobuf 實(shí)現(xiàn),性能高,穩(wěn)定性強(qiáng);

MMKV 基本原理

內(nèi)存準(zhǔn)備:通過 mmap 內(nèi)存映射文件,提供一段可供隨時寫入的內(nèi)存塊,App 只管往里面寫數(shù)據(jù),由操作系統(tǒng)負(fù)責(zé)將內(nèi)存回寫到文件,不必?fù)?dān)心 crash 導(dǎo)致數(shù)據(jù)丟失;

數(shù)據(jù)組織:數(shù)據(jù)序列化方面我們選用 protobuf 協(xié)議,pb 在性能和空間占用上都有不錯的表現(xiàn);

寫入優(yōu)化:考慮到主要使用場景是頻繁地進(jìn)行寫入更新,我們需要有增量更新的能力,考慮將增量 kv 對象序列化后,append 到內(nèi)存末尾;

空間增長:使用 append 實(shí)現(xiàn)增量更新帶來了一個新的問題,就是不斷 append 的話,文件大小會增長得不可控,我們需要在性能和空間上做個折中;

2、MMKV的使用

使用前請初始化:

  1. MMKV.initialize(this) 

mmkv寫入鍵值對;

  1. var mmkv = MMKV.defaultMMKV() 
  2. mmkv.encode("bool",true
  3. mmkv.encode("int",1) 
  4. mmkv.encode("String","test"
  5. mmkv.encode("float",1.0f) 
  6. mmkv.encode("double",1.0) 

mmkv除了能夠?qū)懭脒@些基本類型,只要SharePrefences支持的,它也一定能夠支持;

mmkv讀取鍵值對;

  1. var mmkv = MMKV.defaultMMKV() 
  2. var bo = mmkv.decodeBool("bool"
  3. Log.e(TAG,"bool:${bo}"
  4. var i = mmkv.decodeInt("int"
  5. Log.e(TAG,"int:${i}"
  6. var s = mmkv.decodeString("String"
  7. Log.e(TAG,"String:${s}"
  8. var f = mmkv.decodeFloat("float"
  9. Log.e(TAG,"float:${f}"
  10. var d = mmkv.decodeDouble("double"
  11. Log.e(TAG,"double:$tvxjj7j"

每一個key讀取的數(shù)據(jù)類型就是decodexxx對應(yīng)的類型名字;

mmkv 刪除鍵值對和查鍵值對;

  1. var mmkv = MMKV.defaultMMKV() 
  2. mmkv.removeValueForKey("String"
  3. mmkv.removeValuesForKeys(arrayOf("int","bool")) 
  4. mmkv.containsKey("String"

能夠刪除單個key對應(yīng)的value,也能刪除多個key分別對應(yīng)的value;

containsKey判斷mmkv的磁盤緩存中是否存在對應(yīng)的key;

二、MMKV 源碼解析

1、初始化

通過 MMKV.initialize 方法可以實(shí)現(xiàn) MMKV 的初始化:

  1. public static String initialize(Context context) { 
  2.     String root = context.getFilesDir().getAbsolutePath() + "/mmkv"
  3.     return initialize(root); 

它采用了內(nèi)部存儲空間下的 mmkv 文件夾作為根目錄,之后調(diào)用了 initialize 方法;

  1. public static String initialize(String rootDir) { 
  2.     MMKV.rootDir = rootDir; 
  3.     jniInitialize(MMKV.rootDir); 
  4.     return rootDir; 

調(diào)用到了 jniInitialize 這個 Native 方法進(jìn)行 Native 層的初始化:

  1. extern "C" JNIEXPORT JNICALL void 
  2. Java_com_tencent_mmkv_MMKV_jniInitialize(JNIEnv *env, jobject obj, jstring rootDir) { 
  3.     if (!rootDir) { 
  4.         return
  5.     } 
  6.     const char *kstr = env->GetStringUTFChars(rootDir, nullptr); 
  7.     if (kstr) { 
  8.         MMKV::initializeMMKV(kstr); 
  9.         env->ReleaseStringUTFChars(rootDir, kstr); 
  10.     } 

這里通過 MMKV::initializeMMKV 對 MMKV 類進(jìn)行了初始化:

  1. void MMKV::initializeMMKV(const std::string &rootDir) { 
  2.     static pthread_once_t once_control = PTHREAD_ONCE_INIT; 
  3.     pthread_once(&once_control, initialize); 
  4.     g_rootDir = rootDir; 
  5.     char *path = strdup(g_rootDir.c_str()); 
  6.     mkPath(path); 
  7.     free(path); 
  8.     MMKVInfo("root dir: %s", g_rootDir.c_str()); 

實(shí)際上就是記錄下了 rootDir 并創(chuàng)建對應(yīng)的根目錄,由于 mkPath 方法創(chuàng)建目錄時會修改字符串的內(nèi)容,因此需要復(fù)制一份字符串進(jìn)行;

2、獲取 MMKV 對象

通過 mmkvWithID 方法可以獲取 MMKV 對象,它傳入的 mmapID 就對應(yīng)了 SharedPreferences 中的 name,代表了一個文件對應(yīng)的 name,而 relativePath 則對應(yīng)了一個相對根目錄的相對路徑;

  1. @Nullable 
  2. public static MMKV mmkvWithID(String mmapID, String relativePath) { 
  3.     if (rootDir == null) { 
  4.         throw new IllegalStateException("You should Call MMKV.initialize() first."); 
  5.     } 
  6.     long handle = getMMKVWithID(mmapID, SINGLE_PROCESS_MODE, null, relativePath); 
  7.     if (handle == 0) { 
  8.         return null
  9.     } 
  10.     return new MMKV(handle); 

它調(diào)用到了 getMMKVWithId 這個 Native 方法,并獲取到了一個 handle 構(gòu)造了 Java 層的 MMKV 對象返回;

Java 層通過持有 Native 層對象的地址從而與 Native 對象通信;

  1. extern "C" JNIEXPORT JNICALL jlong Java_com_tencent_mmkv_MMKV_getMMKVWithID( 
  2.     JNIEnv *env, jobject obj, jstring mmapID, jint mode, jstring cryptKey, jstring relativePath) { 
  3.     MMKV *kv = nullptr; 
  4.       // mmapID 為 null 返回空指針 
  5.     if (!mmapID) { 
  6.         return (jlong) kv; 
  7.     } 
  8.     string str = jstring2string(env, mmapID); 
  9.     bool done = false
  10.       // 如果需要進(jìn)行加密,獲取用于加密的 key,最后調(diào)用 MMKV::mmkvWithID 
  11.     if (cryptKey) { 
  12.         string crypt = jstring2string(env, cryptKey); 
  13.         if (crypt.length() > 0) { 
  14.             if (relativePath) { 
  15.                 string path = jstring2string(env, relativePath); 
  16.                 kv = MMKV::mmkvWithID(str, DEFAULT_MMAP_SIZE, (MMKVMode) mode, &crypt, &path); 
  17.             } else { 
  18.                 kv = MMKV::mmkvWithID(str, DEFAULT_MMAP_SIZE, (MMKVMode) mode, &crypt, nullptr); 
  19.             } 
  20.             done = true
  21.         } 
  22.     } 
  23.       // 如果不需要加密,則調(diào)用 mmkvWithID 不傳入加密 key,表示不進(jìn)行加密 
  24.     if (!done) { 
  25.         if (relativePath) { 
  26.             string path = jstring2string(env, relativePath); 
  27.             kv = MMKV::mmkvWithID(str, DEFAULT_MMAP_SIZE, (MMKVMode) mode, nullptr, &path); 
  28.         } else { 
  29.             kv = MMKV::mmkvWithID(str, DEFAULT_MMAP_SIZE, (MMKVMode) mode, nullptr, nullptr); 
  30.         } 
  31.     } 
  32.     return (jlong) kv; 

這里實(shí)際上調(diào)用了 MMKV::mmkvWithID 方法,它根據(jù)是否傳入用于加密的 key 以及是否使用相對路徑調(diào)用了不同的方法;

  1. MMKV *MMKV::mmkvWithID( 
  2.     const std::string &mmapID, int size, MMKVMode mode, string *cryptKey, string *relativePath) { 
  3.     if (mmapID.empty()) { 
  4.         return nullptr; 
  5.     } 
  6.       // 加鎖 
  7.     SCOPEDLOCK(g_instanceLock); 
  8.       // 將 mmapID 與 relativePath 結(jié)合生成 mmapKey 
  9.     auto mmapKey = mmapedKVKey(mmapID, relativePath); 
  10.       // 通過 mmapKey 在 map 中查找對應(yīng)的 MMKV 對象并返回 
  11.     auto itr = g_instanceDic->find(mmapKey); 
  12.     if (itr != g_instanceDic->end()) { 
  13.         MMKV *kv = itr->second
  14.         return kv; 
  15.     } 
  16.       // 如果找不到,構(gòu)建路徑后構(gòu)建 MMKV 對象并加入 map 
  17.     if (relativePath) { 
  18.         auto filePath = mappedKVPathWithID(mmapID, mode, relativePath); 
  19.         if (!isFileExist(filePath)) { 
  20.             if (!createFile(filePath)) { 
  21.                 return nullptr; 
  22.             } 
  23.         } 
  24.         MMKVInfo("prepare to load %s (id %s) from relativePath %s", mmapID.c_str(), mmapKey.c_str(), 
  25.                  relativePath->c_str()); 
  26.     } 
  27.     auto kv = new MMKV(mmapID, size, mode, cryptKey, relativePath); 
  28.     (*g_instanceDic)[mmapKey] = kv; 
  29.     return kv; 

這里的步驟如下:

  • 通過 mmapedKVKey 方法對 mmapID 及 relativePath 進(jìn)行結(jié)合生成了對應(yīng)的 mmapKey,它會將它們兩者的結(jié)合經(jīng)過 md5 從而生成對應(yīng)的 key,主要目的是為了支持不同相對路徑下的同名 mmapID;
  • 通過 mmapKey 在 g_instanceDic 這個 map 中查找對應(yīng)的 MMKV 對象,如果找到直接返回;
  • 如果找不到對應(yīng)的 MMKV 對象,構(gòu)建一個新的 MMKV 對象,加入 map 后返回;
  • 構(gòu)造 MMKV 對象;

MMKV 的構(gòu)造函數(shù):

  1. MMKV::MMKV( 
  2.     const std::string &mmapID, int size, MMKVMode mode, string *cryptKey, string *relativePath) 
  3.     : m_mmapID(mmapedKVKey(mmapID, relativePath)) 
  4.     // ...) { 
  5.     // ... 
  6.     if (m_isAshmem) { 
  7.         m_ashmemFile = new MmapedFile(m_mmapID, static_cast<size_t>(size), MMAP_ASHMEM); 
  8.         m_fd = m_ashmemFile->getFd(); 
  9.     } else { 
  10.         m_ashmemFile = nullptr; 
  11.     } 
  12.         // 通過加密 key 構(gòu)建 AES 加密對象 AESCrypt 
  13.     if (cryptKey && cryptKey->length() > 0) { 
  14.         m_crypter = new AESCrypt((const unsigned char *) cryptKey->data(), cryptKey->length()); 
  15.     } 
  16.         // 賦值操作 
  17.     // 加鎖后調(diào)用 loadFromFile 加載數(shù)據(jù) 
  18.     { 
  19.         SCOPEDLOCK(m_sharedProcessLock); 
  20.         loadFromFile(); 
  21.     } 
  • 進(jìn)行了一些賦值操作,之后如果需要加密則根據(jù)用于加密的 cryptKey 生成對應(yīng)的 AESCrypt 對象用于 AES 加密;
  • 加鎖后通過 loadFromFile 方法從文件中讀取數(shù)據(jù),這里的鎖是一個跨進(jìn)程的文件共享鎖;

3、從文件加載數(shù)據(jù)loadFromFile

我們都知道,MMKV 是基于 mmap 實(shí)現(xiàn)的,通過內(nèi)存映射在高效率的同時保證了數(shù)據(jù)的同步寫入文件,loadFromFile 中就會真正進(jìn)行內(nèi)存映射:

  1. void MMKV::loadFromFile() { 
  2.     // ... 
  3.       // 打開對應(yīng)的文件 
  4.     m_fd = open(m_path.c_str(), O_RDWR | O_CREAT, S_IRWXU); 
  5.     if (m_fd < 0) { 
  6.         MMKVError("fail to open:%s, %s", m_path.c_str(), strerror(errno)); 
  7.     } else { 
  8.           // 獲取文件大小 
  9.         m_size = 0; 
  10.         struct stat st = {0}; 
  11.         if (fstat(m_fd, &st) != -1) { 
  12.             m_size = static_cast<size_t>(st.st_size); 
  13.         } 
  14.         // 將文件大小對齊到頁大小的整數(shù)倍,用 0 填充不足的部分 
  15.         if (m_size < DEFAULT_MMAP_SIZE || (m_size % DEFAULT_MMAP_SIZE != 0)) { 
  16.             size_t oldSize = m_size; 
  17.             m_size = ((m_size / DEFAULT_MMAP_SIZE) + 1) * DEFAULT_MMAP_SIZE; 
  18.             if (ftruncate(m_fd, m_size) != 0) { 
  19.                 MMKVError("fail to truncate [%s] to size %zu, %s", m_mmapID.c_str(), m_size, 
  20.                           strerror(errno)); 
  21.                 m_size = static_cast<size_t>(st.st_size); 
  22.             } 
  23.             zeroFillFile(m_fd, oldSize, m_size - oldSize); 
  24.         } 
  25.           // 通過 mmap 將文件映射到內(nèi)存 
  26.         m_ptr = (char *) mmap(nullptr, m_size, PROT_READ | PROT_WRITE, MAP_SHARED, m_fd, 0); 
  27.         if (m_ptr == MAP_FAILED) { 
  28.             MMKVError("fail to mmap [%s], %s", m_mmapID.c_str(), strerror(errno)); 
  29.         } else { 
  30.             memcpy(&m_actualSize, m_ptr, Fixed32Size); 
  31.             MMKVInfo("loading [%s] with %zu size in total, file size is %zu", m_mmapID.c_str(), 
  32.                      m_actualSize, m_size); 
  33.             bool loadFromFile = false, needFullWriteback = false
  34.             if (m_actualSize > 0) { 
  35.                 if (m_actualSize < m_size && m_actualSize + Fixed32Size <= m_size) { 
  36.                       // 對文件進(jìn)行 CRC 校驗(yàn),如果失敗根據(jù)策略進(jìn)行不同對處理 
  37.                     if (checkFileCRCValid()) { 
  38.                         loadFromFile = true
  39.                     } else { 
  40.                           // CRC 校驗(yàn)失敗,如果策略是錯誤時恢復(fù),則繼續(xù)讀取,并且最后需要進(jìn)行回寫 
  41.                         auto strategic = onMMKVCRCCheckFail(m_mmapID); 
  42.                         if (strategic == OnErrorRecover) { 
  43.                             loadFromFile = true
  44.                             needFullWriteback = true
  45.                         } 
  46.                     } 
  47.                 } else { 
  48.                       // 文件大小有誤,若策略是錯誤時恢復(fù),則繼續(xù)讀取,并且最后需要進(jìn)行回寫 
  49.                     auto strategic = onMMKVFileLengthError(m_mmapID); 
  50.                     if (strategic == OnErrorRecover) { 
  51.                         loadFromFile = true
  52.                         needFullWriteback = true
  53.                     } 
  54.                 } 
  55.             } 
  56.               // 從文件中讀取內(nèi)容 
  57.             if (loadFromFile) { 
  58.                 MMKVInfo("loading [%s] with crc %u sequence %u", m_mmapID.c_str(), 
  59.                          m_metaInfo.m_crcDigest, m_metaInfo.m_sequence); 
  60.                   // 讀取 MMBuffer 
  61.                 MMBuffer inputBuffer(m_ptr + Fixed32Size, m_actualSize, MMBufferNoCopy); 
  62.                 // 如果需要解密,對文件進(jìn)行解密 
  63.                   if (m_crypter) { 
  64.                     decryptBuffer(*m_crypter, inputBuffer); 
  65.                 } 
  66.                   // 通過 MiniPBCoder 將 MMBuffer 轉(zhuǎn)換為 Map 
  67.                 m_dic.clear(); 
  68.                 MiniPBCoder::decodeMap(m_dic, inputBuffer); 
  69.                   // 構(gòu)造用于輸出的 CodeOutputData 
  70.                 m_output = new CodedOutputData(m_ptr + Fixed32Size + m_actualSize, 
  71.                                                m_size - Fixed32Size - m_actualSize); 
  72.                 if (needFullWriteback) { 
  73.                     fullWriteback(); 
  74.                 } 
  75.             } else { 
  76.                 SCOPEDLOCK(m_exclusiveProcessLock); 
  77.                 if (m_actualSize > 0) { 
  78.                     writeAcutalSize(0); 
  79.                 } 
  80.                 m_output = new CodedOutputData(m_ptr + Fixed32Size, m_size - Fixed32Size); 
  81.                 recaculateCRCDigest(); 
  82.             } 
  83.             MMKVInfo("loaded [%s] with %zu values", m_mmapID.c_str(), m_dic.size()); 
  84.         } 
  85.     } 
  86.     if (!isFileValid()) { 
  87.         MMKVWarning("[%s] file not valid", m_mmapID.c_str()); 
  88.     } 
  89.     m_needLoadFromFile = false

步驟如下: 

  • 打開文件并獲取文件大小,將文件的大小對齊到頁的整數(shù)倍,不足則補(bǔ) 0(與內(nèi)存映射的原理有關(guān),內(nèi)存映射是基于頁的換入換出機(jī)制實(shí)現(xiàn)的);
  • 通過 mmap 函數(shù)將文件映射到內(nèi)存中,得到指向該區(qū)域的指針 m_ptr;
  • 對文件進(jìn)行長度校驗(yàn)及 CRC 校驗(yàn)(循環(huán)冗余校驗(yàn),可以校驗(yàn)文件完整性),在失敗的情況下會根據(jù)當(dāng)前策略進(jìn)行抉擇,如果策略是失敗時恢復(fù),則繼續(xù)讀取,并且在最后將 map 中的內(nèi)容回寫到文件;
  • 通過 m_ptr 構(gòu)造出一塊用于管理 MMKV 映射內(nèi)存的 MMBuffer 對象,如果需要解密,通過之前構(gòu)造的 AESCrypt 進(jìn)行解密;
  • 由于 MMKV 使用了 protobuf 進(jìn)行序列化,通過 MiniPBCoder::decodeMap 方法將 protobuf 轉(zhuǎn)換成對應(yīng)的 map;
  • 構(gòu)造用于輸出的 CodedOutputData 類,如果需要回寫(CRC 校驗(yàn)或文件長度校驗(yàn)失敗),則調(diào)用 fullWriteback 方法將 map 中的數(shù)據(jù)回寫到文件;

4、數(shù)據(jù)寫入 

Java 層的 MMKV 對象繼承了 SharedPreferences 及 SharedPreferences.Editor 接口并實(shí)現(xiàn)了一系列如 putInt、putLong 的方法用于對存儲的數(shù)據(jù)進(jìn)行修改;

  1. @Override 
  2. public Editor putInt(String keyint value) { 
  3.     encodeInt(nativeHandle, key, value); 
  4.     return this; 

它調(diào)用到了 encodeInt 這個 Native 方法: 

  1. extern "C" JNIEXPORT JNICALL jboolean Java_com_tencent_mmkv_MMKV_encodeInt( 
  2.     JNIEnv *env, jobject obj, jlong handle, jstring oKey, jint value) { 
  3.     MMKV *kv = reinterpret_cast<MMKV *>(handle); 
  4.     if (kv && oKey) { 
  5.         string key = jstring2string(env, oKey); 
  6.         return (jboolean) kv->setInt32(value, key); 
  7.     } 
  8.     return (jboolean) false

這里將 Java 層持有的 NativeHandle 轉(zhuǎn)為了對應(yīng)的 MMKV 對象,之后調(diào)用了其 setInt32 方法:

  1. bool MMKV::setInt32(int32_t value, const std::string &key) { 
  2.     if (key.empty()) { 
  3.         return false
  4.     } 
  5.       // 構(gòu)造值對應(yīng)的 MMBuffer,通過 CodedOutputData 將其寫入 Buffer 
  6.     size_t size = pbInt32Size(value); 
  7.     MMBuffer data(size); 
  8.     CodedOutputData output(data.getPtr(), size); 
  9.     output.writeInt32(value); 
  10.     return setDataForKey(std::move(data), key); 
  • 獲取到了寫入的 value 在 protobuf 中所占據(jù)的大小,之后為其構(gòu)造了對應(yīng)的 MMBuffer 并將數(shù)據(jù)寫入了這段 Buffer,最后調(diào)用到了 setDataForKey 方法;
  • 同時可以發(fā)現(xiàn) CodedOutputData 是與 Buffer 交互的橋梁,可以通過它實(shí)現(xiàn)向 MMBuffer 中寫入數(shù)據(jù);
  1. bool MMKV::setDataForKey(MMBuffer &&data, const std::string &key) { 
  2.     if (data.length() == 0 || key.empty()) { 
  3.         return false
  4.     } 
  5.       // 獲取寫鎖 
  6.     SCOPEDLOCK(m_lock); 
  7.     SCOPEDLOCK(m_exclusiveProcessLock); 
  8.       // 確保數(shù)據(jù)已讀入內(nèi)存 
  9.     checkLoadData(); 
  10.     // 將 data 寫入 map 中 
  11.     auto itr = m_dic.find(key); 
  12.     if (itr == m_dic.end()) { 
  13.         itr = m_dic.emplace(key, std::move(data)).first
  14.     } else { 
  15.         itr->second = std::move(data); 
  16.     } 
  17.     m_hasFullWriteback = false
  18.     return appendDataWithKey(itr->secondkey); 

數(shù)據(jù)已讀入內(nèi)存的情況下將 data 寫入了對應(yīng)的 map,之后調(diào)用了 appendDataWithKey 方法:

  1. bool MMKV::appendDataWithKey(const MMBuffer &data, const std::string &key) { 
  2.     size_t keyLength = key.length(); 
  3.       // 計算寫入到映射空間中的 size 
  4.     size_t size = keyLength + pbRawVarint32Size((int32_t) keyLength); 
  5.     size += data.length() + pbRawVarint32Size((int32_t) data.length()); 
  6.       // 要寫入,獲取寫鎖 
  7.     SCOPEDLOCK(m_exclusiveProcessLock); 
  8.       // 確定剩余映射空間足夠 
  9.     bool hasEnoughSize = ensureMemorySize(size); 
  10.     if (!hasEnoughSize || !isFileValid()) { 
  11.         return false
  12.     } 
  13.     if (m_actualSize == 0) { 
  14.         auto allData = MiniPBCoder::encodeDataWithObject(m_dic); 
  15.         if (allData.length() > 0) { 
  16.             if (m_crypter) { 
  17.                 m_crypter->reset(); 
  18.                 auto ptr = (unsigned char *) allData.getPtr(); 
  19.                 m_crypter->encrypt(ptr, ptr, allData.length()); 
  20.             } 
  21.             writeAcutalSize(allData.length()); 
  22.             m_output->writeRawData(allData); // note: don't write size of data 
  23.             recaculateCRCDigest(); 
  24.             return true
  25.         } 
  26.         return false
  27.     } else { 
  28.         writeAcutalSize(m_actualSize + size); 
  29.         m_output->writeString(key); 
  30.         m_output->writeData(data); // note: write size of data 
  31.         auto ptr = (uint8_t *) m_ptr + Fixed32Size + m_actualSize - size
  32.         if (m_crypter) { 
  33.             m_crypter->encrypt(ptr, ptr, size); 
  34.         } 
  35.         updateCRCDigest(ptr, size, KeepSequence); 
  36.         return true
  37.     } 
  • 首先計算了即將寫入到映射空間的內(nèi)容大小,之后調(diào)用了 ensureMemorySize 方法確保剩余映射空間足夠;
  • 如果 m_actualSize 為 0,則會通過 MiniPBCoder::encodeDataWithObject 將整個 map 轉(zhuǎn)換為對應(yīng)的 MMBuffer,加密后通過 CodedOutputData 寫入,最后重新計算 CRC 校驗(yàn)碼。否則會將 key 和對應(yīng) data 寫入,最后更新 CRC 校驗(yàn)碼;
  • m_actualSize 是位于文件的首部的,因此是否為 0 取決于文件對應(yīng)位置;

注意的是:由于 protobuf 不支持增量更新,為了避免全量寫入帶來的性能問題,MMKV 在文件中的寫入并不是通過修改文件對應(yīng)的位置,而是直接在后面 append 一條新的數(shù)據(jù),即使是修改了已存在的 key。而讀取時只記錄最后一條對應(yīng) key 的數(shù)據(jù),這樣顯然會在文件中存在冗余的數(shù)據(jù)。這樣設(shè)計的原因我認(rèn)為是出于性能的考量,MMKV 中存在著一套內(nèi)存重整機(jī)制用于對冗余的 key-value 數(shù)據(jù)進(jìn)行處理。它正是在確保內(nèi)存充足時實(shí)現(xiàn)的;

5、內(nèi)存重整ensureMemorySize

我們接下來看看 ensureMemorySize 是如何確保映射空間是否足夠的:

  1. bool MMKV::ensureMemorySize(size_t newSize) { 
  2.     // ... 
  3.     if (newSize >= m_output->spaceLeft()) { 
  4.         // 如果內(nèi)存剩余大小不足以寫入,嘗試進(jìn)行內(nèi)存重整,將 map 中的數(shù)據(jù)重新寫入 protobuf 文件 
  5.         static const int offset = pbFixed32Size(0); 
  6.         MMBuffer data = MiniPBCoder::encodeDataWithObject(m_dic); 
  7.         size_t lenNeeded = data.length() + offset + newSize; 
  8.         if (m_isAshmem) { 
  9.             if (lenNeeded > m_size) { 
  10.                 MMKVWarning("ashmem %s reach size limit:%zu, consider configure with larger size"
  11.                             m_mmapID.c_str(), m_size); 
  12.                 return false
  13.             } 
  14.         } else { 
  15.             size_t avgItemSize = lenNeeded / std::max<size_t>(1, m_dic.size()); 
  16.             size_t futureUsage = avgItemSize * std::max<size_t>(8, (m_dic.size() + 1) / 2); 
  17.             // 如果內(nèi)存重整后仍不足以寫入,則將大小不斷乘2直至足夠?qū)懭耄詈笸ㄟ^ mmap 重新映射文件 
  18.             if (lenNeeded >= m_size || (lenNeeded + futureUsage) >= m_size) { 
  19.                 size_t oldSize = m_size; 
  20.                 do { 
  21.                       // double 空間直至足夠 
  22.                     m_size *= 2; 
  23.                 } while (lenNeeded + futureUsage >= m_size); 
  24.                    // ... 
  25.                 if (ftruncate(m_fd, m_size) != 0) { 
  26.                     MMKVError("fail to truncate [%s] to size %zu, %s", m_mmapID.c_str(), m_size, 
  27.                               strerror(errno)); 
  28.                     m_size = oldSize; 
  29.                     return false
  30.                 } 
  31.                   // 用零填充不足部分 
  32.                 if (!zeroFillFile(m_fd, oldSize, m_size - oldSize)) { 
  33.                     MMKVError("fail to zeroFile [%s] to size %zu, %s", m_mmapID.c_str(), m_size, 
  34.                               strerror(errno)); 
  35.                     m_size = oldSize; 
  36.                     return false
  37.                 } 
  38.                                 // unmap 
  39.                 if (munmap(m_ptr, oldSize) != 0) { 
  40.                     MMKVError("fail to munmap [%s], %s", m_mmapID.c_str(), strerror(errno)); 
  41.                 } 
  42.                                 // 重新通過 mmap 映射 
  43.                 m_ptr = (char *) mmap(m_ptr, m_size, PROT_READ | PROT_WRITE, MAP_SHARED, m_fd, 0); 
  44.                 if (m_ptr == MAP_FAILED) { 
  45.                     MMKVError("fail to mmap [%s], %s", m_mmapID.c_str(), strerror(errno)); 
  46.                 } 
  47.                 // check if we fail to make more space 
  48.                 if (!isFileValid()) { 
  49.                     MMKVWarning("[%s] file not valid", m_mmapID.c_str()); 
  50.                     return false
  51.                 } 
  52.             } 
  53.         } 
  54.           // 加密數(shù)據(jù) 
  55.         if (m_crypter) { 
  56.             m_crypter->reset(); 
  57.             auto ptr = (unsigned char *) data.getPtr(); 
  58.             m_crypter->encrypt(ptr, ptr, data.length()); 
  59.         } 
  60.           // 重新構(gòu)建并寫入數(shù)據(jù) 
  61.         writeAcutalSize(data.length()); 
  62.         delete m_output; 
  63.         m_output = new CodedOutputData(m_ptr + offset, m_size - offset); 
  64.         m_output->writeRawData(data); 
  65.         recaculateCRCDigest(); 
  66.         m_hasFullWriteback = true
  67.     } 
  68.     return true

內(nèi)存重整步驟如下:

  • 當(dāng)剩余映射空間不足以寫入需要寫入的內(nèi)容,嘗試進(jìn)行內(nèi)存重整;
  • 內(nèi)存重整會將文件清空,將 map 中的數(shù)據(jù)重新寫入文件,從而去除冗余數(shù)據(jù);
  • 若內(nèi)存重整后剩余映射空間仍然不足,不斷將映射空間 double 直到足夠,并用 mmap 重新映射;

6、刪除remove

通過 Java 層 MMKV 的 remove 方法可以實(shí)現(xiàn)刪除操作:

  1. @Override 
  2. public Editor remove(String key) { 
  3.     removeValueForKey(key); 
  4.     return this; 

它調(diào)用了 removeValueForKey 這個 Native 方法:

  1. extern "C" JNIEXPORT JNICALL void Java_com_tencent_mmkv_MMKV_removeValueForKey(JNIEnv *env, 
  2.                                                                                jobject instance, 
  3.                                                                                jlong handle, 
  4.                                                                                jstring oKey) { 
  5.     MMKV *kv = reinterpret_cast<MMKV *>(handle); 
  6.     if (kv && oKey) { 
  7.         string key = jstring2string(env, oKey); 
  8.         kv->removeValueForKey(key); 
  9.     } 

這里調(diào)用了 Native 層 MMKV 的 removeValueForKey 方法:

  1. void MMKV::removeValueForKey(const std::string &key) { 
  2.     if (key.empty()) { 
  3.         return
  4.     } 
  5.     SCOPEDLOCK(m_lock); 
  6.     SCOPEDLOCK(m_exclusiveProcessLock); 
  7.     checkLoadData(); 
  8.     removeDataForKey(key); 

它在數(shù)據(jù)讀入內(nèi)存的前提下,調(diào)用了 removeDataForKey 方法:

  1. bool MMKV::removeDataForKey(const std::string &key) { 
  2.     if (key.empty()) { 
  3.         return false
  4.     } 
  5.     auto deleteCount = m_dic.erase(key); 
  6.     if (deleteCount > 0) { 
  7.         m_hasFullWriteback = false
  8.         static MMBuffer nan(0); 
  9.         return appendDataWithKey(nan, key); 
  10.     } 
  11.     return false
  • 這里實(shí)際上是構(gòu)造了一條 size 為 0 的 MMBuffer 并調(diào)用 appendDataWithKey 將其 append 到 protobuf 文件中,并將 key 對應(yīng)的內(nèi)容從 map 中刪除;
  • 讀取時發(fā)現(xiàn)它的 size 為 0,則會認(rèn)為這條數(shù)據(jù)已經(jīng)刪除;

7、讀取

我們通過 getInt、getLong 等操作可以實(shí)現(xiàn)對數(shù)據(jù)的讀取,我們以 getInt 為例:

  1. @Override 
  2. public int getInt(String keyint defValue) { 
  3.     return decodeInt(nativeHandle, key, defValue); 

它調(diào)用到了 decodeInt 這個 Native 方法:

  1. extern "C" JNIEXPORT JNICALL jint Java_com_tencent_mmkv_MMKV_decodeInt( 
  2.     JNIEnv *env, jobject obj, jlong handle, jstring oKey, jint defaultValue) { 
  3.     MMKV *kv = reinterpret_cast<MMKV *>(handle); 
  4.     if (kv && oKey) { 
  5.         string key = jstring2string(env, oKey); 
  6.         return (jint) kv->getInt32ForKey(key, defaultValue); 
  7.     } 
  8.     return defaultValue; 

它調(diào)用到了 MMKV.getInt32ForKey 方法:

  1. int32_t MMKV::getInt32ForKey(const std::string &key, int32_t defaultValue) { 
  2.     if (key.empty()) { 
  3.         return defaultValue; 
  4.     } 
  5.     SCOPEDLOCK(m_lock); 
  6.     auto &data = getDataForKey(key); 
  7.     if (data.length() > 0) { 
  8.         CodedInputData input(data.getPtr(), data.length()); 
  9.         return input.readInt32(); 
  10.     } 
  11.     return defaultValue; 

調(diào)用了 getDataForKey 方法獲取到了 key 對應(yīng)的 MMBuffer,之后通過 CodedInputData 將數(shù)據(jù)讀出并返回;

長度為 0 時會將其視為不存在,返回默認(rèn)值;

  1. const MMBuffer &MMKV::getDataForKey(const std::string &key) { 
  2.     checkLoadData(); 
  3.     auto itr = m_dic.find(key); 
  4.     if (itr != m_dic.end()) { 
  5.         return itr->second
  6.     } 
  7.     static MMBuffer nan(0); 
  8.     return nan; 

這里實(shí)際上是通過在 Map 中尋找從而實(shí)現(xiàn),找不到會返回 size 為 0 的 Buffer;

MMKV讀寫是直接讀寫到mmap文件映射的內(nèi)存上,繞開了普通讀寫io需要進(jìn)入內(nèi)核,寫到磁盤的過程;

總結(jié)

MMKV使用的注意事項(xiàng)

1.保證每一個文件存儲的數(shù)據(jù)都比較小,也就說需要把數(shù)據(jù)根據(jù)業(yè)務(wù)線存儲分散。這要就不會把虛擬內(nèi)存消耗過快;

2.適當(dāng)?shù)臅r候釋放一部分內(nèi)存數(shù)據(jù),比如在App中監(jiān)聽onTrimMemory方法,在Java內(nèi)存吃緊的情況下進(jìn)行MMKV的trim操作; 

3.不需要使用的時候,最好把MMKV給close掉,甚至調(diào)用exit方法。

 

責(zé)任編輯:武曉燕 來源: Android開發(fā)編程
相關(guān)推薦

2022-02-12 21:05:11

異步爬蟲框架

2016-09-22 15:50:38

JavascriptRedux源碼解析

2019-09-18 18:12:35

騰訊開源物聯(lián)網(wǎng)

2021-09-09 06:55:43

AndroidViewDragHel原理

2021-09-01 06:48:16

AndroidGlide緩存

2019-03-10 20:55:11

瀏覽器Midori開源

2022-06-01 09:28:43

??Umami??開源

2021-07-03 08:51:30

源碼Netty選擇器

2021-05-17 09:50:06

Kubebuilde源碼CURD

2019-05-07 14:42:03

深度學(xué)習(xí)編程人工智能

2021-09-02 07:00:01

Glide流程Android

2022-08-11 08:35:40

Hanko開源

2013-02-20 14:54:03

C#.NETNDatabase

2025-02-19 09:55:39

2009-07-17 14:38:51

輕量級Swing組件

2009-07-14 18:05:28

輕量級Swing組件

2015-09-16 09:10:27

Java源碼解析

2010-01-06 14:19:47

JSON輕量級

2022-05-20 10:32:49

事件循環(huán)器事件隊列鴻蒙

2022-05-16 07:37:58

SQL 編輯器數(shù)據(jù)庫管理工具
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

欧美一级理论性理论a| 99re这里只有精品视频首页| 日韩视频精品在线| 国产在线视频三区| 麻豆免费在线| 亚洲国产成人自拍| 亚洲一区二区中文| 毛片视频网站在线观看| 成人羞羞视频播放网站| 精品免费一区二区三区| 又色又爽又高潮免费视频国产| 欧美激情二区| 99精品桃花视频在线观看| 国产精品嫩草影院一区二区| 精品无码久久久久久久| av一区二区高清| 亚洲精品aⅴ中文字幕乱码 | 大地资源二中文在线影视观看| 欧美a视频在线| 黄色成人在线免费| 亚洲av综合色区| a√在线中文网新版址在线| 成人精品亚洲人成在线| 91精品国产综合久久香蕉| 国偷自拍第113页| 欧美激情日韩| 久久久av一区| 美国黄色特级片| 日本一道高清一区二区三区| 日韩欧美你懂的| 91小视频在线播放| gogo亚洲高清大胆美女人体 | 波多野结衣一区二区在线| 欧美三区不卡| 欧美大肥婆大肥bbbbb| 波多野结衣一二三四区| 亚洲影院天堂中文av色| 亚洲第一精品福利| avtt中文字幕| 日韩欧美久久| 日韩欧美国产麻豆| 四虎1515hh.com| 国产激情综合| 91精品国产麻豆国产自产在线| 女性隐私黄www网站视频| 国产资源在线观看入口av| 一区二区三区丝袜| 可以在线看黄的网站| 九义人在线观看完整免费版电视剧| 国产欧美精品一区aⅴ影院| 欧美久久久久久一卡四| 人成免费电影一二三区在线观看| 99久久婷婷国产综合精品 | 在线观看国产中文字幕| 怡红院成人在线| 欧美性色欧美a在线播放| 日av中文字幕| julia一区二区三区中文字幕| 岛国视频午夜一区免费在线观看| 国产一区二区在线视频播放| 韩日毛片在线观看| 日本久久一区二区三区| 国产a级片免费观看| 99re66热这里只有精品4| 欧美亚洲一区二区三区四区| 日韩一区二区三区不卡视频| 成人国产精品入口免费视频| 欧美日韩国产高清一区二区| 五月六月丁香婷婷| 成人春色在线观看免费网站| 欧美精品一区二区蜜臀亚洲| 国产麻豆天美果冻无码视频| 成人短片线上看| 久久视频在线观看免费| 欧美日韩人妻精品一区二区三区| 亚洲视频综合| 国产99久久久欧美黑人| 亚洲一级黄色大片| 国产成人免费视频一区| 久久久久久国产精品免费免费| 国产中文在线视频| 亚洲精选在线视频| 黄色片视频在线免费观看| 中文另类视频| 日韩精品专区在线影院重磅| 亚洲欧美在线不卡| 欧美日韩在线播放视频| 欧美人与性动交a欧美精品| 国产又色又爽又黄的| 日韩精品五月天| 亚洲a中文字幕| 色一情一乱一乱一区91av| 久久精品日产第一区二区三区高清版| 一区二区三区不卡在线| 日韩专区av| 欧美伊人久久久久久午夜久久久久| av在线网站免费观看| 亚洲理论电影片| 美女av一区二区三区| 六月丁香激情综合| 国产一区二区三区综合| 欧美久久久久久久| 欧美另类tv| 欧美三级资源在线| 国产亚洲无码精品| 欧美 日韩 国产精品免费观看| 欧美一级片免费在线| av中文字幕第一页| 日本一区二区三区四区 | 日韩电影免费一区| 国产精品乱码| 成人video亚洲精品| 色综合 综合色| 四虎成人免费视频| 国产韩国精品一区二区三区| 96精品视频在线| 99久久精品无免国产免费| 久久久噜噜噜久久人人看| 玖玖精品在线视频| 国产一区二区色噜噜| 亚洲国产精品人人爽夜夜爽| 国产波霸爆乳一区二区| 美女看a上一区| 日本一区二区三区视频免费看| heyzo高清在线| 日韩欧美的一区二区| 国产成人精品视频免费| 日韩国产一区二| 久久视频在线观看中文字幕| 国产又色又爽又黄刺激在线视频| 欧美日韩精品福利| 亚洲图片第一页| 日韩中文字幕91| 九九九九九精品| 97在线视频免费观看完整版| 日韩欧美一区二区在线视频| 希岛爱理中文字幕| 九九九久久久精品| 91手机视频在线| 四虎国产精品免费久久5151| 丝袜情趣国产精品| 97国产精品久久久| 中文字幕日韩欧美一区二区三区| 91香蕉视频导航| 欧美少妇xxxx| 国产精品日韩精品| 日本在线人成| 在线播放一区二区三区| 久久久99久久精品女同性| 99久久99久久精品国产| 韩国三级电影一区二区| 在线观看成人一级片| 欧美高清影院| www.日本久久久久com.| 国产理论视频在线观看| 亚洲免费资源在线播放| 国产又黄又嫩又滑又白| 欧美国产91| 国产伦精品一区二区三区免 | 天堂网av在线播放| 午夜精品一区二区三区免费视频 | 伊人久久成人网| 日韩毛片一二三区| 国产无套精品一区二区三区| 国产精品99一区二区| 国产精品乱码| 国产精品迅雷| 色阁综合伊人av| 91片黄在线观看喷潮| 亚洲精品久久久久久国产精华液| 91人人澡人人爽| 亚洲黄色影院| 日本电影一区二区三区| 日韩另类视频| 色综合男人天堂| 天天操天天干天天操| 91福利社在线观看| 精品无码久久久久成人漫画| 处破女av一区二区| 欧在线一二三四区| 亚洲91视频| 麻豆一区区三区四区产品精品蜜桃| 台湾佬成人网| 久久国产天堂福利天堂| 深夜福利视频在线免费观看| 欧美性猛交xxxx乱大交3| 五月婷婷婷婷婷| 国产成人精品一区二| 欧美一级黄色片视频| 亚洲老妇激情| 欧美精品一区二区三区在线看午夜| 外国成人毛片| 97超级碰在线看视频免费在线看| 国产高清一区在线观看| 日韩免费观看高清完整版| www.久久久久久久| 亚洲激情校园春色| 久久久精品成人| 成人h动漫精品一区二| 成年网站免费在线观看| 亚洲激情一区| 午夜啪啪福利视频| 一本色道久久综合狠狠躁的番外| 91精品在线国产| 人人视频精品| 欧美福利视频在线| 97在线观看免费观看高清| 精品久久久久久无| 在线播放精品视频| 色诱视频网站一区| 天天操天天射天天爽| 国产精品美女久久久久久久 | 国产乱码久久久久| 色婷婷国产精品综合在线观看| 农村妇女精品一区二区| 欧美国产禁国产网站cc| 精品无码在线视频| 顶级嫩模精品视频在线看| 手机看片一级片| 日韩专区欧美专区| 免费毛片小视频| 狠狠综合久久av一区二区老牛| 亚洲综合第一| 精品国产中文字幕第一页| 精品国产乱码久久久久久郑州公司| 精品欧美视频| 成人春色激情网| 国内自拍亚洲| 国产精品成久久久久三级| 蜜桃视频m3u8在线观看| 久久久噜噜噜久噜久久| 国产乱码在线| 欧美成人网在线| 国产人成网在线播放va免费| 在线成人一区二区| 国产三级视频在线| 亚洲男人av电影| 免费看男男www网站入口在线 | 污片免费在线观看| 成人美女在线观看| 国产精品熟妇一区二区三区四区| 国产乱淫av一区二区三区| 日韩a一级欧美一级| 精品一区二区久久| 四川一级毛毛片| 国产91丝袜在线播放0| 久久久久久婷婷| 成人激情免费电影网址| 大乳护士喂奶hd| 91免费看`日韩一区二区| 欧美深性狂猛ⅹxxx深喉 | 久久国产精品精品国产色婷婷| 好吊妞国产欧美日韩免费观看网站| 成人在线免费观看一区| 嗯用力啊快一点好舒服小柔久久| 不卡一卡2卡3卡4卡精品在| www.豆豆成人网.com| 国产一区二区视频在线免费观看 | 亚洲午夜在线观看| 婷婷综合视频| 欧美久久久久久久久久久久久久| 欧美日本三区| 久久综合九色综合88i| 香蕉久久a毛片| xx欧美撒尿嘘撒尿xx| 经典三级在线一区| 国产精品亚洲一区二区无码| 97久久精品人人做人人爽| 亚洲欧洲久久久| 中文字幕精品一区二区精品绿巨人| 一区二区三区在线播放视频| 亚洲日本欧美天堂| 日本在线免费观看| 在线欧美小视频| 91久久精品无码一区二区| 欧美一区二区在线免费观看| 欧美 日韩 国产 在线| 亚洲男人天堂视频| 麻豆传媒在线免费| 97色伦亚洲国产| 久久人体av| 国产乱子伦精品| 日韩精品午夜| 国产美女在线一区| 日本中文字幕一区二区视频| 韩国三级在线播放| www国产精品av| 四虎影院中文字幕| 五月天激情综合| 一级黄色录像大片| 日韩成人在线视频观看| 美女隐私在线观看| 51ⅴ精品国产91久久久久久| 亚洲一区二区av| 欧美日韩精品免费观看视一区二区| 国产精品99久久精品| 男人日女人逼逼| 国产自产2019最新不卡| 久久久久久久久免费看无码| 国产精品天干天干在线综合| 日本在线观看视频网站| 欧美高清视频在线高清观看mv色露露十八| 高清毛片aaaaaaaaa片| 中文字幕亚洲一区二区三区五十路| 牛牛电影国产一区二区| 国产精品免费久久久久影院| 卡通动漫国产精品| 国产在线无码精品| 日本中文字幕一区二区有限公司| 精品一区二区视频在线观看| 亚洲三级免费电影| 最新黄色网址在线观看| 亚洲经典中文字幕| 黄色污污视频在线观看| 成人激情视频在线| 国产日产精品一区二区三区四区的观看方式 | 精品国产99| 亚洲熟妇无码另类久久久| 国产一区视频导航| 91无套直看片红桃在线观看| 日韩欧美中文免费| 偷拍自拍在线视频| 国语自产精品视频在免费| 视频在线亚洲| 日本一二三区视频在线| 久热成人在线视频| 国产一级久久久久毛片精品| 精品久久久久久久久久国产| 老熟妇高潮一区二区高清视频| 成人97在线观看视频| 色999韩欧美国产综合俺来也| 日韩欧美在线电影| 日韩高清在线电影| 免费一级做a爰片久久毛片潮| 性做久久久久久久久| 黄色片一区二区三区| 久久99久久久久久久噜噜| 美女日韩一区| 天堂av在线中文| 国产激情一区二区三区| 91成人福利视频| 日韩免费视频一区| 蜜臀av在线| 国产一区二区三区四区五区加勒比| 韩日成人av| 偷偷色噜狠狠狠狠的777米奇| 亚洲成av人影院| 午夜性色福利视频| 欧美亚洲另类制服自拍| 亚洲v天堂v手机在线| 精品视频一区二区在线| 久久影院视频免费| 国产主播第一页| 中文字幕日韩综合av| 日本a人精品| 日本a级片在线观看| 国产91精品精华液一区二区三区| 日本少妇性高潮| 国产视频精品免费播放| 中文.日本.精品| 小说区视频区图片区| 国产传媒久久文化传媒| 一级aaa毛片| 伊人伊成久久人综合网小说| 日日夜夜亚洲| 国产精品一线二线三线| 久久亚洲春色中文字幕久久久| 无码人妻av免费一区二区三区 | 久草免费在线色站| 精品亚洲一区二区三区四区五区高| 羞羞答答国产精品www一本| 精品亚洲aⅴ无码一区二区三区| 欧美群妇大交群的观看方式| 日韩激情av| 色涩成人影视在线播放| 国产曰批免费观看久久久| 人人干人人干人人干| 永久555www成人免费| 日韩欧美中文字幕在线视频 | 麻豆成人久久精品二区三区红 | 在线观看免费小视频| 91精品国产综合久久福利| 交100部在线观看| 亚洲资源视频| 26uuu亚洲| 国产精品一区二区av白丝下载| 午夜精品一区二区三区在线视频| 青青草综合网| 四虎精品一区二区| www国产在线| 欧美理论片在线观看| 天堂99x99es久久精品免费| 不卡的在线视频| 欧美日韩午夜视频在线观看| 免费网站免费进入在线| 韩国精品一区二区三区六区色诱| 久久精品国产免费看久久精品| 成人免费看片98| 视频直播国产精品| 九九亚洲视频|