日期:2014-05-16  浏览次数:20685 次

Oracle JDBC 连接时的一个BUG

前两天接到一个工作,某网站无法访问了,重启后也一直挂在哪里,起不来。

我刚开始怀疑是程序配置问题,后来发现,只要初始化Spring内容服务就起不来。看了一下spring配置文件里面配置了两个数据源,都是由tomcat管理的jndi数据源。查看了一下发现有一个数据源有问题,使用小程序测试,发现无法连接该数据库,同时也发现一个问题,程序一直挂在哪里,是否能连接却不再返回结果。

?

网上也有人遇到过这类问题,看来都没有很好的解决方案,当然,我也没有。

我现在只是说明一下问题,并复现一下场景。

你可以使用附件内调试工具或自己下载一个TCP调试工具,然后开启一个端口,然后写一个小程序把这个端口当成Oracle服务端口去访问。

?

?

package com;
import java.sql.Connection;
import java.util.Properties;
import oracle.jdbc.driver.OracleDriver;
/**
 * 数据库连接对象管理类
 * @说明
 * @author cuisuqiang
 * @version 1.0
 * @since
 */
public class ConnectionManager {
	private static final String url = "jdbc:oracle:thin:@192.168.1.155:1521:orcl";
	private static final String username = "scott";
	private static final String userpass = "bi123";
	public static void main(String[] args) throws Exception {
		Connection conn = getConnection("1");
		if (null != conn) {
			System.out.println(conn.toString());
		} else {
			System.out.println("NO LINK");
		}
	}
	public static Connection getConnection(String tar) {
		Connection conn = null;
		try {
			OracleDriver driver = new OracleDriver();
			Properties properties = new Properties();
			properties.put("user", username);
			properties.put("password", userpass);
			conn = driver.connect(url, properties);
		} catch (Exception e) {
			e.printStackTrace();
		}
		return conn;
	}
}

?

看服务窗口:

?

可以看到,程序发来了登录信息,但是我们不回复,然后就会看到,程序死在了:

conn = driver.connect(url, properties);

?获得连接这里!

?

希望有志之士提供解决方案!

26 楼 cuisuqiang 24 小时前  
看来大家看惯了:
Class.forName(driver);
conn = DriverManager.getConnection(url, username, userpass);

,我这么写就有点水土不服了:
OracleDriver driver = new OracleDriver();
Properties properties = new Properties();
properties.put("user", username);
properties.put("password", userpass);
conn = driver.connect(url, properties);

首先肯定的说,不是写法问题!

其次,程序一直运行,突然出问题,我也是从各个角度排查后才发现电信方面数据库的问题!可能是我太相信他们了!

其次,说到解决问题,问题已经解决。但是Oracle驱动发出登录消息后就一直死等,这种做法是否欠妥?
27 楼 xurichusheng 24 小时前  
Oracle驱动发出登录消息后就一直死等,这样的情况,我能想到的原因有:
1. 由于某种原因,oracle被锁住了,或者说,oracle拒绝了访问,可能是oracle的锁机制造成的;
2. 多线程下的同步问题;
3. oracle的某个服务由于某种原因被停止了;

在调试多线程的时候,经常会遇到这样的问题
debug的时候,就死在某行代码下,再也运行不了了,
当前线程变成了守护线程,死锁了
这个时候程序是不会报异常的,你得把所有相关的类、方法全都调试
而且有可能是你调试1天都找不到原因
这种让人抓狂的问题,得是对线程或者说对java相关API类有深入了解的人才能解决
28 楼 huaye2007 23 小时前  
http://blog.csdn.net/downloadsunlight2009/article/details/7579058
在jdbc的标准接口中,有DriverManager.setLoginTimeout方法,用来设置建立数据库连接的超时.我在对sql   server进行过测试,该方法是能够生效的.但是如果对于oracle(8i,9i),该方法没有效果.查阅oracle的官方文档,结果竟然是 "Oracle   JDBC   does   not   support   login   timeouts.   Calling   the   static   DriverManager.setLoginTimeout()   method   will   have   no   effect ",没有想到oralce   jdbc   driver竟然没有实现这个功能,
29 楼 rensanning 20 小时前  
properties.put("oracle.jdbc.ReadTimeout", "10000"); 
properties.put("oracle.net.CONNECT_TIMEOUT", "10000"); 
30 楼 masuwen