久久精品国产精品国产精品污,男人扒开添女人下部免费视频,一级国产69式性姿势免费视频,夜鲁夜鲁很鲁在线视频 视频,欧美丰满少妇一区二区三区,国产偷国产偷亚洲高清人乐享,中文 在线 日韩 亚洲 欧美,熟妇人妻无乱码中文字幕真矢织江,一区二区三区人妻制服国产

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > 数据库 >内容正文

数据库

Redis高级客户端Lettuce详解

發布時間:2025/10/17 数据库 6 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Redis高级客户端Lettuce详解 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

前提

Lettuce是一個Redis的Java驅動包,初識她的時候是使用RedisTemplate的時候遇到點問題Debug到底層的一些源碼,發現spring-data-redis的驅動包在某個版本之后替換為Lettuce。Lettuce翻譯為生菜,沒錯,就是吃的那種生菜,所以它的Logo長這樣:

既然能被Spring生態所認可,Lettuce想必有過人之處,于是筆者花時間閱讀她的官方文檔,整理測試示例,寫下這篇文章。編寫本文時所使用的版本為Lettuce 5.1.8.RELEASE,SpringBoot 2.1.8.RELEASE,JDK [8,11]。超長警告:這篇文章斷斷續續花了兩周完成,超過4萬字.....

Lettuce簡介

Lettuce是一個高性能基于Java編寫的Redis驅動框架,底層集成了Project Reactor提供天然的反應式編程,通信框架集成了Netty使用了非阻塞IO,5.x版本之后融合了JDK1.8的異步編程特性,在保證高性能的同時提供了十分豐富易用的API,5.1版本的新特性如下:

  • 支持Redis的新增命令ZPOPMIN, ZPOPMAX, BZPOPMIN, BZPOPMAX。
  • 支持通過Brave模塊跟蹤Redis命令執行。
  • 支持Redis Streams。
  • 支持異步的主從連接。
  • 支持異步連接池。
  • 新增命令最多執行一次模式(禁止自動重連)。
  • 全局命令超時設置(對異步和反應式命令也有效)。
  • ......等等

注意一點:Redis的版本至少需要2.6,當然越高越好,API的兼容性比較強大。

只需要引入單個依賴就可以開始愉快地使用Lettuce:

  • Maven
<dependency><groupId>io.lettuce</groupId><artifactId>lettuce-core</artifactId><version>5.1.8.RELEASE</version> </dependency>
  • Gradle
dependencies {compile 'io.lettuce:lettuce-core:5.1.8.RELEASE' }

連接Redis

單機、哨兵、集群模式下連接Redis需要一個統一的標準去表示連接的細節信息,在Lettuce中這個統一的標準是RedisURI。可以通過三種方式構造一個RedisURI實例:

  • 定制的字符串URI語法:
RedisURI uri = RedisURI.create("redis://localhost/");
  • 使用建造器(RedisURI.Builder):
RedisURI uri = RedisURI.builder().withHost("localhost").withPort(6379).build();
  • 直接通過構造函數實例化:
RedisURI uri = new RedisURI("localhost", 6379, 60, TimeUnit.SECONDS);

