亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > Java > 正文

spring boot + quartz集群搭建的完整步驟

2024-07-14 08:40:39
字體:
來源:轉載
供稿:網友

quartz集群能力:

  • quartz集群分為水平集群和垂直集群,水平集群即將定時任務節點部署在不同的服務器,水平集群最大的問題就是時鐘同步問題,
    quartz集群強烈要求時鐘同步,若時鐘不能同步,則會導致集群中各個節點狀態紊亂,造成不可預知的后果,請自行搜索服務器時鐘同步,
    若能保證時鐘同步,水平集群能保證服務的可靠性,其中一個節點掛掉或其中一個服務器宕機,其他節點依然正常服務;垂直集群則是集群各節點部署在同一臺服務器,
    時鐘同步自然不是問題,但存在單點故障問題,服務器宕機會嚴重影響服務的可用性。因此,要結合實際情況來考慮集群方案
  • 由于集群中強烈要求時鐘同步,因此不管是垂直集群還是水平集群,本地開發決不能連接線上環境(本地也是集群模式),這樣的話勢必會破壞集群,但本地若是非集群模式,
    則可以依情況來連接線上環境。
  • quartz集群和redis這樣的集群實現方式不一樣,redis集群需要節點之間通信,各節點需要知道其他節點的狀況,而quartz集群的實現
    方式在于11張表,集群節點相互之間不通信,而是通過定時任務持久化加鎖的方式來實現集群。
  • 破壞集群后果一般是死鎖或者狀態紊亂每個節點都不可用或其中某些節點能用部分或全部的定時任務

以上是個人的一些見解,下面開始本文的正文,本文主要介紹了關于spring boot + quartz集群搭建的相關內容,分享出來供大家參考學習,下面話不多說了,來一起看看詳細的介紹吧。

步驟如下:

spring boot bean配置:

