色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

一種輕量分表方案-MyBatis攔截器分表實(shí)踐

京東云 ? 來源:京東零售 張均杰 ? 作者:京東零售 張均杰 ? 2024-12-03 11:19 ? 次閱讀

作者:京東零售 張均杰

背景

部門內(nèi)有一些億級(jí)別核心業(yè)務(wù)表增速非常快,增量日均100W,但線上業(yè)務(wù)只依賴近一周的數(shù)據(jù)。隨著數(shù)據(jù)量的迅速增長(zhǎng),慢SQL頻發(fā),數(shù)據(jù)庫(kù)性能下降,系統(tǒng)穩(wěn)定性受到嚴(yán)重影響。本篇文章,將分享如何使用MyBatis攔截器低成本的提升數(shù)據(jù)庫(kù)穩(wěn)定性。

業(yè)界常見方案

針對(duì)冷數(shù)據(jù)多的大表,常用的策略有以2種:

刪除/歸檔舊數(shù)據(jù)。

分表。

歸檔/刪除舊數(shù)據(jù)

定期將冷數(shù)據(jù)移動(dòng)到歸檔表或者冷存儲(chǔ)中,或定期對(duì)表進(jìn)行刪除,以減少表的大小。此策略邏輯簡(jiǎn)單,只需要編寫一個(gè)JOB定期執(zhí)行SQL刪除數(shù)據(jù)。我們開始也是用這種方案,但此方案也有一些副作用:

1.數(shù)據(jù)刪除會(huì)影響數(shù)據(jù)庫(kù)性能,引發(fā)慢sql,多張表并行刪除,數(shù)據(jù)庫(kù)壓力會(huì)更大。

2.頻繁刪除數(shù)據(jù),會(huì)產(chǎn)生數(shù)據(jù)庫(kù)碎片,影響數(shù)據(jù)庫(kù)性能,引發(fā)慢SQL。

綜上,此方案有一定風(fēng)險(xiǎn),為了規(guī)避這種風(fēng)險(xiǎn),我們決定采用另一種方案:分表。

分表

我們決定按日期對(duì)表進(jìn)行橫向拆分,實(shí)現(xiàn)讓系統(tǒng)每周生成一張周期表,表內(nèi)只存近一周的數(shù)據(jù),規(guī)避單表過大帶來的風(fēng)險(xiǎn)。

分表方案選型

經(jīng)調(diào)研,考慮2種分表方案:Sharding-JDBC、利用Mybatis自帶的攔截器特性。

經(jīng)過對(duì)比后,決定采用Mybatis攔截器來實(shí)現(xiàn)分表,原因如下:

1.JAVA生態(tài)中很常用的分表框架是Sharding-JDBC,雖然功能強(qiáng)大,但需要一定的接入成本,并且很多功能暫時(shí)用不上。

2.系統(tǒng)本身已經(jīng)在使用Mybatis了,只需要添加一個(gè)mybaits攔截器,把SQL表名替換為新的周期表就可以了,沒有接入新框架的成本,開發(fā)成本也不高。

分表具體實(shí)現(xiàn)代碼

分表配置對(duì)象

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

import java.util.Date;

@Data
@AllArgsConstructor
@NoArgsConstructor
public class ShardingProperty {
    // 分表周期天數(shù),配置7,就是一周一分
    private Integer days;
    // 分表開始日期,需要用這個(gè)日期計(jì)算周期表名
    private Date beginDate;
    // 需要分表的表名
    private String tableName;
}

分表配置類

import java.util.concurrent.ConcurrentHashMap;

public class ShardingPropertyConfig {

    public static final ConcurrentHashMap SHARDING_TABLE = new ConcurrentHashMap();

    static {
        ShardingProperty orderInfoShardingConfig = new ShardingProperty(15, DateUtils.string2Date("20231117"), "order_info");
        ShardingProperty userInfoShardingConfig = new ShardingProperty(7, DateUtils.string2Date("20231117"), "user_info");

        SHARDING_TABLE.put(orderInfoShardingConfig.getTableName(), orderInfoShardingConfig);
        SHARDING_TABLE.put(userInfoShardingConfig.getTableName(), userInfoShardingConfig);
    }
}

