菜园子

BlogJava 首页 新随笔 联系 聚合 管理
  7 Posts :: 1 Stories :: 31 Comments :: 0 Trackbacks

常用链接

留言簿(1)

随笔档案

文章分类

文章档案

搜索

最新评论

阅读排行榜

评论排行榜

在我们开发的一个系统中,有定时任务,自然就想到了Quartz,由于框架采用的Spring,Quartz跟Spring的集成也非常简单,所以就把Quartz配置到框架中,当系统启动后,定时任务也就自动启动。在开发的过程中一直没有发现问题,但是最后上线的时候,采用的是weblogic cluster,启动了4个节点,发现有的定时任务执行了不止一次,才恍然大悟,4个节点启动了4个应用,也就启动了4个定时任务,所以在同一个时间定时任务执行了不止一次。去网上搜索,发现Quartz也支持cluster,但是我觉得就我们的系统而言,没有必要采用cluster的定时任务,也许是比较懒吧,就想让定时任务只执行一次。在网上搜到了robbin的一篇文章(http://robbin.iteye.com/blog/40989 ),发现把quartz集中到webapp当中还是有一定的风险,同时同一个时间点执行也不止一次。Robbin的解决办法就是自己单独启动一个Job Server,来quartz跑job,不要部署在web容器中。 

我也比较同意这个办法。鉴于时间比较紧,就想有没有比较方便的方法。其实把原来的webapp当做一个quartz的容器就可以了。可以自己写一个线程来跑应用,再写一个command启动这个线程就可以了。线程类很简单,如下:

public class StartServer {

    public static void main(String[] args) throws Exception {

        ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext(
new String[] { "/spring/context-annotation.xml","/spring/context-transaction.xml",
"/spring/context-hibernate.xml",
"/spring/context-quartz.xml"});

        System.out.println("start server....");

        while (true) {

            try {

                Thread.sleep(900);

            } catch (InterruptedException ex) {

            }

        }

    };

}

去掉了系统的controller配置servlet.xml,运行这个类就可以了。

在web-inf目录下写一个command来启动这个java类:

setlocal ENABLEDELAYEDEXPANSION

if defined CLASSPATH (set CLASSPATH=%CLASSPATH%;.) else (set CLASSPATH=.)

FOR /R .\lib %%G IN (*.jar) DO set CLASSPATH=!CLASSPATH!;%%G

Echo The Classpath definition is==== %CLASSPATH%

set CLASSPATH=./classes;%CLASSPATH%

java com.company.job.StartServer

这个command需要把需要的jar(web-inf/lib中)包都放到classpath中。

每次启动的时候执行这个command就可以了。跟原来的应用分开了,调试起定时任务也不用影响到原来的应用,还是比较方便的。部署的时候原样拷贝一份,然后执行这个command就好了,部署起来也比较方便。

 



QQ:24889356
posted on 2011-09-13 12:53 GhostZhang 阅读(2647) 评论(3)  编辑  收藏

Feedback

# re: 一次定时任务 2011-09-14 08:55 tb
恩 不错   回复  更多评论
  

# re: 一次定时任务 2011-09-15 18:19 Mister4
可以根据服务器的iP做个判断  回复  更多评论
  

# re: 一次定时任务[未登录] 2011-09-30 15:57 BucketLi
这个简单处理方式有很多. 数据库搞张任务表,放一条记录,每个节点先取这条记录(任务状态是可执行),然后再通过update将value+1并且更新状态,带上先前查询出来的value作为查询条件,这样相当于加了一把乐观锁,因为数据库底层是原子的,所以只有一台机器会更新成功. 这样就达到目的了. 还有稍微复杂点的是通过zk来维持一把任务锁,这样执行任务的只有一台机器,挂掉后另外一台机器抢到锁开始做事情. 当然还有其他方法  回复  更多评论
  


只有注册用户登录后才能发表评论。


网站导航: