【问题标题】:Oracle 12c database connection using thin driver throws IO error使用瘦驱动程序的 Oracle 12c 数据库连接引发 IO 错误
【发布时间】:2015-06-20 17:23:43
【问题描述】:

我正在遵循 JDBC 开发人员指南并尝试使用一个简短的 java 程序测试 JDBC 瘦驱动程序连接。

import java.sql.*;
import oracle.jdbc.*;
import oracle.jdbc.pool.OracleDataSource;
class JDBCVersion
{
public static void main (String args[]) throws SQLException
{
        OracleDataSource ods = new OracleDataSource();
        ods.setURL("jdbc:oracle:thin:hr/hr@localhost:1522:orcl");
        Connection conn = ods.getConnection();
        // Create Oracle DatabaseMetaData object
        DatabaseMetaData meta = conn.getMetaData();
        // gets driver info:
        System.out.println("JDBC driver version is " + meta.getDriverVersion());
}
} //<host>:<port>:<service>

我已经尝试了所有可能的&lt;host&gt;:&lt;port&gt;:&lt;service&gt; 组合,但仍然得到java.sql.SQLRecoverableException: IO Error: The Network Adapter could not establish the connection

我已经使用教程中包含的另一个程序成功地测试了 OCI 驱动程序....但是无法让这个程序工作。我的应用程序将使用瘦驱动程序连接到数据库,所以我的挫败感是......爬。

感谢任何帮助。

【问题讨论】:

  • 即使您提供了连接的用户名和密码,是否也会出现同样的异常?
  • 是的,同样的错误。它需要用户名和密码吗?我读过几个相互矛盾的
  • 连接成功,是的。经过重新思考,我猜这个例外会有不同的措辞。你检查过主机、端口和SID吗?您将哪个连接 URL 用于 OCI?
  • OCI 使用 TNS,不需要其他信息。我会用 un/pw 再试一次并报告。
  • 您是否将主机、端口和 SID 与 TNS 名称中的值进行了比较?

标签: java sql jdbc oracle12c ojdbc


【解决方案1】:

我可以使用用户名/密码连接到我的容器数据库(包含我的表、包等)。

返回:

JDBC driver version is 12.1.0.2.0

仍然无法连接到 oracle 12c 安装附带的教程“HR”PDB 以及 JDBC 教程使用的教程。

编辑:

使用以下方法让它工作:

import java.sql.*;
import oracle.jdbc.*;
import oracle.jdbc.pool.OracleDataSource;
class JDBCVersion
{
public static void main (String args[]) throws SQLException
{       
        OracleDataSource ods = new OracleDataSource();
        ods.setURL("jdbc:oracle:thin:@//localhost:1522/pdborcl.global.XXXXXXXX.com");
        ods.setUser("hr");
        ods.setPassword("hr");
        Connection conn = ods.getConnection();
        // Create Oracle DatabaseMetaData object
        DatabaseMetaData meta = conn.getMetaData();
        // gets driver info:
        System.out.println("JDBC driver version is " + meta.getDriverVersion());
}
}

仍然不明白为什么我需要完整的全局名称而不是实例名称。

【讨论】:

    【解决方案2】:

    也许以下 cmets 可以解释为什么您需要服务名称而不是 URL 中的 SID。

    • Oracle JDBC FAQ 提到 SIDs will be cease to be supported in one of the next few releases of the database

    • Oracle JDBC devolopers guide提到Always connect to a service. Never use instance_name or SID because these do not direct to known good instances and SID is deprecated

    • Oracle 2 day + Java developer tutorial 提到语法 jdbc:oracle:driver_type:[username/password]@//host_name:port_number:SID,它似乎是 SID 和服务名称 URL 的混合体(在其他文档和您的工作示例之后)

    • 相比之下,OracleDriver 的 javadoc 仅提及 SID 语法

    • Oracle FAQ wiki 提到这两种语法

    .

    jdbc:oracle:thin:[USER/PASSWORD]@[HOST][:PORT]:SID
    jdbc:oracle:thin:[USER/PASSWORD]@//[HOST][:PORT]/SERVICE
    

    【讨论】:

    • 完美,感谢您提供的信息。我对数据库非常陌生,并且发现了很多相互矛盾的教程。我目前正在研究:JDBC Developer's Guide 12c Release 1 (12.1)
    【解决方案3】:

    连接到 PDB 时,您应该始终在连接字符串中使用 PDB 的服务名称。看起来您的 PDB 的服务是“pdborcl.global.XXXXXXXX.com”,所以您需要使用它来直接连接 PDB。

    我个人觉得使用长 URL 格式更容易:

    "jdbc:oracle:thin:@(DESCRIPTION=(ADDRESS=(PROTOCOL=tcp)(HOST=localhost)(PORT=1522))(CONNECT_DATA=(SERVICE_NAME=pdborcl.global.XXXXXXXX.com)))"

    很明显,您使用的是服务名称而不是 SID。

    它的美妙之处在于您还可以使用 sqlplus 轻松测试您的连接字符串:

    sqlplus "hr/hr@(DESCRIPTION=(ADDRESS=(PROTOCOL=tcp)(HOST=localhost)(PORT=1522))(CONNECT_DATA=(SERVICE_NAME=pdborcl.global.XXXXXXXX.com)))"

    如果 sqlplus 可以工作,那么 JDBC Thin 驱动程序没有理由不工作。

    最后你也可以使用特权用户连接根数据库,然后执行“ALTER SESSION SET CONTAINER=pdb”切换到PDB。如果您决定这样做,则必须先修改连接字符串以连接到根容器。它应该有自己的服务名称。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 2013-01-09
      • 2010-11-05
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2013-12-23
      • 2012-11-20
      相关资源
      最近更新 更多