定制的連接URI語法

  • 單機(前綴為redis://)
格式:redis://[password@]host[:port][/databaseNumber][?[timeout=timeout[d|h|m|s|ms|us|ns]] 完整:redis://mypassword@127.0.0.1:6379/0?timeout=10s 簡單:redis://localhost
  • 單機并且使用SSL(前綴為rediss://) <== 注意后面多了個s
格式:rediss://[password@]host[:port][/databaseNumber][?[timeout=timeout[d|h|m|s|ms|us|ns]] 完整:rediss://mypassword@127.0.0.1:6379/0?timeout=10s 簡單:rediss://localhost
  • 單機Unix Domain Sockets模式(前綴為redis-socket://)
格式:redis-socket://path[?[timeout=timeout[d|h|m|s|ms|us|ns]][&_database=database_]] 完整:redis-socket:///tmp/redis?timeout=10s&_database=0
  • 哨兵(前綴為redis-sentinel://)
格式:redis-sentinel://[password@]host[:port][,host2[:port2]][/databaseNumber][?[timeout=timeout[d|h|m|s|ms|us|ns]]#sentinelMasterId 完整:redis-sentinel://mypassword@127.0.0.1:6379,127.0.0.1:6380/0?timeout=10s#mymaster

超時時間單位:

  • d 天
  • h 小時
  • m 分鐘
  • s 秒鐘
  • ms 毫秒
  • us 微秒
  • ns 納秒

個人建議使用RedisURI提供的建造器,畢竟定制的URI雖然簡潔,但是比較容易出現人為錯誤。鑒于筆者沒有SSL和Unix Domain Socket的使用場景,下面不對這兩種連接方式進行列舉。

基本使用

Lettuce使用的時候依賴于四個主要組件:

  • RedisURI:連接信息。
  • RedisClient:Redis客戶端,特殊地,集群連接有一個定制的RedisClusterClient。
  • Connection:Redis連接,主要是StatefulConnection或者StatefulRedisConnection的子類,連接的類型主要由連接的具體方式(單機、哨兵、集群、訂閱發布等等)選定,比較重要。
  • RedisCommands:Redis命令API接口,基本上覆蓋了Redis發行版本的所有命令,提供了同步(sync)、異步(async)、反應式(reative)的調用方式,對于使用者而言,會經常跟RedisCommands系列接口打交道。

一個基本使用例子如下:

@Test public void testSetGet() throws Exception {RedisURI redisUri = RedisURI.builder() // <1> 創建單機連接的連接信息.withHost("localhost").withPort(6379).withTimeout(Duration.of(10, ChronoUnit.SECONDS)).build();RedisClient redisClient = RedisClient.create(redisUri); // <2> 創建客戶端StatefulRedisConnection<String, String> connection = redisClient.connect(); // <3> 創建線程安全的連接RedisCommands<String, String> redisCommands = connection.sync(); // <4> 創建同步命令SetArgs setArgs = SetArgs.Builder.nx().ex(5);String result = redisCommands.set("name", "throwable", setArgs);Assertions.assertThat(result).isEqualToIgnoringCase("OK");result = redisCommands.get("name");Assertions.assertThat(result).isEqualTo("throwable");// ... 其他操作connection.close(); // <5> 關閉連接redisClient.shutdown(); // <6> 關閉客戶端 }

注意:

  • <5>:關閉連接一般在應用程序停止之前操作,一個應用程序中的一個Redis驅動實例不需要太多的連接(一般情況下只需要一個連接實例就可以,如果有多個連接的需要可以考慮使用連接池,其實Redis目前處理命令的模塊是單線程,在客戶端多個連接多線程調用理論上沒有效果)。
  • <6>:關閉客戶端一般應用程序停止之前操作,如果條件允許的話,基于后開先閉原則,客戶端關閉應該在連接關閉之后操作。

API

Lettuce主要提供三種API:

  • 同步(sync):RedisCommands。
  • 異步(async):RedisAsyncCommands。
  • 反應式(reactive):RedisReactiveCommands。

先準備好一個單機Redis連接備用:

private static StatefulRedisConnection<String, String> CONNECTION; private static RedisClient CLIENT;@BeforeClass public static void beforeClass() {RedisURI redisUri = RedisURI.builder().withHost("localhost").withPort(6379).withTimeout(Duration.of(10, ChronoUnit.SECONDS)).build();CLIENT = RedisClient.create(redisUri);CONNECTION = CLIENT.connect(); }@AfterClass public static void afterClass() throws Exception {CONNECTION.close();CLIENT.shutdown(); }

Redis命令API的具體實現可以直接從StatefulRedisConnection實例獲取,見其接口定義:

public interface StatefulRedisConnection<K, V> extends StatefulConnection<K, V> {boolean isMulti();RedisCommands<K, V> sync();RedisAsyncCommands<K, V> async();RedisReactiveCommands<K, V> reactive(); }

值得注意的是,在不指定編碼解碼器RedisCodec的前提下,RedisClient創建的StatefulRedisConnection實例一般是泛型實例StatefulRedisConnection<String,String>,也就是所有命令API的KEY和VALUE都是String類型,這種使用方式能滿足大部分的使用場景。當然,必要的時候可以定制編碼解碼器RedisCodec<K,V>。

同步API

先構建RedisCommands實例:

private static RedisCommands<String, String> COMMAND;@BeforeClass public static void beforeClass() {COMMAND = CONNECTION.sync(); }

基本使用:

@Test public void testSyncPing() throws Exception {String pong = COMMAND.ping();Assertions.assertThat(pong).isEqualToIgnoringCase("PONG"); }@Test public void testSyncSetAndGet() throws Exception {SetArgs setArgs = SetArgs.Builder.nx().ex(5);COMMAND.set("name", "throwable", setArgs);String value = COMMAND.get("name");log.info("Get value: {}", value); }// Get value: throwable

同步API在所有命令調用之后會立即返回結果。如果熟悉Jedis的話,RedisCommands的用法其實和它相差不大。

異步API

先構建RedisAsyncCommands實例:

private static RedisAsyncCommands<String, String> ASYNC_COMMAND;@BeforeClass public static void beforeClass() {ASYNC_COMMAND = CONNECTION.async(); }

基本使用:

@Test public void testAsyncPing() throws Exception {RedisFuture<String> redisFuture = ASYNC_COMMAND.ping();log.info("Ping result:{}", redisFuture.get()); } // Ping result:PONG

RedisAsyncCommands所有方法執行返回結果都是RedisFuture實例,而RedisFuture接口的定義如下:

public interface RedisFuture<V> extends CompletionStage<V>, Future<V> {String getError();boolean await(long timeout, TimeUnit unit) throws InterruptedException; }

也就是,RedisFuture可以無縫使用Future或者JDK1.8中引入的CompletableFuture提供的方法。舉個例子:

@Test public void testAsyncSetAndGet1() throws Exception {SetArgs setArgs = SetArgs.Builder.nx().ex(5);RedisFuture<String> future = ASYNC_COMMAND.set("name", "throwable", setArgs);// CompletableFuture#thenAccept()future.thenAccept(value -> log.info("Set命令返回:{}", value));// Future#get()future.get(); } // Set命令返回:OK@Test public void testAsyncSetAndGet2() throws Exception {SetArgs setArgs = SetArgs.Builder.nx().ex(5);CompletableFuture<Void> result =(CompletableFuture<Void>) ASYNC_COMMAND.set("name", "throwable", setArgs).thenAcceptBoth(ASYNC_COMMAND.get("name"),(s, g) -> {log.info("Set命令返回:{}", s);log.info("Get命令返回:{}", g);});result.get(); } // Set命令返回:OK // Get命令返回:throwable

如果能熟練使用CompletableFuture和函數式編程技巧,可以組合多個RedisFuture完成一些列復雜的操作。

反應式API

Lettuce引入的反應式編程框架是Project Reactor,如果沒有反應式編程經驗可以先自行了解一下Project Reactor。

構建RedisReactiveCommands實例:

private static RedisReactiveCommands<String, String> REACTIVE_COMMAND;@BeforeClass public static void beforeClass() {REACTIVE_COMMAND = CONNECTION.reactive(); }

根據Project Reactor,RedisReactiveCommands的方法如果返回的結果只包含0或1個元素,那么返回值類型是Mono,如果返回的結果包含0到N(N大于0)個元素,那么返回值是Flux。舉個例子:

@Test public void testReactivePing() throws Exception {Mono<String> ping = REACTIVE_COMMAND.ping();ping.subscribe(v -> log.info("Ping result:{}", v));Thread.sleep(1000); } // Ping result:PONG@Test public void testReactiveSetAndGet() throws Exception {SetArgs setArgs = SetArgs.Builder.nx().ex(5);REACTIVE_COMMAND.set("name", "throwable", setArgs).block();REACTIVE_COMMAND.get("name").subscribe(value -> log.info("Get命令返回:{}", value));Thread.sleep(1000); } // Get命令返回:throwable@Test public void testReactiveSet() throws Exception {REACTIVE_COMMAND.sadd("food", "bread", "meat", "fish").block();Flux<String> flux = REACTIVE_COMMAND.smembers("food");flux.subscribe(log::info);REACTIVE_COMMAND.srem("food", "bread", "meat", "fish").block();Thread.sleep(1000); } // meat // bread // fish

舉個更加復雜的例子,包含了事務、函數轉換等:

@Test public void testReactiveFunctional() throws Exception {REACTIVE_COMMAND.multi().doOnSuccess(r -> {REACTIVE_COMMAND.set("counter", "1").doOnNext(log::info).subscribe();REACTIVE_COMMAND.incr("counter").doOnNext(c -> log.info(String.valueOf(c))).subscribe();}).flatMap(s -> REACTIVE_COMMAND.exec()).doOnNext(transactionResult -> log.info("Discarded:{}", transactionResult.wasDiscarded())).subscribe();Thread.sleep(1000); } // OK // 2 // Discarded:false

這個方法開啟一個事務,先把counter設置為1,再將counter自增1。

發布和訂閱

非集群模式下的發布訂閱依賴于定制的連接StatefulRedisPubSubConnection,集群模式下的發布訂閱依賴于定制的連接StatefulRedisClusterPubSubConnection,兩者分別來源于RedisClient#connectPubSub()系列方法和RedisClusterClient#connectPubSub():

  • 非集群模式:
// 可能是單機、普通主從、哨兵等非集群模式的客戶端 RedisClient client = ... StatefulRedisPubSubConnection<String, String> connection = client.connectPubSub(); connection.addListener(new RedisPubSubListener<String, String>() { ... });// 同步命令 RedisPubSubCommands<String, String> sync = connection.sync(); sync.subscribe("channel");// 異步命令 RedisPubSubAsyncCommands<String, String> async = connection.async(); RedisFuture<Void> future = async.subscribe("channel");// 反應式命令 RedisPubSubReactiveCommands<String, String> reactive = connection.reactive(); reactive.subscribe("channel").subscribe();reactive.observeChannels().doOnNext(patternMessage -> {...}).subscribe()
  • 集群模式:
// 使用方式其實和非集群模式基本一致 RedisClusterClient clusterClient = ... StatefulRedisClusterPubSubConnection<String, String> connection = clusterClient.connectPubSub(); connection.addListener(new RedisPubSubListener<String, String>() { ... }); RedisPubSubCommands<String, String> sync = connection.sync(); sync.subscribe("channel"); // ...

這里用單機同步命令的模式舉一個Redis鍵空間通知(Redis Keyspace Notifications)的例子:

@Test public void testSyncKeyspaceNotification() throws Exception {RedisURI redisUri = RedisURI.builder().withHost("localhost").withPort(6379)// 注意這里只能是0號庫.withDatabase(0).withTimeout(Duration.of(10, ChronoUnit.SECONDS)).build();RedisClient redisClient = RedisClient.create(redisUri);StatefulRedisConnection<String, String> redisConnection = redisClient.connect();RedisCommands<String, String> redisCommands = redisConnection.sync();// 只接收鍵過期的事件redisCommands.configSet("notify-keyspace-events", "Ex");StatefulRedisPubSubConnection<String, String> connection = redisClient.connectPubSub();connection.addListener(new RedisPubSubAdapter<>() {@Overridepublic void psubscribed(String pattern, long count) {log.info("pattern:{},count:{}", pattern, count);}@Overridepublic void message(String pattern, String channel, String message) {log.info("pattern:{},channel:{},message:{}", pattern, channel, message);}});RedisPubSubCommands<String, String> commands = connection.sync();commands.psubscribe("__keyevent@0__:expired");redisCommands.setex("name", 2, "throwable");Thread.sleep(10000);redisConnection.close();connection.close();redisClient.shutdown(); } // pattern:__keyevent@0__:expired,count:1 // pattern:__keyevent@0__:expired,channel:__keyevent@0__:expired,message:name

實際上,在實現RedisPubSubListener的時候可以單獨抽離,盡量不要設計成匿名內部類的形式。

事務和批量命令執行

事務相關的命令就是WATCH、UNWATCH、EXEC、MULTI和DISCARD,在RedisCommands系列接口中有對應的方法。舉個例子:

// 同步模式 @Test public void testSyncMulti() throws Exception {COMMAND.multi();COMMAND.setex("name-1", 2, "throwable");COMMAND.setex("name-2", 2, "doge");TransactionResult result = COMMAND.exec();int index = 0;for (Object r : result) {log.info("Result-{}:{}", index, r);index++;} } // Result-0:OK // Result-1:OK

Redis的Pipeline也就是管道機制可以理解為把多個命令打包在一次請求發送到Redis服務端,然后Redis服務端把所有的響應結果打包好一次性返回,從而節省不必要的網絡資源(最主要是減少網絡請求次數)。Redis對于Pipeline機制如何實現并沒有明確的規定,也沒有提供特殊的命令支持Pipeline機制。Jedis中底層采用BIO(阻塞IO)通訊,所以它的做法是客戶端緩存將要發送的命令,最后需要觸發然后同步發送一個巨大的命令列表包,再接收和解析一個巨大的響應列表包。Pipeline在Lettuce中對使用者是透明的,由于底層的通訊框架是Netty,所以網絡通訊層面的優化Lettuce不需要過多干預,換言之可以這樣理解:Netty幫Lettuce從底層實現了Redis的Pipeline機制。但是,Lettuce的異步API也提供了手動Flush的方法:

@Test public void testAsyncManualFlush() {// 取消自動flushASYNC_COMMAND.setAutoFlushCommands(false);List<RedisFuture<?>> redisFutures = Lists.newArrayList();int count = 5000;for (int i = 0; i < count; i++) {String key = "key-" + (i + 1);String value = "value-" + (i + 1);redisFutures.add(ASYNC_COMMAND.set(key, value));redisFutures.add(ASYNC_COMMAND.expire(key, 2));}long start = System.currentTimeMillis();ASYNC_COMMAND.flushCommands();boolean result = LettuceFutures.awaitAll(10, TimeUnit.SECONDS, redisFutures.toArray(new RedisFuture[0]));Assertions.assertThat(result).isTrue();log.info("Lettuce cost:{} ms", System.currentTimeMillis() - start); } // Lettuce cost:1302 ms

上面只是從文檔看到的一些理論術語,但是現實是骨感的,對比了下Jedis的Pipeline提供的方法,發現了Jedis的Pipeline執行耗時比較低:

@Test public void testJedisPipeline() throws Exception {Jedis jedis = new Jedis();Pipeline pipeline = jedis.pipelined();int count = 5000;for (int i = 0; i < count; i++) {String key = "key-" + (i + 1);String value = "value-" + (i + 1);pipeline.set(key, value);pipeline.expire(key, 2);}long start = System.currentTimeMillis();pipeline.syncAndReturnAll();log.info("Jedis cost:{} ms", System.currentTimeMillis() - start); } // Jedis cost:9 ms

個人猜測Lettuce可能底層并非合并所有命令一次發送(甚至可能是單條發送),具體可能需要抓包才能定位。依此來看,如果真的有大量執行Redis命令的場景,不妨可以使用Jedis的Pipeline。

注意:由上面的測試推斷RedisTemplate的executePipelined()方法是假的Pipeline執行方法,使用RedisTemplate的時候請務必注意這一點。

Lua腳本執行

Lettuce中執行Redis的Lua命令的同步接口如下:

public interface RedisScriptingCommands<K, V> {<T> T eval(String var1, ScriptOutputType var2, K... var3);<T> T eval(String var1, ScriptOutputType var2, K[] var3, V... var4);<T> T evalsha(String var1, ScriptOutputType var2, K... var3);<T> T evalsha(String var1, ScriptOutputType var2, K[] var3, V... var4);List<Boolean> scriptExists(String... var1);String scriptFlush();String scriptKill();String scriptLoad(V var1);String digest(V var1); }

異步和反應式的接口方法定義差不多,不同的地方就是返回值類型,一般我們常用的是eval()、evalsha()和scriptLoad()方法。舉個簡單的例子:

private static RedisCommands<String, String> COMMANDS; private static String RAW_LUA = "local key = KEYS[1]\n" +"local value = ARGV[1]\n" +"local timeout = ARGV[2]\n" +"redis.call('SETEX', key, tonumber(timeout), value)\n" +"local result = redis.call('GET', key)\n" +"return result;"; private static AtomicReference<String> LUA_SHA = new AtomicReference<>();@Test public void testLua() throws Exception {LUA_SHA.compareAndSet(null, COMMANDS.scriptLoad(RAW_LUA));String[] keys = new String[]{"name"};String[] args = new String[]{"throwable", "5000"};String result = COMMANDS.evalsha(LUA_SHA.get(), ScriptOutputType.VALUE, keys, args);log.info("Get value:{}", result); } // Get value:throwable

高可用和分片

為了Redis的高可用,一般會采用普通主從(Master/Replica,這里筆者稱為普通主從模式,也就是僅僅做了主從復制,故障需要手動切換)、哨兵和集群。普通主從模式可以獨立運行,也可以配合哨兵運行,只是哨兵提供自動故障轉移和主節點提升功能。普通主從和哨兵都可以使用MasterSlave,通過入參包括RedisClient、編碼解碼器以及一個或者多個RedisURI獲取對應的Connection實例。

這里注意一點,MasterSlave中提供的方法如果只要求傳入一個RedisURI實例,那么Lettuce會進行拓撲發現機制,自動獲取Redis主從節點信息;如果要求傳入一個RedisURI集合,那么對于普通主從模式來說所有節點信息是靜態的,不會進行發現和更新。

拓撲發現的規則如下:

  • 對于普通主從(Master/Replica)模式,不需要感知RedisURI指向從節點還是主節點,只會進行一次性的拓撲查找所有節點信息,此后節點信息會保存在靜態緩存中,不會更新。
  • 對于哨兵模式,會訂閱所有哨兵實例并偵聽訂閱/發布消息以觸發拓撲刷新機制,更新緩存的節點信息,也就是哨兵天然就是動態發現節點信息,不支持靜態配置。

拓撲發現機制的提供API為TopologyProvider,需要了解其原理的可以參考具體的實現。

對于集群(Cluster)模式,Lettuce提供了一套獨立的API。

另外,如果Lettuce連接面向的是非單個Redis節點,連接實例提供了數據讀取節點偏好(ReadFrom)設置,可選值有:

  • MASTER:只從Master節點中讀取。
  • MASTER_PREFERRED:優先從Master節點中讀取。
  • SLAVE_PREFERRED:優先從Slavor節點中讀取。
  • SLAVE:只從Slavor節點中讀取。
  • NEAREST:使用最近一次連接的Redis實例讀取。

普通主從模式

假設現在有三個Redis服務形成樹狀主從關系如下:

  • 節點一:localhost:6379,角色為Master。
  • 節點二:localhost:6380,角色為Slavor,節點一的從節點。
  • 節點三:localhost:6381,角色為Slavor,節點二的從節點。

首次動態節點發現主從模式的節點信息需要如下構建連接:

@Test public void testDynamicReplica() throws Exception {// 這里只需要配置一個節點的連接信息,不一定需要是主節點的信息,從節點也可以RedisURI uri = RedisURI.builder().withHost("localhost").withPort(6379).build();RedisClient redisClient = RedisClient.create(uri);StatefulRedisMasterSlaveConnection<String, String> connection = MasterSlave.connect(redisClient, new Utf8StringCodec(), uri);// 只從從節點讀取數據connection.setReadFrom(ReadFrom.SLAVE);// 執行其他Redis命令connection.close();redisClient.shutdown(); }

如果需要指定靜態的Redis主從節點連接屬性,那么可以這樣構建連接:

@Test public void testStaticReplica() throws Exception {List<RedisURI> uris = new ArrayList<>();RedisURI uri1 = RedisURI.builder().withHost("localhost").withPort(6379).build();RedisURI uri2 = RedisURI.builder().withHost("localhost").withPort(6380).build();RedisURI uri3 = RedisURI.builder().withHost("localhost").withPort(6381).build();uris.add(uri1);uris.add(uri2);uris.add(uri3);RedisClient redisClient = RedisClient.create();StatefulRedisMasterSlaveConnection<String, String> connection = MasterSlave.connect(redisClient,new Utf8StringCodec(), uris);// 只從主節點讀取數據connection.setReadFrom(ReadFrom.MASTER);// 執行其他Redis命令connection.close();redisClient.shutdown(); }

哨兵模式

由于Lettuce自身提供了哨兵的拓撲發現機制,所以只需要隨便配置一個哨兵節點的RedisURI實例即可:

@Test public void testDynamicSentinel() throws Exception {RedisURI redisUri = RedisURI.builder().withPassword("你的密碼").withSentinel("localhost", 26379).withSentinelMasterId("哨兵Master的ID").build();RedisClient redisClient = RedisClient.create();StatefulRedisMasterSlaveConnection<String, String> connection = MasterSlave.connect(redisClient, new Utf8StringCodec(), redisUri);// 只允許從從節點讀取數據connection.setReadFrom(ReadFrom.SLAVE);RedisCommands<String, String> command = connection.sync();SetArgs setArgs = SetArgs.Builder.nx().ex(5);command.set("name", "throwable", setArgs);String value = command.get("name");log.info("Get value:{}", value); } // Get value:throwable

集群模式

鑒于筆者對Redis集群模式并不熟悉,Cluster模式下的API使用本身就有比較多的限制,所以這里只簡單介紹一下怎么用。先說幾個特性:

下面的API提供跨槽位(Slot)調用的功能

  • RedisAdvancedClusterCommands。
  • RedisAdvancedClusterAsyncCommands。
  • RedisAdvancedClusterReactiveCommands。

靜態節點選擇功能:

  • masters:選擇所有主節點執行命令。
  • slaves:選擇所有從節點執行命令,其實就是只讀模式。
  • all nodes:命令可以在所有節點執行。

集群拓撲視圖動態更新功能:

  • 手動更新,主動調用RedisClusterClient#reloadPartitions()。
  • 后臺定時更新。
  • 自適應更新,基于連接斷開和MOVED/ASK命令重定向自動更新。

Redis集群搭建詳細過程可以參考官方文檔,假設已經搭建好集群如下(192.168.56.200是筆者的虛擬機Host):

  • 192.168.56.200:7001 => 主節點,槽位0-5460。
  • 192.168.56.200:7002 => 主節點,槽位5461-10922。
  • 192.168.56.200:7003 => 主節點,槽位10923-16383。
  • 192.168.56.200:7004 => 7001的從節點。
  • 192.168.56.200:7005 => 7002的從節點。
  • 192.168.56.200:7006 => 7003的從節點。

簡單的集群連接和使用方式如下:

@Test public void testSyncCluster(){RedisURI uri = RedisURI.builder().withHost("192.168.56.200").build();RedisClusterClient redisClusterClient = RedisClusterClient.create(uri);StatefulRedisClusterConnection<String, String> connection = redisClusterClient.connect();RedisAdvancedClusterCommands<String, String> commands = connection.sync();commands.setex("name",10, "throwable");String value = commands.get("name");log.info("Get value:{}", value); } // Get value:throwable

節點選擇:

@Test public void testSyncNodeSelection() {RedisURI uri = RedisURI.builder().withHost("192.168.56.200").withPort(7001).build();RedisClusterClient redisClusterClient = RedisClusterClient.create(uri);StatefulRedisClusterConnection<String, String> connection = redisClusterClient.connect();RedisAdvancedClusterCommands<String, String> commands = connection.sync(); // commands.all(); // 所有節點 // commands.masters(); // 主節點// 從節點只讀NodeSelection<String, String> replicas = commands.slaves();NodeSelectionCommands<String, String> nodeSelectionCommands = replicas.commands();// 這里只是演示,一般應該禁用keys *命令Executions<List<String>> keys = nodeSelectionCommands.keys("*");keys.forEach(key -> log.info("key: {}", key));connection.close();redisClusterClient.shutdown(); }

定時更新集群拓撲視圖(每隔十分鐘更新一次,這個時間自行考量,不能太頻繁):

@Test public void testPeriodicClusterTopology() throws Exception {RedisURI uri = RedisURI.builder().withHost("192.168.56.200").withPort(7001).build();RedisClusterClient redisClusterClient = RedisClusterClient.create(uri);ClusterTopologyRefreshOptions options = ClusterTopologyRefreshOptions.builder().enablePeriodicRefresh(Duration.of(10, ChronoUnit.MINUTES)).build();redisClusterClient.setOptions(ClusterClientOptions.builder().topologyRefreshOptions(options).build());StatefulRedisClusterConnection<String, String> connection = redisClusterClient.connect();RedisAdvancedClusterCommands<String, String> commands = connection.sync();commands.setex("name", 10, "throwable");String value = commands.get("name");log.info("Get value:{}", value);Thread.sleep(Integer.MAX_VALUE);connection.close();redisClusterClient.shutdown(); }

自適應更新集群拓撲視圖:

@Test public void testAdaptiveClusterTopology() throws Exception {RedisURI uri = RedisURI.builder().withHost("192.168.56.200").withPort(7001).build();RedisClusterClient redisClusterClient = RedisClusterClient.create(uri);ClusterTopologyRefreshOptions options = ClusterTopologyRefreshOptions.builder().enableAdaptiveRefreshTrigger(ClusterTopologyRefreshOptions.RefreshTrigger.MOVED_REDIRECT,ClusterTopologyRefreshOptions.RefreshTrigger.PERSISTENT_RECONNECTS).adaptiveRefreshTriggersTimeout(Duration.of(30, ChronoUnit.SECONDS)).build();redisClusterClient.setOptions(ClusterClientOptions.builder().topologyRefreshOptions(options).build());StatefulRedisClusterConnection<String, String> connection = redisClusterClient.connect();RedisAdvancedClusterCommands<String, String> commands = connection.sync();commands.setex("name", 10, "throwable");String value = commands.get("name");log.info("Get value:{}", value);Thread.sleep(Integer.MAX_VALUE);connection.close();redisClusterClient.shutdown(); }

動態命令和自定義命令

自定義命令是Redis命令有限集,不過可以更細粒度指定KEY、ARGV、命令類型、編碼解碼器和返回值類型,依賴于dispatch()方法:

// 自定義實現PING方法 @Test public void testCustomPing() throws Exception {RedisURI redisUri = RedisURI.builder().withHost("localhost").withPort(6379).withTimeout(Duration.of(10, ChronoUnit.SECONDS)).build();RedisClient redisClient = RedisClient.create(redisUri);StatefulRedisConnection<String, String> connect = redisClient.connect();RedisCommands<String, String> sync = connect.sync();RedisCodec<String, String> codec = StringCodec.UTF8;String result = sync.dispatch(CommandType.PING, new StatusOutput<>(codec));log.info("PING:{}", result);connect.close();redisClient.shutdown(); } // PING:PONG// 自定義實現Set方法 @Test public void testCustomSet() throws Exception {RedisURI redisUri = RedisURI.builder().withHost("localhost").withPort(6379).withTimeout(Duration.of(10, ChronoUnit.SECONDS)).build();RedisClient redisClient = RedisClient.create(redisUri);StatefulRedisConnection<String, String> connect = redisClient.connect();RedisCommands<String, String> sync = connect.sync();RedisCodec<String, String> codec = StringCodec.UTF8;sync.dispatch(CommandType.SETEX, new StatusOutput<>(codec),new CommandArgs<>(codec).addKey("name").add(5).addValue("throwable"));String result = sync.get("name");log.info("Get value:{}", result);connect.close();redisClient.shutdown(); } // Get value:throwable

動態命令是基于Redis命令有限集,并且通過注解和動態代理完成一些復雜命令組合的實現。主要注解在io.lettuce.core.dynamic.annotation包路徑下。簡單舉個例子:

public interface CustomCommand extends Commands {// SET [key] [value]@Command("SET ?0 ?1")String setKey(String key, String value);// SET [key] [value]@Command("SET :key :value")String setKeyNamed(@Param("key") String key, @Param("value") String value);// MGET [key1] [key2]@Command("MGET ?0 ?1")List<String> mGet(String key1, String key2);/*** 方法名作為命令*/@CommandNaming(strategy = CommandNaming.Strategy.METHOD_NAME)String mSet(String key1, String value1, String key2, String value2); }@Test public void testCustomDynamicSet() throws Exception {RedisURI redisUri = RedisURI.builder().withHost("localhost").withPort(6379).withTimeout(Duration.of(10, ChronoUnit.SECONDS)).build();RedisClient redisClient = RedisClient.create(redisUri);StatefulRedisConnection<String, String> connect = redisClient.connect();RedisCommandFactory commandFactory = new RedisCommandFactory(connect);CustomCommand commands = commandFactory.getCommands(CustomCommand.class);commands.setKey("name", "throwable");commands.setKeyNamed("throwable", "doge");log.info("MGET ===> " + commands.mGet("name", "throwable"));commands.mSet("key1", "value1","key2", "value2");log.info("MGET ===> " + commands.mGet("key1", "key2"));connect.close();redisClient.shutdown(); } // MGET ===> [throwable, doge] // MGET ===> [value1, value2]

高階特性

Lettuce有很多高階使用特性,這里只列舉個人認為常用的兩點:

  • 配置客戶端資源。
  • 使用連接池。

更多其他特性可以自行參看官方文檔。

配置客戶端資源

客戶端資源的設置與Lettuce的性能、并發和事件處理相關。線程池或者線程組相關配置占據客戶端資源配置的大部分(EventLoopGroups和EventExecutorGroup),這些線程池或者線程組是連接程序的基礎組件。一般情況下,客戶端資源應該在多個Redis客戶端之間共享,并且在不再使用的時候需要自行關閉。筆者認為,客戶端資源是面向Netty的。注意除非特別熟悉或者花長時間去測試調整下面提到的參數,否則在沒有經驗的前提下憑直覺修改默認值,有可能會踩坑

客戶端資源接口是ClientResources,實現類是DefaultClientResources。

構建DefaultClientResources實例:

// 默認 ClientResources resources = DefaultClientResources.create();// 建造器 ClientResources resources = DefaultClientResources.builder().ioThreadPoolSize(4).computationThreadPoolSize(4).build()

使用:

ClientResources resources = DefaultClientResources.create(); // 非集群 RedisClient client = RedisClient.create(resources, uri); // 集群 RedisClusterClient clusterClient = RedisClusterClient.create(resources, uris); // ...... client.shutdown(); clusterClient.shutdown(); // 關閉資源 resources.shutdown();

客戶端資源基本配置:

屬性描述默認值
ioThreadPoolSizeI/O線程數Runtime.getRuntime().availableProcessors()
computationThreadPoolSize任務線程數Runtime.getRuntime().availableProcessors()

客戶端資源高級配置:

屬性描述默認值
eventLoopGroupProviderEventLoopGroup提供商-
eventExecutorGroupProviderEventExecutorGroup提供商-
eventBus事件總線DefaultEventBus
commandLatencyCollectorOptions命令延時收集器配置DefaultCommandLatencyCollectorOptions
commandLatencyCollector命令延時收集器DefaultCommandLatencyCollector
commandLatencyPublisherOptions命令延時發布器配置DefaultEventPublisherOptions
dnsResolverDNS處理器JDK或者Netty提供
reconnectDelay重連延時配置Delay.exponential()
nettyCustomizerNetty自定義配置器-
tracing軌跡記錄器-

非集群客戶端RedisClient的屬性配置:

Redis非集群客戶端RedisClient本身提供了配置屬性方法:

RedisClient client = RedisClient.create(uri); client.setOptions(ClientOptions.builder().autoReconnect(false).pingBeforeActivateConnection(true).build());

非集群客戶端的配置屬性列表:

屬性描述默認值
pingBeforeActivateConnection連接激活之前是否執行PING命令false
autoReconnect是否自動重連true
cancelCommandsOnReconnectFailure重連失敗是否拒絕命令執行false
suspendReconnectOnProtocolFailure底層協議失敗是否掛起重連操作false
requestQueueSize請求隊列容量2147483647(Integer#MAX_VALUE)
disconnectedBehavior失去連接時候的行為DEFAULT
sslOptionsSSL配置-
socketOptionsSocket配置10 seconds Connection-Timeout, no keep-alive, no TCP noDelay
timeoutOptions超時配置-
publishOnScheduler發布反應式信號數據的調度器使用I/O線程

集群客戶端屬性配置:

Redis集群客戶端RedisClusterClient本身提供了配置屬性方法:

RedisClusterClient client = RedisClusterClient.create(uri); ClusterTopologyRefreshOptions topologyRefreshOptions = ClusterTopologyRefreshOptions.builder().enablePeriodicRefresh(refreshPeriod(10, TimeUnit.MINUTES)).enableAllAdaptiveRefreshTriggers().build();client.setOptions(ClusterClientOptions.builder().topologyRefreshOptions(topologyRefreshOptions).build());

集群客戶端的配置屬性列表:

屬性描述默認值
enablePeriodicRefresh是否允許周期性更新集群拓撲視圖false
refreshPeriod更新集群拓撲視圖周期60秒
enableAdaptiveRefreshTrigger設置自適應更新集群拓撲視圖觸發器RefreshTrigger-
adaptiveRefreshTriggersTimeout自適應更新集群拓撲視圖觸發器超時設置30秒
refreshTriggersReconnectAttempts自適應更新集群拓撲視圖觸發重連次數5
dynamicRefreshSources是否允許動態刷新拓撲資源true
closeStaleConnections是否允許關閉陳舊的連接true
maxRedirects集群重定向次數上限5
validateClusterNodeMembership是否校驗集群節點的成員關系true

使用連接池

引入連接池依賴commons-pool2:

<dependency><groupId>org.apache.commons</groupId><artifactId>commons-pool2</artifactId><version>2.7.0</version> </dependency

基本使用如下:

@Test public void testUseConnectionPool() throws Exception {RedisURI redisUri = RedisURI.builder().withHost("localhost").withPort(6379).withTimeout(Duration.of(10, ChronoUnit.SECONDS)).build();RedisClient redisClient = RedisClient.create(redisUri);GenericObjectPoolConfig poolConfig = new GenericObjectPoolConfig();GenericObjectPool<StatefulRedisConnection<String, String>> pool= ConnectionPoolSupport.createGenericObjectPool(redisClient::connect, poolConfig);try (StatefulRedisConnection<String, String> connection = pool.borrowObject()) {RedisCommands<String, String> command = connection.sync();SetArgs setArgs = SetArgs.Builder.nx().ex(5);command.set("name", "throwable", setArgs);String n = command.get("name");log.info("Get value:{}", n);}pool.close();redisClient.shutdown(); }

其中,同步連接的池化支持需要用ConnectionPoolSupport,異步連接的池化支持需要用AsyncConnectionPoolSupport(Lettuce5.1之后才支持)。

幾個常見的漸進式刪除例子

漸進式刪除Hash中的域-屬性:

@Test public void testDelBigHashKey() throws Exception {// SCAN參數ScanArgs scanArgs = ScanArgs.Builder.limit(2);// TEMP游標ScanCursor cursor = ScanCursor.INITIAL;// 目標KEYString key = "BIG_HASH_KEY";prepareHashTestData(key);log.info("開始漸進式刪除Hash的元素...");int counter = 0;do {MapScanCursor<String, String> result = COMMAND.hscan(key, cursor, scanArgs);// 重置TEMP游標cursor = ScanCursor.of(result.getCursor());cursor.setFinished(result.isFinished());Collection<String> fields = result.getMap().values();if (!fields.isEmpty()) {COMMAND.hdel(key, fields.toArray(new String[0]));}counter++;} while (!(ScanCursor.FINISHED.getCursor().equals(cursor.getCursor()) && ScanCursor.FINISHED.isFinished() == cursor.isFinished()));log.info("漸進式刪除Hash的元素完畢,迭代次數:{} ...", counter); }private void prepareHashTestData(String key) throws Exception {COMMAND.hset(key, "1", "1");COMMAND.hset(key, "2", "2");COMMAND.hset(key, "3", "3");COMMAND.hset(key, "4", "4");COMMAND.hset(key, "5", "5"); }

漸進式刪除集合中的元素:

@Test public void testDelBigSetKey() throws Exception {String key = "BIG_SET_KEY";prepareSetTestData(key);// SCAN參數ScanArgs scanArgs = ScanArgs.Builder.limit(2);// TEMP游標ScanCursor cursor = ScanCursor.INITIAL;log.info("開始漸進式刪除Set的元素...");int counter = 0;do {ValueScanCursor<String> result = COMMAND.sscan(key, cursor, scanArgs);// 重置TEMP游標cursor = ScanCursor.of(result.getCursor());cursor.setFinished(result.isFinished());List<String> values = result.getValues();if (!values.isEmpty()) {COMMAND.srem(key, values.toArray(new String[0]));}counter++;} while (!(ScanCursor.FINISHED.getCursor().equals(cursor.getCursor()) && ScanCursor.FINISHED.isFinished() == cursor.isFinished()));log.info("漸進式刪除Set的元素完畢,迭代次數:{} ...", counter); }private void prepareSetTestData(String key) throws Exception {COMMAND.sadd(key, "1", "2", "3", "4", "5"); }

漸進式刪除有序集合中的元素:

@Test public void testDelBigZSetKey() throws Exception {// SCAN參數ScanArgs scanArgs = ScanArgs.Builder.limit(2);// TEMP游標ScanCursor cursor = ScanCursor.INITIAL;// 目標KEYString key = "BIG_ZSET_KEY";prepareZSetTestData(key);log.info("開始漸進式刪除ZSet的元素...");int counter = 0;do {ScoredValueScanCursor<String> result = COMMAND.zscan(key, cursor, scanArgs);// 重置TEMP游標cursor = ScanCursor.of(result.getCursor());cursor.setFinished(result.isFinished());List<ScoredValue<String>> scoredValues = result.getValues();if (!scoredValues.isEmpty()) {COMMAND.zrem(key, scoredValues.stream().map(ScoredValue<String>::getValue).toArray(String[]::new));}counter++;} while (!(ScanCursor.FINISHED.getCursor().equals(cursor.getCursor()) && ScanCursor.FINISHED.isFinished() == cursor.isFinished()));log.info("漸進式刪除ZSet的元素完畢,迭代次數:{} ...", counter); }private void prepareZSetTestData(String key) throws Exception {COMMAND.zadd(key, 0, "1");COMMAND.zadd(key, 0, "2");COMMAND.zadd(key, 0, "3");COMMAND.zadd(key, 0, "4");COMMAND.zadd(key, 0, "5"); }

在SpringBoot中使用Lettuce

個人認為,spring-data-redis中的API封裝并不是很優秀,用起來比較重,不夠靈活,這里結合前面的例子和代碼,在SpringBoot腳手架項目中配置和整合Lettuce。先引入依賴:

<dependencyManagement><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-dependencies</artifactId><version>2.1.8.RELEASE</version><type>pom</type><scope>import</scope></dependency></dependencies> </dependencyManagement> <dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>io.lettuce</groupId><artifactId>lettuce-core</artifactId><version>5.1.8.RELEASE</version></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><version>1.18.10</version><scope>provided</scope></dependency> </dependencies>

一般情況下,每個應用應該使用單個Redis客戶端實例和單個連接實例,這里設計一個腳手架,適配單機、普通主從、哨兵和集群四種使用場景。對于客戶端資源,采用默認的實現即可。對于Redis的連接屬性,比較主要的有Host、Port和Password,其他可以暫時忽略。基于約定大于配置的原則,先定制一系列屬性配置類(其實有些配置是可以完全共用,但是考慮到要清晰描述類之間的關系,這里拆分多個配置屬性類和多個配置方法):

@Data @ConfigurationProperties(prefix = "lettuce") public class LettuceProperties {private LettuceSingleProperties single;private LettuceReplicaProperties replica;private LettuceSentinelProperties sentinel;private LettuceClusterProperties cluster;}@Data public class LettuceSingleProperties {private String host;private Integer port;private String password; }@EqualsAndHashCode(callSuper = true) @Data public class LettuceReplicaProperties extends LettuceSingleProperties {}@EqualsAndHashCode(callSuper = true) @Data public class LettuceSentinelProperties extends LettuceSingleProperties {private String masterId; }@EqualsAndHashCode(callSuper = true) @Data public class LettuceClusterProperties extends LettuceSingleProperties {}

配置類如下,主要使用@ConditionalOnProperty做隔離,一般情況下,很少有人會在一個應用使用一種以上的Redis連接場景:

@RequiredArgsConstructor @Configuration @ConditionalOnClass(name = "io.lettuce.core.RedisURI") @EnableConfigurationProperties(value = LettuceProperties.class) public class LettuceAutoConfiguration {private final LettuceProperties lettuceProperties;@Bean(destroyMethod = "shutdown")public ClientResources clientResources() {return DefaultClientResources.create();}@Bean@ConditionalOnProperty(name = "lettuce.single.host")public RedisURI singleRedisUri() {LettuceSingleProperties singleProperties = lettuceProperties.getSingle();return RedisURI.builder().withHost(singleProperties.getHost()).withPort(singleProperties.getPort()).withPassword(singleProperties.getPassword()).build();}@Bean(destroyMethod = "shutdown")@ConditionalOnProperty(name = "lettuce.single.host")public RedisClient singleRedisClient(ClientResources clientResources, @Qualifier("singleRedisUri") RedisURI redisUri) {return RedisClient.create(clientResources, redisUri);}@Bean(destroyMethod = "close")@ConditionalOnProperty(name = "lettuce.single.host")public StatefulRedisConnection<String, String> singleRedisConnection(@Qualifier("singleRedisClient") RedisClient singleRedisClient) {return singleRedisClient.connect();}@Bean@ConditionalOnProperty(name = "lettuce.replica.host")public RedisURI replicaRedisUri() {LettuceReplicaProperties replicaProperties = lettuceProperties.getReplica();return RedisURI.builder().withHost(replicaProperties.getHost()).withPort(replicaProperties.getPort()).withPassword(replicaProperties.getPassword()).build();}@Bean(destroyMethod = "shutdown")@ConditionalOnProperty(name = "lettuce.replica.host")public RedisClient replicaRedisClient(ClientResources clientResources, @Qualifier("replicaRedisUri") RedisURI redisUri) {return RedisClient.create(clientResources, redisUri);}@Bean(destroyMethod = "close")@ConditionalOnProperty(name = "lettuce.replica.host")public StatefulRedisMasterSlaveConnection<String, String> replicaRedisConnection(@Qualifier("replicaRedisClient") RedisClient replicaRedisClient,@Qualifier("replicaRedisUri") RedisURI redisUri) {return MasterSlave.connect(replicaRedisClient, new Utf8StringCodec(), redisUri);}@Bean@ConditionalOnProperty(name = "lettuce.sentinel.host")public RedisURI sentinelRedisUri() {LettuceSentinelProperties sentinelProperties = lettuceProperties.getSentinel();return RedisURI.builder().withPassword(sentinelProperties.getPassword()).withSentinel(sentinelProperties.getHost(), sentinelProperties.getPort()).withSentinelMasterId(sentinelProperties.getMasterId()).build();}@Bean(destroyMethod = "shutdown")@ConditionalOnProperty(name = "lettuce.sentinel.host")public RedisClient sentinelRedisClient(ClientResources clientResources, @Qualifier("sentinelRedisUri") RedisURI redisUri) {return RedisClient.create(clientResources, redisUri);}@Bean(destroyMethod = "close")@ConditionalOnProperty(name = "lettuce.sentinel.host")public StatefulRedisMasterSlaveConnection<String, String> sentinelRedisConnection(@Qualifier("sentinelRedisClient") RedisClient sentinelRedisClient,@Qualifier("sentinelRedisUri") RedisURI redisUri) {return MasterSlave.connect(sentinelRedisClient, new Utf8StringCodec(), redisUri);}@Bean@ConditionalOnProperty(name = "lettuce.cluster.host")public RedisURI clusterRedisUri() {LettuceClusterProperties clusterProperties = lettuceProperties.getCluster();return RedisURI.builder().withHost(clusterProperties.getHost()).withPort(clusterProperties.getPort()).withPassword(clusterProperties.getPassword()).build();}@Bean(destroyMethod = "shutdown")@ConditionalOnProperty(name = "lettuce.cluster.host")public RedisClusterClient redisClusterClient(ClientResources clientResources, @Qualifier("clusterRedisUri") RedisURI redisUri) {return RedisClusterClient.create(clientResources, redisUri);}@Bean(destroyMethod = "close")@ConditionalOnProperty(name = "lettuce.cluster")public StatefulRedisClusterConnection<String, String> clusterConnection(RedisClusterClient clusterClient) {return clusterClient.connect();} }

最后為了讓IDE識別我們的配置,可以添加IDE親緣性,/META-INF文件夾下新增一個文件spring-configuration-metadata.json,內容如下:

{"properties": [{"name": "lettuce.single","type": "club.throwable.spring.lettuce.LettuceSingleProperties","description": "單機配置","sourceType": "club.throwable.spring.lettuce.LettuceProperties"},{"name": "lettuce.replica","type": "club.throwable.spring.lettuce.LettuceReplicaProperties","description": "主從配置","sourceType": "club.throwable.spring.lettuce.LettuceProperties"},{"name": "lettuce.sentinel","type": "club.throwable.spring.lettuce.LettuceSentinelProperties","description": "哨兵配置","sourceType": "club.throwable.spring.lettuce.LettuceProperties"},{"name": "lettuce.single","type": "club.throwable.spring.lettuce.LettuceClusterProperties","description": "集群配置","sourceType": "club.throwable.spring.lettuce.LettuceProperties"}] }

如果想IDE親緣性做得更好,可以添加/META-INF/additional-spring-configuration-metadata.json進行更多細節定義。簡單使用如下:

@Slf4j @Component public class RedisCommandLineRunner implements CommandLineRunner {@Autowired@Qualifier("singleRedisConnection")private StatefulRedisConnection<String, String> connection;@Overridepublic void run(String... args) throws Exception {RedisCommands<String, String> redisCommands = connection.sync();redisCommands.setex("name", 5, "throwable");log.info("Get value:{}", redisCommands.get("name"));} } // Get value:throwable

小結

本文算是基于Lettuce的官方文檔,對它的使用進行全方位的分析,包括主要功能、配置都做了一些示例,限于篇幅部分特性和配置細節沒有分析。Lettuce已經被spring-data-redis接納作為官方的Redis客戶端驅動,所以值得信賴,它的一些API設計確實比較合理,擴展性高的同時靈活性也高。個人建議,基于Lettuce包自行添加配置到SpringBoot應用用起來會得心應手,畢竟RedisTemplate實在太笨重,而且還屏蔽了Lettuce一些高級特性和靈活的API。

參考資料:

  • Lettuce Reference Guide

鏈接

  • Github Page:http://www.throwable.club/2019/09/28/redis-client-driver-lettuce-usage
  • Coding Page:http://throwable.coding.me/2019/09/28/redis-client-driver-lettuce-usage

(本文完 c-14-d e-a-20190928 最近事太多...)

轉載于:https://www.cnblogs.com/throwable/p/11601538.html

總結

以上是生活随笔為你收集整理的Redis高级客户端Lettuce详解的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

一本一道久久综合久久 | 老司机亚洲精品影院 | 国产精品.xx视频.xxtv | 性啪啪chinese东北女人 | 正在播放东北夫妻内射 | 亚洲国产精品久久人人爱 | 少妇被黑人到高潮喷出白浆 | 天天拍夜夜添久久精品大 | 理论片87福利理论电影 | 无遮无挡爽爽免费视频 | 在线欧美精品一区二区三区 | av无码不卡在线观看免费 | 最新国产麻豆aⅴ精品无码 | 久久人人爽人人爽人人片ⅴ | 中文字幕无码视频专区 | 粉嫩少妇内射浓精videos | 日日摸夜夜摸狠狠摸婷婷 | 欧美人与牲动交xxxx | 偷窥村妇洗澡毛毛多 | 少妇久久久久久人妻无码 | 男人和女人高潮免费网站 | 国产97在线 | 亚洲 | 日本一区二区更新不卡 | 国产农村乱对白刺激视频 | www国产精品内射老师 | 曰本女人与公拘交酡免费视频 | 丰满人妻精品国产99aⅴ | 日本精品少妇一区二区三区 | 少妇人妻大乳在线视频 | 丰满岳乱妇在线观看中字无码 | 亚洲の无码国产の无码步美 | 成熟人妻av无码专区 | 18禁止看的免费污网站 | 亚洲综合在线一区二区三区 | 久久久久成人片免费观看蜜芽 | 婷婷五月综合激情中文字幕 | 中文亚洲成a人片在线观看 | 又色又爽又黄的美女裸体网站 | 在线观看国产午夜福利片 | 亚洲一区二区三区香蕉 | 88国产精品欧美一区二区三区 | 天天综合网天天综合色 | 麻豆人妻少妇精品无码专区 | 国产真实夫妇视频 | 性做久久久久久久免费看 | 中文无码成人免费视频在线观看 | 亚洲国产精品美女久久久久 | 麻花豆传媒剧国产免费mv在线 | 精品厕所偷拍各类美女tp嘘嘘 | 天天躁日日躁狠狠躁免费麻豆 | 中文字幕乱码亚洲无线三区 | 中国女人内谢69xxxx | 午夜不卡av免费 一本久久a久久精品vr综合 | 国产 精品 自在自线 | 7777奇米四色成人眼影 | 国产一区二区三区精品视频 | 奇米影视7777久久精品人人爽 | 欧美老人巨大xxxx做受 | 高清无码午夜福利视频 | 成熟女人特级毛片www免费 | 久久无码专区国产精品s | 成人试看120秒体验区 | 大肉大捧一进一出视频出来呀 | 亚洲七七久久桃花影院 | 成人免费视频在线观看 | 99麻豆久久久国产精品免费 | 亚洲国产精品久久人人爱 | 蜜桃av抽搐高潮一区二区 | 国产69精品久久久久app下载 | 综合人妻久久一区二区精品 | 丰满少妇弄高潮了www | 动漫av一区二区在线观看 | 国产亚洲人成a在线v网站 | 国产精品国产三级国产专播 | 澳门永久av免费网站 | 黑人大群体交免费视频 | 久久久无码中文字幕久... | 日本精品高清一区二区 | 久久精品人人做人人综合 | 少妇性l交大片欧洲热妇乱xxx | 少妇邻居内射在线 | 亚洲中文字幕无码一久久区 | 黑森林福利视频导航 | 国产小呦泬泬99精品 | 亚洲成色www久久网站 | 免费国产黄网站在线观看 | 天天拍夜夜添久久精品大 | 少妇性l交大片欧洲热妇乱xxx | 又粗又大又硬又长又爽 | 麻豆av传媒蜜桃天美传媒 | 亚洲欧洲中文日韩av乱码 | 美女极度色诱视频国产 | 亚洲成在人网站无码天堂 | 国产精品亚洲五月天高清 | 国内老熟妇对白xxxxhd | 无码成人精品区在线观看 | 亚洲中文字幕乱码av波多ji | 呦交小u女精品视频 | 婷婷丁香五月天综合东京热 | 台湾无码一区二区 | 在线看片无码永久免费视频 | 激情内射亚州一区二区三区爱妻 | 少妇高潮喷潮久久久影院 | 国产婷婷色一区二区三区在线 | 亚洲精品无码人妻无码 | 欧美亚洲国产一区二区三区 | 国产精品丝袜黑色高跟鞋 | 亚洲精品久久久久久久久久久 | 久久久久久国产精品无码下载 | 日韩精品无码一区二区中文字幕 | 亚拍精品一区二区三区探花 | 亚洲欧美色中文字幕在线 | 99er热精品视频 | 曰本女人与公拘交酡免费视频 | 精品国产福利一区二区 | 国产suv精品一区二区五 | 麻豆果冻传媒2021精品传媒一区下载 | 色窝窝无码一区二区三区色欲 | 亚洲日本va午夜在线电影 | 国语自产偷拍精品视频偷 | 国产 浪潮av性色四虎 | 久久无码专区国产精品s | 国产偷抇久久精品a片69 | 窝窝午夜理论片影院 | 人妻人人添人妻人人爱 | 捆绑白丝粉色jk震动捧喷白浆 | 人妻互换免费中文字幕 | 强奷人妻日本中文字幕 | 亚洲欧美综合区丁香五月小说 | 中文字幕乱码人妻二区三区 | 国产精品久免费的黄网站 | 一本色道婷婷久久欧美 | 欧美黑人性暴力猛交喷水 | 免费乱码人妻系列无码专区 | 日本精品人妻无码77777 天堂一区人妻无码 | 亚洲娇小与黑人巨大交 | 亚洲精品国产第一综合99久久 | 熟妇人妻无乱码中文字幕 | 欧美xxxx黑人又粗又长 | 成人女人看片免费视频放人 | 久久午夜无码鲁丝片午夜精品 | 国产亚洲精品久久久久久久久动漫 | 亚洲欧洲中文日韩av乱码 | 欧美一区二区三区视频在线观看 | 18禁黄网站男男禁片免费观看 | 精品乱子伦一区二区三区 | 色婷婷久久一区二区三区麻豆 | 亚洲无人区一区二区三区 | 日日碰狠狠躁久久躁蜜桃 | 领导边摸边吃奶边做爽在线观看 | 性做久久久久久久免费看 | 一本久道久久综合狠狠爱 | 狠狠色欧美亚洲狠狠色www | 天天燥日日燥 | 中文毛片无遮挡高清免费 | 中文字幕无码免费久久9一区9 | 97夜夜澡人人爽人人喊中国片 | 2020久久香蕉国产线看观看 | 亚洲一区二区观看播放 | 少妇性荡欲午夜性开放视频剧场 | 超碰97人人射妻 | 麻豆av传媒蜜桃天美传媒 | 思思久久99热只有频精品66 | 婷婷综合久久中文字幕蜜桃三电影 | 99久久精品无码一区二区毛片 | 青草青草久热国产精品 | 伊人久久大香线蕉av一区二区 | 国产精品美女久久久 | 99久久精品日本一区二区免费 | 无码人妻久久一区二区三区不卡 | 午夜福利电影 | 小sao货水好多真紧h无码视频 | 在线观看欧美一区二区三区 | 成人综合网亚洲伊人 | 国产成人精品三级麻豆 | 久久久久免费看成人影片 | 国产午夜福利亚洲第一 | 精品成在人线av无码免费看 | 牲交欧美兽交欧美 | 久久99精品久久久久婷婷 | www一区二区www免费 | 亚洲另类伦春色综合小说 | 夜先锋av资源网站 | 色婷婷欧美在线播放内射 | 在线亚洲高清揄拍自拍一品区 | 亚洲乱码国产乱码精品精 | 无码吃奶揉捏奶头高潮视频 | 国产麻豆精品精东影业av网站 | 亚洲国产精品一区二区第一页 | 久久久久久亚洲精品a片成人 | 亚洲va中文字幕无码久久不卡 | 欧美野外疯狂做受xxxx高潮 | 日本熟妇大屁股人妻 | 2020最新国产自产精品 | 又大又紧又粉嫩18p少妇 | 国产成人精品三级麻豆 | 成年美女黄网站色大免费视频 | 日本免费一区二区三区最新 | 樱花草在线播放免费中文 | 麻豆果冻传媒2021精品传媒一区下载 | 国产精品久久久一区二区三区 | 成熟妇人a片免费看网站 | 亚无码乱人伦一区二区 | 久久 国产 尿 小便 嘘嘘 | 欧美性猛交内射兽交老熟妇 | 亚洲中文字幕在线观看 | www一区二区www免费 | 无码纯肉视频在线观看 | 欧美亚洲日韩国产人成在线播放 | 欧美日本免费一区二区三区 | 人妻少妇精品无码专区动漫 | 欧美大屁股xxxxhd黑色 | 伊人久久大香线蕉av一区二区 | 国产在线aaa片一区二区99 | 日本熟妇乱子伦xxxx | 亚洲中文字幕va福利 | 牲欲强的熟妇农村老妇女视频 | 亚洲色欲色欲欲www在线 | 亚洲另类伦春色综合小说 | 亚洲综合无码一区二区三区 | 亚洲自偷自偷在线制服 | 久久99热只有频精品8 | 欧美高清在线精品一区 | 欧美真人作爱免费视频 | 亚洲日韩中文字幕在线播放 | 欧美 日韩 亚洲 在线 | 精品日本一区二区三区在线观看 | 未满成年国产在线观看 | 亚洲 a v无 码免 费 成 人 a v | 荫蒂添的好舒服视频囗交 | 国内精品人妻无码久久久影院蜜桃 | 十八禁视频网站在线观看 | 少妇性l交大片欧洲热妇乱xxx | 亚洲精品国偷拍自产在线观看蜜桃 | 久久精品女人天堂av免费观看 | 国产精品亚洲一区二区三区喷水 | 欧美日韩综合一区二区三区 | 扒开双腿吃奶呻吟做受视频 | 性做久久久久久久久 | 无码国产乱人伦偷精品视频 | 超碰97人人射妻 | 精品无码一区二区三区爱欲 | 国产一精品一av一免费 | 欧美性猛交xxxx富婆 | 青青草原综合久久大伊人精品 | 沈阳熟女露脸对白视频 | 成人精品一区二区三区中文字幕 | 国产欧美精品一区二区三区 | 日日橹狠狠爱欧美视频 | 精品久久8x国产免费观看 | 国产成人一区二区三区在线观看 | 女人被男人躁得好爽免费视频 | 精品厕所偷拍各类美女tp嘘嘘 | 欧美性猛交xxxx富婆 | 国产后入清纯学生妹 | 麻豆成人精品国产免费 | 久久人人97超碰a片精品 | 亚洲综合久久一区二区 | 精品无人区无码乱码毛片国产 | 亚洲中文字幕无码一久久区 | 装睡被陌生人摸出水好爽 | 在线精品国产一区二区三区 | 国产亚洲精品精品国产亚洲综合 | 欧美xxxxx精品 | 精品国精品国产自在久国产87 | 老熟女乱子伦 | 亚洲精品一区二区三区在线 | 乱码av麻豆丝袜熟女系列 | 国内精品久久久久久中文字幕 | 亚洲国产欧美日韩精品一区二区三区 | 国产精品人人爽人人做我的可爱 | 性做久久久久久久久 | 欧洲vodafone精品性 | 国产精品二区一区二区aⅴ污介绍 | 激情五月综合色婷婷一区二区 | 大乳丰满人妻中文字幕日本 | 人人爽人人澡人人人妻 | 日本一卡二卡不卡视频查询 | 国产精品第一国产精品 | 无码播放一区二区三区 | 亚洲精品国产精品乱码不卡 | 亚洲 另类 在线 欧美 制服 | 国产乱人偷精品人妻a片 | 国产精品美女久久久 | 国产 浪潮av性色四虎 | 亚洲va欧美va天堂v国产综合 | 午夜精品一区二区三区的区别 | 男人的天堂2018无码 | 人人妻人人澡人人爽欧美一区 | 久久综合九色综合欧美狠狠 | 国内少妇偷人精品视频 | 免费国产成人高清在线观看网站 | 国语精品一区二区三区 | 亚洲人成网站免费播放 | 麻豆av传媒蜜桃天美传媒 | 奇米综合四色77777久久 东京无码熟妇人妻av在线网址 | 成人精品一区二区三区中文字幕 | 国产无遮挡又黄又爽又色 | 激情亚洲一区国产精品 | 男人的天堂2018无码 | 精品欧美一区二区三区久久久 | 国产精品亚洲а∨无码播放麻豆 | 综合人妻久久一区二区精品 | www成人国产高清内射 | 一区二区三区乱码在线 | 欧洲 | 国内老熟妇对白xxxxhd | 玩弄中年熟妇正在播放 | 亚洲一区二区三区四区 | 国产做国产爱免费视频 | 丁香花在线影院观看在线播放 | 亚洲一区二区三区四区 | 99re在线播放 | 九九热爱视频精品 | 亚洲中文字幕在线观看 | 国产无av码在线观看 | 国产av一区二区三区最新精品 | 强奷人妻日本中文字幕 | ass日本丰满熟妇pics | 精品国产青草久久久久福利 | 国产综合久久久久鬼色 | 久久久婷婷五月亚洲97号色 | 高潮毛片无遮挡高清免费 | 国产精品高潮呻吟av久久4虎 | 免费看少妇作爱视频 | 色综合天天综合狠狠爱 | 国产精品二区一区二区aⅴ污介绍 | 国产精品亚洲专区无码不卡 | 久久视频在线观看精品 | 久久人人爽人人爽人人片av高清 | 中文字幕人妻无码一夲道 | 少妇一晚三次一区二区三区 | 色 综合 欧美 亚洲 国产 | 国产又粗又硬又大爽黄老大爷视 | www国产亚洲精品久久网站 | 国产性生大片免费观看性 | 熟女体下毛毛黑森林 | 久久综合九色综合欧美狠狠 | 亚洲 欧美 激情 小说 另类 | 成人性做爰aaa片免费看不忠 | 色欲综合久久中文字幕网 | 强奷人妻日本中文字幕 | 国产疯狂伦交大片 | 日韩av无码一区二区三区 | 激情亚洲一区国产精品 | 草草网站影院白丝内射 | 领导边摸边吃奶边做爽在线观看 | 亚洲色大成网站www国产 | 正在播放老肥熟妇露脸 | 国产精品高潮呻吟av久久 | 熟女体下毛毛黑森林 | 国产艳妇av在线观看果冻传媒 | 亚洲日韩乱码中文无码蜜桃臀网站 | 日韩精品一区二区av在线 | 天天躁日日躁狠狠躁免费麻豆 | 熟妇激情内射com | 日本爽爽爽爽爽爽在线观看免 | 十八禁真人啪啪免费网站 | 西西人体www44rt大胆高清 | 中文字幕乱码人妻无码久久 | 成 人 网 站国产免费观看 | 日韩人妻无码中文字幕视频 | 国产午夜精品一区二区三区嫩草 | 狠狠色丁香久久婷婷综合五月 | 少妇被粗大的猛进出69影院 | 99久久久无码国产精品免费 | 小鲜肉自慰网站xnxx | 无码人妻丰满熟妇区五十路百度 | 图片小说视频一区二区 | 色综合久久久久综合一本到桃花网 | 狂野欧美性猛xxxx乱大交 | 精品亚洲韩国一区二区三区 | 久久亚洲中文字幕精品一区 | 亚洲日本va午夜在线电影 | 色 综合 欧美 亚洲 国产 | 日本va欧美va欧美va精品 | 亚洲男人av香蕉爽爽爽爽 | 伊人久久大香线焦av综合影院 | 麻花豆传媒剧国产免费mv在线 | 东京热无码av男人的天堂 | 久久久久久九九精品久 | 日本一卡二卡不卡视频查询 | 国产成人精品必看 | 国产激情综合五月久久 | 免费国产黄网站在线观看 | 嫩b人妻精品一区二区三区 | 成人精品天堂一区二区三区 | 色综合久久久无码网中文 | 性欧美牲交xxxxx视频 | 亚洲精品成人av在线 | 亚洲日韩av一区二区三区中文 | 亚洲欧美精品aaaaaa片 | 国产成人综合色在线观看网站 | 久久国语露脸国产精品电影 | 亚洲精品国产第一综合99久久 | 初尝人妻少妇中文字幕 | 国产69精品久久久久app下载 | 欧美熟妇另类久久久久久多毛 | 人妻少妇精品久久 | 麻豆蜜桃av蜜臀av色欲av | 人人妻人人澡人人爽人人精品浪潮 | 国产高清不卡无码视频 | 内射后入在线观看一区 | 鲁一鲁av2019在线 | 日日干夜夜干 | 亚洲中文字幕乱码av波多ji | 大色综合色综合网站 | 东京热一精品无码av | 日产精品高潮呻吟av久久 | 又大又紧又粉嫩18p少妇 | 国产舌乚八伦偷品w中 | 国产成人精品优优av | 高潮毛片无遮挡高清免费 | 国产69精品久久久久app下载 | 一本精品99久久精品77 | a国产一区二区免费入口 | 国产女主播喷水视频在线观看 | 在线 国产 欧美 亚洲 天堂 | 色 综合 欧美 亚洲 国产 | 女人被爽到呻吟gif动态图视看 | av在线亚洲欧洲日产一区二区 | 偷窥村妇洗澡毛毛多 | 亚洲人成网站在线播放942 | 免费中文字幕日韩欧美 | 精品亚洲韩国一区二区三区 | 无码av免费一区二区三区试看 | 无码免费一区二区三区 | 99riav国产精品视频 | 国内精品人妻无码久久久影院 | 亚洲中文字幕在线无码一区二区 | 亚洲小说春色综合另类 | 成人精品视频一区二区三区尤物 | 亚洲最大成人网站 | 国产人妻精品一区二区三区 | 日本一区二区更新不卡 | 伊在人天堂亚洲香蕉精品区 | 97精品人妻一区二区三区香蕉 | 成人欧美一区二区三区黑人 | 人人爽人人爽人人片av亚洲 | 亚洲精品国产第一综合99久久 | 国产激情精品一区二区三区 | 国产成人无码a区在线观看视频app | 一本色道久久综合狠狠躁 | 精品一区二区三区无码免费视频 | 成人无码视频在线观看网站 | 熟妇人妻激情偷爽文 | 67194成是人免费无码 | 国产精品对白交换视频 | 成人免费视频在线观看 | 国产av一区二区精品久久凹凸 | 日本va欧美va欧美va精品 | 精品国产一区二区三区四区在线看 | 香蕉久久久久久av成人 | 亚洲 另类 在线 欧美 制服 | 国产成人精品一区二区在线小狼 | 内射巨臀欧美在线视频 | 亚洲国精产品一二二线 | 国产精品国产自线拍免费软件 | 中文久久乱码一区二区 | 亚洲精品国产a久久久久久 | 沈阳熟女露脸对白视频 | 国语自产偷拍精品视频偷 | 亚洲一区av无码专区在线观看 | 亚洲乱码中文字幕在线 | 一个人看的视频www在线 | 亚洲午夜福利在线观看 | 秋霞成人午夜鲁丝一区二区三区 | www国产精品内射老师 | 女人被男人爽到呻吟的视频 | 亚洲成av人影院在线观看 | 宝宝好涨水快流出来免费视频 | 欧美黑人性暴力猛交喷水 | 久久97精品久久久久久久不卡 | 精品国产麻豆免费人成网站 | 亚无码乱人伦一区二区 | 俺去俺来也www色官网 | 无码国产激情在线观看 | 无码精品国产va在线观看dvd | 久久精品国产亚洲精品 | 久久久av男人的天堂 | 免费播放一区二区三区 | 无码人妻av免费一区二区三区 | 丰满岳乱妇在线观看中字无码 | 欧美 丝袜 自拍 制服 另类 | 亚洲精品国产a久久久久久 | 夜夜影院未满十八勿进 | 国产亚洲精品久久久久久国模美 | 国产高清不卡无码视频 | 国产亚洲精品久久久ai换 | 亚洲爆乳精品无码一区二区三区 | 鲁一鲁av2019在线 | 丰满妇女强制高潮18xxxx | 亚洲狠狠色丁香婷婷综合 | 国产成人综合色在线观看网站 | 日韩在线不卡免费视频一区 | 天天躁日日躁狠狠躁免费麻豆 | 又黄又爽又色的视频 | 曰本女人与公拘交酡免费视频 | 色综合视频一区二区三区 | 国产精品美女久久久久av爽李琼 | 国内精品久久久久久中文字幕 | 午夜理论片yy44880影院 | 成人精品一区二区三区中文字幕 | 青青青手机频在线观看 | 亚洲熟妇色xxxxx欧美老妇 | 欧美国产亚洲日韩在线二区 | 精品成在人线av无码免费看 | 国产成人久久精品流白浆 | 色诱久久久久综合网ywww | 特级做a爰片毛片免费69 | 偷窥村妇洗澡毛毛多 | 亚洲中文字幕乱码av波多ji | 水蜜桃色314在线观看 | 日韩av无码一区二区三区不卡 | 国产午夜精品一区二区三区嫩草 | 日本xxxx色视频在线观看免费 | 牛和人交xxxx欧美 | 激情五月综合色婷婷一区二区 | 无码人中文字幕 | 图片区 小说区 区 亚洲五月 | 东京热无码av男人的天堂 | 成人女人看片免费视频放人 | 成人精品视频一区二区三区尤物 | 鲁大师影院在线观看 | 青青青手机频在线观看 | 国产成人久久精品流白浆 | 男人的天堂av网站 | 亚洲成a人片在线观看无码 | 97人妻精品一区二区三区 | 乱人伦人妻中文字幕无码久久网 | 久久精品国产精品国产精品污 | 特黄特色大片免费播放器图片 | 精品 日韩 国产 欧美 视频 | 99国产精品白浆在线观看免费 | 在线天堂新版最新版在线8 | 午夜福利一区二区三区在线观看 | 图片小说视频一区二区 | 老熟妇乱子伦牲交视频 | 精品亚洲成av人在线观看 | 久久综合狠狠综合久久综合88 | 俺去俺来也在线www色官网 | 亚洲 a v无 码免 费 成 人 a v | 日本精品少妇一区二区三区 | 中文字幕无线码免费人妻 | 日日天干夜夜狠狠爱 | 久久无码专区国产精品s | 久久久精品456亚洲影院 | 久久综合狠狠综合久久综合88 | 97无码免费人妻超级碰碰夜夜 | 无码人妻少妇伦在线电影 | 色婷婷综合激情综在线播放 | 男人的天堂av网站 | 无码成人精品区在线观看 | 亚洲国产欧美国产综合一区 | 四虎国产精品免费久久 | аⅴ资源天堂资源库在线 | 久久久精品国产sm最大网站 | 国产精品沙发午睡系列 | 精品国偷自产在线 | 好屌草这里只有精品 | 国产黄在线观看免费观看不卡 | 青青草原综合久久大伊人精品 | 丁香啪啪综合成人亚洲 | 99久久人妻精品免费二区 | 中文精品无码中文字幕无码专区 | 无人区乱码一区二区三区 | 亚洲综合色区中文字幕 | 国产精品沙发午睡系列 | 亚洲精品一区二区三区大桥未久 | ass日本丰满熟妇pics | 国产av一区二区精品久久凹凸 | 夜夜夜高潮夜夜爽夜夜爰爰 | 中国女人内谢69xxxx | 国产精品va在线观看无码 | 成人试看120秒体验区 | 精品偷拍一区二区三区在线看 | 国产性生交xxxxx无码 | 精品国产麻豆免费人成网站 | 精品久久8x国产免费观看 | 97久久国产亚洲精品超碰热 | 超碰97人人射妻 | 日韩精品无码一区二区中文字幕 | 无码精品国产va在线观看dvd | 日本大乳高潮视频在线观看 | 又黄又爽又色的视频 | 国产手机在线αⅴ片无码观看 | 欧美熟妇另类久久久久久不卡 | 国产电影无码午夜在线播放 | 成在人线av无码免费 | 欧美人妻一区二区三区 | 亚洲 日韩 欧美 成人 在线观看 | 国产成人久久精品流白浆 | 日韩精品无码免费一区二区三区 | 亚洲色大成网站www国产 | 亚洲人交乣女bbw | 欧美精品一区二区精品久久 | 人人妻人人澡人人爽欧美一区九九 | 色一情一乱一伦 | 色情久久久av熟女人妻网站 | 人人妻人人澡人人爽人人精品 | 人人爽人人澡人人高潮 | 中文字幕无码av激情不卡 | 纯爱无遮挡h肉动漫在线播放 | 免费人成网站视频在线观看 | aa片在线观看视频在线播放 | 成人免费视频在线观看 | 鲁一鲁av2019在线 | 色婷婷综合激情综在线播放 | 九月婷婷人人澡人人添人人爽 | 国产综合在线观看 | 国产美女精品一区二区三区 | 久久亚洲日韩精品一区二区三区 | 精品国偷自产在线 | 免费看男女做好爽好硬视频 | www一区二区www免费 | 无码福利日韩神码福利片 | 国产精品igao视频网 | 亚洲最大成人网站 | 我要看www免费看插插视频 | 国精产品一区二区三区 | 99久久精品国产一区二区蜜芽 | 国产精品久久久久久久9999 | 九九久久精品国产免费看小说 | 四十如虎的丰满熟妇啪啪 | 亚洲成a人片在线观看无码 | 亚洲一区二区三区在线观看网站 | 婷婷色婷婷开心五月四房播播 | 精品人人妻人人澡人人爽人人 | 欧美人与禽zoz0性伦交 | 精品无人区无码乱码毛片国产 | 亚洲一区av无码专区在线观看 | 精品国产精品久久一区免费式 | 色婷婷香蕉在线一区二区 | 亚洲啪av永久无码精品放毛片 | 国产午夜亚洲精品不卡下载 | 国产麻豆精品精东影业av网站 | 岛国片人妻三上悠亚 | 亚洲毛片av日韩av无码 | 亚洲の无码国产の无码步美 | 综合人妻久久一区二区精品 | 好爽又高潮了毛片免费下载 | 丝袜美腿亚洲一区二区 | 红桃av一区二区三区在线无码av | 国产精品无码mv在线观看 | 高潮毛片无遮挡高清免费 | 天干天干啦夜天干天2017 | 久久精品人人做人人综合 | 中文字幕+乱码+中文字幕一区 | 国产精品va在线播放 | av人摸人人人澡人人超碰下载 | 日韩人妻少妇一区二区三区 | 国产熟妇高潮叫床视频播放 | 亚洲日韩av一区二区三区四区 | 亚洲午夜福利在线观看 | 免费人成在线观看网站 | 成人亚洲精品久久久久 | 给我免费的视频在线观看 | 亚洲另类伦春色综合小说 | 男女作爱免费网站 | 亚洲精品久久久久avwww潮水 | 高清不卡一区二区三区 | 中文字幕日产无线码一区 | 国产乱人伦偷精品视频 | 麻豆精品国产精华精华液好用吗 | 亚洲国产成人a精品不卡在线 | 日欧一片内射va在线影院 | 久久国语露脸国产精品电影 | 1000部啪啪未满十八勿入下载 | 在线观看免费人成视频 | 97精品国产97久久久久久免费 | 粗大的内捧猛烈进出视频 | 日日碰狠狠躁久久躁蜜桃 | 亚洲国产av美女网站 | 久久久久亚洲精品男人的天堂 | 亚洲中文字幕无码中文字在线 | 精品国精品国产自在久国产87 | 麻豆蜜桃av蜜臀av色欲av | 亚洲无人区午夜福利码高清完整版 | 久久久精品人妻久久影视 | 亚洲精品久久久久中文第一幕 | 亚洲日本在线电影 | 97久久国产亚洲精品超碰热 | 性欧美熟妇videofreesex | 夜夜高潮次次欢爽av女 | 一本久道久久综合狠狠爱 | 亚洲s码欧洲m码国产av | 精品 日韩 国产 欧美 视频 | 51国偷自产一区二区三区 | 婷婷色婷婷开心五月四房播播 | 日韩av无码一区二区三区 | 国精品人妻无码一区二区三区蜜柚 | 久久久精品成人免费观看 | 欧美猛少妇色xxxxx | 亚洲 另类 在线 欧美 制服 | 97人妻精品一区二区三区 | 精品人妻中文字幕有码在线 | 中文精品无码中文字幕无码专区 | 无码成人精品区在线观看 | 国产小呦泬泬99精品 | 日本熟妇乱子伦xxxx | 日本护士xxxxhd少妇 | 国产乱人偷精品人妻a片 | 久久精品人人做人人综合试看 | 国内少妇偷人精品视频 | 中国女人内谢69xxxxxa片 | 国产麻豆精品精东影业av网站 | 四虎永久在线精品免费网址 | 精品国产一区二区三区四区 | 久久久久99精品国产片 | 国产热a欧美热a在线视频 | 亚洲人成影院在线无码按摩店 | 国产成人久久精品流白浆 | 中文字幕乱码人妻无码久久 | 国产另类ts人妖一区二区 | 国产av一区二区三区最新精品 | 日本精品久久久久中文字幕 | 中文字幕人妻无码一区二区三区 | 男女爱爱好爽视频免费看 | 国产内射爽爽大片视频社区在线 | 狠狠色噜噜狠狠狠7777奇米 | 日本一卡二卡不卡视频查询 | 亚洲精品国产精品乱码不卡 | 少妇久久久久久人妻无码 | 国产人妖乱国产精品人妖 | 免费无码的av片在线观看 | 久久久av男人的天堂 | 亚洲国产av美女网站 | 欧美真人作爱免费视频 | 中文字幕av日韩精品一区二区 | 天天躁夜夜躁狠狠是什么心态 | 色五月五月丁香亚洲综合网 | 久久久www成人免费毛片 | 大地资源中文第3页 | 欧美人与物videos另类 | 天堂久久天堂av色综合 | 国产 精品 自在自线 | 久久久精品456亚洲影院 | 男人和女人高潮免费网站 | 日韩少妇白浆无码系列 | 亚洲精品美女久久久久久久 | 国产精品二区一区二区aⅴ污介绍 | 国产精品亚洲综合色区韩国 | 日韩人妻无码一区二区三区久久99 | 久久久久久亚洲精品a片成人 | 清纯唯美经典一区二区 | 18无码粉嫩小泬无套在线观看 | 一本无码人妻在中文字幕免费 | 久久99久久99精品中文字幕 | 国产亚洲美女精品久久久2020 | 中文字幕无码免费久久9一区9 | 国产69精品久久久久app下载 | 日韩少妇白浆无码系列 | 国产在线一区二区三区四区五区 | 巨爆乳无码视频在线观看 | 国产偷抇久久精品a片69 | 欧美阿v高清资源不卡在线播放 | 亚洲精品久久久久中文第一幕 | 内射老妇bbwx0c0ck | 国产农村妇女高潮大叫 | 1000部啪啪未满十八勿入下载 | 国产办公室秘书无码精品99 | 东京热无码av男人的天堂 | 日本va欧美va欧美va精品 | aa片在线观看视频在线播放 | 在线a亚洲视频播放在线观看 | 国产成人无码区免费内射一片色欲 | 国产 精品 自在自线 | 国产无套粉嫩白浆在线 | 精品欧洲av无码一区二区三区 | 日韩人妻无码一区二区三区久久99 | 精品久久综合1区2区3区激情 | 人人爽人人澡人人高潮 | 午夜福利试看120秒体验区 | 啦啦啦www在线观看免费视频 | 成人一在线视频日韩国产 | 日欧一片内射va在线影院 | 国产亚洲人成a在线v网站 | 美女黄网站人色视频免费国产 | 四虎国产精品一区二区 | 亚洲国产精品一区二区美利坚 | 青青草原综合久久大伊人精品 | 亚洲成在人网站无码天堂 | 伊人久久大香线蕉亚洲 | 国产小呦泬泬99精品 | 欧美日韩在线亚洲综合国产人 | 两性色午夜视频免费播放 | 四虎国产精品免费久久 | 色综合久久久久综合一本到桃花网 | 欧美 亚洲 国产 另类 | 国产69精品久久久久app下载 | 老司机亚洲精品影院无码 | 欧美肥老太牲交大战 | 伊在人天堂亚洲香蕉精品区 | 高清无码午夜福利视频 | 国产手机在线αⅴ片无码观看 | 中文字幕久久久久人妻 | 国内精品人妻无码久久久影院蜜桃 | 中文精品久久久久人妻不卡 | 中国女人内谢69xxxx | 97久久超碰中文字幕 | 扒开双腿吃奶呻吟做受视频 | 国产成人午夜福利在线播放 | 大肉大捧一进一出视频出来呀 | 欧美黑人巨大xxxxx | 蜜臀av在线观看 在线欧美精品一区二区三区 | 4hu四虎永久在线观看 | 在线播放免费人成毛片乱码 | 131美女爱做视频 | 国产成人一区二区三区在线观看 | 欧美成人午夜精品久久久 | 中文字幕无码乱人伦 | 日本熟妇人妻xxxxx人hd | 欧美性生交活xxxxxdddd | 久久久精品成人免费观看 | 老头边吃奶边弄进去呻吟 | 波多野结衣av一区二区全免费观看 | 日本精品人妻无码免费大全 | 国产激情精品一区二区三区 | 纯爱无遮挡h肉动漫在线播放 | 国产亚洲美女精品久久久2020 | 人人澡人摸人人添 | 曰本女人与公拘交酡免费视频 | 精品厕所偷拍各类美女tp嘘嘘 | 无码任你躁久久久久久久 | 国产国产精品人在线视 | 午夜福利一区二区三区在线观看 | 麻豆国产人妻欲求不满 | 国产熟妇高潮叫床视频播放 | 日本丰满护士爆乳xxxx | 国产精华av午夜在线观看 | 国产极品视觉盛宴 | 狠狠亚洲超碰狼人久久 | 亚洲人成网站免费播放 | 亚洲精品午夜国产va久久成人 | 国产超碰人人爽人人做人人添 | 久久久久亚洲精品男人的天堂 | 无码精品人妻一区二区三区av | 国产女主播喷水视频在线观看 | 中文字幕 人妻熟女 | 国产人妻久久精品二区三区老狼 | 日本丰满熟妇videos | 欧美日韩人成综合在线播放 | 97久久国产亚洲精品超碰热 | √8天堂资源地址中文在线 | 丰满岳乱妇在线观看中字无码 | 曰本女人与公拘交酡免费视频 | 天堂无码人妻精品一区二区三区 | 最新国产麻豆aⅴ精品无码 | 男女猛烈xx00免费视频试看 | 东京一本一道一二三区 | 欧洲vodafone精品性 | 亚洲娇小与黑人巨大交 | 成 人影片 免费观看 | 一本色道久久综合狠狠躁 | 中文字幕乱妇无码av在线 | 亚洲国产精品无码一区二区三区 | 欧美 日韩 人妻 高清 中文 | 国产小呦泬泬99精品 | 亚洲精品国偷拍自产在线观看蜜桃 | 久久精品中文闷骚内射 | 国产疯狂伦交大片 | 精品国精品国产自在久国产87 | 亚洲欧美日韩综合久久久 | 无码吃奶揉捏奶头高潮视频 | 无码人妻黑人中文字幕 | 亚洲一区二区观看播放 | 成在人线av无码免观看麻豆 | 久久午夜无码鲁丝片午夜精品 | 国产小呦泬泬99精品 | 国产疯狂伦交大片 | 在线播放免费人成毛片乱码 | 好男人社区资源 | 人妻无码久久精品人妻 | 亚洲国产精品久久久久久 | 乱码av麻豆丝袜熟女系列 | 久久亚洲国产成人精品性色 | 自拍偷自拍亚洲精品被多人伦好爽 | 久青草影院在线观看国产 | 国产精品毛多多水多 | 男女下面进入的视频免费午夜 | 成人欧美一区二区三区黑人免费 | 中文字幕av无码一区二区三区电影 | 水蜜桃av无码 | 精品久久久久久亚洲精品 | 国产亚洲日韩欧美另类第八页 | 国产又粗又硬又大爽黄老大爷视 | 国产口爆吞精在线视频 | 午夜熟女插插xx免费视频 | 日日碰狠狠躁久久躁蜜桃 | 一二三四社区在线中文视频 | 青春草在线视频免费观看 | 青青青爽视频在线观看 | 亚洲成a人片在线观看日本 | 久久久中文字幕日本无吗 | 一本色道久久综合亚洲精品不卡 | 狠狠色色综合网站 | 狠狠色丁香久久婷婷综合五月 | 少妇久久久久久人妻无码 | 欧美freesex黑人又粗又大 | 国产97色在线 | 免 | 中文字幕+乱码+中文字幕一区 | 成人精品视频一区二区三区尤物 | 国产香蕉尹人视频在线 | 日韩少妇内射免费播放 | 国产美女精品一区二区三区 | 99久久精品无码一区二区毛片 | 国产一精品一av一免费 | 日本xxxx色视频在线观看免费 | 97精品国产97久久久久久免费 | 久久www免费人成人片 | 麻豆果冻传媒2021精品传媒一区下载 | 日韩人妻无码中文字幕视频 | 亚洲综合久久一区二区 | 四虎永久在线精品免费网址 | 亚洲码国产精品高潮在线 | 亚洲人交乣女bbw | 日日碰狠狠躁久久躁蜜桃 | 久久久精品456亚洲影院 | 正在播放东北夫妻内射 | 思思久久99热只有频精品66 | 熟女少妇在线视频播放 | 国产成人无码专区 | 99国产精品白浆在线观看免费 | 欧美午夜特黄aaaaaa片 | 曰本女人与公拘交酡免费视频 | 国产成人精品三级麻豆 | 久久综合九色综合欧美狠狠 | 亚洲一区二区三区无码久久 | 亚洲精品一区二区三区在线观看 | 国产亚洲欧美日韩亚洲中文色 | 在线天堂新版最新版在线8 | 国产成人无码a区在线观看视频app | 波多野结衣高清一区二区三区 | 久久熟妇人妻午夜寂寞影院 | 国产黑色丝袜在线播放 | 88国产精品欧美一区二区三区 | 国产热a欧美热a在线视频 | 国产精品无码永久免费888 | 清纯唯美经典一区二区 | 国产精品久久久一区二区三区 | 老司机亚洲精品影院无码 | 婷婷综合久久中文字幕蜜桃三电影 | 欧美性生交xxxxx久久久 | 色诱久久久久综合网ywww | 黑人巨大精品欧美一区二区 | 国产免费观看黄av片 | 少妇太爽了在线观看 | 狂野欧美性猛xxxx乱大交 | 欧美国产日韩久久mv | 熟妇人妻激情偷爽文 | 中文字幕无码免费久久9一区9 | 久久久久免费精品国产 | 亚洲 日韩 欧美 成人 在线观看 | 丰满护士巨好爽好大乳 | 国精品人妻无码一区二区三区蜜柚 | 亚洲娇小与黑人巨大交 | 中文字幕 人妻熟女 | 国产成人精品视频ⅴa片软件竹菊 | 特黄特色大片免费播放器图片 | 欧美大屁股xxxxhd黑色 | 久久综合给久久狠狠97色 | 日本va欧美va欧美va精品 | 俺去俺来也www色官网 | 日韩视频 中文字幕 视频一区 | 国产精品内射视频免费 | 国产成人人人97超碰超爽8 | 爆乳一区二区三区无码 | 亚洲自偷精品视频自拍 | 丝袜 中出 制服 人妻 美腿 | 黑人玩弄人妻中文在线 | 精品夜夜澡人妻无码av蜜桃 | 无码人中文字幕 | 亚洲s色大片在线观看 | 欧美黑人乱大交 | 亚洲а∨天堂久久精品2021 | 国产性猛交╳xxx乱大交 国产精品久久久久久无码 欧洲欧美人成视频在线 | 丝袜足控一区二区三区 | 亚洲午夜久久久影院 | 黑人巨大精品欧美黑寡妇 | 日本高清一区免费中文视频 | 亚洲日本在线电影 | 欧美freesex黑人又粗又大 | 国产99久久精品一区二区 | 鲁大师影院在线观看 | 亚洲日韩av一区二区三区四区 | 丰满少妇人妻久久久久久 | 我要看www免费看插插视频 | 久久久久99精品成人片 | 人妻无码αv中文字幕久久琪琪布 | 亚洲人交乣女bbw | 最近中文2019字幕第二页 | 乱码午夜-极国产极内射 | 欧美丰满熟妇xxxx | 水蜜桃亚洲一二三四在线 | 麻豆果冻传媒2021精品传媒一区下载 | aa片在线观看视频在线播放 | 色婷婷综合激情综在线播放 | 国产偷国产偷精品高清尤物 | 青春草在线视频免费观看 | 国产 精品 自在自线 | 精品一区二区不卡无码av | 天海翼激烈高潮到腰振不止 | 亚洲の无码国产の无码步美 | 国内精品久久毛片一区二区 | a在线亚洲男人的天堂 | 亚洲区小说区激情区图片区 | 妺妺窝人体色www在线小说 | 成人综合网亚洲伊人 | 亚洲a无码综合a国产av中文 | 学生妹亚洲一区二区 | 欧美日韩人成综合在线播放 | 亚洲成av人在线观看网址 | 亚洲日韩一区二区 | 人人妻人人藻人人爽欧美一区 | 国产无套粉嫩白浆在线 | 又大又紧又粉嫩18p少妇 | 久久久av男人的天堂 | 日本乱人伦片中文三区 | 纯爱无遮挡h肉动漫在线播放 | 香港三级日本三级妇三级 | 欧美性猛交内射兽交老熟妇 | 蜜臀aⅴ国产精品久久久国产老师 | 亚洲精品无码人妻无码 | 国产成人无码av片在线观看不卡 | 水蜜桃亚洲一二三四在线 | 国产一区二区三区四区五区加勒比 | 久青草影院在线观看国产 | 中文字幕av日韩精品一区二区 | 中文字幕乱码亚洲无线三区 | 一个人免费观看的www视频 | 亚洲中文字幕成人无码 | 18禁止看的免费污网站 | 国产色xx群视频射精 | 午夜精品一区二区三区的区别 | 国产偷抇久久精品a片69 | 乱码午夜-极国产极内射 | 97夜夜澡人人双人人人喊 | 亚洲国产综合无码一区 | 国产无av码在线观看 | 成人精品视频一区二区 | 2020久久超碰国产精品最新 | 日日鲁鲁鲁夜夜爽爽狠狠 | 国产香蕉97碰碰久久人人 | 一本久久伊人热热精品中文字幕 | 亚洲精品一区二区三区四区五区 | 爆乳一区二区三区无码 | 久久人人爽人人爽人人片av高清 | 人妻夜夜爽天天爽三区 | 又色又爽又黄的美女裸体网站 | 国产精品亚洲一区二区三区喷水 | 天堂а√在线地址中文在线 | 精品人妻中文字幕有码在线 | 久久亚洲精品中文字幕无男同 | 俄罗斯老熟妇色xxxx | 久久亚洲国产成人精品性色 | 一本无码人妻在中文字幕免费 | 亚洲精品一区二区三区大桥未久 | 国产精品人人爽人人做我的可爱 | 欧美精品国产综合久久 | 国内综合精品午夜久久资源 | 狠狠色丁香久久婷婷综合五月 | 精品久久综合1区2区3区激情 | 国产色在线 | 国产 | 免费无码的av片在线观看 | 无码免费一区二区三区 | 十八禁视频网站在线观看 | 国产精品资源一区二区 | 国产午夜手机精彩视频 | 亚洲日韩av一区二区三区中文 | 伊人久久婷婷五月综合97色 | 人妻无码久久精品人妻 | 东京热一精品无码av | 久久婷婷五月综合色国产香蕉 | 伊人色综合久久天天小片 | 日产精品高潮呻吟av久久 | 亚洲日韩精品欧美一区二区 | 精品久久久中文字幕人妻 | 国产精品爱久久久久久久 | 亚洲熟妇自偷自拍另类 | 成人欧美一区二区三区黑人 | 人妻少妇精品无码专区二区 | 无套内射视频囯产 | 性欧美疯狂xxxxbbbb | 国产成人无码专区 | 欧美激情综合亚洲一二区 | 人人妻人人藻人人爽欧美一区 | 小泽玛莉亚一区二区视频在线 | 国产亚洲精品久久久久久久 | 欧美大屁股xxxxhd黑色 | 国产精品对白交换视频 | 久久精品国产亚洲精品 | 亚洲欧美日韩成人高清在线一区 | 亚洲色无码一区二区三区 | a在线观看免费网站大全 | 亚洲啪av永久无码精品放毛片 | 99久久精品无码一区二区毛片 | 中文字幕亚洲情99在线 | 精品国产青草久久久久福利 | 亚洲精品www久久久 | 少妇性荡欲午夜性开放视频剧场 | 日韩视频 中文字幕 视频一区 | 2020久久香蕉国产线看观看 | 久久人人97超碰a片精品 | 亚洲自偷自偷在线制服 | 亚洲国产欧美日韩精品一区二区三区 | 高潮毛片无遮挡高清免费视频 | 日韩无码专区 | 午夜福利电影 | 国产精品自产拍在线观看 | 内射巨臀欧美在线视频 | 少妇激情av一区二区 | 亚洲人成影院在线无码按摩店 | 天堂亚洲免费视频 | 欧美三级不卡在线观看 | 秋霞成人午夜鲁丝一区二区三区 | 天天av天天av天天透 | 好男人www社区 | 久久人妻内射无码一区三区 | 久久99热只有频精品8 | 极品嫩模高潮叫床 | 最新版天堂资源中文官网 | 国产午夜福利100集发布 | 久久亚洲中文字幕精品一区 | 久久久精品456亚洲影院 | 天堂亚洲2017在线观看 | 久久天天躁狠狠躁夜夜免费观看 | 扒开双腿疯狂进出爽爽爽视频 | 欧洲欧美人成视频在线 | 免费无码的av片在线观看 | 18禁黄网站男男禁片免费观看 | 欧美刺激性大交 | 日韩人妻无码中文字幕视频 | 在线观看国产一区二区三区 | 色情久久久av熟女人妻网站 | 日韩成人一区二区三区在线观看 | 久久久国产精品无码免费专区 | 少妇性l交大片欧洲热妇乱xxx | 欧美自拍另类欧美综合图片区 | 日本欧美一区二区三区乱码 | 国产高清不卡无码视频 | 国产特级毛片aaaaaa高潮流水 | 中文字幕无码人妻少妇免费 | 俺去俺来也在线www色官网 | 波多野结衣乳巨码无在线观看 | 99麻豆久久久国产精品免费 | 人妻无码αv中文字幕久久琪琪布 | 精品 日韩 国产 欧美 视频 | 亚洲欧美日韩成人高清在线一区 | 人妻无码αv中文字幕久久琪琪布 | 动漫av网站免费观看 | 亚洲а∨天堂久久精品2021 | 精品国产一区av天美传媒 | 亚洲欧美国产精品久久 | 国产电影无码午夜在线播放 | 午夜免费福利小电影 | 少妇被粗大的猛进出69影院 | 少妇无码一区二区二三区 | 99久久久国产精品无码免费 | 综合网日日天干夜夜久久 | 少妇愉情理伦片bd | 久久精品99久久香蕉国产色戒 | 丝袜 中出 制服 人妻 美腿 | 成人综合网亚洲伊人 | 欧美日韩人成综合在线播放 | 日本一卡二卡不卡视频查询 | 帮老师解开蕾丝奶罩吸乳网站 | 76少妇精品导航 | 国产真实伦对白全集 | 国产成人av免费观看 | 在线精品亚洲一区二区 | 欧美丰满熟妇xxxx | 国产99久久精品一区二区 | 亚洲男女内射在线播放 | 国产电影无码午夜在线播放 | 2019午夜福利不卡片在线 | 久久精品国产99久久6动漫 | 久久综合九色综合97网 | 国产舌乚八伦偷品w中 | 成年美女黄网站色大免费全看 | 国产极品美女高潮无套在线观看 | www国产精品内射老师 | 久久99精品久久久久久动态图 | 奇米综合四色77777久久 东京无码熟妇人妻av在线网址 | 国产黄在线观看免费观看不卡 | 精品国产一区av天美传媒 | 国产精品毛多多水多 | 久久国产精品精品国产色婷婷 | 99精品视频在线观看免费 | 少妇邻居内射在线 | 噜噜噜亚洲色成人网站 | 国产精华av午夜在线观看 | 国产亚洲精品久久久ai换 | 99精品视频在线观看免费 | 久久熟妇人妻午夜寂寞影院 | 国产精品久久久久久久影院 | 色五月五月丁香亚洲综合网 | 激情内射亚州一区二区三区爱妻 | 亚洲一区二区观看播放 | 在线亚洲高清揄拍自拍一品区 | 国产精品人妻一区二区三区四 | 国产av无码专区亚洲a∨毛片 | 极品嫩模高潮叫床 | 亚洲精品午夜无码电影网 | 色诱久久久久综合网ywww | 午夜免费福利小电影 | 久久久久久a亚洲欧洲av冫 | 精品无码国产一区二区三区av | 国产精品亚洲一区二区三区喷水 | 亚洲无人区午夜福利码高清完整版 | 精品国产一区av天美传媒 | 高潮毛片无遮挡高清免费 | 欧美人与禽猛交狂配 | 一区二区三区高清视频一 | 秋霞特色aa大片 | 亚洲伊人久久精品影院 | 高潮毛片无遮挡高清免费 | 色综合久久88色综合天天 | 97色伦图片97综合影院 | 精品久久久久香蕉网 | 国产真实乱对白精彩久久 | 国产麻豆精品一区二区三区v视界 | 精品久久8x国产免费观看 | 中文精品久久久久人妻不卡 | 国产热a欧美热a在线视频 | 亚洲成a人片在线观看日本 | 国产亚洲精品久久久久久久 | 蜜桃av抽搐高潮一区二区 | 老子影院午夜伦不卡 | 国产熟女一区二区三区四区五区 | 亚洲国产精品无码久久久久高潮 | 婷婷五月综合激情中文字幕 | 东京无码熟妇人妻av在线网址 | 亚洲成a人片在线观看无码 | 国产午夜亚洲精品不卡 | 鲁大师影院在线观看 | 欧美猛少妇色xxxxx | 欧美一区二区三区视频在线观看 | 亚洲性无码av中文字幕 | 亚洲精品无码国产 | 一本久久a久久精品vr综合 | 亚洲天堂2017无码中文 | 丰满护士巨好爽好大乳 | 欧美熟妇另类久久久久久不卡 | 国产精品无码一区二区三区不卡 | 一区二区传媒有限公司 | 亚洲一区二区三区在线观看网站 | 无码福利日韩神码福利片 | 国产三级久久久精品麻豆三级 | 天天躁日日躁狠狠躁免费麻豆 | 狂野欧美激情性xxxx | 大胆欧美熟妇xx | 欧美乱妇无乱码大黄a片 | 亚洲成av人片天堂网无码】 | 久久久www成人免费毛片 | 日日摸天天摸爽爽狠狠97 | 欧洲熟妇色 欧美 | 国产精品a成v人在线播放 | 性生交大片免费看女人按摩摩 | 天堂久久天堂av色综合 | 乌克兰少妇xxxx做受 | 久久综合香蕉国产蜜臀av | 国産精品久久久久久久 | 人人爽人人爽人人片av亚洲 | 天堂无码人妻精品一区二区三区 | 亚洲va欧美va天堂v国产综合 | 国产精品.xx视频.xxtv | 久久久久久a亚洲欧洲av冫 | 久久国产精品_国产精品 | 久久久精品欧美一区二区免费 | 国产舌乚八伦偷品w中 | 四虎4hu永久免费 | 亚洲国产精品美女久久久久 | 国产免费观看黄av片 | 亚洲爆乳精品无码一区二区三区 | 久久精品国产日本波多野结衣 | 2019nv天堂香蕉在线观看 | 欧美人与牲动交xxxx | 亚洲成av人综合在线观看 | 欧美日韩亚洲国产精品 | 未满成年国产在线观看 | 久久久中文字幕日本无吗 | 亚洲一区av无码专区在线观看 | 国产色xx群视频射精 | 国产舌乚八伦偷品w中 | 国产xxx69麻豆国语对白 | 亚洲一区二区观看播放 | 午夜精品一区二区三区在线观看 | 97资源共享在线视频 | 久久99国产综合精品 | 国产av一区二区精品久久凹凸 | 欧美日韩视频无码一区二区三 | 亚洲精品国产精品乱码不卡 | 天堂久久天堂av色综合 | 久久精品国产一区二区三区 | 亚洲色偷偷男人的天堂 | 理论片87福利理论电影 | 国产亚洲精品精品国产亚洲综合 | 5858s亚洲色大成网站www | 伊人久久大香线蕉亚洲 | 国产精品免费大片 | 国产精品亚洲а∨无码播放麻豆 | 精品久久综合1区2区3区激情 | 国产成人久久精品流白浆 | 无人区乱码一区二区三区 | 亚洲国精产品一二二线 | 日本一区二区更新不卡 | 日本xxxx色视频在线观看免费 | 亚洲国产午夜精品理论片 | 中文字幕无码av激情不卡 | 强奷人妻日本中文字幕 | 4hu四虎永久在线观看 | 狠狠综合久久久久综合网 | 国产精品第一国产精品 | 少妇久久久久久人妻无码 | 亚洲国产精品无码久久久久高潮 | 国产绳艺sm调教室论坛 | 午夜时刻免费入口 | 国产成人精品三级麻豆 | 成人aaa片一区国产精品 | 欧美性猛交内射兽交老熟妇 | 玩弄中年熟妇正在播放 | 亚洲中文字幕乱码av波多ji | 国产亚洲精品久久久久久久久动漫 | 久久综合狠狠综合久久综合88 | 国产免费久久精品国产传媒 | 亚洲成熟女人毛毛耸耸多 | 国产精品久久久午夜夜伦鲁鲁 | 动漫av网站免费观看 | 色欲人妻aaaaaaa无码 | 亚洲人亚洲人成电影网站色 | yw尤物av无码国产在线观看 | 九九热爱视频精品 | 图片小说视频一区二区 | 无遮挡啪啪摇乳动态图 | 娇妻被黑人粗大高潮白浆 | 国产成人无码区免费内射一片色欲 | 国产精品久久久久无码av色戒 | 亚洲综合精品香蕉久久网 | 女人被爽到呻吟gif动态图视看 | 国产九九九九九九九a片 | 又大又硬又黄的免费视频 | 亚洲精品一区二区三区在线观看 | 欧美freesex黑人又粗又大 | 亚洲a无码综合a国产av中文 | 欧美日韩一区二区综合 | 久久精品国产大片免费观看 | 奇米综合四色77777久久 东京无码熟妇人妻av在线网址 | 日韩人妻少妇一区二区三区 | 领导边摸边吃奶边做爽在线观看 | 丰满少妇人妻久久久久久 | 国内揄拍国内精品少妇国语 | 色婷婷综合中文久久一本 | 国产亚洲精品久久久久久久久动漫 | 玩弄中年熟妇正在播放 | 亚洲色欲色欲天天天www | 日韩人妻系列无码专区 | 国产sm调教视频在线观看 | 无码av最新清无码专区吞精 | 性欧美疯狂xxxxbbbb | √天堂中文官网8在线 | 99精品国产综合久久久久五月天 | 国产乱人偷精品人妻a片 | 玩弄少妇高潮ⅹxxxyw | 中国女人内谢69xxxxxa片 | 人妻少妇精品无码专区动漫 | 青青久在线视频免费观看 | 欧美乱妇无乱码大黄a片 | 无码人妻少妇伦在线电影 | 国产精品无码一区二区三区不卡 | 一本色道婷婷久久欧美 | 野外少妇愉情中文字幕 | 成人精品视频一区二区三区尤物 | 国产真人无遮挡作爱免费视频 | 波多野结衣av在线观看 | 久久精品人妻少妇一区二区三区 | 又大又硬又黄的免费视频 | 久久aⅴ免费观看 | 国产人妖乱国产精品人妖 | 少女韩国电视剧在线观看完整 | 色妞www精品免费视频 | 性欧美熟妇videofreesex | 久久久久久久人妻无码中文字幕爆 | 成 人 免费观看网站 | 亚洲欧洲日本综合aⅴ在线 | 国产无套内射久久久国产 | 亚洲人亚洲人成电影网站色 | 久久国产精品萌白酱免费 | 性史性农村dvd毛片 | 亚洲中文字幕在线观看 | 免费人成网站视频在线观看 | 377p欧洲日本亚洲大胆 | 亚洲第一网站男人都懂 | 亚洲欧美日韩综合久久久 | 久久久无码中文字幕久... | 亚洲一区二区三区香蕉 | 激情内射日本一区二区三区 | 欧美日本免费一区二区三区 | 色综合久久久无码中文字幕 | 欧美人与物videos另类 | 欧美大屁股xxxxhd黑色 | 日本欧美一区二区三区乱码 | 国产高清不卡无码视频 | 亚洲国产高清在线观看视频 | 国产偷国产偷精品高清尤物 | 国产69精品久久久久app下载 | 国产亚洲精品久久久久久大师 | 中文无码伦av中文字幕 | 精品一区二区三区波多野结衣 | 无人区乱码一区二区三区 | 久久亚洲国产成人精品性色 | 国产精品亚洲一区二区三区喷水 | 领导边摸边吃奶边做爽在线观看 | 国产人妻精品一区二区三区不卡 | 国产免费久久精品国产传媒 | 高潮毛片无遮挡高清免费 | 欧美人与牲动交xxxx | 亚洲日韩av一区二区三区中文 | 亚洲日韩av片在线观看 | 99久久婷婷国产综合精品青草免费 | 亚洲男人av天堂午夜在 | 男女下面进入的视频免费午夜 | 乌克兰少妇性做爰 | 蜜桃臀无码内射一区二区三区 | 男女爱爱好爽视频免费看 | 国产精品毛多多水多 | 亚洲国产成人av在线观看 | 亚洲成av人在线观看网址 | 精品国产av色一区二区深夜久久 | 久久久久久av无码免费看大片 | 亚洲另类伦春色综合小说 | 国产精华av午夜在线观看 | 秋霞成人午夜鲁丝一区二区三区 | 天天爽夜夜爽夜夜爽 | 蜜臀av无码人妻精品 | 国产精品毛片一区二区 | 午夜男女很黄的视频 | 荫蒂被男人添的好舒服爽免费视频 | 夜夜夜高潮夜夜爽夜夜爰爰 | 中文字幕+乱码+中文字幕一区 | 无码播放一区二区三区 | 暴力强奷在线播放无码 | 国产欧美熟妇另类久久久 | 亚洲 另类 在线 欧美 制服 | 免费男性肉肉影院 | v一区无码内射国产 | 亚洲中文字幕无码中文字在线 | 亚洲人成人无码网www国产 | 久久视频在线观看精品 | 亚洲综合伊人久久大杳蕉 | 性欧美熟妇videofreesex | 亚洲 欧美 激情 小说 另类 | 在线精品国产一区二区三区 | 亚洲精品久久久久avwww潮水 | 亚洲中文字幕成人无码 | 日韩欧美成人免费观看 | 亚洲一区二区观看播放 | 精品久久久久久亚洲精品 | 中文字幕精品av一区二区五区 | 99精品久久毛片a片 | 日韩精品无码免费一区二区三区 | 亚洲成av人影院在线观看 | 丁香花在线影院观看在线播放 | 国产女主播喷水视频在线观看 | a在线观看免费网站大全 | 国产精品久久久久9999小说 | 国产色在线 | 国产 | 国产免费久久久久久无码 | 亚洲中文字幕va福利 | 国产色精品久久人妻 | 最近免费中文字幕中文高清百度 | 草草网站影院白丝内射 | 亚洲精品久久久久久一区二区 | 亚洲国产欧美国产综合一区 | 久久亚洲日韩精品一区二区三区 | 天天摸天天透天天添 | av人摸人人人澡人人超碰下载 | 99精品国产综合久久久久五月天 | 国产精品国产自线拍免费软件 | 亚洲aⅴ无码成人网站国产app | 国产无套粉嫩白浆在线 | 婷婷丁香五月天综合东京热 | 国产超碰人人爽人人做人人添 | 亚洲日本在线电影 | 久久精品国产一区二区三区肥胖 | 97夜夜澡人人爽人人喊中国片 | 99久久久无码国产精品免费 | 九九久久精品国产免费看小说 | 欧美日韩久久久精品a片 | 人人妻人人澡人人爽欧美精品 | 精品偷自拍另类在线观看 | 国精品人妻无码一区二区三区蜜柚 | 久久精品人妻少妇一区二区三区 | 色婷婷香蕉在线一区二区 | 黑人巨大精品欧美一区二区 | 欧美日韩色另类综合 | 日韩无套无码精品 | 一个人看的www免费视频在线观看 | 一个人免费观看的www视频 | 国精品人妻无码一区二区三区蜜柚 | 最近免费中文字幕中文高清百度 | 少妇人妻大乳在线视频 | 十八禁真人啪啪免费网站 | 天堂久久天堂av色综合 | 99精品久久毛片a片 | 精品久久久中文字幕人妻 | 日日摸日日碰夜夜爽av | 免费国产成人高清在线观看网站 | а√资源新版在线天堂 | 亚洲精品成人av在线 | 性欧美熟妇videofreesex | 国产无遮挡又黄又爽又色 | 美女黄网站人色视频免费国产 | 精品无码成人片一区二区98 | 国产成人精品必看 | www国产亚洲精品久久网站 | 国产乱码精品一品二品 | 国内丰满熟女出轨videos | 中文字幕人妻丝袜二区 | 无套内谢的新婚少妇国语播放 | aⅴ亚洲 日韩 色 图网站 播放 | 无码国模国产在线观看 | 永久免费观看国产裸体美女 | 成人毛片一区二区 | 亚洲国产精品成人久久蜜臀 | 精品人妻人人做人人爽 | 人妻体内射精一区二区三四 | 国产激情无码一区二区app | 任你躁国产自任一区二区三区 | 成人欧美一区二区三区黑人 | 免费无码的av片在线观看 | 中文字幕中文有码在线 | 丰满少妇高潮惨叫视频 | 亚洲一区二区三区 | 国产女主播喷水视频在线观看 | 国产成人无码午夜视频在线观看 | 久久天天躁夜夜躁狠狠 | 丰满少妇弄高潮了www | 国产97人人超碰caoprom | 亚洲国产精品无码久久久久高潮 | 日日摸夜夜摸狠狠摸婷婷 | 激情综合激情五月俺也去 | 日韩无套无码精品 | 乱人伦中文视频在线观看 | 高清国产亚洲精品自在久久 | 欧美三级a做爰在线观看 | 一个人看的视频www在线 | 高潮毛片无遮挡高清免费视频 | 在线成人www免费观看视频 | 中文字幕无码乱人伦 | 麻豆精品国产精华精华液好用吗 | 亚洲国产精华液网站w | 久久久成人毛片无码 | 乱人伦中文视频在线观看 | 久久99精品久久久久久 | 国内少妇偷人精品视频 | 无码毛片视频一区二区本码 | 色婷婷综合中文久久一本 | 国产成人精品一区二区在线小狼 | 精品无码国产一区二区三区av | 亚洲爆乳精品无码一区二区三区 | 午夜精品一区二区三区在线观看 | 窝窝午夜理论片影院 | 久久久精品456亚洲影院 | 成人一在线视频日韩国产 | 纯爱无遮挡h肉动漫在线播放 | 国产精品欧美成人 | 欧美人与禽zoz0性伦交 | 丰满护士巨好爽好大乳 | 亚洲va欧美va天堂v国产综合 | 久久精品国产大片免费观看 | 日韩精品久久久肉伦网站 | 青青草原综合久久大伊人精品 | 中文字幕久久久久人妻 | 亚洲熟熟妇xxxx | 中文无码精品a∨在线观看不卡 | 国产精品爱久久久久久久 | 中文字幕av日韩精品一区二区 | 天干天干啦夜天干天2017 | 亚洲 高清 成人 动漫 | 中文字幕色婷婷在线视频 | 人妻无码久久精品人妻 | 欧美日韩一区二区综合 | 国产亚洲精品久久久ai换 | 国产97人人超碰caoprom | 国产又粗又硬又大爽黄老大爷视 | 妺妺窝人体色www婷婷 | 秋霞成人午夜鲁丝一区二区三区 | 少妇无码av无码专区在线观看 | 国产真实夫妇视频 | 亚洲啪av永久无码精品放毛片 | 中文久久乱码一区二区 | 少妇被粗大的猛进出69影院 | 色一情一乱一伦一区二区三欧美 | 国产精品久久福利网站 | 亚洲s码欧洲m码国产av | 日本成熟视频免费视频 | 又大又紧又粉嫩18p少妇 | 色综合久久久无码中文字幕 | 亚洲乱码日产精品bd | 亚洲另类伦春色综合小说 | 野外少妇愉情中文字幕 | 国产亚洲精品久久久ai换 | 日日鲁鲁鲁夜夜爽爽狠狠 | 亚洲国产精品毛片av不卡在线 | 成熟妇人a片免费看网站 | 久久精品丝袜高跟鞋 | 国产午夜无码精品免费看 | 日韩av无码一区二区三区 | 久久99热只有频精品8 | 夫妻免费无码v看片 | 少女韩国电视剧在线观看完整 | 久久精品女人天堂av免费观看 | 午夜理论片yy44880影院 | 国内少妇偷人精品视频免费 | 狠狠噜狠狠狠狠丁香五月 | 性生交大片免费看l | 国内精品人妻无码久久久影院 | 成人性做爰aaa片免费看不忠 | 美女毛片一区二区三区四区 | 成人综合网亚洲伊人 | 亚洲 高清 成人 动漫 | 无遮挡国产高潮视频免费观看 | 97久久国产亚洲精品超碰热 | 人妻插b视频一区二区三区 | 嫩b人妻精品一区二区三区 | 天天摸天天碰天天添 | 欧美一区二区三区视频在线观看 | 久久天天躁狠狠躁夜夜免费观看 | 色综合久久久久综合一本到桃花网 | 色噜噜亚洲男人的天堂 | 少妇激情av一区二区 | 福利一区二区三区视频在线观看 | 亚洲日韩一区二区 | 性欧美videos高清精品 | 久久久精品欧美一区二区免费 | 久久久久免费精品国产 | 国产97在线 | 亚洲 | 性欧美牲交xxxxx视频 | 人妻少妇精品无码专区动漫 | 日本精品高清一区二区 | 久久精品国产精品国产精品污 | 亚洲精品久久久久avwww潮水 | 日韩人妻少妇一区二区三区 | 亚洲人亚洲人成电影网站色 | 少妇人妻av毛片在线看 | 性欧美牲交xxxxx视频 | 精品无码av一区二区三区 | 亚洲国产午夜精品理论片 | 亚洲中文字幕无码中文字在线 | 欧美三级a做爰在线观看 | www国产亚洲精品久久久日本 | 久久亚洲国产成人精品性色 | 无码国产激情在线观看 | 真人与拘做受免费视频 | 久久99国产综合精品 | 国模大胆一区二区三区 | 久久人人97超碰a片精品 | 国产精品久久久 | 国产午夜视频在线观看 | 免费观看的无遮挡av | 亚洲 日韩 欧美 成人 在线观看 | 午夜精品久久久久久久久 | 麻豆国产丝袜白领秘书在线观看 | 1000部啪啪未满十八勿入下载 | 欧洲vodafone精品性 | 久久久久99精品国产片 | 中文字幕亚洲情99在线 | 亚洲欧美国产精品久久 | 成人精品视频一区二区三区尤物 | 97人妻精品一区二区三区 | 鲁一鲁av2019在线 | 久在线观看福利视频 | 日韩av无码一区二区三区不卡 | 国产人妻精品午夜福利免费 | a在线观看免费网站大全 | 无码任你躁久久久久久久 | 学生妹亚洲一区二区 |