日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

android 日志文件LogUtils實例

瀏覽:154日期:2022-09-22 18:14:24

背景

這是好久之前在網上找的一個常用類,已經忘記原文鏈接了,但是覺得很好用一直都在用,可以將日志寫到file里面也可以定位你是在哪個類哪一行打印的日志,保存到文件的路徑就是android/data/你的包名/files/目錄下,然后我們就可以愉快的找問題了

import android.text.TextUtils;import android.util.Log;import com.smartlink.suixing.App;import com.smartlink.suixing.BuildConfig;import java.io.BufferedWriter;import java.io.File;import java.io.FileOutputStream;import java.io.IOException;import java.io.OutputStreamWriter;import java.text.SimpleDateFormat;import java.util.Date;import java.util.Formatter;import java.util.Locale;public class LogUtils {public static StringcustomTagPrefix= 'log';// 自定義Tag的前綴,可以是作者名private static final booleanisSaveLog= true;// 是否把保存日志到SD卡中private static StringcacheDirPath;private LogUtils() {}// 容許打印日志的類型,默認是true,設置為false則不打印public static booleanallowD= BuildConfig.DEBUG;public static booleanallowE= BuildConfig.DEBUG;public static booleanallowI= BuildConfig.DEBUG;public static booleanallowV= BuildConfig.DEBUG;public static booleanallowW= BuildConfig.DEBUG;public static booleanallowWtf= BuildConfig.DEBUG;// public static boolean allowD = true;// public static boolean allowE = true;// public static boolean allowI = true;// public static boolean allowV = true;// public static boolean allowW = true;// public static boolean allowWtf = true;private static String generateTag(StackTraceElement caller) {String tag = '%s.%s(Line:%d)'; // 占位符String callerClazzName = caller.getClassName(); // 獲取到類名callerClazzName = callerClazzName.substring(callerClazzName.lastIndexOf('.') + 1);tag = String.format(tag, callerClazzName, caller.getMethodName(), caller.getLineNumber()); // 替換tag = TextUtils.isEmpty(customTagPrefix) ? tag : customTagPrefix + ':' + tag;return tag;}/*** * 打印控制臺顯示不了那么長的日志問題 * * @param msg */public static void logE(String msg) { // 信息太長,分段打印if (!allowE) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);// 因為String的length是字符數量不是字節數量所以為了防止中文字符過多,// 把4*1024的MAX字節打印長度改為2001字符數int max_str_length = 2001 - tag.length();// 大于4000時while (msg.length() > max_str_length) {// Log.e(tag, msg.substring(0, max_str_length));LogUtils.e(msg.substring(0, max_str_length));msg = msg.substring(max_str_length);}// 剩余部分// Log.e(tag, msg);LogUtils.e(msg);}/** * 自定義的logger */public static CustomLogger customLogger;public interface CustomLogger {void d(String tag, String content);void d(String tag, String content, Throwable tr);void e(String tag, String content);void e(String tag, String content, Throwable tr);void i(String tag, String content);void i(String tag, String content, Throwable tr);void v(String tag, String content);void v(String tag, String content, Throwable tr);void w(String tag, String content);void w(String tag, String content, Throwable tr);void w(String tag, Throwable tr);void wtf(String tag, String content);void wtf(String tag, String content, Throwable tr);void wtf(String tag, Throwable tr);}public static void d(String content) {if (!allowD) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.d(tag, content);} else {Log.d(tag, content);}}public static void d(String content, Throwable tr) {if (!allowD) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.d(tag, content, tr);} else {Log.d(tag, content, tr);}}public static void e(String content) {if (!allowE) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.e(tag, content);} else {Log.e(tag, content);}if (isSaveLog) {point(cacheDirPath, tag, content);}}public static void e(String content, Throwable tr) {if (!allowE) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.e(tag, content, tr);} else {Log.e(tag, content, tr);}if (isSaveLog) {point(cacheDirPath, tag, tr.getMessage());}}public static void e(Throwable tr) {if (!allowE) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.e(tag, '', tr);} else {Log.e(tag, '', tr);}if (isSaveLog) {point(cacheDirPath, tag, tr.getMessage());}}public static void i(String content) {if (!allowI) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.i(tag, content);} else {Log.i(tag, content);}}public static void i(String content, Throwable tr) {if (!allowI) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.i(tag, content, tr);} else {Log.i(tag, content, tr);}}public static void v(String content) {if (!allowV) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.v(tag, content);} else {Log.v(tag, content);}}public static void v(String content, Throwable tr) {if (!allowV) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.v(tag, content, tr);} else {Log.v(tag, content, tr);}}public static void w(String content) {if (!allowW) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.w(tag, content);} else {Log.w(tag, content);}}public static void w(String content, Throwable tr) {if (!allowW) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.w(tag, content, tr);} else {Log.w(tag, content, tr);}}public static void w(Throwable tr) {if (!allowW) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.w(tag, tr);} else {Log.w(tag, tr);}}public static void wtf(String content) {if (!allowWtf) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.wtf(tag, content);} else {Log.wtf(tag, content);}}public static void wtf(String content, Throwable tr) {if (!allowWtf) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.wtf(tag, content, tr);} else {Log.wtf(tag, content, tr);}}public static void wtf(Throwable tr) {if (!allowWtf) return;StackTraceElement caller = getCallerStackTraceElement();String tag = generateTag(caller);if (customLogger != null) {customLogger.wtf(tag, tr);} else {Log.wtf(tag, tr);}}private static StackTraceElement getCallerStackTraceElement() {return Thread.currentThread().getStackTrace()[4];}public static void point(String path, String tag, String msg) {if (isSDAva()) {path = cacheDirPath;Date date = new Date();SimpleDateFormat dateFormat = new SimpleDateFormat('', Locale.SIMPLIFIED_CHINESE);dateFormat.applyPattern('yyyy');path = path + dateFormat.format(date) + '/';dateFormat.applyPattern('MM');path += dateFormat.format(date) + '/';dateFormat.applyPattern('dd');path += dateFormat.format(date) + '.log';dateFormat.applyPattern('[yyyy-MM-dd HH:mm:ss]');String time = dateFormat.format(date);File file = new File(path);if (!file.exists()) createDipPath(path);BufferedWriter out = null;try {out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(file, true)));out.write(time + ' ' + tag + ' ' + msg + 'rn');} catch (Exception e) {e.printStackTrace();} finally {try {if (out != null) {out.close();}} catch (IOException e) {e.printStackTrace();}}}}/** * 根據文件路徑 遞歸創建文件 * * @param file */public static void createDipPath(String file) {String parentFile = file.substring(0, file.lastIndexOf('/'));File file1 = new File(file);File parent = new File(parentFile);if (!file1.exists()) {parent.mkdirs();try {file1.createNewFile();LogUtils.e('日志文件的路徑是' + file1.getAbsolutePath());} catch (IOException e) {e.printStackTrace();}}}/** * A little trick to reuse a formatter in the same thread */private static class ReusableFormatter {private Formatterformatter;private StringBuilderbuilder;public ReusableFormatter() {builder = new StringBuilder();formatter = new Formatter(builder);}public String format(String msg, Object... args) {formatter.format(msg, args);String s = builder.toString();builder.setLength(0);return s;}}private static final ThreadLocal<ReusableFormatter> thread_local_formatter = new ThreadLocal<ReusableFormatter>() {protected ReusableFormatter initialValue() {return new ReusableFormatter();}};public static String format(String msg, Object... args) {ReusableFormatter formatter = thread_local_formatter.get();return formatter.format(msg, args);}public static boolean isSDAva() {if (cacheDirPath == null) cacheDirPath = App.getAppContext().getExternalFilesDir('log').getAbsolutePath();if (!TextUtils.isEmpty(cacheDirPath)) {return true;} else {return false;}}}

補充知識:Android日志打印類LogUtils,能夠定位到類名,方法名以及出現錯誤的行數并保存日志文件

在開發中,我們常常用打印log的方式來調試我們的應用。在Java中我們常常使用方法System.out.println()來在控制臺打印日志,以便我們的調試。Android中有一個專門的類Log來實現在Android系統下日志的打印,更加方便我們定位程序出現問題的地方。

但是Android官方提供的Log類在實際項目使用中,也不是非常方便。當程序出現錯誤時,我們最希望的就是這個Log類能幫我們定位到是哪個類的哪個方法,甚至于是那一行出現了錯誤。這樣就能給我們的調試帶來很大的便利。

同時我們也應該想到為了應用程序的安全起見,在app正式上線之前,我們應該要把打印日志的功能關閉掉,以防別人通過Log來破解你的應用。生產模式的下打印Log,正式模式就把打印日志的功能關閉掉,要做到Log的靈活關閉與打開,也需要在原生的Log類上進行一些封裝。

還有一種時候,當我們的程序出現問題崩潰了,我們希望能夠收集到出現異常的原因進行分析,所以可以把Log日志保存到一個文件中,放在SD卡程序創建的目錄下。也可以在用戶聯網的情況下,在程序的后臺把出異常的Log日志文件上傳到服務端,方便程序員進行分析,解決bug。

今天就給大家分享一個做項目中很實用的一個Log類LogUtils,這個類是從xUtils中提取出來,稍作修改的,有注釋。

示例:

我們在MainActivity中調用一些LogUtils中的方法,注意看行數。 1

接著看看控制臺打印的日志是不是像MainActivity調用的那樣,Log中有這個類的名字和oncreate方法名,已及當前行數;  2

看到上圖中的Log日志是不是很方便定位程序中的問題了,出現異常也能快速的定位到。然后把下面的Log類型在程序的任何地方設置為false則不會打印日志,使用起來相當的方便。 

 // 容許打印日志的類型,默認是true,設置為false則不打印 public static boolean allowD = true; public static boolean allowE = true; public static boolean allowI = true; public static boolean allowV = true; public static boolean allowW = true; public static boolean allowWtf = true;

代碼貼在下面:

package com.finddreams.log;import java.io.BufferedWriter;import java.io.File;import java.io.FileOutputStream;import java.io.IOException;import java.io.OutputStreamWriter;import java.text.SimpleDateFormat;import java.util.Date;import java.util.Formatter;import java.util.Locale;import android.os.Environment;import android.text.TextUtils;import android.util.Log;/** * Log工具,類似android.util.Log。 tag自動產生,格式: * customTagPrefix:className.methodName(Line:lineNumber), * customTagPrefix為空時只輸出:className.methodName(Line:lineNumber)。 */public class LogUtils { public static String customTagPrefix = 'finddreams'; // 自定義Tag的前綴,可以是作者名 private static final boolean isSaveLog = false; // 是否把保存日志到SD卡中 public static final String ROOT = Environment.getExternalStorageDirectory() .getPath() + '/finddreams/'; // SD卡中的根目錄 private static final String PATH_LOG_INFO = ROOT + 'info/'; private LogUtils() { } // 容許打印日志的類型,默認是true,設置為false則不打印 public static boolean allowD = true; public static boolean allowE = true; public static boolean allowI = true; public static boolean allowV = true; public static boolean allowW = true; public static boolean allowWtf = true; private static String generateTag(StackTraceElement caller) { String tag = '%s.%s(Line:%d)'; // 占位符 String callerClazzName = caller.getClassName(); // 獲取到類名 callerClazzName = callerClazzName.substring(callerClazzName .lastIndexOf('.') + 1); tag = String.format(tag, callerClazzName, caller.getMethodName(), caller.getLineNumber()); // 替換 tag = TextUtils.isEmpty(customTagPrefix) ? tag : customTagPrefix + ':' + tag; return tag; } /** * 自定義的logger */ public static CustomLogger customLogger; public interface CustomLogger { void d(String tag, String content); void d(String tag, String content, Throwable tr); void e(String tag, String content); void e(String tag, String content, Throwable tr); void i(String tag, String content); void i(String tag, String content, Throwable tr); void v(String tag, String content); void v(String tag, String content, Throwable tr); void w(String tag, String content); void w(String tag, String content, Throwable tr); void w(String tag, Throwable tr); void wtf(String tag, String content); void wtf(String tag, String content, Throwable tr); void wtf(String tag, Throwable tr); } public static void d(String content) { if (!allowD) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.d(tag, content); } else { Log.d(tag, content); } } public static void d(String content, Throwable tr) { if (!allowD) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.d(tag, content, tr); } else { Log.d(tag, content, tr); } } public static void e(String content) { if (!allowE) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.e(tag, content); } else { Log.e(tag, content); } if (isSaveLog) { point(PATH_LOG_INFO, tag, content); } } public static void e(String content, Throwable tr) { if (!allowE) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.e(tag, content, tr); } else { Log.e(tag, content, tr); } if (isSaveLog) { point(PATH_LOG_INFO, tag, tr.getMessage()); } } public static void i(String content) { if (!allowI) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.i(tag, content); } else { Log.i(tag, content); } } public static void i(String content, Throwable tr) { if (!allowI) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.i(tag, content, tr); } else { Log.i(tag, content, tr); } } public static void v(String content) { if (!allowV) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.v(tag, content); } else { Log.v(tag, content); } } public static void v(String content, Throwable tr) { if (!allowV) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.v(tag, content, tr); } else { Log.v(tag, content, tr); } } public static void w(String content) { if (!allowW) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.w(tag, content); } else { Log.w(tag, content); } } public static void w(String content, Throwable tr) { if (!allowW) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.w(tag, content, tr); } else { Log.w(tag, content, tr); } } public static void w(Throwable tr) { if (!allowW) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.w(tag, tr); } else { Log.w(tag, tr); } } public static void wtf(String content) { if (!allowWtf) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.wtf(tag, content); } else { Log.wtf(tag, content); } } public static void wtf(String content, Throwable tr) { if (!allowWtf) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.wtf(tag, content, tr); } else { Log.wtf(tag, content, tr); } } public static void wtf(Throwable tr) { if (!allowWtf) return; StackTraceElement caller = getCallerStackTraceElement(); String tag = generateTag(caller); if (customLogger != null) { customLogger.wtf(tag, tr); } else { Log.wtf(tag, tr); } } private static StackTraceElement getCallerStackTraceElement() { return Thread.currentThread().getStackTrace()[4]; } public static void point(String path, String tag, String msg) { if (isSDAva()) { Date date = new Date(); SimpleDateFormat dateFormat = new SimpleDateFormat('', Locale.SIMPLIFIED_CHINESE); dateFormat.applyPattern('yyyy'); path = path + dateFormat.format(date) + '/'; dateFormat.applyPattern('MM'); path += dateFormat.format(date) + '/'; dateFormat.applyPattern('dd'); path += dateFormat.format(date) + '.log'; dateFormat.applyPattern('[yyyy-MM-dd HH:mm:ss]'); String time = dateFormat.format(date); File file = new File(path); if (!file.exists()) createDipPath(path); BufferedWriter out = null; try { out = new BufferedWriter(new OutputStreamWriter( new FileOutputStream(file, true))); out.write(time + ' ' + tag + ' ' + msg + 'rn'); } catch (Exception e) { e.printStackTrace(); } finally { try { if (out != null) { out.close(); } } catch (IOException e) { e.printStackTrace(); } } } } /** * 根據文件路徑 遞歸創建文件 * * @param file */ public static void createDipPath(String file) { String parentFile = file.substring(0, file.lastIndexOf('/')); File file1 = new File(file); File parent = new File(parentFile); if (!file1.exists()) { parent.mkdirs(); try { file1.createNewFile(); } catch (IOException e) { e.printStackTrace(); } } } /** * A little trick to reuse a formatter in the same thread */ private static class ReusableFormatter { private Formatter formatter; private StringBuilder builder; public ReusableFormatter() { builder = new StringBuilder(); formatter = new Formatter(builder); } public String format(String msg, Object... args) { formatter.format(msg, args); String s = builder.toString(); builder.setLength(0); return s; } } private static final ThreadLocal<ReusableFormatter> thread_local_formatter = new ThreadLocal<ReusableFormatter>() { protected ReusableFormatter initialValue() { return new ReusableFormatter(); } }; public static String format(String msg, Object... args) { ReusableFormatter formatter = thread_local_formatter.get(); return formatter.format(msg, args); } public static boolean isSDAva() { if (Environment.getExternalStorageState().equals( Environment.MEDIA_MOUNTED) || Environment.getExternalStorageDirectory().exists()) { return true; } else { return false; } }}

以上這篇android 日志文件LogUtils實例就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持好吧啦網。

標簽: Android
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
日韩中文字幕麻豆| 国产综合色产| 亚洲欧洲日韩| 中文字幕一区日韩精品| 亚洲一区二区三区免费在线观看| 91成人网在线观看| 亚洲精品午夜av福利久久蜜桃| 狠狠操综合网| 国产精品女主播一区二区三区| 日韩午夜一区| 热久久国产精品| 日韩亚洲国产欧美| 日本一区二区中文字幕| 视频在线观看一区| 日韩av一二三| 欧美国产先锋| 精品视频亚洲| 91精品韩国| 99国产精品自拍| 一区二区三区午夜视频| 国产日韩一区| 黄色成人精品网站| 日韩成人精品一区二区三区 | 精品色999| 日本国产欧美| 麻豆免费精品视频| 国产精品精品| 91成人精品视频| 亚洲综合中文| 欧美午夜网站| 国产成人77亚洲精品www| 91精品国产乱码久久久久久久| 五月天久久777| 日韩极品在线观看| 国内不卡的一区二区三区中文字幕| 久久在线免费| 日韩欧美一区二区三区在线观看| 国产精品v一区二区三区| 蜜桃久久久久| 香蕉成人av| 亚洲三级在线| 国产一区二区三区四区五区| 亚洲精品一二三区区别| 亚洲高清激情| 免费视频最近日韩| 国产精品一卡| 性欧美xxxx免费岛国不卡电影| 综合一区在线| 91看片一区| 日韩国产91| 天堂中文在线播放| 亚洲男人在线| 欧美va天堂| 97精品国产99久久久久久免费| 欧美极品一区二区三区| 好吊一区二区三区| 91亚洲无吗| 99国产精品免费视频观看| 免费观看在线综合| 最新中文字幕在线播放| 亚洲欧美网站| 久久精品亚洲| 久久一二三区| 麻豆久久一区| 亚洲专区欧美专区| 国语精品一区| 一区二区三区网站| 国产一区二区三区天码| 日韩专区欧美专区| 国产色播av在线| 婷婷亚洲成人| 欧美性感美女一区二区| 国产精东传媒成人av电影| 日韩视频免费| 日本一区二区免费高清| 日韩区一区二| 欧美va亚洲va日韩∨a综合色| 国产精品一区二区三区美女| 欧美综合社区国产| 在线亚洲一区| 日韩一区亚洲二区| 国产日韩中文在线中文字幕| 不卡av一区二区| 国内在线观看一区二区三区| 色在线中文字幕| 欧美在线看片| 亚洲精品麻豆| 天堂√8在线中文| 国产日产一区| 久久国产福利| 欧美va天堂在线| 欧美国产偷国产精品三区| 亚洲精品自拍| 午夜欧美精品| 播放一区二区| 精品视频亚洲| 欧美日本不卡高清| 激情自拍一区| 在线看片国产福利你懂的| 日本欧美在线| 亚洲三级视频| 日韩视频在线一区二区三区| 少妇精品导航| 色在线中文字幕| 国产一区不卡| 精品免费av| 国产激情欧美| 国产精品最新自拍| 日本va欧美va精品发布| 日韩欧美高清一区二区三区| 综合亚洲视频| 亚洲资源在线| 久久亚洲一区| 午夜精品亚洲| 日韩一级欧洲| 亚洲综合另类| 99久久婷婷这里只有精品| 免费在线小视频| yellow在线观看网址| 国产精品成人a在线观看| 麻豆久久一区| 欧美激情99| 精品一区二区三区亚洲| 久久av网址| 成人精品动漫一区二区三区| 国产一区二区三区不卡av| 精品久久电影| 97精品中文字幕| 国产aa精品| 人人精品亚洲| 91国语精品自产拍| 久久国产99| 日韩美女国产精品| 日韩国产在线不卡视频| 亚洲午夜久久| 日韩和欧美一区二区三区| 国产精品欧美在线观看| 日韩av福利| 99久久婷婷| 欧美综合国产| 亚洲九九精品| 国产精品美女久久久久久不卡| 久久av网址| 久久久免费人体| 91视频一区| 国产一区二区三区自拍| 亚洲欧美综合| 蜜臀精品久久久久久蜜臀| 日韩黄色免费网站| 麻豆91精品91久久久的内涵| 日韩欧美一区二区三区在线视频 | 日韩av有码| 水蜜桃久久夜色精品一区| 欧美天堂视频| 99视频一区| 日韩综合一区二区三区| 国产精品一线| 99久久视频| 亚洲精品影院在线观看| 国产日产精品_国产精品毛片| 精品国产午夜肉伦伦影院| 久久久久网站| 日韩精品一级中文字幕精品视频免费观看 | 日韩精品视频一区二区三区| 国产精品hd| 成人午夜毛片| 不卡中文一二三区| 日韩精品高清不卡| 激情久久99| 黄页网站一区| 国产九九精品| 美女网站视频一区| 巨乳诱惑日韩免费av| 欧美久久精品| 欧洲精品一区二区三区| 亚洲免费观看高清完整版在线观| 欧美国产不卡| 国产手机视频一区二区| 国产精品入口久久| 亚洲视频综合| 国产精品尤物| 女同性一区二区三区人了人一| 欧美日韩精品一区二区三区视频| 日韩伦理在线一区| 亚洲精品日韩久久| 蜜桃视频在线网站| 日韩免费小视频| 丝袜亚洲另类欧美| 精品国产乱码久久久久久樱花| 蜜桃国内精品久久久久软件9| 国产欧美一区二区三区国产幕精品| 日韩高清成人| 欧美一级久久| 久久精品卡一| 国产亚洲一区| 99国产精品久久久久久久成人热| 欧美激情久久久久久久久久久| 国产精品试看| 福利一区和二区| 日韩1区2区3区|