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

開源日志庫Logger的剖析

開源
上一篇介紹了開源日志庫Logger的使用,今天主要來分析Logger實現的原理。

上一篇介紹了開源日志庫Logger的使用,今天主要來分析Logger實現的原理。

庫的整體架構圖

 詳細剖析

我們從使用的角度來對Logger庫抽繭剝絲:

  1. String userName = "Jerry"
  2. Logger.i(userName);  

看看Logger.i()這個方法: 

  1. public static void i(String message, Object... args) {       
  2.     printer.i(message, args); 
  3.  

還有個可變參數,來看看printer.i(message, args)是啥:

  1. public Interface Printer{ 
  2.     void i(String message, Object... args); 

 是個接口,那我們就要找到這個接口的實現類,找到printer對象在Logger類中聲明的地方: 

  1. private static Printer printer = new LoggerPrinter(); 

實現類是LoggerPrinter,而且這還是個靜態的成員變量,這個靜態是有用處的,后面會講到,那就繼續跟蹤LoggerPrinter類的i(String message, Object... args)方法的實現: 

  1. @Override public void i(String message, Object... args) {   
  2.     log(INFO, null, message, args); 
  3. /**  
  4. * This method is synchronized in order to avoid messy of logs' order.  
  5. */ 
  6. private synchronized void log(int priority, Throwable throwable, String msg, Object... args) { 
  7.     // 判斷當前設置的日志級別,為NONE則不打印日志   
  8.     if (settings.getLogLevel() == LogLevel.NONE) {     
  9.         return;   
  10.     } 
  11.     // 獲取tag 
  12.     String tag = getTag();  
  13.     // 創建打印的消息 
  14.     String message = createMessage(msg, args);       
  15.     // 打印 
  16.     log(priority, tag, message, throwable); 
  17.  
  18. public enum LogLevel {   
  19.     /**    
  20.     * Prints all logs    
  21.     */   
  22.     FULL,   
  23.     /**    
  24.     * No log will be printed    
  25.     */   
  26.     NONE 
  27.  
  • 首先,log方法是一個線程安全的同步方法,為了防止日志打印時候順序的錯亂,在多線程環境下,這是非常有必要的。
  • 其次,判斷日志配置的打印級別,FULL打印全部日志,NONE不打印日志。
  • 再來,getTag(): 
    1. private final ThreadLocal<String> localTag = new ThreadLocal<>(); 
    2. /**  
    3. * @return the appropriate tag based on local or global */ 
    4. private String getTag() {   
    5.     // 從ThreadLocal<String> localTag里獲取本地一個緩存的tag 
    6.     String tag = localTag.get();   
    7.     if (tag != null) {     
    8.         localTag.remove();     
    9.         return tag;   
    10.     }   
    11.     return this.tag; 
    12.  

這個方法是獲取本地或者全局的tag值,當localTag中有tag的時候就返回出去,并且清空localTag的值,關于ThreadLocal還不是很清楚的可以參考主席的文章:http://blog.csdn.net/singwhat...

接著,createMessage方法: 

  1. private String createMessage(String message, Object... args) {  
  2.     return args == null || args.length == 0 ? message : String.format(message, args); 

 這里就很清楚了,為什么我們用Logger.i(message, args)的時候沒有寫args,也就是null,也可以打印,而且是直接打印的message消息的原因。同樣博主上一篇文章也提到了: 

  1. Logger.i("博主今年才%d,英文名是%s", 16, "Jerry"); 

像這樣的可以拼接不同格式的數據的打印日志,原來實現的方式是用String.format方法,這個想必小伙伴們在開發Android應用的時候String.xml里的動態字符占位符用的也不少,應該很容易理解這個format方法的用法。

重頭戲,我們把tag,打印級別,打印的消息處理好了,接下來該打印出來了: 

  1. @Override public synchronized void log(int priority, String tag, String message, Throwable throwable) { 
  2.     // 同樣判斷一次庫配置的打印開關,為NONE則不打印日志 
  3.     if (settings.getLogLevel() == LogLevel.NONE) {     
  4.         return;   
  5.     } 
  6.     // 異常和消息不為空的時候,獲取異常的原因轉換成字符串后拼接到打印的消息中   
  7.     if (throwable != null && message != null) {     
  8.         message += " : " + Helper.getStackTraceString(throwable);   
  9.     }   
  10.     if (throwable != null && message == null) {     
  11.         message = Helper.getStackTraceString(throwable);   
  12.     }   
  13.     if (message == null) {     
  14.         message = "No message/exception is set";   
  15.     }   
  16.     // 獲取方法數 
  17.     int methodCount = getMethodCount();  
  18.     // 判斷消息是否為空  
  19.     if (Helper.isEmpty(message)) {     
  20.         message = "Empty/NULL log message";   
  21.     }   
  22.     // 打印日志體的上邊界 
  23.     logTopBorder(priority, tag); 
  24.     // 打印日志體的頭部內容   
  25.     logHeaderContent(priority, tag, methodCount);   
  26.     //get bytes of message with system's default charset (which is UTF-8 for Android)   
  27.     byte[] bytes = message.getBytes();   
  28.     int length = bytes.length;   
  29.     // 消息字節長度小于等于4000 
  30.     if (length <= CHUNK_SIZE) {     
  31.         if (methodCount > 0) {   
  32.             // 方法數大于0,打印出分割線     
  33.             logDivider(priority, tag);     
  34.         }     
  35.         // 打印消息內容 
  36.         logContent(priority, tag, message); 
  37.         // 打印日志體底部邊界 
  38.         logBottomBorder(priority, tag);     
  39.         return;   
  40.     }   
  41.     if (methodCount > 0) {     
  42.         logDivider(priority, tag);   
  43.     }   
  44.     for (int i = 0; i < length; i += CHUNK_SIZE) {     
  45.         int count = Math.min(length - i, CHUNK_SIZE); 
  46.         //create a new String with system's default charset (which is UTF-8 for Android)     
  47.         logContent(priority, tag, new String(bytes, i, count));   
  48.     }   
  49.     logBottomBorder(priority, tag); 

 我們重點來看看logHeaderContent方法和logContent方法: 

  1. @SuppressWarnings("StringBufferReplaceableByString"
  2. private void logHeaderContent(int logType, String tag, int methodCount) {   
  3.   // 獲取當前線程堆棧跟蹤元素數組 
  4.   //(里面存儲了虛擬機調用的方法的一些信息:方法名、類名、調用此方法在文件中的行數) 
  5.   // 這也是這個庫的 “核心” 
  6.   StackTraceElement[] trace = Thread.currentThread().getStackTrace(); 
  7.   // 判斷庫的配置是否顯示線程信息   
  8.   if (settings.isShowThreadInfo()) { 
  9.       // 獲取當前線程的名稱,并且打印出來,然后打印分割線     
  10.       logChunk(logType, tag, HORIZONTAL_DOUBLE_LINE + "Thread: " + Thread.currentThread().getName());    logDivider(logType, tag);   
  11.   }   
  12.   String level = "";   
  13.   // 獲取追蹤棧的方法起始位置 
  14.   int stackOffset = getStackOffset(trace) + settings.getMethodOffset();   
  15.   //corresponding method count with the current stack may exceeds the stack trace. Trims the count   
  16.   // 打印追蹤的方法數超過了當前線程能夠追蹤的方法數,總的追蹤方法數扣除偏移量(從調用日志的起算扣除的方法數),就是需要打印的方法數量 
  17.   if (methodCount + stackOffset > trace.length) {     
  18.       methodCount = trace.length - stackOffset - 1;   
  19.   }   
  20.   for (int i = methodCount; i > 0; i--) {    
  21.       int stackIndex = i + stackOffset;     
  22.       if (stackIndex >= trace.length) {       
  23.           continue;     
  24.       }     
  25.       // 拼接方法堆棧調用路徑追蹤字符串 
  26.       StringBuilder builder = new StringBuilder();  
  27.       builder.append("║ ")         
  28.       .append(level)      
  29.       .append(getSimpleClassName(trace[stackIndex].getClassName()))  // 追蹤到的類名 
  30.       .append(".")  
  31.       .append(trace[stackIndex].getMethodName())  // 追蹤到的方法名       
  32.       .append(" ")         
  33.       .append(" (")        
  34.       .append(trace[stackIndex].getFileName()) // 方法所在的文件名 
  35.       .append(":")         
  36.       .append(trace[stackIndex].getLineNumber())  // 在文件中的行號       
  37.       .append(")");     
  38.       level += "   ";     
  39.       // 打印出頭部信息 
  40.       logChunk(logType, tag, builder.toString());  
  41.   } 

 接下來看logContent方法: 

  1. private void logContent(int logType, String tag, String chunk) {   
  2.     // 這個作用就是獲取換行符數組,getProperty方法獲取的就是"\\n"的意思 
  3.     String[] lines = chunk.split(System.getProperty("line.separator"));   
  4.     for (String line : lines) {     
  5.         // 打印出包含換行符的內容 
  6.         logChunk(logType, tag, HORIZONTAL_DOUBLE_LINE + " " + line);   
  7.     } 

 如上圖來說內容是字符串數組,本身里面是沒用換行符的,所以不需要換行,打印出來的效果就是一行,但是json、xml這樣的格式是有換行符的,所以打印呈現出來的效果就是:

 上面說了大半天,都還沒看到具體的打印是啥,現在來看看logChunk方法: 

  1. private void logChunk(int logType, String tag, String chunk) { 
  2.     // ***格式化下tag   
  3.     String finalTag = formatTag(tag);   
  4.     // 根據不同的日志打印類型,然后交給LogAdapter這個接口來打印 
  5.     switch (logType) {     
  6.         case ERROR:       
  7.             settings.getLogAdapter().e(finalTag, chunk);       
  8.         break;     
  9.         case INFO:       
  10.             settings.getLogAdapter().i(finalTag, chunk);       
  11.         break;     
  12.         case VERBOSE:       
  13.             settings.getLogAdapter().v(finalTag, chunk);       
  14.         break;     
  15.         case WARN:       
  16.             settings.getLogAdapter().w(finalTag, chunk);       
  17.         break;    
  18.         case ASSERT:       
  19.             settings.getLogAdapter().wtf(finalTag, chunk);       
  20.         break;     
  21.         case DEBUG:       
  22.             // Fall through, log debug by default     
  23.         default:             
  24.             settings.getLogAdapter().d(finalTag, chunk);       
  25.         break;   
  26.     } 
  27.  

這個方法很簡單,就是***格式化tag,然后根據不同的日志類型把打印的工作交給LogAdapter接口來處理,我們來看看settings.getLogAdapter()這個方法(Settings.java文件): 

  1. public LogAdapter getLogAdapter() {   
  2.     if (logAdapter == null) { 
  3.         // 最終的實現類是AndroidLogAdapter 
  4.         logAdapter = new AndroidLogAdapter();   
  5.     }   
  6.     return logAdapter; 
  7.  

找到AndroidLogAdapter類:

 原來繞了一大圈,最終打印還是使用了:系統的Log。

好了Logger日志框架的源碼解析完了,有沒有更清晰呢,也許小伙伴會說這個最終的日志打印,我不想用系統的Log,是不是可以換呢。這是自然的,看開篇的那種整體架構圖,這個LogAdapter是個接口,只要實現這個接口,里面做你自己想要打印的方式,然后通過Settings 的logAdapter(LogAdapter logAdapter)方法設置進去就可以。

以上就是博主分析一個開源庫的思路,從使用的角度出發抽繭剝絲,基本上一個庫的核心部分都能搞懂。畫畫整個框架的大概類圖,對分析庫非常有幫助,每一個輪子都有值得學習的地方,吸收了就是進步的開始,耐心的分析完一個庫,還是非常有成就感的。

感謝你耐心看完,以后博主還會繼續努力分析其它輪子的。

責任編輯:龐桂玉 來源: segmentfault
相關推薦

2016-09-22 19:31:30

開源日志庫Logger

2012-12-03 10:44:00

開源

2012-07-10 16:22:01

開源架構

2010-02-05 15:33:29

Android JDK

2010-02-04 11:05:56

ibmdw虛擬化

2009-10-28 13:44:40

linux庫文件路徑

2019-11-27 14:41:50

Java技術語言

2019-04-22 15:40:33

2010-02-03 11:26:28

2010-04-21 11:43:33

Oracle數據庫

2010-01-07 18:03:03

Linux動態庫

2019-01-30 18:00:21

開源Python庫

2024-10-23 16:06:50

2018-09-27 11:25:07

開源日志聚合

2021-08-10 08:52:15

微軟GCToolkit工具

2010-05-31 17:56:27

2009-06-05 09:45:44

Struts優缺點開源

2016-12-06 10:39:54

剖析建模開源工具

2010-05-20 18:05:38

2010-04-12 15:17:40

dump Oracle
點贊
收藏

51CTO技術棧公眾號

男人天堂av电影| 免费不卡av在线| 中文字幕你懂的| 一区二区免费不卡在线| 精品成人一区二区三区四区| 岛国大片在线播放| 欧美18xxxxx| 蜜桃久久精品一区二区| 久久国产精品亚洲| 国产呦小j女精品视频| 婷婷激情一区| 亚洲精品免费看| 激情伦成人综合小说| 波多野结衣视频在线看| 欧美高清不卡| 一本大道久久加勒比香蕉| 亚欧美一区二区三区| 成人欧美大片| 亚洲综合一区二区三区| 日韩福利二区| 隣の若妻さん波多野结衣| 日精品一区二区三区| 欧美日韩第一视频| 538精品视频| 成人自拍在线| 欧美日韩国产a| 狠狠干 狠狠操| 欧美三级电影一区二区三区| 91丨porny丨中文| 91手机视频在线观看| 波多野结衣视频观看| 亚洲高清毛片| 欧美精品在线观看| 国产精品综合激情| 精品一区在线| 日韩电视剧在线观看免费网站| 中文字幕 日韩 欧美| 中文字幕乱码在线播放| 亚洲一级二级在线| www.69av| 亚洲淫性视频| 亚洲视频综合在线| 亚洲一区二区三区免费看| 欧美偷拍视频| 91蜜桃在线免费视频| 国产精品二区在线| www.av网站| 国产一区视频网站| 成人在线国产精品| 一卡二卡在线观看| 另类小说视频一区二区| 国产精品电影久久久久电影网| 日本少妇bbwbbw精品| 国内精品久久久久久久97牛牛| 欧美美女15p| 日韩影院一区二区| 欧美一区激情| 欧美高清性猛交| 久久国产露脸精品国产| 亚洲午夜激情在线| 久久久久女教师免费一区| 国产成人无码aa精品一区| 一区二区影视| 欧美激情精品久久久久久黑人| 青青草原在线免费观看| 国产精品激情| 97欧美精品一区二区三区| 日本少妇激情舌吻| 乱码第一页成人| 国产精品久久久久久一区二区 | 91av视频在线免费观看| 久草视频精品在线| 中国女人久久久| 日韩av免费在线播放| 国产精品传媒在线观看| 久久精品国产精品亚洲精品| 亚洲a中文字幕| www.精品视频| 99精品一区二区| 色噜噜一区二区| av在线电影网| 亚洲精品免费看| 无码人妻精品一区二区三区在线| 神马午夜在线视频| 欧美视频中文字幕| 精品人妻一区二区三| 欧美理伦片在线播放| 亚洲视屏在线播放| 91香蕉视频在线播放| 欧美久久99| 欧美制服第一页| 亚洲最大成人av| 国产丶欧美丶日本不卡视频| 国产精品日韩一区二区| 免费一级在线观看播放网址| 国产精品乱人伦一区二区| 精品一区二区三区毛片| 亚洲小少妇裸体bbw| 欧美日韩国产免费一区二区| 中文字幕在线国产| 精品国产一区二区三区av片| 日韩视频一区在线| 青青草av在线播放| 激情六月婷婷久久| 蜜桃av色综合| 91小视频xxxx网站在线| 日韩欧美精品中文字幕| 国产黑丝在线视频| 精品美女在线视频| 国内成人精品一区| 一区二区三区免费在线视频| 99久久国产综合色|国产精品| 亚洲精品8mav| 欧美巨大丰满猛性社交| 91.成人天堂一区| 国产精品无码一区二区三区免费| 外国成人激情视频| 日本精品一区二区三区在线| 成 人 黄 色 片 在线播放| 中文字幕乱码亚洲精品一区| 成人免费aaa| 国产一区二区三区| 在线亚洲男人天堂| 成人毛片18女人毛片| 国产精品中文字幕一区二区三区| 日本不卡二区| 韩国成人二区| 精品国产免费人成在线观看| 肉色超薄丝袜脚交69xx图片| 视频一区二区不卡| 久久99九九| 日本动漫同人动漫在线观看| 欧美日韩不卡一区| 性猛交ⅹxxx富婆video| 国产精品一二| 国内外成人免费视频| 中文字幕在线观看网站| 欧美日韩精品一区二区在线播放| 一区二区黄色片| 国产模特精品视频久久久久| 国产亚洲二区| free性m.freesex欧美| 日韩区在线观看| 国产高潮流白浆| 狠狠v欧美v日韩v亚洲ⅴ| 午夜精品美女久久久久av福利 | 国产欧美日韩中文字幕| 国产美女性感在线观看懂色av| 欧美日韩午夜激情| aa片在线观看视频在线播放| 亚洲久久一区| 激情五月综合色婷婷一区二区 | 91精品专区| 欧美午夜精品久久久久久超碰| 公肉吊粗大爽色翁浪妇视频| 日韩精品欧美精品| 亚洲不卡1区| 欧美极品免费| 色999日韩欧美国产| 国产精品欧美亚洲| 亚洲日本在线天堂| 女人扒开腿免费视频app| 红桃视频国产精品| 九九九九九精品| 综合在线影院| 日韩在线观看免费av| 国产aⅴ爽av久久久久成人| 亚洲精品自拍动漫在线| 蜜臀视频在线观看| 亚洲女人av| 亚洲日本japanese丝袜| 国产美女亚洲精品7777| 欧美日韩国产成人| 无码国产色欲xxxx视频| 在线一区二区三区四区五区| 国产精品一区二区亚洲| 狠狠色丁香婷婷综合久久片| r级无码视频在线观看| 希岛爱理av免费一区二区| 国产精品美女www爽爽爽视频| 91在线看黄| 欧美成人video| 九九精品免费视频| 国产精品久久国产精麻豆99网站| 日本精品一区在线| 激情六月综合| 午夜一区二区三区| 天堂久久av| 国产精品777| h片在线免费观看| 日韩精品在线免费观看| 在线观看毛片av| 亚洲午夜久久久久久久久电影院| 97超碰在线资源| 韩国三级中文字幕hd久久精品| 青草青青在线视频| 成人激情视频| 国产一区二区三区四区五区在线| 99九九久久| 国内精品久久久久影院 日本资源| 男女视频在线观看免费| 欧美电影免费观看完整版| 蜜臀精品一区二区三区| 亚洲精品视频在线| 免费在线观看a视频| 成人动漫精品一区二区| 中文字幕天天干| av成人国产| 一本二本三本亚洲码| 日韩精品免费一区二区夜夜嗨| 91啪国产在线| 成人网ww555视频免费看| 国内精品视频久久| 国产福利视频在线观看| 伊人久久久久久久久久久| 刘亦菲久久免费一区二区| 777奇米成人网| 国产精品sm调教免费专区| 天天色综合天天| 免费人成视频在线| 中文字幕一区二区5566日韩| av女人的天堂| 99精品偷自拍| 一级黄色片毛片| 国产精品99久久久久久有的能看 | 日本视频精品一区| 久久综合五月婷婷| 91中文字精品一区二区| 亚洲欧美专区| 国产日本欧美一区二区三区| 午夜日韩成人影院| 欧美在线观看日本一区| av资源网在线播放| 久久人人97超碰精品888| 在线观看av免费| 久久亚洲精品一区二区| 亚洲1卡2卡3卡4卡乱码精品| 国产一区二区三区精品久久久 | 日韩中文字幕在线视频观看| 激情亚洲网站| 无码毛片aaa在线| 国产精品久久占久久| 亚洲三区在线| 久久精品不卡| 国产精品无码乱伦| 亚洲区小说区图片区qvod按摩| 久久riav二区三区| 性欧美lx╳lx╳| 蜜桃狠狠色伊人亚洲综合网站| 精品国产导航| 久久av免费一区| 日韩精品福利一区二区三区| 久久影视中文粉嫩av| 网红女主播少妇精品视频| 欧美日本亚洲| 欧美色女视频| 一区二区不卡在线视频 午夜欧美不卡' | 狠色狠色综合久久| 99在线免费视频观看| 亚洲乱码视频| 成熟老妇女视频| 免费观看日韩电影| 免费精品99久久国产综合精品应用| 美女看a上一区| 91精品国产三级| 国产91对白在线观看九色| 大桥未久恸哭の女教师| 91在线观看污| 国产黄色录像视频| 亚洲同性gay激情无套| 九九在线观看视频| 精品成人乱色一区二区| 日本视频在线观看免费| 欧美三级视频在线观看 | 亚洲福利视频网站| 日本国产在线| 色偷偷偷亚洲综合网另类| 成人欧美在线| 51精品在线观看| 久久久精品一区二区毛片免费看| 亚洲最大的免费| 欧美综合自拍| 亚洲人成人77777线观看| 国产精品chinese| www黄色av| 国产一区视频网站| 不卡一区二区在线观看| 综合久久一区二区三区| 国产无精乱码一区二区三区| 在线观看亚洲成人| 性欧美一区二区三区| 亚洲欧美精品中文字幕在线| 米奇777四色精品人人爽| 性日韩欧美在线视频| 精品裸体bbb| 国产传媒一区| 日韩精品久久久久久久电影99爱| 久久久久99精品成人片| 免费美女久久99| 在线精品视频播放| 17c精品麻豆一区二区免费| 国产主播在线观看| 欧美日韩国产综合一区二区三区| 日本韩国在线观看| xvideos亚洲人网站| 在线免费日韩片| 成人做爰66片免费看网站| 成人激情电影在线| 无码人妻丰满熟妇区96| 国产乱码精品一品二品| 91精品久久久久久久久久久久| 午夜久久久久久久久久一区二区| 一级做a爱片久久毛片| 亚洲精品午夜精品| 黄页网站在线| 成人黄色免费网站在线观看| 国产中文字幕一区二区三区| 国产九九九九九| 成人一区二区三区视频| 国产麻豆视频在线观看| 在线观看一区二区精品视频| 天堂网在线播放| 欧美成人久久久| 91九色成人| 视频一区二区精品| 亚洲中字黄色| 亚洲啪av永久无码精品放毛片| 1024亚洲合集| 91精品国产乱码久久久| 伊人久久精品视频| 日韩欧美一区二区三区免费观看| 久久www免费人成精品| 亚洲视频狠狠| 又色又爽又黄18网站| 亚洲激情第一区| 99久久精品日本一区二区免费| 神马久久久久久| 激情久久一区二区| 日韩在线第一区| 青草av.久久免费一区| 色哟哟精品观看| 日本高清不卡aⅴ免费网站| 深夜福利在线视频| 欧洲日本亚洲国产区| 亚洲第一论坛sis| 欧美日韩国产精品激情在线播放| 成人av在线电影| av大片在线免费观看| 亚洲精品99久久久久| 麻豆免费版在线观看| 精品国产一区二区三区四区精华| 亚洲人成免费| 美国黄色一级毛片| 日韩欧美主播在线| 欧美在线观看在线观看| 国产成人精品日本亚洲| 久久国产电影| 黄色a级三级三级三级| 亚洲综合色婷婷| 狠狠躁日日躁夜夜躁av| 91精品国产91久久久久福利| 日韩精选在线| 欧美视频免费播放| 亚洲国产精品成人综合| 国产一区二区视频免费观看 | 男人av在线播放| 欧美精品尤物在线| 久久精品国产精品亚洲综合| 日本妇女毛茸茸| 亚洲国产欧美一区二区三区同亚洲 | 视频小说一区二区| mm1313亚洲国产精品无码试看| 国产精品久久国产精麻豆99网站| 国产日产亚洲系列最新| 高清视频欧美一级| 一个色免费成人影院| 国产三级国产精品国产专区50| 亚洲欧美区自拍先锋| 免费看av毛片| 国产精品免费久久久久久| 亚洲欧美日韩高清在线| 日韩免费高清一区二区| 欧洲av在线精品| 污影院在线观看| 欧美一二三区| 国产在线看一区| 国产免费观看av| 一区二区欧美久久| 中文无码日韩欧| 99草草国产熟女视频在线| 亚洲色图19p| 男人天堂资源在线| 91情侣在线视频| 久久一区中文字幕| 欧美精品入口蜜桃| 中文字幕日韩欧美在线视频| 成人动漫视频| 手机av在线免费| 欧美性猛交xxxx免费看漫画| 高h视频在线观看|