SSH综合项目实战(快递) -- day14 quartz定时器

一、Quartz入门案例

1、创建maven project,打包方式为jar,继承common-parent


2、编写自定义任务类

package com.itheima.job;

/**
 * 自定义任务类
 * @author Administrator
 *
 */
public class MyJob {

	public void sendMsg(){
		System.out.println("定时发送邮件");
	}
}

3、编写配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
	xmlns:context="http://www.springframework.org/schema/context"
	xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:tx="http://www.springframework.org/schema/tx"
	xmlns:jpa="http://www.springframework.org/schema/data/jpa" xmlns:task="http://www.springframework.org/schema/task"
	xmlns:jaxws="http://cxf.apache.org/jaxws" xmlns:soap="http://cxf.apache.org/bindings/soap"
	xsi:schemaLocation="
						http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
						http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd
						http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
						http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc.xsd
						http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd
						http://www.springframework.org/schema/data/jpa 
						http://www.springframework.org/schema/data/jpa/spring-jpa.xsd
						http://cxf.apache.org/bindings/soap 
						http://cxf.apache.org/schemas/configuration/soap.xsd
						http://cxf.apache.org/jaxws 
						http://cxf.apache.org/schemas/jaxws.xsd
						">
	<!-- 1.注册自定义任务对象 -->
	<bean id="myJob" class="com.itheima.job.MyJob"></bean>
	
	<!-- 2.配置JobDetail,用于通过反射调用myJob类 -->
	<bean id="jobDetail" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
		<!-- 注入目标对象 -->
		<property name="targetObject" ref="myJob"></property>
		<!-- 注入目标方法 -->
		<property name="targetMethod" value="sendMsg"></property>
	</bean>
	
	<!-- 3.配置触发器,指定任务的触发时间 -->
	<bean id="myTrigger" class="org.springframework.scheduling.quartz.CronTriggerFactoryBean">
		<!-- 注入jobDetail -->
		<property name="jobDetail" ref="jobDetail"></property>
		<!-- 通过表达式指定任务触发的时间 -->
		<property name="cronExpression">
			<value>0/5 * * * * ?</value>
		</property>
	</bean>
	
	<!-- 4.配置调度工厂 -->
	<bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
		<!-- 配置触发器,可以有多个 -->
		<property name="triggers">
			<list>
				<ref bean="myTrigger"/>
			</list>
		</property>
	</bean>
</beans>

4、编写测试类

package com.itheima.test;

import org.springframework.context.support.ClassPathXmlApplicationContext;

public class JobTest {

	public static void main(String[] args) {
		//加载配置文件
		new ClassPathXmlApplicationContext("applicationContext.xml");
		
	}
}

二、定时器任务的cron表达式

1、cron表达式介绍

        Quartz cron 表达式的格式十分类似于 UNIX cron 格式,但还是有少许明显的区别。区别之一就是 Quartz 的格式向下支持到秒级别的计划,而 UNIX cron 计划仅支持至分钟级。许多我们的触发计划要基于秒级递增的(例如,每45秒),因此这是一个非常好的差异。在 UNIX cron 里,要执行的作业(或者说命令)是存放在 cron 表达式中的,在第六个域位置上。Quartz 用 cron 表达式存放执行计划。引用了 cron 表达式的 CronTrigger 在计划的时间里会与 job 关联上。

2、cron表达式支持的7个域


月份和星期的名称是不区分大小写的。FRI 和 fri 是一样的。 

域之间有空格分隔,这和 UNIX cron 一样。无可争辩的,我们能写的最简单的表达式看起来就是这个了: 

        * * * ? * * 

这个表达会每秒钟(每分种的、每小时的、每天的)激发一个部署的 job。

3、cron表达式中的特殊字符

(1)、* 星号

使用星号(*) 指示着你想在这个域上包含所有合法的值。例如,在月份域上使用星号意味着每个月都会触发这个 trigger。

表达式样例: 

0 * 17 * * ? 

意义:每天从下午5点到下午5:59中的每分钟激发一次 trigger。它停在下午 5:59 是因为值 17 在小时域上,在下午 6 点时,小时变为 18 了,也就不再理会这个 trigger,直到下一天的下午5点。 

在你希望 trigger 在该域的所有有效值上被激发时使用 * 字符。 

