`

java设置一段代码执行的超时时间的简单方法 间接实现获取Connection超时问题

阅读更多
最近有个需求, 当DB压力过大时获取Connction的时间过慢长时间不返回的话, 就不连接DB了, 研究了好久,DataSource里面的setLoginTimeOut 根本没法用, 刚开始一直纠结在大google搜索"java get connection 超时"答案上, 但始终找不到答案, 偶然尝试了下"java 设置超时" 问题就迎刃而解了.

java早已经给我们提供了解决方案。jdk1.5自带的并发库中Future类就能满足这个需求。Future类中重要方法包括get()和cancel()。get()获取数据对象,如果数据没有加载,就会阻塞直到取到数据,而 cancel()是取消数据加载。另外一个get(timeout)操作,表示如果在timeout时间内没有取到就失败返回,而不再阻塞。
看来不能一直纠结在一条道上, 偶尔换个思路还是很有帮助的,  不多说了, 解决方案如下

我的code:
	public boolean checkDBStatus() {
		boolean bdStatus = false;

		final ExecutorService exec = Executors.newFixedThreadPool(1);
		Callable<String> call = new Callable<String>() {
			public String call() throws Exception {
				DataSource dataSource = getJdbcTemplate().getDataSource();
				Connection connection = dataSource.getConnection();
				Statement statement = connection.createStatement();
				statement.executeQuery("select * from citirisk_menu_node");
				return "true";
			}
		};

		try {
			Future<String> future = exec.submit(call);
			// set db connection timeout to 10 seconds
			String obj = future.get(1000 * 10, TimeUnit.MILLISECONDS); 
			bdStatus = Boolean.parseBoolean(obj);
			System.out.println("the return value from call is :" + obj);
		} catch (TimeoutException ex) {
			System.out.println("====================task time out===============");
			ex.printStackTrace();
			bdStatus = false;
		} catch (Exception e) {
			System.out.println("failed to handle.");
			e.printStackTrace();
			bdStatus = false;
		}
		// close thread pool
		exec.shutdown();

		return bdStatus;
	}


在Java中,如果需要设定代码执行的最长时间,即超时,可以用Java线程池ExecutorService类配合Future接口来实现。 Future接口是Java标准API的一部分,在java.util.concurrent包中。Future接口是Java线程Future模式的实现,可以来进行异步计算。

Future模式可以这样来描述:我有一个任务,提交给了Future,Future替我完成这个任务。期间我自己可以去做任何想做的事情。一段时间之后,我就便可以从Future那儿取出结果。就相当于下了一张订货单,一段时间后可以拿着提订单来提货,这期间可以干别的任何事情。其中Future 接口就是订货单,真正处理订单的是Executor类,它根据Future接口的要求来生产产品。

Future接口提供方法来检测任务是否被执行完,等待任务执行完获得结果,也可以设置任务执行的超时时间。这个设置超时的方法就是实现Java程序执行超时的关键。

Future接口是一个泛型接口,严格的格式应该是Future<V>,其中V代表了Future执行的任务返回值的类型。 Future接口的方法介绍如下:

    boolean cancel (boolean mayInterruptIfRunning) 取消任务的执行。参数指定是否立即中断任务执行,或者等等任务结束
    boolean isCancelled () 任务是否已经取消,任务正常完成前将其取消,则返回 true
    boolean isDone () 任务是否已经完成。需要注意的是如果任务正常终止、异常或取消,都将返回true
    V get () throws InterruptedException, ExecutionException  等待任务执行结束,然后获得V类型的结果。InterruptedException 线程被中断异常, ExecutionException任务执行异常,如果任务被取消,还会抛出CancellationException
    V get (long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException 同上面的get功能一样,多了设置超时时间。参数timeout指定超时时间,uint指定时间的单位,在枚举类TimeUnit中有相关的定义。如果计算超时,将抛出TimeoutException

Future的实现类有java.util.concurrent.FutureTask<V>即 javax.swing.SwingWorker<T,V>。通常使用FutureTask来处理我们的任务。FutureTask类同时又实现了Runnable接口,所以可以直接提交给Executor执行。使用FutureTask实现超时执行的代码如下:

    ExecutorService executor = Executors.newSingleThreadExecutor();  
    FutureTask<String> future =  
           new FutureTask<String>(new Callable<String>() {//使用Callable接口作为构造参数  
             public String call() {  
               //真正的任务在这里执行,这里的返回值类型为String,可以为任意类型  
           }});  
    executor.execute(future);  
    //在这里可以做别的任何事情  
    try {  
        result = future.get(5000, TimeUnit.MILLISECONDS); //取得结果,同时设置超时执行时间为5秒。同样可以用future.get(),不设置执行超时时间取得结果  
    } catch (InterruptedException e) {  
        futureTask.cancel(true);  
    } catch (ExecutionException e) {  
        futureTask.cancel(true);  
    } catch (TimeoutException e) {  
        futureTask.cancel(true);  
    } finally {  
        executor.shutdown();  
    }  



不直接构造Future对象,也可以使用ExecutorService.submit方法来获得Future对象,submit方法即支持以 Callable接口类型,也支持Runnable接口作为参数,具有很大的灵活性。使用示例如下:
    ExecutorService executor = Executors.newSingleThreadExecutor();  
    FutureTask<String> future = executor.submit(  
       new Callable<String>() {//使用Callable接口作为构造参数  
           public String call() {  
          //真正的任务在这里执行,这里的返回值类型为String,可以为任意类型  
       }});  
    //在这里可以做别的任何事情  
    //同上面取得结果的代码  
分享到:
评论
5 楼 leoning12 2015-01-15  
leoning12 写道
     

4 楼 leoning12 2015-01-15  
     
3 楼 leoning12 2015-01-15  
     
2 楼 leoning12 2015-01-15  
    
1 楼 leoning12 2015-01-15  
   

相关推荐

    java通过线程控制程序执行超时(新)

    反射则允许我们在运行时动态获取类、方法和字段信息,这在一些复杂的超时控制场景中可能会用到,例如,我们需要监控某个特定方法的执行时间。 总结起来,Java通过线程控制程序执行超时是通过结合线程机制、Future/...

    java通过线程控制程序执行超时

    例如,我们可能需要在超时时停止一个正在执行的长时间方法,这可以通过反射获取该方法的`Method`对象,并调用`invoke()`方法来实现。 总结来说,Java通过线程和定时器可以有效地控制程序执行超时,结合反射可以在...

    java超时代码处理:以正则表达式设置超时时间为例

    java超时取消正则表达式匹配方法,代码超时处理,设置代码执行时间,超棒的工具类 lambda,Callable,ExecutorService,超过执行5秒退出

    Java线程超时监控

    在Java编程中,多线程是并发执行任务的重要方式,然而在实际应用中,我们可能会遇到某些线程执行时间过长或死锁的情况,这可能导致系统资源的浪费甚至整体性能下降。因此,对线程进行超时监控是必要的,以确保程序的...

    weblogic设置session超时时间

    WebLogic 是一个功能强大且流行的 Java 企业级应用服务器,支持多种方式来设置 Session 的超时时间。在本文中,我们将介绍如何在 WebLogic 中设置 Session 超时时间。 Method 1: 使用 web.xml 文件 在 Web 应用...

    C# 获取js执行之后的网页源代码(使用线程并设置超时功能)_20200712_140337.rar

    在C#编程中,有时我们需要获取网页的源代码,但有些网页的内容是通过JavaScript动态生成的,单纯使用HttpClient或WebClient等方法无法获取到完整的HTML,因为它们不会执行页面上的JavaScript代码。在这种情况下,...

    Sql Server 数据库超时问题的解决方法

    Sql Server 数据库超时问题的解决方法主要包括调整等待响应时间、在企业管理器中调整等待响应时间、在查询分析器中调整等待响应时间、在.NET 中调整等待响应时间等方法。通过这些方法,可以解决 Sql Server 数据库...

    java 访问网络 下载文件 爬虫 超时处理解决方案

    在IT领域,尤其是在Java编程中,访问网络、下载文件、实现爬虫功能以及处理超时问题是非常关键的技术点。本文将深入探讨如何在Java中有效处理这些方面的问题,特别是超时处理解决方案,这对于构建高效、稳定的网络...

    详解Nginx服务器中配置超时时间的方法

    首先,我们需要了解何时需要设置超时时间。在服务器运行过程中,如果一个请求耗时过长,可能导致服务器资源被过度占用,从而影响其他用户的正常访问。通过设定超时时间,可以在特定时间后结束无响应的请求,避免系统...

    Java中实现线程的超时中断方法实例

    Java中实现线程的超时中断方法实例 概述:在 Java 中实现线程的超时中断是非常重要的,特别是在熔断降级组件中。熔断降级组件需要在指定的超时时间内中断请求线程,以避免请求长时间阻塞系统资源。在这篇文章中,...

    Java编程Webservice指定超时时间代码详解

    // 设置超时时间 Map, Object&gt; context = ((BindingProvider) helloService).getRequestContext(); context.put(BindingProviderProperties.CONNECT_TIMEOUT, 5000); // 建立连接超时时间,单位为毫秒 context.put...

    C#函数超时执行事例

    超时处理是一种控制程序执行时间的机制,它允许我们设定一个最大执行时间,如果函数在此时间内未完成,系统将强制中断该函数的执行。这在处理可能会陷入无限循环或者执行时间过长的操作时特别有用。 在C#中,我们...

    cxf超时设置

    可以在创建服务客户端时,通过`ClientProxyFactoryBean`的`setConnectionTimeout`和`setReceiveTimeout`方法来设置超时: ```java ClientProxyFactoryBean factory = new ClientProxyFactoryBean(); factory....

    Java实现任务超时处理方法

    这种方式可以在提交任务时设置超时时间,如果任务超时,就抛出异常,并可以取消任务。 使用延时任务来终止超时操作是通过使用ScheduledExecutorService来实现的。首先,需要创建一个工作线程,例如: ```java ...

    Tcpclient连接服务器连接超时设置

    这段代码定义了一个`TcpClientWithTimeout`类,它封装了异步连接过程并设置了超时。`ConnectAsync`方法接收一个`IPEndPoint`对象,表示服务器的IP地址和端口。它创建了一个新的`Socket`对象,然后同时启动连接任务和...

    cpp-并行执行http请求支持超时设置

    通过`curl_multi_perform`可以并行执行这些请求,同时利用`curl_easy_setopt`设置超时参数,如`CURLOPT_TIMEOUT`来指定单个请求的超时时间。 压缩包中的`paw-master`可能是一个开源项目,它可能包含了实现上述功能...

    BlockingQueue队列自定义超时时间取消线程池任务

    `newFixedThreadPool`是`java.util.concurrent`包中的一个线程池工厂方法,用于创建固定数量线程的线程池。`FutureTask`则是表示异步计算的结果,它实现了`RunnableFuture`接口,提供了对任务状态的查询和结果获取。...

    java超时控制

    `java超时限制2.doc`,`java超时限制3.txt`,`java超时限制2.txt`,`java超时限制4.txt`,`java超时限制.txt`),我们可以获取更多关于Java超时控制的具体实践案例和代码示例,进一步提升我们的编程技能。...

    为jquery的ajax请求添加超时timeout时间的操作方法

    在上述代码中,我们首先使用setTimeout函数实现延时调用,这样可以在发起请求之前等待一段预设的时间。如果checkLoading函数在没有设置延时参数的情况下被调用,那么默认等待1秒钟。 除了设置超时时间以外,我们还...

    c#socket连接池和连接超时时间设置

    提供一个c# socket连接池设计的例子,解决socket并发连接限制的问题,并且提供一种设置连接超时时间的方法,默认连接超时时间是不能设置的,提供了socket网络发送数据的接口.可用于pos小票打印机通讯。

Global site tag (gtag.js) - Google Analytics