Mybatis逆向工程、Quartz框架的定时任务管理详解、Cron表达式

Mybatis逆向工程

  mybatis是目前很流行的持久层框架,很多企业都在采用。但是其复杂繁琐的配置,重复性的实体类创建等等,消耗了程序员大量的精力,同时有些地方如果一个细小的疏忽,可能导致最终功能运行失败。例如:在几十个字段的表中,某一列的列名配置疏忽。
  基于此,mybatis推出了一套jar包,可以依据我们设计好的数据库表,自动生成pojo、mapper以及mapper.xml。有了逆向工程,便大大缩减了我们的开发时间。本章节将介绍借助idea的方式实现mybatis的逆向工程。

环境搭建

(1)配置依赖
创建mybatis_generator工程并导入坐标
在这里插入图片描述

<dependencies>
    <dependency>
        <groupId>org.mybatis.generator</groupId>
        <artifactId>mybatis-generator-core</artifactId>
        <version>1.3.7</version>
    </dependency>
    <dependency>
        <groupId>org.mybatis</groupId>
        <artifactId>mybatis</artifactId>
        <version>3.4.5</version>
    </dependency>
    <dependency>
        <groupId>log4j</groupId>
        <artifactId>log4j</artifactId>
        <version>1.2.17</version>
    </dependency>
</dependencies>
<!-- 配置mybatis-generator-maven-plugin 插件-->
<build>
    <plugins>
        <plugin>
            <groupId>org.mybatis.generator</groupId>
            <artifactId>mybatis-generator-maven-plugin</artifactId>
            <version>1.3.7</version>
            <configuration>
                <verbose>true</verbose>
                <overwrite>true</overwrite>
            </configuration>
        </plugin>
    </plugins>
</build>

(2)添加配置文件
在resources目录下添加名为 generatorConfig.xml 配置文件,内容如下:

<generatorConfiguration>
    <!-- classPathEntry:数据库的 JDBC驱动的jar 包地址 -->
    <classPathEntry
            location="d:\\mysql-connector-java-5.1.32.jar" />
    <context id="caigouTables" targetRuntime="MyBatis3">

        <!-- 配置生成pojo的序列化的插件,mybatis支持很多插件,这些插件都在 org.mybatis.generator.plugins包下  -->
        <plugin type="org.mybatis.generator.plugins.SerializablePlugin" />

        <commentGenerator>
            <!-- 是否去除自动生成的注释 true:是 : false:-->
            <property name="suppressAllComments" value="true" />
        </commentGenerator>
        <!--数据库连接的信息:驱动类、连接地址、用户名、密码 -->
        <jdbcConnection driverClass="com.mysql.jdbc.Driver"
                        connectionURL="jdbc:mysql://127.0.0.1:3306/as-demo" 	   
                        userId="root"
                        password="root">
        </jdbcConnection>
        
        <javaTypeResolver>
            <property name="forceBigDecimals" value="false" />
        </javaTypeResolver>
        <!-- targetProject:生成 POJO 类的位置 -->
        <javaModelGenerator targetPackage="cn.it.domain.cargo" targetProject="src/main/java">
            <!-- enableSubPackages:是否让schema 作为包的后缀 -->
            <property name="enableSubPackages" value="true" />
            <!-- 从数据库返回的值被清理前后的空格 -->
            <property name="trimStrings" value="true" />
        </javaModelGenerator>

        <!-- 生成 XML文件 -->
        <sqlMapGenerator targetPackage="cn.it.dao.cargo"      targetProject="./src/main/resources">
            <property name="enableSubPackages" value="false" />
        </sqlMapGenerator>
        <!--生成接口-->
        <javaClientGenerator type="XMLMAPPER"
                             targetPackage="cn.it.dao.cargo"  targetProject="./src/main/java">
            <property name="enableSubPackages" value="false" />
        </javaClientGenerator>

        <!-- 指定数据库表 -->
        <table  tableName="co_contract" domainObjectName="Contract" mapperName="ContractDao"
               enableCountByExample="false" enableDeleteByExample="false"
               enableSelectByExample="false" enableUpdateByExample="false"/>

        <table  tableName="co_contract_product" domainObjectName="ContractProduct" mapperName="ContractProductDao"
               enableCountByExample="false" enableDeleteByExample="false"
               enableSelectByExample="false" enableUpdateByExample="false"/>

        <table  tableName="co_ext_cproduct" domainObjectName="ExtCproduct" mapperName="ExtCproductDao"
               enableCountByExample="false" enableDeleteByExample="false"
               enableSelectByExample="false" enableUpdateByExample="false"/>

        <table  tableName="co_factory" domainObjectName="Factory" mapperName="FactoryDao"
                enableCountByExample="false" enableDeleteByExample="false"
                enableSelectByExample="false" enableUpdateByExample="false"/>
    </context>
