java web定时服务的几种方法

3 篇文章 0 订阅
1 篇文章 0 订阅
  1. 监听(Listener)
    首先,我们创建一个监听的类,继承ServletContextListener,如下:
    [java]  view plain copy
    1. package com.wxp.thread;  
    2.   
    3. import javax.servlet.ServletContextEvent;  
    4. import javax.servlet.ServletContextListener;  
    5.   
    6. /** 
    7.  * Listener的方式在后台执行一线程 
    8.  *  
    9.  * @author Champion.Wong 
    10.  *  
    11.  */  
    12. public class MyListener implements ServletContextListener {  
    13.     private MyThread myThread;  
    14.   
    15.     public void contextDestroyed(ServletContextEvent e) {  
    16.         if (myThread != null && myThread.isInterrupted()) {  
    17.             myThread.interrupt();  
    18.         }  
    19.     }  
    20.   
    21.     public void contextInitialized(ServletContextEvent e) {  
    22.         String str = null;  
    23.         if (str == null && myThread == null) {  
    24.             myThread = new MyThread();  
    25.             myThread.start(); // servlet 上下文初始化时启动 socket  
    26.         }  
    27.     }  
    28. }  
    29.   
    30. /** 
    31.  * 自定义一个 Class 线程类继承自线程类,重写 run() 方法,用于从后台获取并处理数据 
    32.  *  
    33.  * @author Champion.Wong 
    34.  *  
    35.  */  
    36. class MyThread extends Thread {  
    37.     public void run() {  
    38.         while (!this.isInterrupted()) {// 线程未中断执行循环  
    39.             try {  
    40.                 Thread.sleep(2000); //每隔2000ms执行一次  
    41.             } catch (InterruptedException e) {  
    42.                 e.printStackTrace();  
    43.             }  
    44.               
    45. //           ------------------ 开始执行 ---------------------------  
    46.             System.out.println("____FUCK TIME:" + System.currentTimeMillis());  
    47.         }  
    48.     }  
    49. }  

    然后,在web.xml中配置如下:
    [html]  view plain copy
    1. <listener>  
    2.     <listener-class>com.wxp.thread.MyListener</listener-class>  
    3. </listener>  

    OK,启动项目,我们可以在控制台看到隔时间段输出的文字内容。
  2. 使用Servlet,在项目启动的时候启动它。
    首先,创建一个Servlet,继承HttpServlet
    [java]  view plain copy
    1. package com.wxp.thread;  
    2.   
    3. import java.io.IOException;  
    4. import javax.servlet.ServletException;  
    5. import javax.servlet.http.HttpServlet;  
    6. import javax.servlet.http.HttpServletRequest;  
    7. import javax.servlet.http.HttpServletResponse;  
    8. import com.ite.common.Constants;  
    9.   
    10. public class MyServlet extends HttpServlet{  
    11.   
    12.     private static final long serialVersionUID = 1L;  
    13.     private MyThread1 myThread1;  
    14.       
    15.     public MyServlet(){  
    16.     }  
    17.   
    18.     public void init(){  
    19.         String str = null;  
    20.         if (str == null && myThread1 == null) {  
    21.             myThread1 = new MyThread1();  
    22.             myThread1.start(); // servlet 上下文初始化时启动 socket  
    23.         }  
    24.     }  
    25.   
    26.     public void doGet(HttpServletRequest httpservletrequest, HttpServletResponse httpservletresponse)  
    27.         throws ServletException, IOException{  
    28.     }  
    29.   
    30.     public void destory(){  
    31.         if (myThread1 != null && myThread1.isInterrupted()) {  
    32.             myThread1.interrupt();  
    33.         }  
    34.     }  
    35. }  
    36.   
    37. /** 
    38.  * 自定义一个 Class 线程类继承自线程类,重写 run() 方法,用于从后台获取并处理数据 
    39.  *  
    40.  * @author Champion.Wong 
    41.  *  
    42.  */  
    43. class MyThread1 extends Thread {  
    44.     public void run() {  
    45.         while (!this.isInterrupted()) {// 线程未中断执行循环  
    46.             try {  
    47.                 Thread.sleep(2000);  
    48.             } catch (InterruptedException e) {  
    49.                 e.printStackTrace();  
    50.             }  
    51.               
    52.             // ------------------ 开始执行 ---------------------------  
    53.             System.out.println("____FUCK TIME:" + System.currentTimeMillis());  
    54.         }  
    55.     }  
    56. }  

    然后,在web.xml中配置
    [html]  view plain copy
    1. <!-- LISTENER FOR THREAD -->  
    2. <servlet>  
    3.     <servlet-name>MyListener</servlet-name>  
    4.     <servlet-class>com.ite.wxp.MyServlet</servlet-class>  
    5.     <load-on-startup>9</load-on-startup><!-- 数字越小,启动的优先级越高,必须大于0 -->  
    6. </servlet>  
    7. <servlet-mapping>    
    8.         <servlet-name>MyListener</servlet-name>    
    9.         <url-pattern>/test</url-pattern>     
    10. </servlet-mapping>  

    OK。启动项目,依然可以看到如图所示:


                          两种方法都是在JavaWeb 服务启动时,在后台启动加载一个线程。其中,Listener的方式,该线程肯定是项目中首先启动的,优先于任何一个Servlet。而Servlet的方式,可以设置与其它Servlet启动的顺序。如果有时候需要首先启动一个Servlet或者它们之间的启动顺序有特殊要求的时候,这个就很有作用了。


