Как принудительно установить таймаут для вызова метода DriverManager.getConnection ()? - PullRequest
8 голосов
/ 31 января 2011

У меня есть приложение, которое установит соединение БД с MySQL и выполнит запросы. Иногда вызов метода DriverManager.getConnection() занимает 2 секунды, а иногда - 30 секунд. Есть ли способ контролировать этот метод для тайм-аута через 2 секунды?

DriverManager.setLoginTimeout() не работает.

На самом деле, я могу установить тайм-аут для statement.executeQuery(), спя потоком значение моего тайм-аута и закрывая соединение после пробуждения. Но это часть установления соединения, где я не смог установить время ожидания.

Буду признателен за любую помощь.

Ответы [ 6 ]

5 голосов
/ 31 января 2011

Если других вариантов нет, вы всегда можете просто выполнить вызов в отдельном потоке, который вы прерываете / игнорируете, если он не завершается в течение 2 секунд.пример того, о чем я думал:

public class Dummy extends Thread {
private volatile Connection conn = null;
@Override
public void run() {
    try {
        this.conn = DriverManager.getConnection("foobar") ;
    } catch (SQLException e) {
        e.printStackTrace();
    }
}
static public Connection getConnection() {
    Dummy d = new Dummy() ;
    d.start() ;
    try {
        Thread.sleep(2000) ;
    } catch (InterruptedException e) {}
    return d.conn ;
}
}

Затем вы можете просто вызвать статический метод Dummy.getConnection () в других местах вашего кода.Один недостаток заключается в том, что этот метод всегда будет занимать 2 секунды, но изменить его так, чтобы он возвращался сразу после завершения потока, не так уж сложно.

2 голосов
/ 03 июля 2013

Вы можете использовать ExecutorService интерфейс из Java. Ниже приведен пример того, что вам нужно сделать.

Future<Boolean> future = executor.submit(YOUR_METHOD);
future.get(TIMEOUT_YOU_NEED, TimeUnit.SECONDS);
1 голос
/ 13 мая 2013

Попробуйте установить socketTimeout (время в миллисекундах) в URL-адресе соединения или в пуле соединений (если вы используете пул).Старайтесь не устанавливать это значение слишком низким, иначе оно перезапишет значение тайм-аута оператора.

try {
    this.conn = DriverManager.getConnection("url?socketTimeout=2000") ;
} catch (SQLException e) {
    e.printStackTrace();
}

или

<jdbc-connection-pool 
                     connection-validation-method="table"
                     fail-all-connections="true"
                     idle-timeout-in-seconds="300"
                     is-connection-validation-required="true"
                     statement-timeout-in-seconds="2"
                     validation-table-name="dual"
                     ..... >
   <property name="databaseName" value="...."/>
   <property name="serverName" value="....."/>
   <property name="User" value="....."/>
   <property name="Password" value="......."/>
   <property name="URL" value="jdbc:mysql://...../...."/>
   <property name="driverClass" value="...."/>
   <property name="socketTimeout" value="2000"/>
</jdbc-connection-pool>

Установка этого значения для меня устранена.

0 голосов
/ 02 мая 2015

Я расширил ответы CodeBolt и Anpadia, чтобы сделать следующий полный урок. У него есть только один статический метод, который принимает все необходимые параметры соединения, включая время ожидания. Кроме того, метод также выдает SQLException и ClassNotFoundException, если они происходят.

Использование:

String connectionUrl = "...";
String user = "...";
String password = "...";
String driver = "org.postgresql.Driver"; // for example

int timeoutInSeconds = 5;

Connection myConnection =
    ConnectWithTimeout.getConnection(connectionUrl, user, password, driver, timeoutInSeconds);

Ниже приведена реализация:

import java.sql.DriverManager;
import java.sql.Connection;
import java.sql.SQLException;


public class ConnectWithTimeout extends Thread {

    private static String _url;
    private static String _user;
    private static String _password;
    private static String _driver;

    private static volatile Connection _connection = null;
    private static volatile boolean _sleep = true;
    private static volatile SQLException _sqlException = null;
    private static volatile ClassNotFoundException _classNotFoundException = null;

    @Override
    public void run() {
        try {
            Class.forName(_driver);
            _connection = DriverManager.getConnection(_url, _user, _password);
        }
        catch (SQLException ex) {
            _sqlException = ex;
        }
        catch (ClassNotFoundException ex) {
            _classNotFoundException = ex;
        }
        _sleep = false;
    }

    public static Connection getConnection(String url, 
                                           String user, 
                                           String password, 
                                           String driver, 
                                           int timeoutInSeconds) 
        throws SQLException, ClassNotFoundException {

        checkStringOrThrow(url,      "url");
        checkStringOrThrow(user,     "user");
        checkStringOrThrow(password, "password");
        checkStringOrThrow(driver,   "driver");

        if (timeoutInSeconds < 1) {
            throw new IllegalArgumentException(
                "timeoutInSeconds must be positive");
        }

        _url = url;
        _user = user;
        _password = password;
        _driver = driver;

        ConnectWithTimeout conn = new ConnectWithTimeout();
        conn.start();

        try {
            for (int i = 0; i < timeoutInSeconds; i++) {
                if (_sleep) {
                    Thread.sleep(1000);
                }
            }
        }
        catch (InterruptedException ex) {
        }

        if (_sqlException != null) {
            throw _sqlException;
        }

        if (_classNotFoundException != null) {
            throw _classNotFoundException;
        }

        return _connection;
    }

    private static void checkStringOrThrow(String variable, String variableName) {
        if (variable == null || variable.length() == 0) {
            throw new IllegalArgumentException(
                "String is null or empty: " + variableName);
        }
    }
}
0 голосов
/ 17 июля 2013

Спасибо codebolt, я не знаю, является ли это лучшим решением, но это работает для меня Время ожидания 10 секунд.

public class Dummy extends Thread {
             private volatile java.sql.Connection conn = null;
             private boolean sleep = true;
            @Override
             public void run() {
                 try {

                     String driver = "net.sourceforge.jtds.jdbc.Driver";
                     Class.forName(driver).newInstance();                       
                     //timeout
                     DriverManager.setLoginTimeout(10);
                     this.conn = DriverManager.getConnection(url, user, pwd);
                     sleep = false;
                 } catch (Exception e) {}
             }
             public java.sql.Connection getConnection() {
                 Dummy d = new Dummy() ;
                 d.start() ;
                 try {
                     for(int i=1; i<=10; i++) {
                         //Wait 1 second
                         if (d.sleep){
                             Thread.sleep(1000);  
                         }
                     }  
                 } catch (InterruptedException e) {}
                 return d.conn ;
             }
             }

И звонок:

Dummy a = new Dummy();
connection = a.getConnection();
if (connection != null) {....
0 голосов
/ 31 января 2011
...