</generatorConfiguration>

(3)使用配置的插件生成代码
在这里插入图片描述
(4)生成的代码结构如下
在这里插入图片描述

Quartz定时任务调度

  Quartz框架非常有名,因为是开源的spring基于这套框架后续创建了自己的定时任务管理器(SpringTask/SpringSchdule)

概述

  在实际项目开发中,除了Web应用、SOA服务外,还有一类不可缺少的,那就是定时任务调度。定时任务的场景可以说非常广泛:
某些网站会定时发送优惠邮件;
银行系统还款日信用卡催收款;
某些应用的生日祝福短信等。
  完成这些功能都需要用到定时任务调度。那究竟何为定时任务调度,一句话概括就是:基于给定的时间点、给定的时间间隔、给定的执行次数自动执行的任务。

Quartz框架的介绍(设置作业定时运行规则)

在这里插入图片描述
  Quartz是一个完全由Java编写的开源任务调度的框架,通过触发器设置作业定时运行规则,控制作业的运行时间。其中quartz集群通过故障切换和负载平衡的功能,能给调度器带来高可用性和伸缩性。主要用来执行定时任务,如:定时发送信息、定时生成报表等等。
Quartz框架的主要特点:
· 强大的调度功能,例如丰富多样的调度方法,可以满足各种常规和特殊需求;
· 灵活的应用方式,比如支持任务调度和任务的多种组合,支持数据的多种存储;
· 支持分布式集群,在被Terracotta收购之后,在原来基础上进行了进一步的改造。

Quartz框架的核心元素

  Quartz核心要素有Scheduler、Trigger、Job、JobDetail,其中trigger和job、jobDetail为元数据,而Scheduler为实际进行调度的控制器。
在这里插入图片描述
Job
  Job用来定义任务的执行逻辑,相当于定义了一个任务类
JobDetail
  JobDetail表示一个具体的可执行的调度程序,Job是这个可执行的调度程序所要执行的具体内容,另外JobDetail还包含了这个任务调度的方案和策略,相当于指定任务类中的具体方法
Trigger
  Trigger用于定义调度任务的时间规则
Scheduler
  实际执行调度逻辑的控制器(也可以理解为调度容器),可以将多个JobDetail和Trigger注册到Scheduler中,就可以通过Scheduler进行控制执行

Quartz框架入门

(1)创建一个demo工程 quartz_demo
在这里插入图片描述
(2)添加依赖

<dependencies>
        <dependency>
            <groupId>org.quartz-scheduler</groupId>
            <artifactId>quartz</artifactId>
            <version>2.2.3</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-tx</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context-support</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>
    </dependencies>

(3)配置任务类

public class MyJob {

    public void showTime(){
        System.out.println("当前时间:"+new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()));
    }

    public void showTime2(){
        System.out.println("当前时间:"+new SimpleDateFormat("yyyy-MM-dd").format(new Date()));
    }
    
    public void showTime3(){
        System.out.println("当前时间:"+new SimpleDateFormat("yyyy-MM-dd").format(new Date()));
    }
}