攔截器

import lombok.extern.slf4j.Slf4j;
import o2o.aspect.platform.function.template.service.TemplateMatchService;
import org.apache.commons.lang3.StringUtils;
import org.apache.ibatis.executor.statement.StatementHandler;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.plugin.*;
import org.apache.ibatis.reflection.DefaultReflectorFactory;
import org.apache.ibatis.reflection.MetaObject;
import org.apache.ibatis.reflection.ReflectorFactory;
import org.apache.ibatis.reflection.factory.DefaultObjectFactory;
import org.apache.ibatis.reflection.factory.ObjectFactory;
import org.apache.ibatis.reflection.wrapper.DefaultObjectWrapperFactory;
import org.apache.ibatis.reflection.wrapper.ObjectWrapperFactory;
import org.springframework.stereotype.Component;

import java.sql.Connection;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
import java.util.Date;
import java.util.Properties;

@Slf4j
@Component
@Intercepts({@Signature(type = StatementHandler.class, method = "prepare", args = {Connection.class, Integer.class})})
public class ShardingTableInterceptor implements Interceptor {
    private static final ObjectFactory DEFAULT_OBJECT_FACTORY = new DefaultObjectFactory();
    private static final ObjectWrapperFactory DEFAULT_OBJECT_WRAPPER_FACTORY = new DefaultObjectWrapperFactory();
    private static final ReflectorFactory DEFAULT_REFLECTOR_FACTORY = new DefaultReflectorFactory();
    private static final String MAPPED_STATEMENT = "delegate.mappedStatement";
    private static final String BOUND_SQL = "delegate.boundSql";
    private static final String ORIGIN_BOUND_SQL = "delegate.boundSql.sql";
    private static final DateTimeFormatter FORMATTER = DateTimeFormatter.ofPattern("yyyyMMdd");
    private static final String SHARDING_MAPPER = "com.jd.o2o.inviter.promote.mapper.ShardingMapper";

    private ConfigUtils configUtils = SpringContextHolder.getBean(ConfigUtils.class);

    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        boolean shardingSwitch = configUtils.getBool("sharding_switch", false);
        // 沒開啟分表 直接返回老數(shù)據(jù)
        if (!shardingSwitch) {
            return invocation.proceed();
        }

        StatementHandler statementHandler = (StatementHandler) invocation.getTarget();
        MetaObject metaStatementHandler = MetaObject.forObject(statementHandler, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY, DEFAULT_REFLECTOR_FACTORY);
        MappedStatement mappedStatement = (MappedStatement) metaStatementHandler.getValue(MAPPED_STATEMENT);
        BoundSql boundSql = (BoundSql) metaStatementHandler.getValue(BOUND_SQL);
        String originSql = (String) metaStatementHandler.getValue(ORIGIN_BOUND_SQL);
        if (StringUtils.isBlank(originSql)) {
            return invocation.proceed();
        }

        // 獲取表名
        String tableName = TemplateMatchService.matchTableName(boundSql.getSql().trim());
        ShardingProperty shardingProperty = ShardingPropertyConfig.SHARDING_TABLE.get(tableName);
        if (shardingProperty == null) {
            return invocation.proceed();
        }

        // 新表
        String shardingTable = getCurrentShardingTable(shardingProperty, new Date());
        String rebuildSql = boundSql.getSql().replace(shardingProperty.getTableName(), shardingTable);
        metaStatementHandler.setValue(ORIGIN_BOUND_SQL, rebuildSql);
        if (log.isDebugEnabled()) {
            log.info("rebuildSQL -> {}", rebuildSql);
        }

