Springboot 任务

异步任务

同步,在网站上发送邮件时,后台会去发送邮件,这个过程中前台会造成响应不动,直到邮件发送完毕,响应才会成功。

@EnableAsync开启异步注解功能

package com.tasks;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableAsync;

@SpringBootApplication
@EnableAsync //开启异步注解功能
public class TasksApplication {

    public static void main(String[] args) {
        SpringApplication.run(TasksApplication.class, args);
    }
   
}

@Async告诉Spring这是一个异步方法

package com.tasks.asyn;

import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;

@Service
public class AsyncService {

    @Async //告诉Spring这是一个异步方法
    public void sleep(){
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("发送邮件完成...");
    }
}

使用异步后,网页瞬间响应,后台代码依旧执行

定时任务

项目开发中经常需要执行一些定时任务,比如需要在每天凌晨的时候,分析一次前一天的日志信息,Spring为我们提供了异步执行任务调度的方式,提供了TaskExecutor 、TaskScheduler 两个接口。

  • TaskExecutor 任务调度
  • TaskScheduler任务执行者

两个注解:

  • @EnableScheduling 开启定时任务的注解
  • @Scheduled 什么时候执行

Cron表达式,在Linux定时任务常用

  • 格式:秒 分钟 小时 日 月 周 年
  • 在线Cron表达式生成器

https://www.beejson.com/tool/cron.html

https://cron.qqe2.com/

image-20200823223351004.png
package com.tasks;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.annotation.EnableScheduling;

@SpringBootApplication
@EnableAsync //开启异步注解功能
@EnableScheduling //开启定时任务注解功能
public class TasksApplication {

    public static void main(String[] args) {
        SpringApplication.run(TasksApplication.class, args);
    }

}
package com.tasks.scheduled;

import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Service;

@Service
public class ScheduledService {
    //每秒执行一次
    @Scheduled(cron = "* * * * * ?")
    public void hello(){
        mysqlUtil.backup();
        System.out.println("备份数据库");
    }
}

邮件发送

邮件发送,在我们的日常开发中,也非常的多,Springboot也帮我们做了支持

  • 邮件发送需要引入spring-boot-start-mail
  • SpringBoot 自动配置MailSenderAutoConfiguration
  • 定义MailProperties内容,配置在application.yml中
  • 自动装配JavaMailSender
  • 测试邮件发送
spring:
  mail:
    username: strtz3@163.com
    password: 密钥
    host: smtp.163.com

以下的配置可以解决一些异常

spring.mail.protocol=smtp
spring.mail.properties.mail.smtp.auth=true
spring.mail.properties.mail.smtp.ssl.enable=true
spring.mail.test-connection=true

JavaMailUtils工具类

package com.tasks.utils;

import org.springframework.mail.SimpleMailMessage;
import org.springframework.mail.javamail.JavaMailSenderImpl;
import org.springframework.mail.javamail.MimeMessageHelper;

import javax.mail.MessagingException;
import javax.mail.internet.MimeMessage;
import java.io.File;

public class JavaMailUtils {
    //默认编码
    public static final String DEFAULT_ENCODING = "UTF-8";
    
    /**
     * 简单邮件发送
     *
     * @param javaMailSender
     * @param subject
     * @param content
     * @param target
     * @param form
     */
    public static void sendSimpleMessage(JavaMailSenderImpl javaMailSender, String subject, String content, String target, String form) {
        SimpleMailMessage simpleMailMessage = new SimpleMailMessage();
        //邮件内容
        simpleMailMessage.setSubject(subject);
        simpleMailMessage.setText(content);
        //发送和接收方
        simpleMailMessage.setTo(target);
        simpleMailMessage.setFrom(form);
        javaMailSender.send(simpleMailMessage);
    }

    /**
     * 复杂邮件发送
     *
     * @param javaMailSender
     * @param subject        主体
     * @param body           内容
     * @param isHtml         是否html
     * @param target         发送方
     * @param form           发件方
     * @param attachment     附件
     * @throws MessagingException
     */
    public static void sendMimeMessage(JavaMailSenderImpl javaMailSender, String subject, String body, Boolean isHtml, String target, String form, File attachment
    ) throws MessagingException {
        MimeMessage mimeMessage = javaMailSender.createMimeMessage();
        //创建一个helper来处理复杂邮件封装,true开启文件上传功能
        MimeMessageHelper messageHelper = new MimeMessageHelper(mimeMessage, true);
        //邮件内容
        messageHelper.setSubject(subject);
        messageHelper.setText(body, isHtml);
        messageHelper.addAttachment(attachment.getName(), attachment);
        //发送和接收方
        messageHelper.setFrom(form);
        messageHelper.setTo(target);
        javaMailSender.send(mimeMessage);
    }
}