@Configurationpublic class QuartzConfig { @Value("${quartz.scheduler.instanceName}") private String quartzInstanceName;   @Value("${org.quartz.dataSource.myDS.driver}") private String myDSDriver;   @Value("${org.quartz.dataSource.myDS.URL}") private String myDSURL;   @Value("${org.quartz.dataSource.myDS.user}") private String myDSUser;   @Value("${org.quartz.dataSource.myDS.password}") private String myDSPassword;   @Value("${org.quartz.dataSource.myDS.maxConnections}") private String myDSMaxConnections;     /**  * 設置屬性  * @return  * @throws IOException  */ private Properties quartzProperties() throws IOException {  Properties prop = new Properties();  prop.put("quartz.scheduler.instanceName", quartzInstanceName);  prop.put("org.quartz.scheduler.instanceId", "AUTO");  prop.put("org.quartz.scheduler.skipUpdateCheck", "true");  prop.put("org.quartz.scheduler.jmx.export", "true");     prop.put("org.quartz.jobStore.class", "org.quartz.impl.jdbcjobstore.JobStoreTX");  prop.put("org.quartz.jobStore.driverDelegateClass", "org.quartz.impl.jdbcjobstore.StdJDBCDelegate");  prop.put("org.quartz.jobStore.dataSource", "quartzDataSource");  prop.put("org.quartz.jobStore.tablePrefix", "QRTZ_");  prop.put("org.quartz.jobStore.isClustered", "true");     prop.put("org.quartz.jobStore.clusterCheckinInterval", "20000");  prop.put("org.quartz.jobStore.dataSource", "myDS");  prop.put("org.quartz.jobStore.maxMisfiresToHandleAtATime", "1");  prop.put("org.quartz.jobStore.misfireThreshold", "120000");  prop.put("org.quartz.jobStore.txIsolationLevelSerializable", "true");  prop.put("org.quartz.jobStore.selectWithLockSQL", "SELECT * FROM {0}LOCKS WHERE LOCK_NAME = ? FOR UPDATE");     prop.put("org.quartz.threadPool.class", "org.quartz.simpl.SimpleThreadPool");  prop.put("org.quartz.threadPool.threadCount", "10");  prop.put("org.quartz.threadPool.threadPriority", "5");  prop.put("org.quartz.threadPool.threadsInheritContextClassLoaderOfInitializingThread", "true");     prop.put("org.quartz.dataSource.myDS.driver", myDSDriver);  prop.put("org.quartz.dataSource.myDS.URL", myDSURL);  prop.put("org.quartz.dataSource.myDS.user", myDSUser);  prop.put("org.quartz.dataSource.myDS.password", myDSPassword);  System.out.println("myDSMaxConnections:" + myDSMaxConnections);  prop.put("org.quartz.dataSource.myDS.maxConnections", myDSMaxConnections);     prop.put("org.quartz.plugin.triggHistory.class", "org.quartz.plugins.history.LoggingJobHistoryPlugin");  prop.put("org.quartz.plugin.shutdownhook.class", "org.quartz.plugins.management.ShutdownHookPlugin");  prop.put("org.quartz.plugin.shutdownhook.cleanShutdown", "true");  return prop; }   @Bean  public SchedulerFactoryBean schedulerFactoryBean(@Qualifier("dialogJobTrigger") Trigger cronJobTrigger) throws IOException {   SchedulerFactoryBean factory = new SchedulerFactoryBean();   // this allows to update triggers in DB when updating settings in config file:   //用于quartz集群,QuartzScheduler 啟動時更新己存在的Job,這樣就不用每次修改targetObject后刪除qrtz_job_details表對應記錄了   factory.setOverwriteExistingJobs(true);   //用于quartz集群,加載quartz數據源   //factory.setDataSource(dataSource);   //QuartzScheduler 延時啟動,應用啟動完10秒后 QuartzScheduler 再啟動   factory.setStartupDelay(10);  //用于quartz集群,加載quartz數據源配置   factory.setQuartzProperties(quartzProperties());  factory.setAutoStartup(true);  factory.setApplicationContextSchedulerContextKey("applicationContext");  //注冊觸發器   factory.setTriggers(cronJobTrigger);<br data-filtered="filtered">     //直接使用配置文件//  factory.setConfigLocation(new FileSystemResource(this.getClass().getResource("/quartz.properties").getPath()));  return factory;  }     /**  * 加載job  * @return  */ @Bean  public JobDetailFactoryBean updateDialogStatusJobDetail() {   return createJobDetail(InvokingJobDetailDetailFactory.class, "updateDialogStatusGroup", "dialogJob");  }   /**  * 加載觸發器  * @param jobDetail  * @return  */ @Bean(name = "dialogJobTrigger")  public CronTriggerFactoryBean dialogStatusJobTrigger(@Qualifier("updateDialogStatusJobDetail") JobDetail jobDetail) {   return dialogStatusTrigger(jobDetail, "0 0 0/1 * * ?");  }   /**  * 創建job工廠  * @param jobClass  * @param groupName  * @param targetObject  * @return  */ private static JobDetailFactoryBean createJobDetail(Class<?> jobClass, String groupName, String targetObject) {   JobDetailFactoryBean factoryBean = new JobDetailFactoryBean();   factoryBean.setJobClass(jobClass);   factoryBean.setDurability(true);   factoryBean.setRequestsRecovery(true);  factoryBean.setGroup(groupName);  Map<String, String> map = new HashMap<>();  map.put("targetObject", targetObject);  map.put("targetMethod", "execute");  factoryBean.setJobDataAsMap(map);  return factoryBean;  }   /**  * 創建觸發器工廠  * @param jobDetail  * @param cronExpression  * @return  */ private static CronTriggerFactoryBean dialogStatusTrigger(JobDetail jobDetail, String cronExpression) {   CronTriggerFactoryBean factoryBean = new CronTriggerFactoryBean();   factoryBean.setJobDetail(jobDetail);  factoryBean.setCronExpression (cronExpression);  return factoryBean;  } }

InvokingJobDetailDetailFactory對象:

public class InvokingJobDetailDetailFactory extends QuartzJobBean{ // 計劃任務所在類  private String targetObject; // 具體需要執行的計劃任務  private String targetMethod; private ApplicationContext ctx; @Override protected void executeInternal(JobExecutionContext context) throws JobExecutionException {  try {   Object otargetObject = ctx.getBean(targetObject);   Method m = null;    try {     m = otargetObject.getClass().getMethod(targetMethod);     m.invoke(otargetObject);    } catch (SecurityException e) {     e.printStackTrace();    } catch (NoSuchMethodException e) {     e.printStackTrace();    }   } catch (Exception e) {    throw new JobExecutionException(e);   } }  public void setApplicationContext(ApplicationContext applicationContext) {   this.ctx = applicationContext;  }   public void setTargetObject(String targetObject) {   this.targetObject = targetObject;  }   public void setTargetMethod(String targetMethod) {   this.targetMethod = targetMethod;  }}

備注:set方法不能少,setApplicationContext中的applicationContext與factory.setApplicationContextSchedulerContextKey("applicationContext")填入的值有關,其原理由InvokingJobDetailDetailFactory父類中的BeanWrapper實現。

sql腳本:--

<em id="__mceDel">-- 表的結構 `qrtz_blob_triggers`-- CREATE TABLE IF NOT EXISTS `qrtz_blob_triggers` ( `SCHED_NAME` varchar(120) NOT NULL, `TRIGGER_NAME` varchar(120) NOT NULL, `TRIGGER_GROUP` varchar(120) NOT NULL, `BLOB_DATA` blob) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- -------------------------------------------------------- ---- 表的結構 `qrtz_calendars`-- CREATE TABLE IF NOT EXISTS `qrtz_calendars` ( `SCHED_NAME` varchar(120) NOT NULL, `CALENDAR_NAME` varchar(120) NOT NULL, `CALENDAR` blob NOT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- -------------------------------------------------------- ---- 表的結構 `qrtz_cron_triggers`-- CREATE TABLE IF NOT EXISTS `qrtz_cron_triggers` ( `SCHED_NAME` varchar(120) NOT NULL, `TRIGGER_NAME` varchar(120) NOT NULL, `TRIGGER_GROUP` varchar(120) NOT NULL, `CRON_EXPRESSION` varchar(120) NOT NULL, `TIME_ZONE_ID` varchar(80) DEFAULT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- -------------------------------------------------------- ---- 表的結構 `qrtz_fired_triggers`-- CREATE TABLE IF NOT EXISTS `qrtz_fired_triggers` ( `SCHED_NAME` varchar(120) NOT NULL, `ENTRY_ID` varchar(95) NOT NULL, `TRIGGER_NAME` varchar(120) NOT NULL, `TRIGGER_GROUP` varchar(120) NOT NULL, `INSTANCE_NAME` varchar(120) NOT NULL, `FIRED_TIME` bigint(13) NOT NULL, `SCHED_TIME` bigint(13) NOT NULL, `PRIORITY` int(11) NOT NULL, `STATE` varchar(16) NOT NULL, `JOB_NAME` varchar(120) DEFAULT NULL, `JOB_GROUP` varchar(120) DEFAULT NULL, `IS_NONCONCURRENT` varchar(1) DEFAULT NULL, `REQUESTS_RECOVERY` varchar(1) DEFAULT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- -------------------------------------------------------- ---- 表的結構 `qrtz_job_details`-- CREATE TABLE IF NOT EXISTS `qrtz_job_details` ( `SCHED_NAME` varchar(120) NOT NULL, `JOB_NAME` varchar(120) NOT NULL, `JOB_GROUP` varchar(120) NOT NULL, `DESCRIPTION` varchar(250) DEFAULT NULL, `JOB_CLASS_NAME` varchar(250) NOT NULL, `IS_DURABLE` varchar(1) NOT NULL, `IS_NONCONCURRENT` varchar(1) NOT NULL, `IS_UPDATE_DATA` varchar(1) NOT NULL, `REQUESTS_RECOVERY` varchar(1) NOT NULL, `JOB_DATA` blob) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- -------------------------------------------------------- ---- 表的結構 `qrtz_locks`-- CREATE TABLE IF NOT EXISTS `qrtz_locks` ( `SCHED_NAME` varchar(120) NOT NULL, `LOCK_NAME` varchar(40) NOT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- -------------------------------------------------------- ---- 表的結構 `qrtz_paused_trigger_grps`-- CREATE TABLE IF NOT EXISTS `qrtz_paused_trigger_grps` ( `SCHED_NAME` varchar(120) NOT NULL, `TRIGGER_GROUP` varchar(120) NOT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- -------------------------------------------------------- ---- 表的結構 `qrtz_scheduler_state`-- CREATE TABLE IF NOT EXISTS `qrtz_scheduler_state` ( `SCHED_NAME` varchar(120) NOT NULL, `INSTANCE_NAME` varchar(120) NOT NULL, `LAST_CHECKIN_TIME` bigint(13) NOT NULL, `CHECKIN_INTERVAL` bigint(13) NOT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- -------------------------------------------------------- ---- 表的結構 `qrtz_simple_triggers`-- CREATE TABLE IF NOT EXISTS `qrtz_simple_triggers` ( `SCHED_NAME` varchar(120) NOT NULL, `TRIGGER_NAME` varchar(120) NOT NULL, `TRIGGER_GROUP` varchar(120) NOT NULL, `REPEAT_COUNT` bigint(7) NOT NULL, `REPEAT_INTERVAL` bigint(12) NOT NULL, `TIMES_TRIGGERED` bigint(10) NOT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- -------------------------------------------------------- ---- 表的結構 `qrtz_simprop_triggers`-- CREATE TABLE IF NOT EXISTS `qrtz_simprop_triggers` ( `SCHED_NAME` varchar(120) NOT NULL, `TRIGGER_NAME` varchar(120) NOT NULL, `TRIGGER_GROUP` varchar(120) NOT NULL, `STR_PROP_1` varchar(512) DEFAULT NULL, `STR_PROP_2` varchar(512) DEFAULT NULL, `STR_PROP_3` varchar(512) DEFAULT NULL, `INT_PROP_1` int(11) DEFAULT NULL, `INT_PROP_2` int(11) DEFAULT NULL, `LONG_PROP_1` bigint(20) DEFAULT NULL, `LONG_PROP_2` bigint(20) DEFAULT NULL, `DEC_PROP_1` decimal(13,4) DEFAULT NULL, `DEC_PROP_2` decimal(13,4) DEFAULT NULL, `BOOL_PROP_1` varchar(1) DEFAULT NULL, `BOOL_PROP_2` varchar(1) DEFAULT NULL) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- -------------------------------------------------------- ---- 表的結構 `qrtz_triggers`-- CREATE TABLE IF NOT EXISTS `qrtz_triggers` ( `SCHED_NAME` varchar(120) NOT NULL, `TRIGGER_NAME` varchar(120) NOT NULL, `TRIGGER_GROUP` varchar(120) NOT NULL, `JOB_NAME` varchar(120) NOT NULL, `JOB_GROUP` varchar(120) NOT NULL, `DESCRIPTION` varchar(250) DEFAULT NULL, `NEXT_FIRE_TIME` bigint(13) DEFAULT NULL, `PREV_FIRE_TIME` bigint(13) DEFAULT NULL, `PRIORITY` int(11) DEFAULT NULL, `TRIGGER_STATE` varchar(16) NOT NULL, `TRIGGER_TYPE` varchar(8) NOT NULL, `START_TIME` bigint(13) NOT NULL, `END_TIME` bigint(13) DEFAULT NULL, `CALENDAR_NAME` varchar(200) DEFAULT NULL, `MISFIRE_INSTR` smallint(2) DEFAULT NULL, `JOB_DATA` blob) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; ---- Indexes for dumped tables-- ---- Indexes for table `qrtz_blob_triggers`--ALTER TABLE `qrtz_blob_triggers` ADD PRIMARY KEY (`SCHED_NAME`,`TRIGGER_NAME`,`TRIGGER_GROUP`), ADD KEY `SCHED_NAME` (`SCHED_NAME`,`TRIGGER_NAME`,`TRIGGER_GROUP`); ---- Indexes for table `qrtz_calendars`--ALTER TABLE `qrtz_calendars` ADD PRIMARY KEY (`SCHED_NAME`,`CALENDAR_NAME`); ---- Indexes for table `qrtz_cron_triggers`--ALTER TABLE `qrtz_cron_triggers` ADD PRIMARY KEY (`SCHED_NAME`,`TRIGGER_NAME`,`TRIGGER_GROUP`); ---- Indexes for table `qrtz_fired_triggers`--ALTER TABLE `qrtz_fired_triggers` ADD PRIMARY KEY (`SCHED_NAME`,`ENTRY_ID`), ADD KEY `IDX_QRTZ_FT_TRIG_INST_NAME` (`SCHED_NAME`,`INSTANCE_NAME`), ADD KEY `IDX_QRTZ_FT_INST_JOB_REQ_RCVRY` (`SCHED_NAME`,`INSTANCE_NAME`,`REQUESTS_RECOVERY`), ADD KEY `IDX_QRTZ_FT_J_G` (`SCHED_NAME`,`JOB_NAME`,`JOB_GROUP`), ADD KEY `IDX_QRTZ_FT_JG` (`SCHED_NAME`,`JOB_GROUP`), ADD KEY `IDX_QRTZ_FT_T_G` (`SCHED_NAME`,`TRIGGER_NAME`,`TRIGGER_GROUP`), ADD KEY `IDX_QRTZ_FT_TG` (`SCHED_NAME`,`TRIGGER_GROUP`); ---- Indexes for table `qrtz_job_details`--ALTER TABLE `qrtz_job_details` ADD PRIMARY KEY (`SCHED_NAME`,`JOB_NAME`,`JOB_GROUP`), ADD KEY `IDX_QRTZ_J_REQ_RECOVERY` (`SCHED_NAME`,`REQUESTS_RECOVERY`), ADD KEY `IDX_QRTZ_J_GRP` (`SCHED_NAME`,`JOB_GROUP`); ---- Indexes for table `qrtz_locks`--ALTER TABLE `qrtz_locks` ADD PRIMARY KEY (`SCHED_NAME`,`LOCK_NAME`); ---- Indexes for table `qrtz_paused_trigger_grps`--ALTER TABLE `qrtz_paused_trigger_grps` ADD PRIMARY KEY (`SCHED_NAME`,`TRIGGER_GROUP`); ---- Indexes for table `qrtz_scheduler_state`--ALTER TABLE `qrtz_scheduler_state` ADD PRIMARY KEY (`SCHED_NAME`,`INSTANCE_NAME`); ---- Indexes for table `qrtz_simple_triggers`--ALTER TABLE `qrtz_simple_triggers` ADD PRIMARY KEY (`SCHED_NAME`,`TRIGGER_NAME`,`TRIGGER_GROUP`); ---- Indexes for table `qrtz_simprop_triggers`--ALTER TABLE `qrtz_simprop_triggers` ADD PRIMARY KEY (`SCHED_NAME`,`TRIGGER_NAME`,`TRIGGER_GROUP`); ---- Indexes for table `qrtz_triggers`--ALTER TABLE `qrtz_triggers` ADD PRIMARY KEY (`SCHED_NAME`,`TRIGGER_NAME`,`TRIGGER_GROUP`), ADD KEY `IDX_QRTZ_T_J` (`SCHED_NAME`,`JOB_NAME`,`JOB_GROUP`), ADD KEY `IDX_QRTZ_T_JG` (`SCHED_NAME`,`JOB_GROUP`), ADD KEY `IDX_QRTZ_T_C` (`SCHED_NAME`,`CALENDAR_NAME`(191)), ADD KEY `IDX_QRTZ_T_G` (`SCHED_NAME`,`TRIGGER_GROUP`), ADD KEY `IDX_QRTZ_T_STATE` (`SCHED_NAME`,`TRIGGER_STATE`), ADD KEY `IDX_QRTZ_T_N_STATE` (`SCHED_NAME`,`TRIGGER_NAME`,`TRIGGER_GROUP`,`TRIGGER_STATE`), ADD KEY `IDX_QRTZ_T_N_G_STATE` (`SCHED_NAME`,`TRIGGER_GROUP`,`TRIGGER_STATE`), ADD KEY `IDX_QRTZ_T_NEXT_FIRE_TIME` (`SCHED_NAME`,`NEXT_FIRE_TIME`), ADD KEY `IDX_QRTZ_T_NFT_ST` (`SCHED_NAME`,`TRIGGER_STATE`,`NEXT_FIRE_TIME`), ADD KEY `IDX_QRTZ_T_NFT_MISFIRE` (`SCHED_NAME`,`MISFIRE_INSTR`,`NEXT_FIRE_TIME`), ADD KEY `IDX_QRTZ_T_NFT_ST_MISFIRE` (`SCHED_NAME`,`MISFIRE_INSTR`,`NEXT_FIRE_TIME`,`TRIGGER_STATE`), ADD KEY `IDX_QRTZ_T_NFT_ST_MISFIRE_GRP` (`SCHED_NAME`,`MISFIRE_INSTR`,`NEXT_FIRE_TIME`,`TRIGGER_GROUP`,`TRIGGER_STATE`); ---- 限制導出的表-- ---- 限制表 `qrtz_blob_triggers`--ALTER TABLE `qrtz_blob_triggers` ADD CONSTRAINT `qrtz_blob_triggers_ibfk_1` FOREIGN KEY (`SCHED_NAME`, `TRIGGER_NAME`, `TRIGGER_GROUP`) REFERENCES `qrtz_triggers` (`SCHED_NAME`, `TRIGGER_NAME`, `TRIGGER_GROUP`); ---- 限制表 `qrtz_cron_triggers`--ALTER TABLE `qrtz_cron_triggers` ADD CONSTRAINT `qrtz_cron_triggers_ibfk_1` FOREIGN KEY (`SCHED_NAME`, `TRIGGER_NAME`, `TRIGGER_GROUP`) REFERENCES `qrtz_triggers` (`SCHED_NAME`, `TRIGGER_NAME`, `TRIGGER_GROUP`); ---- 限制表 `qrtz_simple_triggers`--ALTER TABLE `qrtz_simple_triggers` ADD CONSTRAINT `qrtz_simple_triggers_ibfk_1` FOREIGN KEY (`SCHED_NAME`, `TRIGGER_NAME`, `TRIGGER_GROUP`) REFERENCES `qrtz_triggers` (`SCHED_NAME`, `TRIGGER_NAME`, `TRIGGER_GROUP`); ---- 限制表 `qrtz_simprop_triggers`--ALTER TABLE `qrtz_simprop_triggers` ADD CONSTRAINT `qrtz_simprop_triggers_ibfk_1` FOREIGN KEY (`SCHED_NAME`, `TRIGGER_NAME`, `TRIGGER_GROUP`) REFERENCES `qrtz_triggers` (`SCHED_NAME`, `TRIGGER_NAME`, `TRIGGER_GROUP`); ---- 限制表 `qrtz_triggers`--ALTER TABLE `qrtz_triggers` ADD CONSTRAINT `qrtz_triggers_ibfk_1` FOREIGN KEY (`SCHED_NAME`, `JOB_NAME`, `JOB_GROUP`) REFERENCES `qrtz_job_details` (`SCHED_NAME`, `JOB_NAME`, `JOB_GROUP`);<br><br><br></em>

quartz集群實現原理,利用數據庫記錄job行為,并通過鎖機制,使job在同一次中僅運行一次。

JobBean示例

//需要交由spring管理@Service("dialogJob")public class DialogJob { @Autowired private QuestionService questionService;// 方法名在quartz定義 public void execute() throws Exception{  //具體執行業務     questionService.XXXXX(); } }

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VeVb武林網的支持。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美成人在线免费| 九九热精品视频在线播放| 久热在线中文字幕色999舞| 亚洲第一区中文字幕| 成人在线视频网站| 亚洲免费人成在线视频观看| 国产精品永久免费| 欧美性高潮床叫视频| 亚洲精品小视频在线观看| 国产精品高潮呻吟久久av黑人| 亚洲成人激情视频| 亚洲在线视频观看| 日韩在线视频免费观看高清中文| 国产精品视频最多的网站| 欧美福利视频网站| 亚洲女在线观看| 一本久久综合亚洲鲁鲁| 性日韩欧美在线视频| 亚洲欧美一区二区三区久久| 欧美激情视频在线免费观看 欧美视频免费一| 日韩欧美在线国产| 亚洲精品国产综合区久久久久久久| 国内偷自视频区视频综合| 国产精品中文字幕在线观看| 中文字幕视频在线免费欧美日韩综合在线看| 国产日产欧美a一级在线| 97在线视频免费看| 精品一区二区三区三区| 欧美又大粗又爽又黄大片视频| 最近2019中文字幕一页二页| 在线视频国产日韩| 欧美性猛交xxxx| 久热精品视频在线| 色妞一区二区三区| 91在线视频导航| 欧美大奶子在线| 久久国产精品久久国产精品| 亚洲999一在线观看www| 久久躁狠狠躁夜夜爽| 综合网日日天干夜夜久久| 久久久久久综合网天天| 欧美激情videoshd| 国产日韩在线视频| 日本国产欧美一区二区三区| 91精品国产色综合久久不卡98| 国产欧美一区二区三区四区| 欧美激情视频三区| 日本不卡免费高清视频| 国产午夜一区二区| 91精品国产99| 国产精品久久久久久久久久小说| 性欧美长视频免费观看不卡| 精品国产999| 欧美影院成年免费版| 欧美日韩国产一区中文午夜| 亚洲精品福利视频| 久久久久亚洲精品| 日本在线观看天堂男亚洲| 一区二区三区国产视频| 久久影院资源网| 亚洲黄色片网站| 一区二区欧美在线| 韩曰欧美视频免费观看| 一区二区三区视频观看| 夜夜嗨av一区二区三区免费区| 97涩涩爰在线观看亚洲| 国产婷婷成人久久av免费高清| 精品久久久久久久久久| 日韩精品免费在线视频| 国产精品白丝jk喷水视频一区| 国产精品一区专区欧美日韩| 精品国内产的精品视频在线观看| 69av视频在线播放| 亚洲欧美日韩中文在线制服| 亚洲自拍小视频免费观看| 久久精品国产99国产精品澳门| 国产精品中文字幕久久久| 国产欧美一区二区三区视频| 久久偷看各类女兵18女厕嘘嘘| 国产精品久久久久久搜索| 国产在线观看精品| 国产成人精品免费久久久久| 日本老师69xxx| 久久久久久免费精品| 欧美视频在线看| 色偷偷88888欧美精品久久久| 久久久久久999| 亚洲精品永久免费| 亚洲男人天堂网站| 精品一区二区三区四区在线| 日韩美女视频在线观看| 岛国av一区二区三区| 国产精品欧美一区二区| 精品欧美国产一区二区三区| 91影视免费在线观看| 中文亚洲视频在线| 日韩在线观看av| 亚洲免费电影一区| 久精品免费视频| 亚洲第一av网| 日韩精品高清在线| 久久中文字幕在线视频| 日韩国产高清污视频在线观看| 久久久久久九九九| 国产中文字幕亚洲| 国产成人精品最新| 久久国产精品久久久| 亚洲在线视频福利| 琪琪亚洲精品午夜在线| 黄色成人在线免费| 国产+成+人+亚洲欧洲| 91精品国产综合久久男男| 91日韩在线播放| 国产91在线播放九色快色| 亚洲最大av网| 91精品久久久久久久久不口人| 亚洲一区二区三区777| 欧美日韩激情视频8区| 国产精品中文字幕久久久| 夜夜躁日日躁狠狠久久88av| 精品一区二区三区三区| 亚洲一区二区三区777| 亚洲女人初尝黑人巨大| 亚洲天堂网站在线观看视频| 亚洲欧美日韩爽爽影院| 国产精品精品视频一区二区三区| 国内外成人免费激情在线视频| 91视频九色网站| 亚洲国产精品成人va在线观看| 久久99精品久久久久久噜噜| 色偷偷偷亚洲综合网另类| 国产精品高清在线观看| 91精品在线观看视频| 亚洲国产精品女人久久久| 日韩在线观看免费全| 青草热久免费精品视频| 久久69精品久久久久久国产越南| 国产美女精品视频| 国产偷亚洲偷欧美偷精品| 国产91在线高潮白浆在线观看| 国产婷婷色综合av蜜臀av| 福利二区91精品bt7086| 国产午夜精品免费一区二区三区| 中文字幕欧美精品日韩中文字幕| 国产精品一区二区三区毛片淫片| 亚洲人午夜色婷婷| 亚洲欧美一区二区精品久久久| 一道本无吗dⅴd在线播放一区| 日韩av男人的天堂| 国色天香2019中文字幕在线观看| 成人免费激情视频| 欧美xxxx14xxxxx性爽| 人人澡人人澡人人看欧美| 91精品久久久久久久久久久久久久| 久久夜色精品国产亚洲aⅴ| 国产精品视频一| 97国产精品免费视频| 欧美激情网站在线观看| 两个人的视频www国产精品| 亚洲欧美激情四射在线日| 亚洲一区二区久久久久久| 日韩精品在线免费观看| 日本高清不卡的在线| 欧美精品第一页在线播放|