(4)配置applicationContext-quartz.xml文件

 <!-- 定义一个任务类 -->
    <bean id="myJob" class="cn.it.manager.job.MyJob"></bean>
    <!-- 任务类描述 -->
    <bean id="jobDetail" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
        <property name="targetObject" ref="myJob"></property>
        <property name="targetMethod" value="showTime"></property>
    </bean>
    <!-- 触发器  -->
    <bean id="mailTrigger" class="org.springframework.scheduling.quartz.CronTriggerFactoryBean">
        <property name="jobDetail" ref="jobDetail"></property>
        <!-cron表达式定义时间规则-->
        <property name="cronExpression" value="0/3 * * * * ? *"></property>
    </bean>
    <!-- 总管理容器 -->
    <bean id="startQuartz" class="org.springframework.scheduling.quartz.SchedulerFactoryBean" >
        <property name="triggers">
            <list>
                <ref bean="mailTrigger"/>
            </list>
        </property>
    </bean>
</beans>

(5)测试:只需要将容器启动起来即可

public class QuartzTest {
    public static void main(String[] args) throws Exception {
        ClassPathXmlApplicationContext app = new ClassPathXmlApplicationContext("classpath:applicationContext-quartz.xml");
        app.start();
        System.in.read();
    }

结果:(每3秒输出一次:0/3 * * * * ? *)
在这里插入图片描述

Cron表达式(重要)

  对于Quartz,我们使用的时候主要是注重两个方面,一个是定时任务的业务,另一个就是Cron表达式。

Cron表达式格式

在这里插入图片描述
  QuartzCron 表达式支持到七个域
在这里插入图片描述
  月份和星期的名称是不区分大小写的。FRI 和 fri 是一样的。 域之间有空格分隔

特殊字符详解

*星号

使用星号(*) 指示着你想在这个域上包含所有合法的值。例如,在月份域上使用星号意味着每个月都会触发这个 trigger。 表达式样例:
0 * 17 * * ?
意义:每天从下午5点到下午5:59中的每分钟激发一次 trigger。它停在下午 5:59 是因为值 17在小时域上,在下午 6 点时,小时变为 18 了,也就不再理会这个 trigger,直到下一天的下午5点。 在你希望 trigger在该域的所有有效值上被激发时使用 * 字符。

? 问号

? 号只能用在日和周域上,但是不能在这两个域上同时使用。你可以认为 ? 字符是 “我并不关心在该域上是什么值。” 这不同于星号,星号是指示着该域上的每一个值。? 是说不为该域指定值。
不能同时这两个域上指定值的理由是难以解释甚至是难以理解的。基本上,假定同时指定值的话,意义就会变得含混不清了:考虑一下,如果一个表达式在日域上有值11,同时在周域上指定了 WED。那么是要 trigger 仅在每个月的11号,且正好又是星期三那天被激发?还是在每个星期三的11号被激发呢?要去除这种不明确性的办法就是不能同时在这两个域上指定值。 只要记住,假如你为这两域的其中一个指定了值,那就必须在另一个字值上放一个 ?

表达式样例:
0 10,44 14 ? 3 WED
意义:在三月中的每个星期三的下午 2:10 和 下午 2:44 被触发。

,逗号

逗号 (,) 是用来在给某个域上指定一个值列表的。例如,使用值 0,15,30,45 在秒域上意味着每15秒触发一个 trigger。
表达式样例:
0 0,15,30,45 * * * ?
意义:每刻钟触发一次 trigger。

/斜杠

斜杠 (/) 是用于时间表的递增的。我们刚刚用了逗号来表示每15分钟的递增,但是我们也能写成这样 0/15。
表达式样例:
0/15 0/30 * * * ?
意义:在整点和半点时每15秒触发 trigger。

–中划线

中划线 (-) 用于指定一个范围。例如,在小时域上的 3-8 意味着 “3,4,5,6,7 和 8 点。” 域的值不允许回转,所以像 50-10 这样的值是不允许的。
表达式样例:
0 45 3-8 ? * *
意义:在上午的3点至上午的8点的45分时触发 trigger。

L字母

L 说明了某域上允许的最后一个值。它仅被日和周域支持。当用在日域上,表示的是在月域上指定的月份的最后一天。例如,当月域上指定了 JAN 时,在日域上的 L 会促使 trigger 在1月31号被触发。假如月域上是 SEP,那么 L 会预示着在9月30号触发。换句话说,就是不管指定了哪个月,都是在相应月份的时最后一天触发 trigger。
表达式 0 0 8 L * ? 意义是在每个月最后一天的上午 8:00 触发 trigger。在月域上的 * 说明是 “每个月”。
当 L 字母用于周域上,指示着周的最后一天,就是星期六 (或者数字7)。所以如果你需要在每个月的最后一个星期六下午的 11:59 触发 trigger,你可以用这样的表达式 0 59 23 ? * L。
当使用于周域上,你可以用一个数字与 L 连起来表示月份的最后一个星期 X。例如,表达式 0 0 12 ? * 2L 说的是在每个月的最后一个星期一触发 trigger。
不要让范围和列表值与 L 连用
虽然你能用星期数(1-7)与 L 连用,但是不允许你用一个范围值和列表值与 L 连用。这会产生不可预知的结果。

w字母

W 字符代表着平日 (Mon-Fri),并且仅能用于日域中。它用来指定离指定日的最近的一个平日。大部分的商业处理都是基于工作周的,所以 W 字符可能是非常重要的。例如,日域中的 15W 意味着 “离该月15号的最近一个平日。” 假如15号是星期六,那么 trigger 会在14号(星期四)触发,因为距15号最近的是星期一,这个例子中也会是17号(译者Unmi注:不会在17号触发的,如果是15W,可能会是在14号(15号是星期六)或者15号(15号是星期天)触发,也就是只能出现在邻近的一天,如果15号当天为平日直接就会当日执行)。W 只能用在指定的日域为单天,不能是范围或列表值。

#井号

字符仅能用于周域中。它用于指定月份中的第几周的哪一天。例如,如果你指定周域的值为 6#3,它意思是某月的第三个周五 (6=星期五,#3意味着月份中的第三周)。另一个例子 2#1 意思是某月的第一个星期一 (2=星期一,#1意味着月份中的第一周)。注意,假如你指定 #5,然而月份中没有第 5 周,那么该月不会触发。

以下是一些测试用例:

“0 0 12 * *?” 每天中午12点触发
“0 15 10 ? **” 每天上午10:15触发
“0 15 10 * ?" 每天上午10:15触发
"0 15 10 * * ?
” 每天上午10:15触发
“0 15 10 * * ?2005” 2005年的每天上午10:15触发
“0 * 14 * *?” 在每天下午2点到下午2:59期间的每1分钟触发
“0 0/5 14 * *?” 在每天下午2点到下午2:55期间的每5分钟触发
“0 0/5 14,18 ** ?” 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发
“0 0-5 14 * *?” 在每天下午2点到下午2:05期间的每1分钟触发
“0 10,44 14 ? 3WED” 每年三月的星期三的下午2:10和2:44触发
“0 15 10 ? *MON-FRI” 周一至周五的上午10:15触发
“0 15 10 15 *?” 每月15日上午10:15触发
“0 15 10 L *?” 每月最后一日的上午10:15触发
“0 15 10 ? *6L” 每月的最后一个星期五上午10:15触发
“0 15 10 ? * 6L2002-2005” 2002年至2005年的每月的最后一个星期五上午10:15触发
“0 15 10 ? *6#3” 每月的第三个星期五上午10:15触发

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值