首先,需要导入quartz 的jar包 ① applicationContext.xml <!-- 轮询任务 --> <import resource="classpath:/conf/quartz/ctmanage-schedule.xml" /> ② ctmanage-schedule.xml <?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:tx="http://www.springframework.org/schema/tx" xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans-2.0.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd"> <!-- 配置该项目的定时任务 --> <!-- 生成账单-定义调用对象和调用对象的方法 --> <bean id="generateBillTaskMethod" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean"> <property name="targetObject" ref="generateBillService" /> <property name="targetMethod" value="generateBill" /> <property name="concurrent" value="false" /> </bean> <!-- 生成账单-定义触发时间 --> <!-- 0 0 2 1 * ? 每个月1号凌晨2点执行 正式--> <!-- 1/15 * * * * ? 启动就执行 每隔15秒执行一次--> <!-- 0 0/3 * * * ? 每三分钟执行一次 启动服务器三分钟后执行第一次--> <bean id="generateBillTaskMethodTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean"> <property name="jobDetail" ref="generateBillTaskMethod" /> <property name="cronExpression"> <value>0 0 2 1 * ?</value> </property> </bean> <!-- 轮询任务列表 --> <bean id="timerFactory" class="org.springframework.scheduling.quartz.SchedulerFactoryBean"> <property name="triggers"> <list> <ref local="generateBillTaskMethodTrigger" /> </list> </property> <property name="quartzProperties"> <props> <prop key="org.quartz.threadPool.class">org.quartz.simpl.SimpleThreadPool</prop> <prop key="org.quartz.threadPool.threadCount">10</prop> <prop key="org.quartz.threadPool.threadsInheritContextClassLoaderOfInitializingThread">true</prop> </props> </property> <property name="startupDelay" value="5"></property> </bean> </beans> ③ 配置实现类 <!-- 定时任务 配置 --> <bean id="generateBillService" class="com.bontai.ct.manager.helper.timed.GenerateBillService"> <property name="generateBillBiz" ref="generateBillBiz" /> </bean> <bean id="generateBillBiz" class="com.bontai.ct.manager.helper.timed.impl.GenerateBillBizImpl"> <property name="ctUserDAO" ref="ctUserDAO"/> <property name="ctGroupDAO" ref="ctGroupDAO"/> <property name="fmOrderDAO" ref="fmOrderDAO"/> <property name="acBillDAO" ref="acBillDAO"/> <property name="acBillOrderDAO" ref="acBillOrderDAO"/> <property name="billDAO" ref="billDAO"/> </bean> ④ 代码 //service package com.bontai.ct.manager.helper.timed; public class GenerateBillService { private GenerateBillBiz generateBillBiz; public GenerateBillBiz getGenerateBillBiz() { return generateBillBiz; } public void setGenerateBillBiz(GenerateBillBiz generateBillBiz) { this.generateBillBiz = generateBillBiz; } //定时任务-生成账单(每月1日为所有机构自动生成账单) public synchronized void generateBill(){ System.out.println("=================开始执行轮询任务=================="); this.generateBillBiz.generateBill(); } } //biz package com.bontai.ct.manager.helper.timed; public interface GenerateBillBiz { public void generateBill(); } //bizImpl package com.bontai.ct.manager.helper.timed.impl; import java.math.BigDecimal; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Date; import java.util.HashMap; import java.util.List; import java.util.Map; import org.springframework.transaction.annotation.Transactional; import com.bontai.ct.manager.dao.custom.mybatis.BillDAO; import com.bontai.ct.manager.dao.db.AcBillDAO; import com.bontai.ct.manager.dao.db.AcBillOrderDAO; import com.bontai.ct.manager.dao.db.CtGroupDAO; import com.bontai.ct.manager.dao.db.CtUserDAO; import com.bontai.ct.manager.dao.db.FmOrderDAO; import com.bontai.ct.manager.entity.mapping.AcBill; import com.bontai.ct.manager.entity.mapping.AcBillOrder; import com.bontai.ct.manager.entity.mapping.CtUser; import com.bontai.ct.manager.entity.mapping.FmOrder; import com.bontai.ct.manager.helper.timed.DatePoint; import com.bontai.ct.manager.helper.timed.GenerateBillBiz; public class GenerateBillBizImpl implements GenerateBillBiz{ private SimpleDateFormat timeStampFormater = new SimpleDateFormat("yyyyMMddHHmmss"); private SimpleDateFormat batchNOFormat = new SimpleDateFormat("yyyyMM"); private SimpleDateFormat yearFormat = new SimpleDateFormat("yyyy"); private SimpleDateFormat monthFormat = new SimpleDateFormat("MM"); private CtUserDAO ctUserDAO; private CtGroupDAO ctGroupDAO; private FmOrderDAO fmOrderDAO; private AcBillDAO acBillDAO; private AcBillOrderDAO acBillOrderDAO; private BillDAO billDAO; //定时任务 - 生成每月机构账单 @Transactional public void generateBill() { System.out.println("==================== 定时任务 - 生成每月机构账单 start ======================"); //定义账单时间段(上月1日到最后一日) Date startTime = DatePoint.getLastMonthBeginDate(); Date endTime = DatePoint.getLastMonthEndDate(); System.out.println("====================startTime:"+startTime+"======================"); System.out.println("====================endTime:"+endTime+"======================"); //获得年月信息 int year = Integer.parseInt(yearFormat.format(startTime)); int month = Integer.parseInt(monthFormat.format(startTime)); //账单批次号 String batchNo = batchNOFormat.format(startTime); //① 查询所有机构用户 List<CtUser> userList = new ArrayList<CtUser>(); CtUser user = new CtUser(); user.setUserType("1"); userList = this.ctUserDAO.queryCtUser(user); //根据userId循环插入 账单 和 账单订单关系 for(int i = 0;i<userList.size();i++){ Long userId = userList.get(i).getId();//机构用户id Long billId = null; //① 查询用户上月的有效订单个数、账单金额、充值金额 List<AcBill> billInfo = new ArrayList<AcBill>(); Map<String,Object> params = new HashMap<String,Object>(); params.put("userId",userId); params.put("year", year); params.put("month", month); billInfo= this.billDAO.queryOrderCountAndInOutMoneyByMonth(params); //② 开始插入数据 ac_bill 插入账单 if(billInfo.size()>0){ //获得统计值 int orderCount = billInfo.get(0).getOrderCount(); BigDecimal billMoney = billInfo.get(0).getBillMoney(); BigDecimal depositMoney = billInfo.get(0).getDepositMoney(); //插入 AcBill bill = new AcBill(); bill.setUserId(userId); bill.setInvoiceStat("0"); bill.setUserType("1"); bill.setStartTime(startTime); bill.setEndTime(endTime); bill.setBillSeq(timeStampFormater.format(new Date())+userId); bill.setBatchNo(batchNo); bill.setOrderCount(orderCount); bill.setBillMoney(billMoney); bill.setDepositMoney(depositMoney); bill.setBadDebtMoney(BigDecimal.ZERO); bill.setBadDebtId(null); bill.setCrtTime(new Date()); this.acBillDAO.insertNotNull(bill); billId = bill.getId(); } //③ 查询某机构某月的订单 List<FmOrder> orderList = new ArrayList<FmOrder>(); orderList= this.billDAO.queryBillOrderByMonth(params); //④ 插入账单-订单关系 for(int j = 0;j<orderList.size();j++){ Long orderId = orderList.get(j).getId(); AcBillOrder bo = new AcBillOrder(); bo.setOrderId(orderId); bo.setBillId(billId); this.acBillOrderDAO.insertNotNull(bo); } } System.out.println("==================== 定时任务 - 生成每月机构账单 end ======================"); } //getters and setters public CtUserDAO getCtUserDAO() { return ctUserDAO; } public CtGroupDAO getCtGroupDAO() { return ctGroupDAO; } public FmOrderDAO getFmOrderDAO() { return fmOrderDAO; } public AcBillDAO getAcBillDAO() { return acBillDAO; } public AcBillOrderDAO getAcBillOrderDAO() { return acBillOrderDAO; } public void setCtUserDAO(CtUserDAO ctUserDAO) { this.ctUserDAO = ctUserDAO; } public void setCtGroupDAO(CtGroupDAO ctGroupDAO) { this.ctGroupDAO = ctGroupDAO; } public void setFmOrderDAO(FmOrderDAO fmOrderDAO) { this.fmOrderDAO = fmOrderDAO; } public void setAcBillDAO(AcBillDAO acBillDAO) { this.acBillDAO = acBillDAO; } public void setAcBillOrderDAO(AcBillOrderDAO acBillOrderDAO) { this.acBillOrderDAO = acBillOrderDAO; } public BillDAO getBillDAO() { return billDAO; } public void setBillDAO(BillDAO billDAO) { this.billDAO = billDAO; } } //DatePoint //获得上月1号的日期 public static Date getLastMonthBeginDate(){ Calendar cal = Calendar.getInstance(); cal.set(Calendar.MONTH, cal.get(Calendar.MONTH)-1); cal.set(Calendar.DAY_OF_MONTH,1); cal.set(Calendar.HOUR_OF_DAY, 0); cal.set(Calendar.MINUTE, 0); cal.set(Calendar.SECOND, 0); return cal.getTime(); } //获得上个月最后一天的日期 public static Date getLastMonthEndDate(){ Calendar cal = Calendar.getInstance(); cal.set(Calendar.DAY_OF_MONTH, 0); cal.set(Calendar.HOUR_OF_DAY, 23); cal.set(Calendar.MINUTE, 59); cal.set(Calendar.SECOND, 59); return cal.getTime(); } ⑤ 重要sql <!-- 查询某用户某月的完成订单数/账单金额/充值金额 --> <select id="queryOrderCountAndInOutMoneyByMonth" parameterType="java.util.Map" resultType ="com.bontai.ct.manager.entity.mapping.AcBill" > SELECT DISTINCT ifnull( ( SELECT count(*) FROM fm_order o WHERE o.buyer_user_id = #{userId} AND o.deal_stat = ‘09‘ AND YEAR(o.final_time) = #{year} AND MONTH(o.final_time) = #{month} ), ‘0‘ ) as orderCount, ifnull( ( SELECT sum(c.pay_money) FROM fm_order o LEFT JOIN fm_order_cost c ON o.id = c.order_id WHERE o.buyer_user_id = #{userId} AND o.deal_stat = ‘09‘ AND YEAR(o.final_time) = #{year} AND MONTH(o.final_time) = #{month} ), ‘0‘ ) as billMoney, ifnull( ( SELECT sum(d.deposit_money) FROM ac_deposit d WHERE d.user_id = #{userId} AND d.deposit_stat = ‘1‘ AND d.audit_stat = ‘2‘ AND YEAR(d.deposit_time) = #{year} AND MONTH(d.deposit_time) = #{month} ), ‘0‘ ) as depositMoney </select> <!-- 查询某机构某月的订单 --> <select id="queryBillOrderByMonth" parameterType="java.util.Map" resultType ="com.bontai.ct.manager.entity.mapping.FmOrder" > SELECT o.id as id FROM fm_order o WHERE o.buyer_user_id = #{userId} AND o.deal_stat = ‘09‘ AND YEAR(o.final_time) = #{year} AND MONTH(o.final_time) = #{month} </select>