        return invocation.proceed();
    }

    @Override
    public Object plugin(Object target) {
        if (target instanceof StatementHandler) {
            return Plugin.wrap(target, this);
        }
        return target;
    }

    @Override
    public void setProperties(Properties properties) {}

    public static String getCurrentShardingTable(ShardingProperty shardingProperty, Date createTime) {
        String tableName = shardingProperty.getTableName();
        Integer days = shardingProperty.getDays();
        Date beginDate = shardingProperty.getBeginDate();

        Date date;
        if (createTime == null) {
            date = new Date();
        } else {
            date = createTime;
        }
        if (date.before(beginDate)) {
            return null;
        }
        LocalDateTime targetDate = SimpleDateFormatUtils.convertDateToLocalDateTime(date);
        LocalDateTime startDate = SimpleDateFormatUtils.convertDateToLocalDateTime(beginDate);
        LocalDateTime intervalStartDate = DateIntervalChecker.getIntervalStartDate(targetDate, startDate, days);
        LocalDateTime intervalEndDate = intervalStartDate.plusDays(days - 1);
        return tableName + "_" + intervalStartDate.format(FORMATTER) + "_" + intervalEndDate.format(FORMATTER);
    }
}

臨界點(diǎn)數(shù)據(jù)不連續(xù)問題

分表方案有1個(gè)難點(diǎn)需要解決:周期臨界點(diǎn)數(shù)據(jù)不連續(xù)。舉例:假設(shè)要對(duì)operate_log(操作日志表)大表進(jìn)行橫向分表,每周一張表,分表明細(xì)可看下面表格。

第一周(operate_log_20240107_20240108) 第二周(operate_log_20240108_20240114) 第三周(operate_log_20240115_20240121)
1月1號(hào) ~ 1月7號(hào)的數(shù)據(jù) 1月8號(hào) ~ 1月14號(hào)的數(shù)據(jù) 1月15號(hào) ~ 1月21號(hào)的數(shù)據(jù)

1月8號(hào)就是分表臨界點(diǎn),8號(hào)需要切換到第二周的表,但8號(hào)0點(diǎn)剛切換的時(shí)候,表內(nèi)沒有任何數(shù)據(jù),這時(shí)如果業(yè)務(wù)需要查近一周的操作日志是查不到的,這樣就會(huì)引發(fā)線上問題。

我決定采用數(shù)據(jù)冗余的方式來解決這個(gè)痛點(diǎn)。每個(gè)周期表都冗余一份上個(gè)周期的數(shù)據(jù),用雙倍數(shù)據(jù)量實(shí)現(xiàn)數(shù)據(jù)滑動(dòng)的效果,效果見下面表格。

第一周(operate_log_20240107_20240108) 第二周(operate_log_20240108_20240114) 第三周(operate_log_20240115_20240121)
12月25號(hào) ~ 12月31號(hào)的數(shù)據(jù) 1月1號(hào) ~ 1月7號(hào)的數(shù)據(jù) 1月8號(hào) ~ 1月14號(hào)的數(shù)據(jù)
1月1號(hào) ~ 1月7號(hào)的數(shù)據(jù) 1月8號(hào) ~ 1月14號(hào)的數(shù)據(jù) 1月15號(hào) ~ 1月21號(hào)的數(shù)據(jù)

注:表格內(nèi)第一行數(shù)據(jù)就是冗余的上個(gè)周期表的數(shù)據(jù)。

思路有了,接下來就要考慮怎么實(shí)現(xiàn)雙寫(數(shù)據(jù)冗余到下個(gè)周期表) ,有2種方案:

1.在SQL執(zhí)行完成返回結(jié)果前添加邏輯(可以用AspectJ 或 mybatis攔截器),如果SQL內(nèi)的表名是當(dāng)前周期表,就把表名替換為下個(gè)周期表,然后再次執(zhí)行SQL。此方案對(duì)業(yè)務(wù)影響大,相當(dāng)于串行執(zhí)行了2次SQL,有性能損耗。

2.監(jiān)聽增量binlog,京東內(nèi)部有現(xiàn)成的數(shù)據(jù)訂閱中間件DRC,讀者也可以使用cannal等開源中間件來代替DRC,原理大同小異,此方案對(duì)業(yè)務(wù)無影響。

方案對(duì)比后,選擇了對(duì)業(yè)務(wù)性能損耗小的方案二。