3.Spring+Quartz实现定时任务的配置方法

1、Scheduler的配置

<bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">  
       <property name="triggers">  
           <list>  
              <ref bean="testTrigger"/>  
           </list>  
       </property>  
       <property name="autoStartup" value="true"/>  
</bean>  
    说明:Scheduler包含一个Trigger列表,每个Trigger表示一个作业。

2、Trigger的配置

<bean id="testTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">  
       <property name="jobDetail" ref="testJobDetail"/>  
       <property name="cronExpression" value="*/1 * * * * ?"/><!-- 每隔1秒钟触发一次 -->  
</bean>  
    说明:

         1)Cron表达式的格式:秒 分 时 日 月 周 年(可选)。

               字段名                 允许的值                        允许的特殊字符  

               秒                         0-59                               , - * /  

               分                         0-59                               , - * /  

               小时                   0-23                               , - * /  

               日                         1-31                               , - * ? / L W C  

               月                         1-12 or JAN-DEC          , - * /  

               周几                     1-7 or SUN-SAT            , - * ? / L C #  

               年 (可选字段)     empty, 1970-2099      , - * /

               “?”字符:表示不确定的值

               “,”字符:指定数个值

               “-”字符:指定一个值的范围

               “/”字符:指定一个值的增加幅度。n/m表示从n开始,每次增加m

               “L”字符:用在日表示一个月中的最后一天,用在周表示该月最后一个星期X

               “W”字符:指定离给定日期最近的工作日(周一到周五)

               “#”字符:表示该月第几个周X。6#3表示该月第3个周五

         2)Cron表达式范例:

                 每隔5秒执行一次:*/5 * * * * ?

                 每隔1分钟执行一次:0 */1 * * * ?

                 每天23点执行一次:0 0 23 * * ?

                 每天凌晨1点执行一次:0 0 1 * * ?

                 每月1号凌晨1点执行一次:0 0 1 1 * ?

                 每月最后一天23点执行一次:0 0 23 L * ?

                 每周星期天凌晨1点实行一次:0 0 1 ? * L

                 在26分、29分、33分执行一次:0 26,29,33 * * * ?

                 每天的0点、13点、18点、21点都执行一次:0 0 0,13,18,21 * * ?

3、JobDetail的配置

<bean id="testJobDetail" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">   
        <property name="targetObject" ref="testJob"/>  
        <property name="targetMethod" value="execute"/>  
        <property name="concurrent" value="false"/>
        <!-- 是否允许任务并发执行。当值为false时,表示必须等到前一个线程处理完毕后才再启一个新的线程 -->  
</bean>  
4、业务类的配置

<bean id="testJob" class="com.cjm.web.service.quartz.TestJob"/>  

5、业务类源代码

public class TestJob {  
    public void execute(){  
        try{  
              //.......
         }catch(Exception ex){  
             ex.printStackTrace();  
         }  
     }  
}  
    说明:业务类不需要继承任何父类,也不需要实现任何接口,只是一个普通的java类。

注意:

     在Spring配置和Quartz集成内容时,有两点需要注意

           1、在<Beans>中不能够设置default-lazy-init="true",否则定时任务不触发,如果不明确指明default-lazy-init的值,默认是false。

           2、在<Beans>中不能够设置default-autowire="byName"的属性,否则后台会报org.springframework.beans.factory.BeanCreationException错误,这样就不能通过Bean名称自动注入,必须通过明确引用注入

前两种方法转自http://blog.csdn.net/it_wangxiangpan/article/details/7168286