(2)、? 问号 

? 号只能用在日和周域上,但是不能在这两个域上同时使用。你可以认为 ? 字符是 "我并不关心在该域上是什么值。" 这不同于星号,星号是指示着该域上的每一个值。? 是说不为该域指定值。 

不能同时这两个域上指定值的理由是难以解释甚至是难以理解的。基本上,假定同时指定值的话,意义就会变得含混不清了:考虑一下,如果一个表达式在日域上有值11,同时在周域上指定了 WED。那么是要 trigger 仅在每个月的11号,且正好又是星期三那天被激发?还是在每个星期三的11号被激发呢?要去除这种不明确性的办法就是不能同时在这两个域上指定值。 

只要记住,假如你为这两域的其中一个指定了值,那就必须在另一个字值上放一个 ?。 

表达式样例: 

0 10,44 14 ? 3 WEN

意义:在三月中的每个星期三的下午 2:10:00 和 下午 2:44:00 被触发。 

(3)、, 逗号 

逗号 (,) 是用来在给某个域上指定一个值列表的。例如,使用值 0,15,30,45 在秒域上意味着每15秒触发一个 trigger。 

表达式样例: 

0 0,15,30,45 * * * ? 

意义:每刻钟触发一次 trigger。 

(4)、/ 斜杠 

斜杠 (/) 是用于时间表的递增的。我们刚刚用了逗号来表示每15分钟的递增,但是我们也能写成这样 0/15。 

表达式样例: 

0/15 0/30 * * * ? 

意义:在整点和半点时每15秒触发 trigger。 

(5)、- 中划线 

中划线 (-) 用于指定一个范围。例如,在小时域上的 3-8 意味着 "3,4,5,6,7 和 8 点。"  域的值不允许回卷,所以像 50-10 这样的值是不允许的。 

表达式样例: 

0 45 3-8 ? * * 

意义:在上午的3点至上午的8点的45分时触发 trigger。

(6)、L 字母Last

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 连用。这会产生不可预知的结果。

(7)、W 字母 

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

(8)、# 井号 

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

4、表达式意义示例

"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触发

三、在BOS项目中使用Quartz创建定时任务发送邮件

1、引入定时器jar包坐标


2、将MailUtil拷贝到bos_management_util中


2、编写自定义任务类

package com.itheima.bos.jobs;

import java.util.Date;
import java.util.List;

import org.springframework.beans.factory.annotation.Autowired;

import com.itheima.bos.dao.take_delivery.WorkbillDao;
import com.itheima.bos.domain.take_delivery.WorkBill;
import com.itheima.bos.utils.MailUtils;

/**
 * 自定义任务,定时发送邮件
 */
public class MailJob {
	@Autowired
	private WorkbillDao workbillDao;

	/**
	 * 发送邮件,邮件内容为所有的工单信息
	 */
	public void sendMail() {
		System.out.println("发送邮件了:" + new Date());
		// 查询所有工单信息
		List<WorkBill> list = workbillDao.findAll();
		String subject = "工单信息统计";// 邮件主题
		String content = "工单编号   工单类型   取件状态   快递员 <br>";// 邮件内容
		for (WorkBill workBill : list) {
			content += workBill.getId() + "  " + workBill.getType() + "  " + workBill.getPickstate() + "  "
					+ workBill.getCourier().getName() + "<br>";
		}
		String to = "test@itcast.cn";// 收件人
		// 调用工具类发送邮件
		MailUtils.sendMail(subject, content, to);
	}
}

3、在spring配置文件中配置定时任务相关配置

<!-- 1.注册自定义任务对象 -->
	<bean id="myJob" class="com.itheima.bos.jobs.MailJob"></bean>

	<!-- 2.配置JobDetail,用于通过反射调用myJob类 -->
	<bean id="jobDetail"
		class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
		<!-- 注入目标对象 -->
		<property name="targetObject" ref="myJob"></property>
		<!-- 注入目标方法 -->
		<property name="targetMethod" value="sendMail"></property>
	</bean>

	<!-- 3.配置触发器,指定任务的触发时间 -->
	<bean id="myTrigger"
		class="org.springframework.scheduling.quartz.CronTriggerFactoryBean">
		<!-- 注入jobDetail -->
		<property name="jobDetail" ref="jobDetail"></property>
		<!-- 通过表达式指定任务触发的时间 -->
		<property name="cronExpression">
			<value>0/5 * * * * ?</value>
		</property>
	</bean>

	<!-- 4.配置调度工厂 -->
	<bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
		<!-- 配置触发器,可以有多个 -->
		<property name="triggers">
			<list>
				<ref bean="myTrigger" />
			</list>
		</property>
	</bean>

4、启动项目等待即可测试

四、基于POI实现分区数据导出到Excel文件

1、编写sub_area.html页面导出按钮点击事件


2、导入解决下载时出现中文乱码的工具类


package com.itheima.bos.utils;

import java.io.IOException;
import java.net.URLEncoder;

import sun.misc.BASE64Encoder;

public class FileUtils {
		/**
		 * 下载文件时,针对不同浏览器,进行附件名的编码
		 * 
		 * @param filename
		 *            下载文件名
		 * @param agent
		 *            客户端浏览器
		 * @return 编码后的下载附件名
		 * @throws IOException
		 */
		public static String encodeDownloadFilename(String filename, String agent)
				throws IOException {
			if (agent.contains("Firefox")) { // 火狐浏览器
				filename = "=?UTF-8?B?"
						+ new BASE64Encoder().encode(filename.getBytes("utf-8"))
						+ "?=";
				filename = filename.replaceAll("\r\n", "");
			} else { // IE及其他浏览器
				filename = URLEncoder.encode(filename, "utf-8");
				filename = filename.replace("+"," ");
			}
			return filename;
		}
}

3、编写Action层代码

	/**
	 * 导出分区数据
	 * 
	 * @throws Exception
	 */
	@Action(value = "subareaAction_exportXls")
	public String exportXls() throws Exception {
		// 1.查询所有分区数据
		List<SubArea> list = service.findAll();
		// 2.基于POI在内存中创建一个Excel文件
		HSSFWorkbook excel = new HSSFWorkbook();
		// 3.在Excel中创建一个sheet页
		HSSFSheet sheet = excel.createSheet("分区数据统计");
		// 4.在标签页中创建标题行
		HSSFRow titleRow = sheet.createRow(0);
		// 5.在行中创建列
		titleRow.createCell(0).setCellValue("编号");
		titleRow.createCell(1).setCellValue("分区起始编号");
		titleRow.createCell(2).setCellValue("分区结束编号");
		titleRow.createCell(3).setCellValue("分区关键字");
		titleRow.createCell(4).setCellValue("辅助关键字");
		titleRow.createCell(5).setCellValue("区域信息");
		// 6.遍历查询出的区域信息,向Excel中写入数据
		for (SubArea sub : list) {
			// 每个分区对象对应一行数据
			HSSFRow contentRow = sheet.createRow(sheet.getLastRowNum() + 1);// 每次从最后一行开始创建行对象
			contentRow.createCell(0).setCellValue(sub.getId());
			contentRow.createCell(1).setCellValue(sub.getStartNum());
			contentRow.createCell(2).setCellValue(sub.getEndNum());
			contentRow.createCell(3).setCellValue(sub.getKeyWords());
			contentRow.createCell(4).setCellValue(sub.getAssistKeyWords());
			contentRow.createCell(5).setCellValue(sub.getArea().getName());
		}
		// 7.通过输出流写回Excel文件到浏览器,文件下载需要一个输出流,设置两个头信息
		ServletOutputStream outputStream = ServletActionContext.getResponse().getOutputStream();
		// 设置文件名
		String fileName = "分区数据统计结果导出表";
		// 获取客户端使用的浏览器类型
		String agent = ServletActionContext.getRequest().getHeader("User-Agent");
		// 使用工具类处理下载时的中文名乱码问题
		fileName = FileUtils.encodeDownloadFilename(fileName, agent);
		// 根据文件名称获取文件的MIME类型
		String mimeType = ServletActionContext.getServletContext().getMimeType(fileName);
		// 设置相应头
		ServletActionContext.getResponse().setContentType(mimeType);
		// 设置以附件的形式下载
		ServletActionContext.getResponse().setHeader("content-disposition", "attachment;filename=" + fileName);
		// 使用输出流输出到浏览器
		excel.write(outputStream);
		return NONE;
	}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值