测试

package com.tasks;

import com.tasks.utils.JavaMailUtils;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.mail.SimpleMailMessage;
import org.springframework.mail.javamail.JavaMailSenderImpl;

import javax.mail.MessagingException;
import java.io.File;

@SpringBootTest

class TasksApplicationTests {
    @Autowired
    private JavaMailSenderImpl javaMailSender;

    @Test
    void contextLoads() throws MessagingException {
        JavaMailUtils.sendMimeMessage(javaMailSender,"工具类发送复杂","<a href=''>go</a>",true,"strtz3@163.com","strtz3@163.com",new File("C:/Users/str/Desktop/mp.jpg"));
    }

}

https://blog.csdn.net/qq_41741884/article/details/99079716 其他内容可以参考该文章

备份与恢复数据库

  • mysqldump -h127.0.0.1 -uroot -p123456 test > d:/test.sql 备份数据
  • mysql -h127.0.0.1 -uroot -p123456 test< d:/test.sql 恢复数据

在cmd调用命令行,其实是调用 mysql安装路径下面的bin目录下面的msqldump.exe和mysql.exe来完成相应的工作

//linux
String[] command = { "/bin/sh", "-c", command };
//window
String[] command = { "cmd", "/c", command};
//java调用外部软件exe执行命令的api;
Process ps = Runtime.getRuntime().exec(command );
package com.tasks.utils;

import java.io.*;
import java.text.SimpleDateFormat;
import java.util.Date;


/**
 * 数据库备份和还原
 */
public class mysqlUtil {

    /**
     * 备份数据库db
     * mysqldump -hlocalhost -uroot -proot test > d:/db.sql
     *
     * @param root
     * @param pwd
     * @param dbName
     * @param backPath
     * @param backName
     */
    public static void dbBackUp(String root, String pwd, String dbName, String backPath, String backName) throws Exception {
        String pathSql = backPath + backName;
        File fileSql = new File(pathSql);
        //创建备份sql文件
        if (!fileSql.exists()) {
            fileSql.createNewFile();
        }
        //mysqldump -hlocalhost -uroot -p123456 db > /home/back.sql
        StringBuffer sb = new StringBuffer();
        sb.append("mysqldump");
        sb.append(" -hlocalhost");
        sb.append(" -u" + root);
        sb.append(" -p" + pwd);
        sb.append(" " + dbName + " >");
        sb.append(pathSql);
        System.out.println("cmd命令为:" + sb.toString());
        Runtime runtime = Runtime.getRuntime();
        System.out.println("开始备份:" + dbName);
        Process process = runtime.exec("cmd /c" + sb.toString());
        System.out.println("备份成功!");
    }

    /**
     * 恢复数据库
     *
     * @param root
     * @param pwd
     * @param dbName
     * @param filePath mysql -hlocalhost -uroot -proot db < /home/back.sql
     */
    public static void dbRestore(String root, String pwd, String dbName, String filePath) {
        StringBuilder sb = new StringBuilder();
        sb.append("mysql");
        sb.append(" -hlocalhost");
        sb.append(" -u" + root);
        sb.append(" -p" + pwd);
        sb.append(" " + dbName + " <");
        sb.append(filePath);
        System.out.println("cmd命令为:" + sb.toString());
        Runtime runtime = Runtime.getRuntime();
        System.out.println("开始还原数据");
        try {
            Process process = runtime.exec("cmd /c" + sb.toString());
            InputStream is = process.getInputStream();
            BufferedReader bf = new BufferedReader(new InputStreamReader(is, "utf8"));
            String line = null;
            while ((line = bf.readLine()) != null) {
                System.out.println(line);
            }
            is.close();
            bf.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
        System.out.println("还原成功!");
    }


    public static void main(String[] args) throws Exception {
        String backName = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss").format(new Date()) + ".sql";
        mysqlUtil.dbBackUp("root", "root", "myblog", "E:/", backName);
        //dbRestore("root", "root", "myblog", "name.sql");
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,324评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,356评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,328评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,147评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,160评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,115评论 1 296
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,025评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,867评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,307评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,528评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,688评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,409评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,001评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,657评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,811评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,685评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,573评论 2 353