【分布式定时任务】XXL-JOB_2.4.1部署与实战

news2024/11/15 17:45:02

部署

分布式任务调度平台XXL-JOBXXL-JOB是一个轻量级分布式任务调度平台,其核心设计目标是开发迅速、学习简单、轻量级、易扩展。现已开放源代码并接入多家公司线上产品线,开箱即用。icon-default.png?t=N7T8https://www.xuxueli.com/xxl-job/

一 数据库配置

  • 执行初始化数据库SQL

xxl-job/doc/db/tables_xxl_job.sql at 2.4.1 · xuxueli/xxl-job · GitHubA distributed task scheduling framework.(分布式任务调度平台XXL-JOB) - xxl-job/doc/db/tables_xxl_job.sql at 2.4.1 · xuxueli/xxl-jobicon-default.png?t=N7T8https://github.com/xuxueli/xxl-job/blob/2.4.1/doc/db/tables_xxl_job.sql

#
# XXL-JOB v2.4.1
# Copyright (c) 2015-present, xuxueli.

CREATE database if NOT EXISTS `xxl_job` default character set utf8mb4 collate utf8mb4_unicode_ci;
use `xxl_job`;

SET NAMES utf8mb4;

CREATE TABLE `xxl_job_info` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `job_group` int(11) NOT NULL COMMENT '执行器主键ID',
  `job_desc` varchar(255) NOT NULL,
  `add_time` datetime DEFAULT NULL,
  `update_time` datetime DEFAULT NULL,
  `author` varchar(64) DEFAULT NULL COMMENT '作者',
  `alarm_email` varchar(255) DEFAULT NULL COMMENT '报警邮件',
  `schedule_type` varchar(50) NOT NULL DEFAULT 'NONE' COMMENT '调度类型',
  `schedule_conf` varchar(128) DEFAULT NULL COMMENT '调度配置,值含义取决于调度类型',
  `misfire_strategy` varchar(50) NOT NULL DEFAULT 'DO_NOTHING' COMMENT '调度过期策略',
  `executor_route_strategy` varchar(50) DEFAULT NULL COMMENT '执行器路由策略',
  `executor_handler` varchar(255) DEFAULT NULL COMMENT '执行器任务handler',
  `executor_param` varchar(512) DEFAULT NULL COMMENT '执行器任务参数',
  `executor_block_strategy` varchar(50) DEFAULT NULL COMMENT '阻塞处理策略',
  `executor_timeout` int(11) NOT NULL DEFAULT '0' COMMENT '任务执行超时时间,单位秒',
  `executor_fail_retry_count` int(11) NOT NULL DEFAULT '0' COMMENT '失败重试次数',
  `glue_type` varchar(50) NOT NULL COMMENT 'GLUE类型',
  `glue_source` mediumtext COMMENT 'GLUE源代码',
  `glue_remark` varchar(128) DEFAULT NULL COMMENT 'GLUE备注',
  `glue_updatetime` datetime DEFAULT NULL COMMENT 'GLUE更新时间',
  `child_jobid` varchar(255) DEFAULT NULL COMMENT '子任务ID,多个逗号分隔',
  `trigger_status` tinyint(4) NOT NULL DEFAULT '0' COMMENT '调度状态:0-停止,1-运行',
  `trigger_last_time` bigint(13) NOT NULL DEFAULT '0' COMMENT '上次调度时间',
  `trigger_next_time` bigint(13) NOT NULL DEFAULT '0' COMMENT '下次调度时间',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