第三种方法转自http://www.oschina.net/question/8676_9032?fromerr=Cd8yGJlU

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Java Web项目中,可以通过以下步骤来限制接口每天只允许调用5万次: 1. 创建一个计数器类来记录接口的调用次数,并在计数器类中设置一个计数器变量,用于存储接口的调用次数。 2. 在计数器类中创建一个定时任务,用于在每天的零点重置计数器变量。 3. 在每个需要进行接口调用次数限制的接口方法中,首先从计数器类中获取当前接口的调用次数,如果当前次数小于5万,则增加计数器变量的值,并继续执行接口的逻辑;否则,返回一个错误提示信息,告知用户接口调用次数已经达到上限。 4. 在项目启动时,创建计数器类的单例实例,并将其存储在全局变量中,以便在所有的接口方法中共享计数器实例。 下面是一个示例代码,演示如何实现接口每天只允许调用5万次的限制: ```java public class ApiCounter { private static ApiCounter instance = new ApiCounter(); private int count = 0; private Date lastResetTime; private ApiCounter() { reset(); scheduleResetTask(); } public static ApiCounter getInstance() { return instance; } public int getCount() { return count; } public void increaseCount() { count++; } public void reset() { count = 0; lastResetTime = new Date(); } private void scheduleResetTask() { TimerTask task = new TimerTask() { @Override public void run() { reset(); } }; Timer timer = new Timer(); Calendar calendar = Calendar.getInstance(); calendar.set(Calendar.HOUR_OF_DAY, 0); calendar.set(Calendar.MINUTE, 0); calendar.set(Calendar.SECOND, 0); Date firstResetTime = calendar.getTime(); if (firstResetTime.before(new Date())) { calendar.add(Calendar.DAY_OF_MONTH, 1); firstResetTime = calendar.getTime(); } timer.schedule(task, firstResetTime, TimeUnit.MILLISECONDS.convert(1, TimeUnit.DAYS)); } } ``` 在需要进行接口调用次数限制的接口方法中,可以使用如下代码来限制每天最多调用5万次: ```java public void myApiMethod() { ApiCounter counter = ApiCounter.getInstance(); if (counter.getCount() >= 50000) { throw new RuntimeException("The daily limit of 50,000 calls has been reached."); } counter.increaseCount(); // Continue with the rest of the method logic. } ``` 这样,每次调用 myApiMethod 方法时,会先从 ApiCounter 实例中获取当前的接口调用次数,如果超过了5万,则会返回一个错误提示信息,否则会增加计数器变量的值,并继续执行接口的逻辑。同时,定时任务会在每天的零点重置计数器变量,从而实现每天最多调用5万次的限制。 ### 回答2: 在Javaweb项目中,限制接口每天只允许调用5万次可以通过以下步骤实现: 1. 在项目中引入一个计数器的功能,用于记录每次接口的调用次数。 2. 在项目中定义一个计数器变量,用于记录当天已经调用的次数。可以使用静态变量或者数据库中的计数器表来存储这个值。 3. 在每次接口请求的地方,进行计数器的递增操作,即每次接口调用时将计数器变量加1。 4. 在接口调用之前,添加一个计数器检查功能,用于检查当前接口调用次数是否已经达到了上限。 5. 如果计数器检查结果显示接口调用次数已经达到上限(即达到5万次),则禁止继续调用接口,并返回相应的错误提示信息。 6. 如果计数器检查结果显示接口调用次数还未达到上限,那么就允许继续调用接口,继续执行接口的业务逻辑。 7. 在每天的第一个接口调用时,重置计数器变量,将计数器重置为0,重新开始计数。 通过上述步骤,就可以实现对接口每天调用次数的限制,确保每天只允许调用5万次。当然,具体的实现方式还可以依据项目框架的不同而有所调整,例如可以使用拦截器、过滤器等技术来进行计数器的检查和控制。 ### 回答3: 在Javaweb项目中限制接口每天只允许调用5万次的方法有很多种,下面介绍几种常用的方法: 1. 使用计数器:通过在代码中加入计数器,每次接口被调用时,计数器加1。在数据库或缓存中保存计数器的值,并定时清零。当计数器的值达到5万次时,后续的请求都会被拒绝。 2. 使用限流算法:可以使用像漏桶算法或令牌桶算法等限流算法,对接口的调用进行限制。这些算法可以设置令牌生成速率,每次调用接口时,消耗令牌,当令牌数耗尽时拒绝后续请求。 3. 使用定时任务:可以通过定时任务来监控接口的调用次数,并在每天的零点将计数器清零。通过定时任务可以定期重置接口的调用次数,保证每天只允许调用5万次。 4. 使用限流框架:可以借助一些开源的限流框架来实现接口调用次数的限制,如Guava RateLimiter。这些框架提供了方便的接口限流功能,可以设置接口的访问速率,限制每天只允许调用5万次。 需要指出的是,以上方法只是对接口调用次数进行限制,并不能阻止恶意攻击或绕过限流的行为。如果需要更加安全可靠的限制,还需要考虑使用其他方式,如IP限制、验证码等手段来进一步加强接口的访问控制。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值