監(jiān)聽binlog數(shù)據(jù)雙寫注意點(diǎn)

1.提前上線監(jiān)聽程序,提前把老表數(shù)據(jù)同步到新的周期表。分表前只監(jiān)聽老表binlog就可以,分表前只需要把老表數(shù)據(jù)同步到新表。

2.切換到新表的臨界點(diǎn),為了避免丟失積壓的老表binlog,需要同時(shí)處理新表binlog和老表binlog,這樣會(huì)出現(xiàn)死循環(huán)同步的問題,因?yàn)槔媳硇枰叫卤恚卤碛中枰p寫老表。為了打破循環(huán),需要先把雙寫老表消費(fèi)堵上讓消息暫時(shí)積壓,切換新表成功后,再打開雙寫消費(fèi)。

監(jiān)聽binlog數(shù)據(jù)雙寫代碼

注:下面代碼不能直接用,只提供基本思路

/**
 * 監(jiān)聽binlog ,分表雙寫,解決數(shù)據(jù)臨界問題
*/
@Slf4j
@Component
public class BinLogConsumer implements MessageListener {
    
    private MessageDeserialize deserialize = new JMQMessageDeserialize();

    private static final String TABLE_PLACEHOLDER = "%TABLE%";

    @Value("${mq.doubleWriteTopic.topic}")
    private String doubleWriteTopic;

    @Autowired
    private JmqProducerService jmqProducerService;


    @Override
    public void onMessage(List messages) throws Exception {
        if (messages == null || messages.isEmpty()) {
            return;
        }
        List entryMessages = deserialize.deserialize(messages);
        for (EntryMessage entryMessage : entryMessages) {
            try {
                syncData(entryMessage);
            } catch (Exception e) {
                log.error("sharding sync data error", e);
                throw e;
            }
        }
    }

    private void syncData(EntryMessage entryMessage) throws JMQException {
        // 根據(jù)binlog內(nèi)的表名,獲取需要同步的表
        // 3種情況:
        // 1、老表:需要同步當(dāng)前周期表,和下個(gè)周期表。
        // 2、當(dāng)前周期表:需要同步下個(gè)周期表,和老表。
        // 3、下個(gè)周期表:不需要同步。
        List syncTables = getSyncTables(entryMessage.tableName, entryMessage.createTime);
        
        if (CollectionUtils.isEmpty(syncTables)) {
            log.info("table {} is not need sync", tableName);
            return;
        }

        if (entryMessage.getHeader().getEventType() == WaveEntry.EventType.INSERT) {
            String insertTableSqlTemplate = parseSqlForInsert(rowData);
            for (String syncTable : syncTables) {
                String insertSql = insertTableSqlTemplate.replaceAll(TABLE_PLACEHOLDER, syncTable);
                // 雙寫老表發(fā)Q,為了避免出現(xiàn)同步死循環(huán)問題
                if (ShardingPropertyConfig.SHARDING_TABLE.containsKey(syncTable)) {
                    Long primaryKey = getPrimaryKey(rowData.getAfterColumnsList());
                    sendDoubleWriteMsg(insertSql, primaryKey);
                    continue;
                }
                mysqlConnection.executeSql(insertSql);
            }
            continue;
        }
    }

數(shù)據(jù)對(duì)比

為了保證新表和老表數(shù)據(jù)一致,需要編寫對(duì)比程序,在上線前進(jìn)行數(shù)據(jù)對(duì)比,保證binlog同步無問題。

具體實(shí)現(xiàn)代碼不做展示,思路:新表查詢一定量級(jí)數(shù)據(jù),老表查詢相同量級(jí)數(shù)據(jù),都轉(zhuǎn)換成JSON,equals對(duì)比。

審核編輯 黃宇

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • SQL
    SQL
    +關(guān)注

    關(guān)注

    1

    文章

    762

    瀏覽量

    44117
  • binlog
    +關(guān)注

    關(guān)注

    0

    文章

    7

    瀏覽量

    1250
  • mybatis
    +關(guān)注