CREATE TABLE `xxl_job_log` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `job_group` int(11) NOT NULL COMMENT '执行器主键ID',
  `job_id` int(11) NOT NULL COMMENT '任务,主键ID',
  `executor_address` varchar(255) DEFAULT NULL COMMENT '执行器地址,本次执行的地址',
  `executor_handler` varchar(255) DEFAULT NULL COMMENT '执行器任务handler',
  `executor_param` varchar(512) DEFAULT NULL COMMENT '执行器任务参数',
  `executor_sharding_param` varchar(20) DEFAULT NULL COMMENT '执行器任务分片参数,格式如 1/2',
  `executor_fail_retry_count` int(11) NOT NULL DEFAULT '0' COMMENT '失败重试次数',
  `trigger_time` datetime DEFAULT NULL COMMENT '调度-时间',
  `trigger_code` int(11) NOT NULL COMMENT '调度-结果',
  `trigger_msg` text COMMENT '调度-日志',
  `handle_time` datetime DEFAULT NULL COMMENT '执行-时间',
  `handle_code` int(11) NOT NULL COMMENT '执行-状态',
  `handle_msg` text COMMENT '执行-日志',
  `alarm_status` tinyint(4) NOT NULL DEFAULT '0' COMMENT '告警状态:0-默认、1-无需告警、2-告警成功、3-告警失败',
  PRIMARY KEY (`id`),
  KEY `I_trigger_time` (`trigger_time`),
  KEY `I_handle_code` (`handle_code`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

CREATE TABLE `xxl_job_log_report` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `trigger_day` datetime DEFAULT NULL COMMENT '调度-时间',
  `running_count` int(11) NOT NULL DEFAULT '0' COMMENT '运行中-日志数量',
  `suc_count` int(11) NOT NULL DEFAULT '0' COMMENT '执行成功-日志数量',
  `fail_count` int(11) NOT NULL DEFAULT '0' COMMENT '执行失败-日志数量',
  `update_time` datetime DEFAULT NULL,
  PRIMARY KEY (`id`),
  UNIQUE KEY `i_trigger_day` (`trigger_day`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

CREATE TABLE `xxl_job_logglue` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `job_id` int(11) NOT NULL COMMENT '任务,主键ID',
  `glue_type` varchar(50) DEFAULT NULL COMMENT 'GLUE类型',
  `glue_source` mediumtext COMMENT 'GLUE源代码',
  `glue_remark` varchar(128) NOT NULL COMMENT 'GLUE备注',
  `add_time` datetime DEFAULT NULL,
  `update_time` datetime DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

CREATE TABLE `xxl_job_registry` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `registry_group` varchar(50) NOT NULL,
  `registry_key` varchar(255) NOT NULL,
  `registry_value` varchar(255) NOT NULL,
  `update_time` datetime DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `i_g_k_v` (`registry_group`,`registry_key`,`registry_value`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

CREATE TABLE `xxl_job_group` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `app_name` varchar(64) NOT NULL COMMENT '执行器AppName',
  `title` varchar(12) NOT NULL COMMENT '执行器名称',
  `address_type` tinyint(4) NOT NULL DEFAULT '0' COMMENT '执行器地址类型:0=自动注册、1=手动录入',
  `address_list` text COMMENT '执行器地址列表,多地址逗号分隔',
  `update_time` datetime DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

CREATE TABLE `xxl_job_user` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `username` varchar(50) NOT NULL COMMENT '账号',
  `password` varchar(50) NOT NULL COMMENT '密码',
  `role` tinyint(4) NOT NULL COMMENT '角色:0-普通用户、1-管理员',
  `permission` varchar(255) DEFAULT NULL COMMENT '权限:执行器ID列表,多个逗号分割',
  PRIMARY KEY (`id`),
  UNIQUE KEY `i_username` (`username`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

CREATE TABLE `xxl_job_lock` (
  `lock_name` varchar(50) NOT NULL COMMENT '锁名称',
  PRIMARY KEY (`lock_name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

INSERT INTO `xxl_job_group`(`id`, `app_name`, `title`, `address_type`, `address_list`, `update_time`) VALUES (1, 'xxl-job-executor-sample', '示例执行器', 0, NULL, '2018-11-03 22:21:31' );
INSERT INTO `xxl_job_info`(`id`, `job_group`, `job_desc`, `add_time`, `update_time`, `author`, `alarm_email`, `schedule_type`, `schedule_conf`, `misfire_strategy`, `executor_route_strategy`, `executor_handler`, `executor_param`, `executor_block_strategy`, `executor_timeout`, `executor_fail_retry_count`, `glue_type`, `glue_source`, `glue_remark`, `glue_updatetime`, `child_jobid`) VALUES (1, 1, '测试任务1', '2018-11-03 22:21:31', '2018-11-03 22:21:31', 'XXL', '', 'CRON', '0 0 0 * * ? *', 'DO_NOTHING', 'FIRST', 'demoJobHandler', '', 'SERIAL_EXECUTION', 0, 0, 'BEAN', '', 'GLUE代码初始化', '2018-11-03 22:21:31', '');
INSERT INTO `xxl_job_user`(`id`, `username`, `password`, `role`, `permission`) VALUES (1, 'admin', 'e10adc3949ba59abbe56e057f20f883e', 1, NULL);
INSERT INTO `xxl_job_lock` ( `lock_name`) VALUES ( 'schedule_lock');

commit;
  • 创建job_user用户作为数据库管理账号
CREATE USER 'job_user'@'%' IDENTIFIED BY 'xxljob0819';
GRANT ALL PRIVILEGES ON xxl_job.* TO 'job_user'@'%';
FLUSH PRIVILEGES;

二 DOCKER部署

# 拉取镜像
docker pull xuxueli/xxl-job-admin:2.4.1
# 启动
docker run -d \
-e PARAMS="--spring.datasource.url=jdbc:mysql://ip:3306/xxl_job?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&serverTimezone=Asia/Shanghai --spring.datasource.username=job_user --spring.datasource.password=xxljob0819" \
-p 7080:8080 \
-v /opt/docker/xxljob/tmp:/data/applogs \
--name xxl-job \
xuxueli/xxl-job-admin:2.4.1

三 验证

  • 访问:http://ip:port/xxl-job-admin/toLogin
  • 默认用户:admin/123456
  • 重置密码

代码

一 JAVA配置

1 依赖

<dependency>
    <groupId>com.xuxueli</groupId>
    <artifactId>xxl-job-core</artifactId>
    <version>2.4.1</version>
</dependency>

2 配置

### 调度中心部署根地址 [选填]:如调度中心集群部署存在多个地址则用逗号分隔。执行器将会使用该地址进行"执行器心跳注册"和"任务结果回调";为空则关闭自动注册;
xxl.job.admin.addresses=http://ip:port/xxl-job-admin
### 执行器通讯TOKEN [选填]:非空时启用;
xxl.job.accessToken=default_token
### 执行器AppName [选填]:执行器心跳注册分组依据;为空则关闭自动注册
xxl.job.executor.appname=${spring.application.name}
### 执行器注册 [选填]:优先使用该配置作为注册地址,为空时使用内嵌服务 ”IP:PORT“ 作为注册地址。从而更灵活的支持容器类型执行器动态IP和动态映射端口问题。
xxl.job.executor.address=
### 执行器IP [选填]:默认为空表示自动获取IP,多网卡时可手动设置指定IP,该IP不会绑定Host仅作为通讯实用;地址信息用于 "执行器注册" 和 "调度中心请求并触发任务";
xxl.job.executor.ip=
### 执行器端口号 [选填]:小于等于0则自动获取;默认端口为9999,单机部署多个执行器时,注意要配置不同执行器端口;
xxl.job.executor.port=8800
### 执行器运行日志文件存储磁盘路径 [选填] :需要对该路径拥有读写权限;为空则使用默认路径;
xxl.job.executor.logpath=/data/applogs/xxl-job/jobhandler
### 执行器日志文件保存天数 [选填] : 过期日志自动清理, 限制值大于等于3时生效; 否则, 如-1, 关闭自动清理功能;
xxl.job.executor.logretentiondays=30

3 自动装配

package com.xx.core.basic.xxljob.configurations;

import com.xx.core.basic.xxljob.properties.XxlJobProperty;
import com.xxl.job.core.executor.impl.XxlJobSpringExecutor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.autoconfigure.AutoConfiguration;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Bean;

/**
 * XXL-JOB自动装配配置类
 *
 * @author wuzijing
 * @since 2024-08-19
 */
@Slf4j
@AutoConfiguration
@EnableConfigurationProperties({
    XxlJobProperty.class
})
public class XxlJobAutoConfiguration {

    @Bean
    public XxlJobSpringExecutor xxlJobExecutor(XxlJobProperty xxlJobProperty) {
        log.info("xxl-job config init.");
        XxlJobSpringExecutor xxlJobSpringExecutor = new XxlJobSpringExecutor();
        xxlJobSpringExecutor.setAdminAddresses(xxlJobProperty.getAdmin().getAddresses());
        xxlJobSpringExecutor.setAppname(xxlJobProperty.getExecutor().getAppname());
        xxlJobSpringExecutor.setIp(xxlJobProperty.getExecutor().getIp());
        xxlJobSpringExecutor.setPort(xxlJobProperty.getExecutor().getPort());
        xxlJobSpringExecutor.setAccessToken(xxlJobProperty.getAccessToken());
        xxlJobSpringExecutor.setLogPath(xxlJobProperty.getExecutor().getLogpath());
        xxlJobSpringExecutor.setLogRetentionDays(xxlJobProperty.getExecutor().getLogretentiondays());
        return xxlJobSpringExecutor;
    }

}
package com.xx.core.basic.xxljob.properties;

import lombok.Getter;
import lombok.Setter;
import org.springframework.boot.context.properties.ConfigurationProperties;

/**
 * XXL-JOB配置参数类
 *
 * @author wuzijing
 * @since 2024-08-19
 */
@Getter
@Setter
@ConfigurationProperties(prefix = "xxl.job")
public class XxlJobProperty {

    /**
     * 执行器通讯TOKEN [选填]:非空时启用;
     */
    private String accessToken;

    /**
     * 调度中心配置
     */
    private JobAdminProperty admin;

    /**
     * 执行器配置
     */
    private JobExecutorProperty executor;

}
package com.xx.core.basic.xxljob.properties;

import lombok.Getter;
import lombok.Setter;

/**
 * XXL-JOB执行期配置参数类
 *
 * @author wuzijing
 * @since 2024-08-19
 */
@Getter
@Setter
public class JobExecutorProperty {

    /**
     * 执行器AppName [选填]:执行器心跳注册分组依据;为空则关闭自动注册
     */
    private String appname;

    /**
     * 执行器注册 [选填]:优先使用该配置作为注册地址,为空时使用内嵌服务 ”IP:PORT“ 作为注册地址。从而更灵活的支持容器类型执行器动态IP和动态映射端口问题。
     */
    private String address;

    /**
     * 执行器IP [选填]:默认为空表示自动获取IP,多网卡时可手动设置指定IP,该IP不会绑定Host仅作为通讯实用;地址信息用于 "执行器注册" 和 "调度中心请求并触发任务";
     */
    private String ip;

    /**
     * 执行器端口号 [选填]:小于等于0则自动获取;默认端口为9999,单机部署多个执行器时,注意要配置不同执行器端口;
     */
    private Integer port;

    /**
     * 执行器运行日志文件存储磁盘路径 [选填] :需要对该路径拥有读写权限;为空则使用默认路径;
     */
    private String logpath;

    /**
     * 执行器日志文件保存天数 [选填] : 过期日志自动清理, 限制值大于等于3时生效; 否则, 如-1, 关闭自动清理功能;
     */
    private Integer logretentiondays;

}
package com.xx.core.basic.xxljob.properties;

import lombok.Getter;
import lombok.Setter;

/**
 * XXL-JOB调度中心配置参数类
 *
 * @author wuzijing
 * @since 2024-08-19
 */
@Getter
@Setter
public class JobAdminProperty {

    /**
     * 调度中心部署根地址 [选填]:如调度中心集群部署存在多个地址则用逗号分隔。执行器将会使用该地址进行"执行器心跳注册"和"任务结果回调";为空则关闭自动注册;
     */
    private String addresses;

}

二 任务实现

1 代码

package com.xx.storage.module.task;

import cn.hutool.core.date.DatePattern;
import cn.hutool.core.date.LocalDateTimeUtil;
import cn.hutool.core.util.ObjectUtil;
import cn.hutool.core.util.StrUtil;
import com.baomidou.dynamic.datasource.annotation.DSTransactional;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.xx.core.basic.constant.IntegerConstant;
import com.xx.storage.module.bean.StrgConfigDO;
import com.xx.storage.module.bean.StrgFileDO;
import com.xx.storage.module.constants.RedissonLockKeyConstant;
import com.xx.storage.module.dao.StrgConfigDao;
import com.xx.storage.module.dao.StrgFileDao;
import com.xx.storage.module.utils.AmazonS3Util;
import com.xxl.job.core.context.XxlJobHelper;
import com.xxl.job.core.handler.annotation.XxlJob;
import jakarta.annotation.Resource;
import lombok.extern.slf4j.Slf4j;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.springframework.stereotype.Component;

import java.time.LocalDateTime;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
 * 存储文件信息表清理定时任务
 *
 * @author wuzijing
 * @since 2024-08-20
 */
@Slf4j
@Component
@DSTransactional
public class StrgFileCleanTask {

    @Resource
    private StrgConfigDao strgConfigDao;

    @Resource
    private StrgFileDao strgFileDao;

    @Resource
    private RedissonClient redissonClient;

    private record FileGroup(String endpointUrl, String bucketName, String keyId, String keySecret,
                             List<String> objectNameList) {
    }

    /**
     * 存储文件信息表清理定时任务
     * <p>
     * 入参设定:yyyy-MM-dd 清除指定日期(00:00:00)前的文件
     * 无参设定:清除一个月(00:00:00)前的文件
     */
    @XxlJob("strgFileCleanTask")
    public void strgFileCleanTask() {
        // 设置清除时间
        LocalDateTime clearTime;
        String param = XxlJobHelper.getJobParam();
        if (!StrUtil.isBlankOrUndefined(param)) {
            clearTime = LocalDateTimeUtil.beginOfDay(LocalDateTimeUtil.parse(param, DatePattern.NORM_DATE_PATTERN));
        } else {
            clearTime = LocalDateTimeUtil.beginOfDay(LocalDateTimeUtil.now().minusMonths(IntegerConstant.ONE));
        }
        if (ObjectUtil.isNull(clearTime)) {
            XxlJobHelper.handleFail("存储文件信息表清理定时任务执行失败,入参解析异常:param=" + param);
            return;
        }
        RLock rLock = redissonClient.getLock(RedissonLockKeyConstant.FILE_CLEAN_TASK);
        if (!rLock.tryLock()) {
            XxlJobHelper.handleFail("存储文件信息表清理定时任务执行失败,当前任务执行中");
            return;
        }
        try {
            // 获取对应配置
            LambdaQueryWrapper<StrgConfigDO> configWrapper = new LambdaQueryWrapper<>();
            configWrapper.eq(StrgConfigDO::getIsClear, IntegerConstant.ONE);
            List<StrgConfigDO> configDOList = strgConfigDao.selectList(configWrapper);
            if (ObjectUtil.isEmpty(configDOList)) {
                XxlJobHelper.handleSuccess(this.successResult(clearTime, IntegerConstant.ZERO));
                return;
            }
            // 获取待清理文件列表
            LambdaQueryWrapper<StrgFileDO> fileWrapper = new LambdaQueryWrapper<>();
            fileWrapper.in(StrgFileDO::getConfigCode, configDOList.stream().map(StrgConfigDO::getCode));
            fileWrapper.and(and -> {
                and.isNull(StrgFileDO::getBizGuid).or();
                and.eq(StrgFileDO::getBizGuid, StrUtil.EMPTY);
            });
            fileWrapper.le(StrgFileDO::getModifyTime, clearTime);
            List<StrgFileDO> fileDOList = strgFileDao.selectList(fileWrapper);
            if (ObjectUtil.isEmpty(fileDOList)) {
                XxlJobHelper.handleSuccess(this.successResult(clearTime, IntegerConstant.ZERO));
                return;
            }
            // 文件分组
            Map<String, FileGroup> fileGroupMap = new HashMap<>();
            for (StrgFileDO fileDO : fileDOList) {
                FileGroup fileGroup = fileGroupMap.get(fileDO.getEndpointUrl() + fileDO.getBucketName() + fileDO.getKeyId() + fileDO.getKeySecret());
                if (ObjectUtil.isNull(fileGroup)) {
                    fileGroup = new FileGroup(fileDO.getEndpointUrl(), fileDO.getBucketName(), fileDO.getKeyId(), fileDO.getKeySecret(), List.of(fileDO.getPath()));
                } else {
                    fileGroup.objectNameList().add(fileDO.getPath());
                    fileGroup = new FileGroup(fileDO.getEndpointUrl(), fileDO.getBucketName(), fileDO.getKeyId(), fileDO.getKeySecret(), fileGroup.objectNameList());
                }
                fileGroupMap.put(fileDO.getEndpointUrl() + fileDO.getBucketName() + fileDO.getKeyId() + fileDO.getKeySecret(), fileGroup);
            }
            // 开始清理文件
            fileGroupMap.forEach((key, fileGroup) -> {
                AmazonS3Util.deleteFile(fileGroup.endpointUrl(), fileGroup.bucketName(), fileGroup.keyId(), fileGroup.keySecret(), fileGroup.objectNameList());
            });
            XxlJobHelper.handleSuccess(this.successResult(clearTime, fileDOList.size()));
        } catch (Exception error) {
            log.error("存储文件信息表清理定时任务执行异常,错误信息:{}", error.getMessage());
            XxlJobHelper.handleFail("存储文件信息表清理定时任务执行异常,错误信息:" + error.getMessage());
        } finally {
            rLock.unlock();
        }
    }

    /**
     * 成功结果
     *
     * @param time 清理时间
     * @param num  清理成功数量
     * @return 成功结果消息
     */
    private String successResult(LocalDateTime time, Integer num) {
        return "存储文件信息表" + LocalDateTimeUtil.format(time, DatePattern.NORM_DATE_PATTERN) + "前数据清理定时任务执行成功,清理文件数:" + num;
    }

}

2 配置

(1)执行器配置

(2)任务配置

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2090177.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

flutter和原生Android以及IOS开发相比有什么优缺点?

Flutter 是 Google 开发的一个开源移动应用开发框架&#xff0c;它使用 Dart 语言编写。Flutter 的主要目标是使开发者能够从单一的代码库构建高性能、高保真的应用程序&#xff0c;这些应用程序可以在 iOS 和 Android 平台上运行&#xff0c;同时保持原生应用的感觉。 Flutte…

tomcat架构设计分析,核心组件详解

提示&#xff1a;tomcat整体架构分析&#xff0c;tomcat核心组件详解、tomcat请求全流程、tomcat设计模式分析。责任链模式设计、tomcat设计详解、tomcat调优的前置文档 文章目录 前言一、相关概念1、tomcat的概念2、web应用部署的3种方式 二、tomcat的整体架构1、tomcat架构图…

三星与海力士发力决战HBM4

在8月中旬&#xff0c;三星宣布正在加速推进下一代HBM&#xff08;High Bandwidth Memory&#xff09;的研发&#xff0c;目标是在今年年底前完成HBM4的设计定稿&#xff08;tape-out&#xff09;。而SK海力士则继续保持其竞争力&#xff0c;计划在10月份完成HBM4的设计定稿&am…

华为云征文|部署私有云和文档管理系统 Kodcloud

华为云征文&#xff5c;部署私有云和文档管理系统 Kodcloud 一、Flexus云服务器X实例介绍1.1 云服务器介绍1.2 应用场景1.3 对比普通ECS 二、Flexus云服务器X实例配置2.1 重置密码2.2 服务器连接2.3 安全组配置 三、部署 Kodcloud3.1 Jellyfin 介绍3.2 Docker 环境搭建3.3 Jell…

Linux内核6.12新特性:panic之后扫码显示故障信息

Linux 内核 6.12 版本即将引入一项有趣的功能——在内核Panic时显示一个可选的二维码。这一功能将允许用户通过扫描二维码直接访问内核Panic信息的日志&#xff0c;从而更容易地诊断问题所在。 这不是 Linux 第一次尝试使用二维码。早在2014年&#xff0c;就有过关于在内核Pani…

docker-compose容器及Dockerfile的使用

软件资源&#xff1a; 链接: https://pan.baidu.com/s/1iYMP7p7v_HUMc9vakIQfbQ?pwd6fes 提取码: 6fes docker-compose容器 部署nmt---nginxcomposetomcat项目 使用基础的docker指令来创建镜像&#xff0c;实现项目的发布。 Dockerfile--->docker-compose 背景&#x…

[知识技巧] iPhone 15 卡在恢复模式修复方法

如果您发现您的iPhone 15 卡在恢复模式下&#xff0c;这可能是一次相当紧张的经历&#xff0c;尤其是当它阻止您使用手机时。请放心&#xff0c;这是许多 iPhone 用户面临的共同挑战。幸运的是&#xff0c;有一些行之有效的方法可以解决此问题并将您的 iPhone 恢复到正常功能。…

启动盘如何复原

条件&#xff1a; 本教程适合Windows系统 通过Diskpart CMD启动U盘还原U盘在Windows中还有CMD命令可以使用 先“WindowsR”呼出运行窗口&#xff0c;在命令行中搜索"Diskpart"。点击回车输入“list disk”并且回车输入“select disk X”(X为替换可启动u盘的磁盘号)…

Vite项目启动服务器时报错 Error: Cannot find module @rollup/rollup-win32-x64-msvc.

前言&#xff1a; 网上找了很多方法&#xff0c;尝试后都不行。在某篇文章评论区种找到了一种方法解决了&#xff0c;谨以此篇文章记录解决问题的方案&#xff0c;方便日后使用。 解决方法&#xff1a;github链接 Cannot find module rollup/rollup-win32-x64-msvc (rollup 4.…

idea配置连接数据库的操作方法(适配不同版本)

文章目录 一、IDEA找不到database图标的解决方法二、链接数据库 一、IDEA找不到database图标的解决方法 首先很多小伙伴说我左右侧边栏都找了找不到数据库图标 解决办法&#xff1a; 第一种&#xff1a;选择 View --> Tool Windows --> Database 如果你的idea版本在这里…

昂瑞微IPO:华为、小米看重的国产射频芯片,“腾飞”了吗?

从一些科技大厂的投资动向中&#xff0c;往往能够发现一些新的技术力量正在酝酿。 据悉&#xff0c;近日&#xff0c;小米、华为投资的射频芯片供应商——北京昂瑞微电子技术股份有限公司&#xff08;以下简称“昂瑞微”&#xff09;在北京证监局办理辅导备案登记&#xff0c;…

docker简单私有仓库的搭建

示例&#xff1a; 【搭建简单的Registry仓库】 1. 下载 Registry 镜像 [rootdocker ~]# docker pull registry #可以查看开放的端口&#xff0c;需要把端口暴露出来 [rootdocker ~]# docker history registry:latest [rootdocker ~]# docker run -d -p 5000:5000 --restartal…

若依微服务ruoyi-auth在knife4j中不显示问题解决

关于若依微服务ruoyi-auth在knife4j中不显示问题解决 解决办法 一、添加swagger依赖文件 在ruoyi-auth模块下的pom.xml文件中添加ruoyi-common-swagger依赖 <!-- RuoYi Common Swagger --><dependency><groupId>com.ruoy

DS1202ZE - 零碎使用记录

文章目录 DS1202ZE - 零碎使用记录概述笔记测量通道频率是使用频率测量还是内建的频率计测量&#xff1f;如何让Ultra Sigma检测到DS1202ZE?如何消除过多的测量结果?通过UltraScope对示波器屏幕进行截图END DS1202ZE - 零碎使用记录 概述 手头在使用DS1202ZE, 将零碎的操作…

javascript网页设计案例详细讲解

​ 大家好&#xff0c;我是程序员小羊&#xff01; 前言&#xff1a; 我们将以深入讲解一个基于JavaScript的网页设计案例。这个案例不仅涵盖了基础的DOM操作、事件处理和动画效果&#xff0c;还将涉及更高级的主题&#xff0c;如异步数据加载、表单验证、模块化设计和响应式布…

【动态规划】两个数组 / 字符串的dp问题(子序列、子数组问题、匹配问题、字符串问题)

文章目录 前言算法题1.最长公共子序列2.不相交的线3.不同的子序列4.通配符匹配5.正则表达式匹配6.交错字符串7.两个字符串的最小ASCII删除和8.最长重复子数组 前言 两个数组或字符串的动态规划问题通常涉及到比较和匹配元素。以下是两个常见的例子&#xff1a; 最长公共子序列…

「OC」CAlayer——巧用动画实现一个丝滑的折叠cell

「OC」CAlayer——巧用动画实现一个丝滑的折叠cell 前言 在这个暑假集训后的时间&#xff0c;都在家里做着学习笔记的整理&#xff0c;深入学习了CALayer的相关知识&#xff0c;掌握了第三方库Masonry自动布局的用法&#xff0c;以及学习了MVC的相关内容&#xff0c;正好组内…

在Postgresql中计算工单的对应的GPS轨迹距离

一、概述 在某个App开发中&#xff0c;要求记录用户的日常轨迹&#xff0c;在用户巡逻设备的时&#xff0c;将记录的轨迹点当做该设备巡逻时候的轨迹。 由于业务逻辑上没有明确的指示人员巡逻工单-GPS位置之间的关系&#xff0c;所以通过时间关系进行轨迹划定。 二、创建测试表…

Core ML

本文翻译整理自&#xff1a;Core ML : https://developer.apple.com/cn/documentation/coreml/ 文章目录 一、概览二、获取 Core ML 模型三、将 Core ML 模型集成到你的 App 中1、将模型添加到您的Xcode项目2、在代码中创建模型3、获取要传递给模型的输入值4、使用模型进行预测…

vue2踩坑记录:el-select如何绑定对象

页面上的下拉框&#xff1a; 选中人员之后&#xff0c;需要使用人员的其它信息做后续操作。所以不能只绑定用户Id&#xff0c;需要绑定整个item(用户对象)&#xff0c;这样每次change事件所获取到的newValue是整个对象&#xff0c;而且v-model中的变量指向的也是整个对象&#…