IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 大数据 -> flink sql 知其所以然(二)| 自定义 redis 数据维表(附源码) -> 正文阅读

[大数据]flink sql 知其所以然(二)| 自定义 redis 数据维表(附源码)

图片

感谢您的关注 ?+ ?点赞 + 再看,对博主的肯定,会督促博主持续的输出更多的优质实战内容!!!

1.序篇-本文结构

  1. 背景篇-为啥需要 redis 维表

  2. 目标篇-做 redis 维表的预期效果是什么

  3. 难点剖析篇-此框架建设的难点、目前有哪些实现

  4. 维表实现篇-维表实现的过程

  5. 总结与展望篇

本文主要介绍了 flink sql redis 维表的实现过程。

如果想在本地测试下:

大数据羊说

大数据羊说

用数据提升美好事物发生的概率~

29篇原创内容

公众号

  1. 在公众号后台回复flink sql 知其所以然(二)| sql 自定义 redis 数据维表获取源码(源码基于 1.13.1 实现)

  2. 在你的本地安装并打开 redis-server,然后使用 redis-cli 执行命令 set a "{\"score\":3,\"name\":\"namehhh\",\"name1\":\"namehhh112\"}"

  3. 执行源码包中的 flink.examples.sql._03.source_sink.RedisLookupTest 测试类,就可以在 console 中看到结果。

如果想直接在集群环境使用:

  1. 命令行执行 mvn package -DskipTests=true 打包

  2. 将生成的包 flink-examples-0.0.1-SNAPSHOT.jar 引入 flink lib 中即可,无需其它设置。

2.背景篇-为啥需要 redis 维表

2.1.啥是维表?事实表?

Dimension Table 概念多出现于数据仓库里面,维表与事实表相互对应。

给两个场景来看看:

比如需要统计分性别的 DAU:

  1. 客户端上报的日志中(事实表)只有设备 id,只用这个事实表是没法统计出分性别的 DAU 的。

  2. 这时候就需要一张带有设备 id、性别映射的表(这就是维表)来提供性别数据。

  3. 然后使用事实表去 join 这张维表去获取到每一个设备 id 对应的性别,然后就可以统计出分性别的 DAU。相当于一个扩充维度的操作。

https://blog.csdn.net/weixin_47482194/article/details/105855116?spm=1001.2014.3001.5501

比如目前想要统计整体销售额:

  1. 目前已有 “销售统计表”,是一个事实表,其中没有具体销售品项的金额。

  2. “商品价格表” 可以用于提供具体销售品项的金额,这就是销售统计的一个维度表。

事实数据和维度数据的识别必须依据具体的主题问题而定。“事实表” 用来存储事实的度量及指向各个维的外键值。维表用来保存该维的元数据。

参考:https://blog.csdn.net/lindan1984/article/details/96566626

2.2.为啥需要 redis 维表?

目前在实时计算的场景中,熟悉 datastream 的同学大多数都使用过 mysql\Hbase\redis 作为维表引擎存储一些维度数据,然后在 datastream api 中调用 mysql\Hbase\redis 客户端去获取到维度数据进行维度扩充。

而 redis 作为 flink 实时场景中最常用的高速维表引擎,官方是没有提供 flink sql api 的 redis 维表 connector 的。如下图,基于 1.13 版本。

https://ci.apache.org/projects/flink/flink-docs-release-1.13/docs/connectors/table/overview/

图片

1

阿里云 flink 是提供了这个能力的。但是这个需要使用阿里云的产品才能使用。有钱人可以直接上。

https://www.alibabacloud.com/help/zh/faq-detail/122722.htm?spm=a2c63.q38357.a3.7.a1227a53TBMuSY

图片

2

因此本文在介绍怎样自定义一个 sql 数据维表的同时,实现一个 sql redis 来给大家使用。

3.目标篇-做 redis 维表预期效果是什么

redis 作为维表在 datastream 中的最常用的数据结构就是 kv、hmap 两种。本文实现主要实现 kv 结构,map 结构大家可以拿到源码之后进行自定义实现。也就多加几行代码就完事了。