    關(guān)注

    0

    文章

    60

    瀏覽量

    6709
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    介紹一種OpenAtom OpenHarmony系統(tǒng)適配方案

    本文在不改變?cè)邢到y(tǒng)基礎(chǔ)框架的基礎(chǔ)上, 介紹了一種OpenAtom OpenHarmony(以下簡(jiǎn)稱“OpenHarmony”)系統(tǒng)適配方案
    的頭像 發(fā)表于 03-05 09:24 ?1139次閱讀
    介紹<b class='flag-5'>一種</b>OpenAtom OpenHarmony<b class='flag-5'>輕</b><b class='flag-5'>量</b>系統(tǒng)適配<b class='flag-5'>方案</b>

    輸入毫伏

    輸入毫伏--見附件
    發(fā)表于 06-19 00:07

    三豐千分表通訊方案

    套能夠讓電腦每隔0.1秒讀取次千分表讀數(shù)的方案,有意者電聯(lián)***
    發(fā)表于 06-19 14:11

    分庫(kù)是什么?怎么實(shí)現(xiàn)?

    數(shù)據(jù)庫(kù)分庫(kù)、讀寫分離的原理實(shí)現(xiàn),使用場(chǎng)景
    發(fā)表于 10-25 17:24

    mybatis支持?jǐn)?shù)據(jù)庫(kù)兼容的方案

    個(gè)方案, 令mybatis支持?jǐn)?shù)據(jù)庫(kù)兼容
    發(fā)表于 04-09 17:44

    原理及實(shí)際應(yīng)用

    覆蓋率。  1.1功原理  功全稱功率分配器,是一種路輸入信號(hào)能量分成兩路或多路輸出
    發(fā)表于 12-03 15:00

    介紹一種串行接口方案

    折疊式手機(jī)面臨哪些問題?一種滿足手機(jī)高速圖像數(shù)據(jù)傳輸?shù)牟?b class='flag-5'>分串行接口方案
    發(fā)表于 06-01 06:51

    動(dòng)能攔截器六自由度仿真建模研究

    仿真建模技術(shù)是動(dòng)能攔截器制導(dǎo)律研究中的重要技術(shù),文中主要建立動(dòng)能攔截器的軌道運(yùn)動(dòng)動(dòng)力學(xué)以及姿態(tài)運(yùn)動(dòng)動(dòng)力學(xué)模型,并建立完整的制導(dǎo)控制系統(tǒng)數(shù)學(xué)模型。文末,以某型
    發(fā)表于 08-07 08:50 ?14次下載

    利用Mycat實(shí)現(xiàn)MySQL讀寫分離、分庫(kù)最佳實(shí)踐

    利用Mycat實(shí)現(xiàn)MySQL讀寫分離、分庫(kù)最佳實(shí)踐
    發(fā)表于 09-08 10:20 ?14次下載
    利用Mycat實(shí)現(xiàn)MySQL讀寫分離、分庫(kù)<b class='flag-5'>分</b><b class='flag-5'>表</b>最佳<b class='flag-5'>實(shí)踐</b>

    springmvc 自定義攔截器實(shí)現(xiàn)未登錄用戶的攔截

    springmvc自定義攔截器實(shí)現(xiàn)未登錄用戶的攔截
    發(fā)表于 11-25 14:44 ?2519次閱讀
    springmvc 自定義<b class='flag-5'>攔截器</b>實(shí)現(xiàn)未登錄用戶的<b class='flag-5'>攔截</b>

    分表和千分表的區(qū)別是什么

    分表:百分表是利用精密齒條齒輪機(jī)構(gòu)制成的式通用長(zhǎng)度測(cè)量工具。通常由測(cè)頭、桿、防震彈簧、齒條、齒輪、游絲、圓表盤及指針等組成。
    發(fā)表于 01-02 18:16 ?8.9w次閱讀
    百<b class='flag-5'>分表</b>和千<b class='flag-5'>分表</b>的區(qū)別是什么

    什么是分庫(kù)?為什么分庫(kù)?什么情況下會(huì)用分庫(kù)呢?

    分庫(kù)是由分庫(kù)和這兩個(gè)獨(dú)立概念組成的,只不過通常分庫(kù)與的操作會(huì)同時(shí)進(jìn)行,以至于我們習(xí)慣
    的頭像 發(fā)表于 11-30 09:37 ?7537次閱讀

    springboot過濾器和攔截器哪個(gè)先執(zhí)行

    的概念、用途、執(zhí)行順序以及實(shí)際使用中的注意事項(xiàng)。 、過濾器和攔截器的概念和用途 過濾器(Filter) 過濾器是Java Web應(yīng)用程序中的一種組件,它用于攔截客戶端請(qǐng)求并對(duì)其進(jìn)行預(yù)
    的頭像 發(fā)表于 12-03 15:00 ?2528次閱讀

    使用go語(yǔ)言實(shí)現(xiàn)個(gè)grpc攔截器

    在開發(fā)grpc服務(wù)時(shí),我們經(jīng)常會(huì)遇到些通用的需求,比如:日志、鏈路追蹤、鑒權(quán)等。這些需求可以通過grpc攔截器來實(shí)現(xiàn)。本文使用go語(yǔ)言來實(shí)現(xiàn)個(gè) grpc元模式(Unary)
    的頭像 發(fā)表于 12-18 10:13 ?656次閱讀
    使用go語(yǔ)言實(shí)現(xiàn)<b class='flag-5'>一</b>個(gè)grpc<b class='flag-5'>攔截器</b>

    分庫(kù)后復(fù)雜查詢的應(yīng)對(duì)之道:基于DTS實(shí)時(shí)性ES寬構(gòu)建技術(shù)實(shí)踐

    1 問題域 業(yè)務(wù)發(fā)展的初期,我們的數(shù)據(jù)庫(kù)架構(gòu)往往是單庫(kù)單,外加讀寫分離來快速的支撐業(yè)務(wù),隨著用戶和訂單的增加,數(shù)據(jù)庫(kù)的計(jì)算和存儲(chǔ)往往會(huì)成為我們系統(tǒng)的瓶頸,業(yè)界的實(shí)踐多數(shù)采用分而治
    的頭像 發(fā)表于 06-25 18:30 ?856次閱讀
    分庫(kù)<b class='flag-5'>分</b><b class='flag-5'>表</b>后復(fù)雜查詢的應(yīng)對(duì)之道:基于DTS實(shí)時(shí)性ES寬<b class='flag-5'>表</b>構(gòu)建技術(shù)<b class='flag-5'>實(shí)踐</b>
    主站蜘蛛池模板: 国产在线观看码高清视频| 帅哥操美女| 最近中文字幕无吗免费高清| 国产免国产免费| 神马电影我不卡国语版| xxx性欧美在线| 免费麻豆国产黄网站在线观看| 亚洲区 bt下载| 国产综合视频在线观看一区| 四虎永久在线精品国产| se01短视频在线观看| 内射人妻无码色AV麻豆去百度搜| 伊人久久大香线蕉综合色啪| 精品无码久久久久久久动漫| 亚洲精品无码不卡在线播放he| 国产无线乱码一区二三区| 午夜国产视频| 国产乱色伦影片在线观看| 午夜免费体验30分| 国产乱码二卡3卡四卡| 午夜国产福利| 国产午夜伦鲁鲁| 亚洲精品午睡沙发系列| 国色天香视频在线社区| 亚洲欧美一区二区三区九九九| 含羞草影院AE在线观看| 亚洲国产系列一区二区三区| 国产在线高清视频无码不卡| 亚洲免费网站在线观看| 九九热在线观看视频| 与子敌伦刺激对白亂輪亂性 | 久久 这里只精品 免费| 亚洲日韩乱码人人爽人人澡人| 精品AV国产一区二区三区| 一个吃奶两个添下面H| 久久中文电影| 99久久久精品| 日韩成人黄色| 国产精品三级在线观看| 亚洲精品无码午夜福利在线观看| 久久本道久久综合伊人|