NBlog Java定时任务-备份MySQL数据
NBlog部署维护流程记录(持续更新):https://blog.csdn.net/qq_43349112/article/details/136129806
为了避免服务器被攻击,给博客添加了一个MySQL数据备份功能。
此功能是配合博客写的,有些方法直接用的已有的,不会再详细展示代码。
备份大致功能步骤如下:
- 使用mysqldump备份数据(完成)
- 对备份文件进行压缩(完成)
- 压缩文件上传到OSS(完成)
- 文件清理(完成)
- 邮件通知(TODO)
- 适应化改造(TODO)
详细步骤见下文

0.备份任务主逻辑
目前暂时使用定时任务触发,以下仅为核心代码。
/*** 定时任务,每周一凌晨四点,备份MySQL的数据* 备份逻辑:* 1.mysql数据备份到文件* 2.备份文件压缩* 3.压缩文件上传到OSS* 4.残留文件清理* 5.备份结果的邮件通知 //TODO* 6.适应化改造,改成类似NBlog中的定时任务 //TODO*/@Scheduled(cron = "0 0 4 * * 1")public void backUpMySQLData() {checkDir(backupDir);String dateFormat = simpleDateFormat.format(new Date());String fileName = String.format("cblog-%s.sql", dateFormat);String compressedFileName = fileName + ".zip";String dataPath = backupDir + File.separator + fileName;String compressedFilePath = backupDir + File.separator + compressedFileName;try {log.debug("mysql备份开始");// 1.mysql数据备份backupData(dataPath);// 2.文件压缩FileUtils.compressFile(dataPath, compressedFilePath);// 3.上传到OSSString uploadLink = UploadUtils.upload(compressedFilePath);log.info("备份文件({})已上传至OSS({})", compressedFilePath, uploadLink);// 4.清除残留文件FileUtils.delFileByPath(dataPath, compressedFilePath);} catch (IOException e) {log.error("mysql数据备份失败");log.error(e.getMessage());}}
1.mysqldump备份
通过Runtime.getRuntime().exec(xxx)执行备份命令,保存到指定路径下。
由于我的MySQL是docker部署,因此使用了docker exec命令。
命令的执行结果会比较长,日志级别建议低一些,下面用的debug级别。
需要注意exec(cmds)的参数格式,写错的话命令不会执行并且不报错,排查了半个下午。
/*** MySQL数据备份* @param dataPath 备份文件的保存路径* @throws IOException*/private void backupData(String dataPath) throws IOException {long start = System.currentTimeMillis();String cmd = String.format("docker exec mysql mysqldump -u%s -p%s cblog > %s", dataSourceProperties.getUsername(), dataSourceProperties.getPassword(), dataPath);String[] cmds = {"sh", "-c", cmd};log.debug("欲执行命令:{}", cmd);try (InputStream inputStream = Runtime.getRuntime().exec(cmds).getInputStream();InputStreamReader inputStreamReader = new InputStreamReader(inputStream, StandardCharsets.UTF_8);BufferedReader bufferedReader = new BufferedReader(inputStreamReader);) {String line = bufferedReader.readLine();while (line != null) {log.debug(line);line = bufferedReader.readLine();}}long end = System.currentTimeMillis();log.info("mysql备份命令执行成功,耗时:{}ms", end - start);}
2.备份文件压缩
压缩成zip格式,核心代码如下:
* 压缩文件到指定路径* @param oriFilePath 要压缩的原文件路径* @param compressedFilePath 压缩后的文件存放路径* @throws IOException IO异常,不在catch模块捕捉,交给调用方自行处理*/public static void compressFile(String oriFilePath, String compressedFilePath) throws IOException {File file = new File(oriFilePath);File zipFile = new File(compressedFilePath);try (FileInputStream fileInputStream = new FileInputStream(file);ZipOutputStream zipOutputStream = new ZipOutputStream(Files.newOutputStream(zipFile.toPath()))){zipOutputStream.putNextEntry(new ZipEntry(file.getName()));int temp = 0;while ((temp = fileInputStream.read()) != -1) {zipOutputStream.write(temp);}}log.info("文件压缩完成");}
3.上传至OSS
核心代码如下
public String upload(String filepath) throws IOException {File file = new File(filepath);String uploadName = aliyunProperties.getBackupPath() + "/" + file.getName();PutObjectRequest putObjectRequest = new PutObjectRequest(aliyunProperties.getBucketName(), uploadName, file);return uploadByOSS(putObjectRequest, uploadName);}private String uploadByOSS(PutObjectRequest putObjectRequest, String uploadName) throws IOException {try {ossClient.putObject(putObjectRequest);return String.format("https://%s.%s/%s", aliyunProperties.getBucketName(), aliyunProperties.getEndpoint(), uploadName);} catch (Exception e) {throw new RuntimeException("阿里云OSS上传失败");}}
4.清除残留文件
上传后,备份文件和压缩文件已经无用,删除掉即可:
public static void delFileByPath(String... paths) {if (paths == null) {return;}for (String path : paths) {File file = new File(path);del(file);}}private static void del(File file) {String filePath = file.getAbsolutePath();file.delete();log.info("{}文件已清除", filePath);}
5.邮件通知结果(TODO)
6.适应化改造(TODO)
NBlog定时任务可以在前端配置,自由修改触发时间,并且可以直接触发。
目前的暂时写死了,下周改造下。
X、测试
确定定时任务触发后,OSS能看到文件即可

相关文章:
NBlog Java定时任务-备份MySQL数据
NBlog部署维护流程记录(持续更新):https://blog.csdn.net/qq_43349112/article/details/136129806 为了避免服务器被攻击,给博客添加了一个MySQL数据备份功能。 此功能是配合博客写的,有些方法直接用的已有的…...
微信小程序项目实战遇到的问题
我们以学生成绩平台来作为例子。这是我们想得到的效果。 以下是完整代码: index.js // index.js Page({//页面的初始数据data: {hello: 欢迎进入微信小程序的编程世界,score: 80,userArray: [{name: 张三,score: [66, 77, 86, 70, 90]},{name: 李四,score: [88, 7…...
网络原理(3)——TCP协议
目录 一、连接管理 二、三次握手 1、何为三次握手? 2、三次握手有何意义? 三、四次挥手 三次握手和四次挥手的相似之处和不同之处 (1)相似之处 (2)不同之处 四、TCP的状态 建立连接: 断开…...
nginx多级代理配置获取客户端真实ip
流量路径 #mermaid-svg-NX785p8k6RVBngHY {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-NX785p8k6RVBngHY .error-icon{fill:#552222;}#mermaid-svg-NX785p8k6RVBngHY .error-text{fill:#552222;stroke:#552222;}#…...
Django框架的全面指南:从入门到高级【第128篇—Django框架】
👽发现宝藏 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。【点击进入巨牛的人工智能学习网站】。 Django框架的全面指南:从入门到高级 Django是一个高效、功能强大的Python Web框…...
C++类和对象基础
目录 类的认识 访问限定符:public(公有),protected(保护),private(私有)。 类的两种定义方式: 类的实例化: 封装: 类的对象大小的计算: 类成员函数的this指针: C语言是面向过程的语言&am…...
消息队列常见的两种消费模式
一、点对点模式 点对点模式:生产者发送消息到消息队列,消费者从消息队列中接收、处理消息,消息被消费后,就不在消息队列中了。每个消息只能由一个消费者接收和处理。如果有多个消费者监听同一个队列,消息将被发送到其…...
php的伪协议详解
在 PHP 中,伪协议(pseudo-protocols)是一种特殊的语法,用于访问各种资源,如文件、网络、输入/输出流等。伪协议实际上并不是真正的协议,而是一种简便的语法,用于访问不同的资源类型。 以下是一…...
【研发日记】Matlab/Simulink技能解锁(四)——在Simulink Debugger窗口调试
文章目录 前言 Block断点 分解Block步进 Watch Data Value 分析和应用 总结 前言 见《【研发日记】Matlab/Simulink技能解锁(一)——在Simulink编辑窗口Debug》 见《【研发日记】Matlab/Simulink技能解锁(二)——在Function编辑窗口Debug》 见《【研发日记】Matlab/Simul…...
沪深主板打板胜率统计
统计了20100101以来的数据,以中信日K为数据来源。 计算方法: 选出每只股票 (收盘价-开盘价)/开盘价 >0.098的日期,然后往后取3天数据,如果3天内有一天能涨超0.2元,则认为打板成功。 总共打板: 52239次 胜: 43784次…...
Python中的列表推导式(List Comprehension)
Python中的列表推导式(List Comprehension)是一种强大且简洁的语法结构,用于快速创建列表。它通过一行代码就能完成原本需要多行代码才能实现的循环迭代与列表添加操作。列表推导式在Python中非常常用,它使得代码更加简洁、易读和…...
MusicHiFi: Fast High-Fidelity Stereo Vocoding
MusicHiFi: Fast High-Fidelity Stereo Vocoding 相关链接:arxiv github 关键字:音乐生成、高保真立体声、立体声编解码器、生成对抗网络、频带扩展 摘要 MusicHiFi是一种高效的高保真立体声编解码器,它通过将低分辨率的mel频谱图转换为音频…...
完美解决 RabbitMQ可视化界面Overview不显示折线图和队列不显示Messages
问题场景: 今天使用docker部署了一个RabbitMQ,浏览器打开15672可视化页面发送消息后不显示Overview中的折线图,还有队列中的Messages,因为我要看队列中的消息数量。 解决方案: 进入容器内部 docker exec -it 容器id…...
matlab 混沌系统李雅普洛夫指数谱相图分岔图和庞加莱界面
1、内容简介 略 65-可以交流、咨询、答疑 2、内容说明 matlab 混沌系统李雅普洛夫指数谱相图分岔图和庞加莱界面 混沌系统李雅普洛夫指数谱相图分岔图和庞加莱界面 李雅普洛夫指数谱、相图、分岔图、庞加莱界面 3、仿真分析 略 4、参考论文 略...
Linux-docker安装数据库mysql
1、拉去mysql镜像: docker pull mysql2、创建容器挂载路径 mkdir -p /usr/local/jiuxiang/mysql/data # 数据存储位置 mkdir -p /usr/local/jiuxiang/mysql/logs # 日志存储位置 mkdir -p /usr/local/jiuxiang/mysql/conf # 配置文件3、启动容器 docker run -…...
网工内推 | 七险一金,上市公司招信息安全工程师,大牛带队
01 启明星辰信息技术集团股份有限公司 招聘岗位:数据安全服务工程师 职责描述: 1、负责数据安全服务项目的管理,统筹组织并协调资源落实项目交付实施; 3、负责数据安全风险评估、数据分类分级、数据安全管理制度、数据安全体系规划等数据安…...
04.组件的组成和组件间通信
一、scoped解决样式冲突 1.默认情况: 写在组件中的样式会 全局生效 → 因此很容易造成多个组件之间的样式冲突问题。 全局样式: 默认组件中的样式会作用到全局,任何一个组件中都会受到此样式的影响 局部样式: 可以给组件加上scoped 属性,可以让样式只…...
【Sql Server】通过Sql语句批量处理数据,使用变量且遍历数据进行逻辑处理
欢迎来到《小5讲堂》,大家好,我是全栈小5。 这是《Sql Server》系列文章,每篇文章将以博主理解的角度展开讲解, 特别是针对知识点的概念进行叙说,大部分文章将会对这些概念进行实际例子验证,以此达到加深对…...
MySQL中group_concat()用法
函数用法见链接处 https://www.cnblogs.com/mcj123/articles/17189384.html 使用过程问题:group_concat()拼接后的字符串长度默认限制为1024位字节,超长会被自动截取。 解决: 修改group_concat限制长度。 1.数据库直接通过sql修改 查询限制长…...
栈队列数组试题(四)——数组和特殊矩阵
01.对特殊矩阵采用压缩存储的主要目的是( D ). A.表达变得简单 B.对矩阵元素的存取变得简单 C.去掉矩阵中的多余元素 D.减少不必要的存储空间解析:特殊矩阵中含有很多相同元素…...
2026年降AI后重新检测还是偏高怎么处理:多轮降AI完整攻略
2026年降AI后重新检测还是偏高怎么处理:多轮降AI完整攻略 从AI率73%到6%,我花了不到一个晚上。降AI后还是高完整经历记录。 核心工具:嘎嘎降AI(www.aigcleaner.com),4.8元,达标率99.26%。踩坑…...
你的J-Link-OB驱动装对了吗?从驱动安装到MDK5/Keil配置的完整避坑流程
J-Link-OB驱动安装与MDK5配置全流程避坑指南 最近在调试STM32项目时,发现不少开发者卡在了J-Link-OB驱动安装和MDK5配置这个看似简单却暗藏玄机的环节。我自己也曾经因为一个驱动签名问题折腾了大半天,今天就把这些实战经验整理成完整的避坑手册。 1. 驱…...
别再死磕微积分了!用Python的SymPy库5分钟搞定拉普拉斯变换解微分方程
用SymPy解放双手:5分钟自动化求解微分方程的工程实践 微分方程是工程和物理学中的常客,从电路分析到机械振动,它无处不在。传统解法需要记忆变换公式、手工计算代数方程、处理部分分式分解——这些步骤不仅耗时,还容易在符号运算…...
手把手教你用Simulink搭建Buck变换器:从元器件选型到波形分析
手把手教你用Simulink搭建Buck变换器:从元器件选型到波形分析 在电力电子领域,Buck变换器作为最基础的DC-DC降压拓扑,几乎出现在所有电源设计工程师的入门课程中。但很多初学者在理论学习后,面对实际仿真建模时仍会感到无从下手—…...
别再乱用shutdown了!Java线程池优雅关闭的3种正确姿势(附Spring Boot实战代码)
Java线程池优雅关闭实战指南:从原理到Spring Boot最佳实践 当你在凌晨三点被生产环境告警惊醒,发现服务因为线程池关闭不当导致数据丢失时,那种头皮发麻的感觉我太熟悉了。去年我们电商大促期间,就曾因为一个简单的shutdownNow()调…...
新概念英语第二册10_Not for jazz
Lesson 10: Not for jazzKey words and expressions jazz 爵士乐musical 音乐的instrument 乐器clavichord 古钢琴 chord 弦 belong 属于damage 损坏key 琴键string 弦allow 允许touch 触摸 customary adj. /ˈ…...
天赐范式第16天:【硬核反骨】哥本哈根沉默:REM睡眠是大脑在50维相空间的“超决定论”搜索(附Python源码)
摘要:梦境不是随机的噪声,而是意识在混沌边缘的精确计算。本文基于 Kuramoto 高维耦合振子模型,利用纯 Python (NumPy) 模拟了快速动眼期(REM)的神经动力学。实验发现:系统在 李雅普诺夫指数 λ0.0086 的弱…...
PUBG-Logitech压枪脚本深度解析与进阶实战指南
PUBG-Logitech压枪脚本深度解析与进阶实战指南 【免费下载链接】PUBG-Logitech PUBG罗技鼠标宏自动识别压枪 项目地址: https://gitcode.com/gh_mirrors/pu/PUBG-Logitech PUBG-Logitech是一款基于C和Qt框架开发的专业级绝地求生游戏压枪辅助工具,通过先进的…...
番茄小说下载器终极指南:3步永久保存你的数字图书馆
番茄小说下载器终极指南:3步永久保存你的数字图书馆 【免费下载链接】fanqienovel-downloader 下载番茄小说 项目地址: https://gitcode.com/gh_mirrors/fa/fanqienovel-downloader 还在为番茄小说突然下架而烦恼吗?fanqienovel-downloader 番茄小…...
别再死磕GCN了!用RGCN搞定知识图谱的实体分类与链接预测(附PyTorch代码)
知识图谱实战:用RGCN高效解决实体分类与链接预测问题 在知识图谱与推荐系统领域,图神经网络(GNN)正成为处理复杂关系数据的利器。传统GCN在处理多关系数据时往往力不从心,而关系图卷积网络(RGCN)通过引入关系特定权重机制,为知识图…...