预期效果就如阿里云的 flink redis:

下面是我在本地跑的结果,先看看 redis 中存储的数据,只有这一条数据,是 json 字符串:

图片

9

下面是预期 flink sql:

CREATE?TABLE?dimTable?(
????name?STRING,
????name1?STRING,
????score?BIGINT??--?redis?中存储数据的?schema
)?WITH?(
????'connector'?=?'redis',?--?指定?connector?是?redis?类型的
????'hostname'?=?'127.0.0.1',?--?redis?server?ip
????'port'?=?'6379',?--?redis?server?端口
????'format'?=?'json'?--?指定?format?解析格式
????'lookup.cache.max-rows'?=?'500',?--?guava?local?cache?最大条目
????'lookup.cache.ttl'?=?'3600',?--?guava?local?cache?ttl
????'lookup.max-retries'?=?'1'?--?redis?命令执行失败后重复次数
)

SELECT?o.f0,?o.f1,?c.name,?c.name1,?c.score
FROM?leftTable?AS?o
–?维表?join
LEFT?JOIN?dimTable?FOR?SYSTEM_TIME?AS?OF?o.proctime?AS?c
ON?o.f0?=?c.name


![图片](https://mmbiz.qpic.cn/mmbiz_png/DODKOLcDkD3FwnfpdoT3tYiavOybWbUhyJsnvq1yLxneUvaJlBPcH3Ij3nloYQI0jwV4CyuHorhYxMmJj3r8wicg/640?wx_fmt=png)

10

结果如下,后面三列就对应到 `c.name, c.name1, c.score`:

+I[a,?b,?namehhh,?namehhh112,?3]
+I[a,?b,?namehhh,?namehhh112,?3]
+I[a,?b,?namehhh,?namehhh112,?3]
+I[a,?b,?namehhh,?namehhh112,?3]
+I[a,?b,?namehhh,?namehhh112,?3]
+I[a,?b,?namehhh,?namehhh112,?3]
+I[a,?b,?namehhh,?namehhh112,?3]
+I[a,?b,?namehhh,?namehhh112,?3]
+I[a,?b,?namehhh,?namehhh112,?3]


4.难点剖析篇-目前有哪些实现
===============

目前可以从网上搜到的实现、以及可以参考的实现有以下两个:

1.  https://github.com/jeff-zou/flink-connector-redis。但是其没有实现 flink sql redis 维表,只实现了 sink 表,并且使用起来有比较多的限制,包括需要在建表时就指定 key-column,value-column 等,其实博主觉得没必要指定这些字段,这些都可以动态调整。其实现是对 apache-bahir-flink https://github.com/apache/bahir-flink 的二次开发,但与 bahir 原生实现有割裂感,因为这个项目几乎重新实现了一遍,接口也和 bahir 不同。
    
2.  阿里云实现 https://www.alibabacloud.com/help/zh/faq-detail/122722.htm?spm=a2c63.q38357.a3.7.a1227a53TBMuSY。可以参考的只有用法和配置等。但是有些配置项也属于阿里自定义的。
    

因此博主在实现时,就定了一个基调。

1.  复用 connector:复用 bahir 提供的 redis connnector
    
2.  复用 format:复用 flink 目前的 format 机制,目前这个上述两个实现都没有做到
    
3.  简洁性:实现 kv 结构。hget 封装一部分
    
4.  维表 local cache:为避免高频率访问 redis,维表加了 local cache 作为缓存
    

5.维表实现篇-维表实现的过程
===============

在实现 redis 维表之前,不得不谈谈 flink 维表加载和使用机制。

5.1.flink 维表原理
--------------

其实上节已经详细描述了 flink sql 对于 source\\sink 的加载机制,维表属于 source 的中的 lookup 表,在具体 flink 程序运行的过程之中可以简单的理解为一个 map,在 map 中调用 redis-client 接口访问 redis 进行扩充维度的过程。

1.  通过 SPI 机制加载所有的 source\\sink\\format 工厂 `Factory`
    
2.  过滤出 DynamicTableSourceFactory + connector 标识的 source 工厂类
    
3.  通过 source 工厂类创建出对应的 source
    

![图片](https://mmbiz.qpic.cn/mmbiz_png/DODKOLcDkD3FwnfpdoT3tYiavOybWbUhy1AwTziag1zgBnK7lJ2XCboempiasYelwAa9oOObibUicG6963qTk6YwiaLA/640?wx_fmt=png)

7

![图片](https://mmbiz.qpic.cn/mmbiz_png/DODKOLcDkD3FwnfpdoT3tYiavOybWbUhySsExgjp1elzmpTsxNdl1c9VA4XmXLhSjcyS7moUpStcwIEDRFWwjGA/640?wx_fmt=png)

5

如图 source 和 sink 是通过 `FactoryUtil.createTableSource` 和 `FactoryUtil.createTableSink` 创建的

![图片](https://mmbiz.qpic.cn/mmbiz_png/DODKOLcDkD3FwnfpdoT3tYiavOybWbUhyCkicedCjibleQviatyYjAUXQHaiaHVSRTbQtLJLI8MTKnGib9wjicGjib36fg/640?wx_fmt=png)

4

所有通过 SPI 的 source\\sink\\formt 插件都继承自 `Factory`。

整体创建 source 方法的调用链如下图。

![图片](https://mmbiz.qpic.cn/mmbiz_png/DODKOLcDkD3FwnfpdoT3tYiavOybWbUhyKibbkulHfp8V4dVib3eCA8GVUicnS1wlYicmKKfNSMUHYweMsLiasib3G96A/640?wx_fmt=png)

6

5.2.flink 维表实现方案
----------------

先看下博主的最终实现。

总重要的三个实现类:

1.  `RedisDynamicTableFactory`
    
2.  `RedisDynamicTableSource`
    
3.  `RedisRowDataLookupFunction`
    

![图片](https://mmbiz.qpic.cn/mmbiz_png/DODKOLcDkD3FwnfpdoT3tYiavOybWbUhyicwXOlGymwkKTy70xf96Bn6ictdzScwDVWmjicvBNV5vHV4uWwGWPKSeQ/640?wx_fmt=png)

8

具体流程:

1.  定义 SPI 的工厂类 `RedisDynamicTableFactory implements DynamicTableSourceFactory`,并且在 resource\\META-INF 下创建 SPI 的插件文件
    
2.  实现 factoryIdentifier 标识 `redis`
    
3.  实现 `RedisDynamicTableFactory#createDynamicTableSource` 来创建对应的 source `RedisDynamicTableSource`
    
4.  定义 `RedisDynamicTableSource implements LookupTableSource`
    
5.  实现 `RedisDynamicTableFactory#getLookupRuntimeProvider` 方法,创建具体的维表 UDF ?`TableFunction<T>`,定义为 `RedisRowDataLookupFunction`
    
6.  实现 `RedisRowDataLookupFunction` 的 eval 方法,这个方法就是用于访问 redis 扩充维度的。
    

介绍完流程,进入具体实现方案细节:

`RedisDynamicTableFactory` 主要创建 source 的逻辑:

public?class?RedisDynamicTableFactory?implements?DynamicTableSourceFactory?{
????…

@Override
????public?String?factoryIdentifier()?{
????????//?标识?redis
????????return?“redis”;
????}

@Override
????public?DynamicTableSource?createDynamicTableSource(Context?context)?{

//?either?implement?your?custom?validation?logic?here?…
????????//?or?use?the?provided?helper?utility
????????final?FactoryUtil.TableFactoryHelper?helper?=?FactoryUtil.createTableFactoryHelper(this,?context);

//?discover?a?suitable?decoding?format
????????//?format?实现
????????final?DecodingFormat<DeserializationSchema>?decodingFormat?=?helper.discoverDecodingFormat(
????????????????DeserializationFormatFactory.class,
????????????????FactoryUtil.FORMAT);

//?validate?all?options
????????//?所有?option?配置的校验,比如?cache?类参数
????????helper.validate();

//?get?the?validated?options
????????final?ReadableConfig?options?=?helper.getOptions();

final?RedisLookupOptions?redisLookupOptions?=?RedisOptions.getRedisLookupOptions(options);

TableSchema?schema?=?context.getCatalogTable().getSchema();

//?创建?RedisDynamicTableSource
????????return?new?RedisDynamicTableSource(
????????????????schema.toPhysicalRowDataType()
????????????????,?decodingFormat
????????????????,?redisLookupOptions);
????}
}


resources\\META-INF 文件:

![图片](https://mmbiz.qpic.cn/mmbiz_png/DODKOLcDkD3FwnfpdoT3tYiavOybWbUhyTdd0LSh9X1VqqgqMdm8GAvENGVFma4XrzQUcYosu0JcSeHDCvX0GCQ/640?wx_fmt=png)

13

`RedisDynamicTableSource` 主要创建 table udf 的逻辑:

public?class?RedisDynamicTableSource?implements?LookupTableSource?{
????…

@Override
????public?LookupRuntimeProvider?getLookupRuntimeProvider(LookupContext?context)?{

//?初始化?redis?客户端配置
????????FlinkJedisConfigBase?flinkJedisConfigBase?=?new?FlinkJedisPoolConfig.Builder()
????????????????.setHost(this.redisLookupOptions.getHostname())
????????????????.setPort(this.redisLookupOptions.getPort())
????????????????.build();

//?redis?key,value?序列化器
????????LookupRedisMapper?lookupRedisMapper?=?new?LookupRedisMapper(
????????????????this.createDeserialization(context,?this.decodingFormat,?createValueFormatProjection(this.physicalDataType)));

//?创建?table?udf
????????return?TableFunctionProvider.of(new?RedisRowDataLookupFunction(
????????????????flinkJedisConfigBase
????????????????,?lookupRedisMapper
????????????????,?this.redisLookupOptions));
????}
}


`RedisRowDataLookupFunction` table udf 执行维表关联的主要流程:

public?class?RedisRowDataLookupFunction?extends?TableFunction?{
????…

/**
?????*?具体?redis?执行方法
?????*/
????public?void?eval(Object…?objects)?throws?IOException?{

for?(int?retry?=?0;?retry?<=?maxRetryTimes;?retry++)?{
????????????try?{
????????????????//?fetch?result
????????????????this.evaler.accept(objects);
????????????????break;
????????????}?catch?(Exception?e)?{
????????????????LOG.error(String.format(“HBase?lookup?error,?retry?times?=?%d”,?retry),?e);
????????????????if?(retry?>=?maxRetryTimes)?{
????????????????????throw?new?RuntimeException(“Execution?of?Redis?lookup?failed.”,?e);
????????????????}
????????????????try?{
????????????????????Thread.sleep(1000?*?retry);
????????????????}?catch?(InterruptedException?e1)?{
????????????????????throw?new?RuntimeException(e1);
????????????????}
????????????}
????????}
????}

@Override
????public?void?open(FunctionContext?context)?{
????????LOG.info(“start?open?…”);

//?redis?命令执行器,初始化?redis?链接
????????try?{
????????????this.redisCommandsContainer?=
????????????????????RedisCommandsContainerBuilder
????????????????????????????.build(this.flinkJedisConfigBase);
????????????this.redisCommandsContainer.open();
????????}?catch?(Exception?e)?{
????????????LOG.error("Redis?has?not?been?properly?initialized:?",?e);
????????????throw?new?RuntimeException(e);
????????}

//?初始化?local?cache
????????this.cache?=?cacheMaxSize?<=?0?||?cacheExpireMs?<=?0???null?:?CacheBuilder.newBuilder()
????????????????.recordStats()
????????????????.expireAfterWrite(cacheExpireMs,?TimeUnit.MILLISECONDS)
????????????????.maximumSize(cacheMaxSize)
????????????????.build();

if?(cache?!=?null)?{
????????????context.getMetricGroup()
????????????????????.gauge(“lookupCacheHitRate”,?(Gauge)?()?->?cache.stats().hitRate());

this.evaler?=?in?->?{
????????????????RowData?cacheRowData?=?cache.getIfPresent(in);
????????????????if?(cacheRowData?!=?null)?{
????????????????????collect(cacheRowData);
????????????????}?else?{
????????????????????//?fetch?result
????????????????????byte[]?key?=?lookupRedisMapper.serialize(in);

byte[]?value?=?null;

switch?(redisCommand)?{
????????????????????????case?GET:
????????????????????????????value?=?this.redisCommandsContainer.get(key);
????????????????????????????break;
????????????????????????case?HGET:
????????????????????????????value?=?this.redisCommandsContainer.hget(key,?this.additionalKey.getBytes());
????????????????????????????break;
????????????????????????default:
????????????????????????????throw?new?IllegalArgumentException("Cannot?process?such?data?type:?"?+?redisCommand);
????????????????????}

RowData?rowData?=?this.lookupRedisMapper.deserialize(value);

collect(rowData);

cache.put(key,?rowData);
????????????????}
????????????};

}
????????…
????}
}


### 5.2.1.复用 bahir connector

如图是 bahir redis connector 的实现。

![图片](https://mmbiz.qpic.cn/mmbiz_png/DODKOLcDkD3FwnfpdoT3tYiavOybWbUhy4lJc2bcX7SEGASjS2WnAQU4vjyibbNzq1haM8T3xaNc8hkVVFzyTrtQ/640?wx_fmt=png)

11

博主在实现过程中将能复用的都尽力复用。如图是最终实现目录。

![图片](https://mmbiz.qpic.cn/mmbiz_png/DODKOLcDkD3FwnfpdoT3tYiavOybWbUhyflFMuJUYrcFsf7xDGKH2Y9R0pYOeIlTRicCEUXO5h4ulrV7e4ichsh2Q/640?wx_fmt=png)

12

可以看到目录结构是与 bahir redis connector 一致的。

其中 `redis 客户端及其配置` 是直接复用了 bahir redis 的。由于 bahir redis 基本都是 sink 实现,某些实现没法继承复用,所以这里我单独开辟了目录,`redis 命令执行器` 和 `redis 命令定义器`,但是也基本和 bahir 一致。如果你想要在生产环境中进行使用,可以直接将两部分代码合并,成本很低。

### 5.2.2.复用 format

博主直接复用了 flink 本身自带的 format 机制来作为维表反序列化机制。参考 HBase connector 实现将 cache 命中率添加到 metric 中。

public?class?RedisDynamicTableFactory?implements?DynamicTableSourceFactory?{
????…
????@Override
????public?DynamicTableSource?createDynamicTableSource(Context?context)?{
????????…

//?discover?a?suitable?decoding?format
????????//?复用?format?实现
????????final?DecodingFormat<DeserializationSchema>?decodingFormat?=?helper.discoverDecodingFormat(
????????????????DeserializationFormatFactory.class,
????????????????FactoryUtil.FORMAT);
????????…
????}
}


format 同样也是 SPI 机制加载。

源码公众号后台回复**flink sql 知其所以然(二)| sql 自定义 redis 数据维表**获取。

5.2.3.维表 local cache
--------------------

local cache 在初始化时可以指定 cache 大小,缓存时长等。

this.evaler?=?in?->?{
????RowData?cacheRowData?=?cache.getIfPresent(in);
????if?(cacheRowData?!=?null)?{
????????collect(cacheRowData);
????}?else?{
????????//?fetch?result
????????byte[]?key?=?lookupRedisMapper.serialize(in);

byte[]?value?=?null;

switch?(redisCommand)?{
????????????case?GET:
????????????????value?=?this.redisCommandsContainer.get(key);
????????????????break;
????????????case?HGET:
????????????????value?=?this.redisCommandsContainer.hget(key,?this.additionalKey.getBytes());
????????????????break;
????????????default:
????????????????throw?new?IllegalArgumentException("Cannot?process?such?data?type:?"?+?redisCommand);
????????}

RowData?rowData?=?this.lookupRedisMapper.deserialize(value);

collect(rowData);

cache.put(key,?rowData);
????}
};


6.总结与展望篇
========

6.1.总结
------

本文主要是针对 flink sql redis 维表进行了扩展以及实现,并且复用 bahir redis connector 的配置,具有良好的扩展性。如果你正好需要这么一个 connector,直接公众号后台回复**flink sql 知其所以然(二)| sql 自定义 redis 数据维表**获取源码吧。

6.2.展望
------

当然上述只是 redis 维表一个基础的实现,用于生产环境还有很多方面可以去扩展的。

1.  jedis cluster 的扩展:目前 bahir datastream 中已经实现了,可以直接参考,扩展起来非常简单
    
2.  aync lookup 维表的扩展:目前 hbase lookup 表已经实现了,可以直接参考实现
    
3.  异常 AOP,alert 等
    
      
    
      
    

![大数据羊说](http://mmbiz.qpic.cn/mmbiz_png/DODKOLcDkD0pbiahcsQHBKXHStQJiaNZ5bQIluAfFYHZe8KMY61OJbG4ghBfOibrKDPKQ2rmMR44cggr3N0o22nEQ/0?wx_fmt=png)

**大数据羊说**

用数据提升美好事物发生的概率~

29篇原创内容

公众号

往期推荐

[

flink sql 知其所以然(一)| source\\sink 原理



](http://mp.weixin.qq.com/s?__biz=MzkxNjA1MzM5OQ==&mid=2247488486&idx=1&sn=b9bdb56e44631145c8cc6354a093e7c0&chksm=c1549f1ef623160834e3c5661c155ec421699fc18c57f2c63ba14d33bab1d37c5930fdce016b&scene=21#wechat_redirect)

[

揭秘字节跳动埋点数据实时动态处理引擎(附源码)



](http://mp.weixin.qq.com/s?__biz=MzkxNjA1MzM5OQ==&mid=2247488435&idx=1&sn=5d89a0d24603c08af4be342462409230&chksm=c1549f4bf623165d977426d13a0bdbe821ec8738744d2274613a7ad92dec0256d090aea4b815&scene=21#wechat_redirect)

[

字节火山大数据引擎牛逼!!!



](http://mp.weixin.qq.com/s?__biz=MzkxNjA1MzM5OQ==&mid=2247487920&idx=1&sn=efdf259512e2a49606d240b50171ac09&chksm=c1549d48f623145ef91f5a25c8fdbd16b8f1086fadf738b3edeeaba85cc191d0f09ab2318604&scene=21#wechat_redirect)

[

实战 | flink sql 与微博热搜的碰撞!!!



](http://mp.weixin.qq.com/s?__biz=MzkxNjA1MzM5OQ==&mid=2247487817&idx=1&sn=4c0192b0f495f39eaf9ea5a6f837844c&chksm=c1549db1f62314a771c6d1e6a17689b5b0826173fff372d5466f26e7e24f3a764e00ef0e622f&scene=21#wechat_redirect)

[

实时数仓不保障时效还玩个毛?



](http://mp.weixin.qq.com/s?__biz=MzkxNjA1MzM5OQ==&mid=2247487642&idx=1&sn=48547ee68b197f5d38a14e9af5c01798&chksm=c1549c62f6231574c940d0858feb9ff4eba884d7c4696a8a7ec77ae023b6e09cc8232680b788&scene=21#wechat_redirect)

  

**更多 Flink 实时大数据分析相关技术博文,视频。后台回复?“flink”?获取。**  

点个赞+在看,感谢您的肯定?👇

  大数据 最新文章
实现Kafka至少消费一次
亚马逊云科技:还在苦于ETL?Zero ETL的时代
初探MapReduce
【SpringBoot框架篇】32.基于注解+redis实现
Elasticsearch:如何减少 Elasticsearch 集
Go redis操作
Redis面试题
专题五 Redis高并发场景
基于GBase8s和Calcite的多数据源查询
Redis——底层数据结构原理
上一篇文章      下一篇文章      查看所有文章
加:2021-08-23 16:45:39  更:2021-08-23 16:46:25 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/18 19:06:39-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码