1.xxl-job基本介绍
1.官方文档
https://www.xuxueli.com/xxl-job/
2.gitee
https://gitee.com/xuxueli0323/xxl-job
2.本地集成xxl-job
1.下载源码包
https://gitee.com/xuxueli0323/xxl-job/tree/6effc8b98f0fd5b5af3a7b6a8995bdcf30de69fc/
2.导入到项目中
1.作为模块导入
data:image/s3,"s3://crabby-images/63d50/63d502c57940150ecc9c4a7613f9c5574542d8d8" alt=""
2.使其成为maven项目
data:image/s3,"s3://crabby-images/e06e5/e06e5c97f90981dbbff7c2387deb19a04499437e" alt=""
3.创建数据库
1.打开xxl-job-admin的配置文件,查看要创建的数据库
data:image/s3,"s3://crabby-images/6094e/6094e354c126c6103e9d48050ca7ba482fc945f8" alt=""
2.打开tables_xxl_job.sql
data:image/s3,"s3://crabby-images/5b7f5/5b7f5ec08e4ec5a9ad6c1e6bcaf9cc15e26bab45" alt=""
3.执行sql脚本,创建数据库表
data:image/s3,"s3://crabby-images/499e9/499e9da1c008f63638bbda812e40faf14f670999" alt=""
data:image/s3,"s3://crabby-images/4a379/4a3797091dc61284d215308de7f0fa40fe2994c6" alt=""
4.修改application.properties的数据库ip和端口为自己的
4.启动访问
1.启动
data:image/s3,"s3://crabby-images/b98c2/b98c271fd990f8ea7bbe6d755e184e33bf41f8bc" alt=""
2.本地访问
http://localhost:8080/xxl-job-admin/toLogin
账号密码 admin 123456
data:image/s3,"s3://crabby-images/e3a05/e3a05ab67d1ff8a70e50757e40d58acd3b18eab5" alt=""
data:image/s3,"s3://crabby-images/00d45/00d45129899b5982b4cbbb44986500c61ac8f33d" alt=""
5.sun-club-subject配置xxl-job
1. application.yml
yaml
# xxl-job配置
xxl:
job:
admin:
addresses: http://127.0.0.1:8080/xxl-job-admin # xxl-job-admin地址
accessToken: default_token
executor:
appname: sun-club-subjcet # 执行器名称
address:
ip: 127.0.0.1 # 执行器ip
port: 9999 # 执行器端口
logpath: /data/applogs/xxl-job/jobhandler
logretentiondays: 30
2.创建跟配置文件执行器相同名字的执行器 sun-club-subjcet
data:image/s3,"s3://crabby-images/fd2d9/fd2d9e853f06b46e70498a450f77898993c7f4a9" alt=""
3.sun-club-domain引入依赖
xml
<!-- xxl-job -->
<dependency>
<groupId>com.xuxueli</groupId>
<artifactId>xxl-job-core</artifactId>
<version>2.3.1</version>
</dependency>
<!-- 这里因为是domain层,没有引入springboot-context,所以才引入了一下 -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.3.27</version>
</dependency>
4.将配置类粘贴到sun-club-domain的配置包中
data:image/s3,"s3://crabby-images/ddf51/ddf51fcdbea6aa3913aeab961fc64eaea9726c02" alt=""
5.重启xxl-job和sun-club-subject,发现节点可以注册成功!
data:image/s3,"s3://crabby-images/3cd0e/3cd0e9260072297bc0540b5f56ed7d784db66a63" alt=""
data:image/s3,"s3://crabby-images/9adb8/9adb8cb14435e535da09538dc5a3fd4879ead06f" alt=""
6.新增定时任务
1.任务管理->新增
data:image/s3,"s3://crabby-images/c5c76/c5c76ec116cd34c527e25403f9b2b6f3db0c3f17" alt=""
2.配置
data:image/s3,"s3://crabby-images/5ab70/5ab70f1036ce3ad9ffb70f2de5847f515b2ee75f" alt=""
3.com/sunxiansheng/subject/domain/job/SyncLikedJob.java
1.代码
java
package com.sunxiansheng.subject.domain.job;
import com.xxl.job.core.context.XxlJobHelper;
import com.xxl.job.core.handler.annotation.XxlJob;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;
/**
* 同步点赞数任务
*/
@Component
@Slf4j
public class SyncLikedJob {
/**
* 同步点赞数任务
*/
@XxlJob("syncLikedJobHandler") // 这个注解是xxl-job的注解,用于标记这个方法是一个定时任务,必须跟执行器的运行模式一致
public void demoJobHandler() throws Exception {
XxlJobHelper.log("syncLikedJobHandler start");
try {
// todo
log.info("123456");
} catch (Exception e) {
XxlJobHelper.log("syncLikedJobHandler error" + e.getMessage());
}
}
}
2.注意:@XxlJob中的内容必须跟新增定时任务的Handler一致
data:image/s3,"s3://crabby-images/81f40/81f404c86c4452d1f2f49fd151e7134ed763488e" alt=""
3.重启subject模块
4.执行一次,查看控制台
data:image/s3,"s3://crabby-images/2470e/2470ebefb0d42166d065dc40cddbae12c9210698" alt=""
data:image/s3,"s3://crabby-images/a896f/a896f1e462eb6643e628e56aea00aa598dab927c" alt=""
5.启动定时任务,查看控制台
data:image/s3,"s3://crabby-images/cbe76/cbe76d9c46746eb7a9a4450d601b7d5a7626cbeb" alt=""
data:image/s3,"s3://crabby-images/a21b5/a21b503d0bcf6ec5af917bb365f8741727f0b02e" alt=""
3.docker安装xxl-job
1.选定服务器
data:image/s3,"s3://crabby-images/43041/43041818c7bdeb8010be5761e7398f0f818c715b" alt=""
2.拉取镜像
1.应该是镜像的问题
data:image/s3,"s3://crabby-images/c6e9a/c6e9a5180467b10741b5aa58368f8d52c0583fad" alt=""
2.配置一下镜像
1.首先找到daemon.json的位置
sh
sudo find / -name "daemon.json" 2>/dev/null
data:image/s3,"s3://crabby-images/bd26d/bd26d3ff17714d7833f270937d818a355ee51748" alt=""
2.将其删除
sh
rm -rf /etc/docker/daemon.json
3.编辑镜像源文件
sh
vim /etc/docker/daemon.json
4.填写代理镜像仓库地址,将以下内容粘贴到daemon.json中
sh
{
"registry-mirrors": [
"https://9cpn8tt6.mirror.aliyuncs.com",
"https://registry.docker-cn.com",
"https://hub-mirror.c.163.com",
"https://mirror.ccs.tencentyun.com",
"https://reg-mirror.qiniu.com",
"https://mirror.baidubce.com",
"https://docker.mirrors.ustc.edu.cn",
"https://mirrors.huaweicloud.com"
]
}
5.重启docker引擎
sh
systemctl restart docker && systemctl status docker
3.重新拉取镜像
sh
docker pull xuxueli/xxl-job-admin:2.4.0
data:image/s3,"s3://crabby-images/fc39e/fc39e154c16bdcdf90e92a64203bebea66d494e0" alt=""
3.启动容器
-p 8088:8088
:将容器的8088端口映射到主机的8088端口。-
- 左侧的
8088
是主机上的端口。 - 右侧的
8088
是容器内的端口。
- 左侧的
-v /tool/xxl-job/logs:/data/applogs
:将主机目录/tool/xxl-job/logs
挂载到容器的/data/applogs
目录。-v /tool/xxl-job/application.properties:/xxl-job/xxl-job-admin/src/main/resources/application.properties
:将主机上的配置文件挂载到容器内指定路径。-e PARAMS="..."
:使用环境变量PARAMS
来传递启动参数。--server.port=8088
:设置服务的端口为8088。--spring.datasource.url=jdbc:mysql://ip:3306/xxl_job?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&serverTimezone=Asia/Shanghai
:设置数据库连接URL。--spring.datasource.username=root
:设置数据库用户名为root
。--spring.datasource.password=88888
:设置数据库密码为88888
。--name xxl-job-admin
:为容器指定一个名字xxl-job-admin
。xuxueli/xxl-job-admin:2.4.0
:指定使用的镜像和标签(版本)。这里使用的是xuxueli/xxl-job-admin
镜像的2.4.0
版本。
sh
docker run -d \
-p 8088:8088 \
-v /tool/xxl-job/logs:/data/applogs \
-v /tool/xxl-job/application.properties:/xxl-job/xxl-job-admin/src/main/resources/application.properties \
-e PARAMS="--server.port=8088 \
--spring.datasource.url=jdbc:mysql://ip:3306/xxl_job?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&serverTimezone=Asia/Shanghai \
--spring.datasource.username=root \
--spring.datasource.password=88888 \
--name xxl-job-admin \
xuxueli/xxl-job-admin:2.4.0
4.开启端口8088
1.宝塔开启
sh
systemctl start firewalld && firewall-cmd --permanent --add-port=8088/tcp && firewall-cmd --reload && firewall-cmd --query-port=8088/tcp
data:image/s3,"s3://crabby-images/49119/49119d22208d1fe556de8c7f26f43b2aa7fa3e23" alt=""
2.腾讯云开启
data:image/s3,"s3://crabby-images/794ea/794ea5261c220a133d8cbefbc6b8f639bfd92b99" alt=""
5.测试访问
http://ip:8088/xxl-job-admin/toLogin 账号密码 admin 123456
data:image/s3,"s3://crabby-images/9d245/9d245b412aead0dc4ca863fe97167128de559330" alt=""
data:image/s3,"s3://crabby-images/780b3/780b30457e70b492012e3fa8259fee9a55d89c37" alt=""
6.修改sun-club-subject的application.yml
1.修改addresses为xxl-job的ip端口
2.执行器的ip和端口为sun-club-subject服务部署的地方,记得要开启9999端口的防火墙!!!!!!
4.redis的hash扫描同步点赞数据
1.sun-club-domain
1.RedisUtil.java 根据key来将每一个hashKey和hashValue转换为Map类型
java
/**
* Redis中的hash类型,根据key来将每一个hashKey和hashValue转换为Map类型
* @param key
* @return
*/
public Map<Object, Object> getHashAndDelete(String key) {
Map<Object, Object> map = new HashMap<>();
// 扫描hash,指定每一个Entry的类型,这里返回的就是Map的游标,可以进行遍历
Cursor<Map.Entry<Object, Object>> cursor = redisTemplate.opsForHash().scan(key, ScanOptions.NONE);
// 遍历每一条数据,放到map中
while (cursor.hasNext()) {
Map.Entry<Object, Object> next = cursor.next();
Object hashKey = next.getKey();
Object hashValue = next.getValue();
map.put(hashKey, hashValue);
// 每遍历一条就删除
redisTemplate.opsForHash().delete(key, hashKey);
}
return map;
}
2.SyncLikedJob.java
java
package com.sunxiansheng.subject.domain.job;
import com.sunxiansheng.subject.domain.service.SubjectLikedDomainService;
import com.xxl.job.core.context.XxlJobHelper;
import com.xxl.job.core.handler.annotation.XxlJob;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;
import javax.annotation.Resource;
/**
* 同步点赞数任务
*/
@Component
@Slf4j
public class SyncLikedJob {
@Resource
private SubjectLikedDomainService subjectLikedDomainService;
/**
* 同步点赞数任务
*/
@XxlJob("syncLikedJobHandler") // 这个注解是xxl-job的注解,用于标记这个方法是一个定时任务,必须跟执行器的运行模式一致
public void demoJobHandler() throws Exception {
XxlJobHelper.log("syncLikedJobHandler start");
try {
// 同步点赞数据到db
subjectLikedDomainService.syncLiked();
} catch (Exception e) {
XxlJobHelper.log("syncLikedJobHandler error" + e.getMessage());
}
}
}
3.SubjectLikedDomainServiceImpl.java 将从redis中拿到的点赞数据同步到数据库
java
@Override
public void syncLiked() {
Map<Object, Object> subjectLiked = redisUtil.getHashAndDelete(SUBJECT_LIKE_KEY);
// 打日志,输出map
if (log.isInfoEnabled()) {
log.info("syncLiked:{}", JSON.toJSONString(subjectLiked));
}
// 判空
if (subjectLiked.isEmpty()) {
return;
}
// 声明一个list
List<SubjectLiked> subjectLikedList = new ArrayList<>();
// 将map中的数据转换为实体类并添加到list中
subjectLiked.forEach((k, v) -> {
String[] keys = k.toString().split(":");
SubjectLiked subjectLiked1 = new SubjectLiked();
subjectLiked1.setSubjectId(Long.valueOf(keys[0]));
subjectLiked1.setLikeUserId(keys[1]);
subjectLiked1.setStatus(Integer.valueOf(v.toString()));
subjectLikedList.add(subjectLiked1);
});
// 调用service的批量插入方法
subjectLikedService.batchInsert(subjectLikedList);
}
5.内网穿透natapp测试定时任务
1.内网穿透使用指南
https://natapp.cn/article/natapp_newbie
2.注册
data:image/s3,"s3://crabby-images/481fa/481fab0f6af1a22b0a89af7096fb57f78d96aef8" alt=""
3.配置
1.购买免费隧道
data:image/s3,"s3://crabby-images/85903/859037c3846f5e1de00a5957327c85e2b8df1392" alt=""
data:image/s3,"s3://crabby-images/74432/744329a891a4f2730e203cbbe5dca4563843e04c" alt=""
2.进入隧道
data:image/s3,"s3://crabby-images/f7892/f78921b6593796f3f59e5e6ca9063fed2f9ce690" alt=""
3.配置端口为定时任务执行器的端口9999
data:image/s3,"s3://crabby-images/fc022/fc02222503df861cd355d60faa5260d6bcb661dd" alt=""
4.下载客户端
data:image/s3,"s3://crabby-images/36f1e/36f1e8ef28d4637f4aa135885dcdbd2365dcaa80" alt=""
5.启动
1.进入客户端exe文件的cmd
data:image/s3,"s3://crabby-images/321a1/321a1f2754865e5cfc0e472eea716f4acf4b2d4c" alt=""
2.找到自己的authtoken,输入命令启动
sh
start natapp -authtoken 05a35b22673e2788
data:image/s3,"s3://crabby-images/cef54/cef544e56fae0e7ea03e03d558371dc35ebb5950" alt=""
6.application-test.yml 配置执行器的ip和端口
data:image/s3,"s3://crabby-images/6bc54/6bc5467f9e4c7d527bc79377c866a476bb039faa" alt=""
yaml
# xxl-job配置
xxl:
job:
executor:
appname: sun-club-subjcet # 执行器名称
address:
ip: xh7set.natappfree.cc # 执行器ip,该模块的部署ip,这里是内网穿透的ip
port: 9999 # 执行器端口,记得要开防火墙
logpath: /data/applogs/xxl-job/jobhandler
logretentiondays: 30
7.进行同步测试
1.以测试环境启动项目
data:image/s3,"s3://crabby-images/4adb4/4adb48a1e82881d60c23f821e5f310babe8e6a99" alt=""
2.查看redis的点赞数据
data:image/s3,"s3://crabby-images/67cdb/67cdbd850fbd06c8a70c74d9cb41c416fce566cf" alt=""
3.进入xxl-job,执行一次任务
http://ip:8088/xxl-job-admin/toLogin 账号密码 admin 123456
4.发现连接被拒绝了
1.调度日志
data:image/s3,"s3://crabby-images/05e41/05e414d024b7701395a1f2c794b09fdb07f55cac" alt=""
2.关闭本机防火墙
data:image/s3,"s3://crabby-images/ad4d2/ad4d28e919f8b5007f8a57541befdf283f52a4fc" alt=""
3.发现还是不行,原因是内网穿透需要手动录入内网穿透的ip,则直接会映射ip+端口,如果自动映射还会在后面加9999端口,这样就不对了
data:image/s3,"s3://crabby-images/9165f/9165f563f1709bdc2eeda999486136bfa9aec399" alt=""
4.内网穿透再说明
1.内网穿透在配置ip和端口的时候正常配置
2.但是如果访问的话,直接访问内网穿透的ip即可,这个就包含了端口,不要再加端口!!!
5.定时任务执行成功!
data:image/s3,"s3://crabby-images/a0984/a098478da89ab473b393522af258bc6f29d073a5" alt=""