第1部分:应用程序连接数据的简单方法
前言
你想不想写自己的JDBC驱动,或者是否想把现在的JDBC驱动转换成Type3的驱动呢?Type3 的驱动主要用于无需客户端设置的Internet/Intrant应用程序中,并且提供灵活的系统管理工具。本文中Nitin Nanda 和 Sunil Kumar描述了JDBC驱动的架构并概略讲解了Type3 JDBC驱动的三层结构。并演示了如何创建自己的Type3 JDBC驱动。
在Java中访问数据库系统你需要一个JDBC驱动。你可以自己写Type1到Type4的JDBC驱动,可以是纯Java的方式或者使用Java结合本地接口方法的复合方式。产品化的方式趋向于利用纯Java技术创建Type3或Type4的驱动程序。在Internet应用程序中访问数据库的效率和稳定性成为了关键因素,这时Type3 JDBC驱动显示出了它的优势。所以,要开发高效的、易部署的Internet应用,你通常需要将现在的Type1或Type2类型的JDBC替换成Type3类型的。
整个文章分为3个部分:
1. 应用程序连接数据库的简单方法。
2. 使用自定义的JDBC驱动编译、部署、访问数据。
3. 通过高级的日志功能、连接池、预处理数据集增强自定义的JWDriver的性能。
在这三个部分中,我们首先介绍了我们自己的Type3 JDBC驱动的结构和设计(第一部分),然后讲了如何实现和部署整个驱动(第二部分),最后讲解了如何在其中增加一些高级特性,比如SQL日志、连接池等(第三部分)。
注意:
在开始之前,也许你可以先看一下Nitin Nanda's的"JDBC Drivers in the Wild",这更有利于你理解JDBC驱动。
JDBC驱动的结构
JDBC为通用的数据库访问提供了一种程序级的接口。在一个特定的数据库环境下使用JDBC API你需要一个JDBC驱动来完成他们直接的协调运行。JDBC被分为四种类型或者说层次。每一个类型都有一个JDBC实现,以便满足日益增加的对平台独立性、执行效率和可管理性的要求。这四种类型分别是:
Type1:JDBC-ODBC(开方数据库连接)桥。
Type2:本地API,部分的Java驱动。
Type3:网络协议,完全的Java 驱动。
Type4:本地协议,完全的Java驱动。
所有的JDBC驱动都要实现下面四个JDBC类:Driver、Connection、Statement和ResultSet。包含在java.sql包种DriverManager类用于管理被加载的驱动。客户端应用程序通过它获得数据库连接。JDBC的Driver通过下面的方法加载:
Class.forName("com.jw.client.JWDriver");
当这个Driver被加载时它的staic部分的代码就会执行,通过DriverManager注册这个驱动。现在,只要客户端应用程序通过DriverManger.getConnection()方法,DriverManager就会转而调用Driver.connect()方法。每一个JDBC驱动必须实现java.sql.Driver接口。所以,JDBC驱动的Driver.connect方法会首先检测当前的驱动Url是否正确,然后会通过connet()方法返回一个Connection对象。
简单Type3驱动的结构
为了向你展示Type3驱动的内容结构,我创建了一个自己的Type3JDBC驱动。如图1所示,我们的Type3驱动是基于网络协议、完全Java实现的,它分为3个层次,同时向你展示了JDBC是如何通过中间层的网络协议进行通信的。中间层服务讲请求传输(直接或间接的)给特定的数据库本地连接接口,然后通过这个接口将请求传输到数据库服务器。中间层用Java实现,数据库访问的实现通过JDBC-ODBC桥。
图1 JDBC Type3 的结构
对于applet,客户层驱动文件位于中间层服务器,并且可以直接下载。在JDBC驱动种RMI提供网络协议服务,负责驱动的客户端和服务端的通信。JDBC-ODBC桥使得JDBC请求可以传输到数据库服务器。
驱动的客户层为客户端程序提供了标准的JDBC接口,包括实现了java.sql.Driver接口的Driver类。同时还包含了JDBC的Connection、Statement、Result接口的实现。
客户端应用程序,比如applet,可以使用Type3驱动的客户层类进行开发,因为它实现了标准的JDBC接口拥有标准的JDBC功能。客户层类的Driver在内部会维护与中间层提供的远程接口之间的通信。这些远程接口提供了客户端传输JDBC请求要用到的基本方法。应用程序会调用已经实现了的Driver中的JDBC方法,随后这些调用会通过RMI发送到中间层。所有Driver类的客户层通过中间层来管理内部的RMI通信。
驱动的服务层是一个RMI服务器,它通过JDBC-ODBC桥来和数据库进行通信。驱动的服务层包含4个远程接口以及它们的实现,这些远程接口分别为JDBC的Driver、Connection、Statement和Result提供了访问接口。实现这些远程接口的类会在内部维护JDBC-ODBC的桥驱动的Connection、Statement和Result对象。当客户层类调用远程接口的时候,远程接口的实现使用内部的JDBC对象与数据库通讯。
现在你已经了解了Type3 JDBC驱动的接口,下面让我们进一步的深入到客户层和服务器层中。
JDBC类驱动图
为了实现Type3 JDBC驱动,我们必须创建客户层和中间层。客户层类位于com.jw.client包中,中间层位于com.jw.server包中,让我们先来看看客户层。
客户层类
客户层包com.jw.client中包含下面几个类:
com.jw.client.JWDriver类:JDBC驱动(Driver)的实现类。
com.jw.client.JWConnection类:JDBC连接(Connection)的实现类。
com.jw.client.JWStatement类:JDBC表达式(Statement)的实现类。
com.jw.client.JWRseultSet类:JDBC结果集(ResultSet)的实现。
如图2所示,它描述了客户层和中间层Driver和Connection类之间的关系。
图2:Driver和Connection类图
让我们仔细的看看JWDriver类的内部:
JWDriver类
com.jw.client.JWDriver类实现了java.sql.Driver接口,它提供了通过DriverManager类注册自己并创建数据库连接的方法。这个类实现了远程Driver的封装以便提供JDBC驱动接口。在应用程序中通过下面的方法调用可以加载JWDriver:
Class.forName("com.jw.client.JWDriver");
当它通过DriverManager类加载后,上面的代码中forName方法会调用JWDriver类的static部分的代码。下面显示的代码中的static部分:
static
{
try
{
// Register the JWDriver with DriverManager
JWDriver driverInst = new JWDriver();
DriverManager.registerDriver(driverInst);
System.setSecurityManager(new RMISecurityManager());
}
...
}
JWDriver类同时还维护一个远程驱动的引用,com.jw.server.IremoteDriver,它位于中间层服务器。远程驱动的引用为JWDriver创建了数据库的连接。所以当应用程序需要获得一个数据库连接的时候就调用DriverManager.getConnection()方法,这时DriverManager的JWDriver.connect()方法会通过远程驱动的引用获得一个数据库连接。
简而言之,JWDriver.connect()方法完成如下超作:
? 比较客户端程序传递的URL如果是一个不匹配的URL就返回一个null。
? 如果不存在远程驱动引用就通过Naming.lookup()方法创建一个。JWDriver.connet()方法控制远程驱动来创建数据库连接。:
if(remoteDriver == null)
{
remoteDriver= RemoteDriver)Naming.lookup
("rmi://"+serverName
+":1099"+"/RemoteDriver");
}
? 创建、使用如上方法创建的数据库连接,并将创建的数据库连接返回给调用它的应用程序。RemoteDriver.getConnetion()方法是中间层JDBC驱动的远程调用,从中间层获得的远程Connection作为一个应用保存在JWConnection类对象中。JWConnection类可以通过下面的方法获得:
{
IRemoteConnection remoteConInstance =
(IRemoteConnection)remoteDriver.getConnection();
localConInstance = new JWConnection
(remoteConInstance);
.
..
return (Connection)localConInstance;
}
下面我们一起来仔细的看看JWConnection类。
JWConnection类
Com.jw.client.JWConnection类实现了Connection接口,而且它还包含一个远程服务器IremoteConnection接口的引用。
客户端程序中的JWDriver.connect方法可以得到JWConnection对象的引用并创建一个JWConnection对象。这样客户端就可以通过JWConnection对象任意的调用JDBC Connection接口的方法了。JWConnection在内部调用远程服务器的Connection。比如当客户端调用conn.createStatement()方法时(conn是一个JWConnection的引用),其内部会调用一个RemoteConnection.createConnection()方法,并返回一个远程的Statement引用;JWConnectoin的createStatement方法会创建一个JWStatement对象;最后一个JWStatement对象会返回个客户端,这个对象中包含的Statement的引用:
public Statement createStatement() throws
SQLException
{
try
{
IRemoteStatement remStmt =
(IRemoteStatement) remoteConnection.createStatement();
JWStatement localStmtInstance = new JWStatement(remStmt);
return (Statement)localStmtInstance;
}
}
图3描述了客户层和中间层中Connection和Statemennt类之间的关系。
图 Connection和Statement之间的关系
下面再让我们来看看JWStatement类
JWStatement类
com.jw.clent.JWStatement类实现了JDBC Statement接口,同时引用了远程服务器的Statement接口。这个类封装了远程Statement存根,它提供了JDBC Statement接口,JWConnection.createStatemnt()方法创建了JWStatement对象并返回给客户端。这样客户端就可以使用JDBC Statement接口提供的所有方法,在内部JWStatement调用了远程的Statement。比如客户端调用stmt.executeQuery()方法,这个方法内部会调用RemoteStatement.executeQuery()方法并返回一个ResulstSet引用。最终,JWResultSet对象会返回到客户端,这个对象包含了ResultSet的引用:
public ResultSet executeQuery(String sqlQuery)
throws SQLException
{
try
{
IRemoteResultSet remoteRsInstance =
(IRemoteResultSet) remoteStmt.executeQuery(sqlQuery);
JWResultSet localRsInstance = new JWResultSet(remoteRsInstance);
return (ResultSet)localRsInstance;
}
catch(RemoteException ex)
{
throw(new SQLException(ex.getMessage()));
}
}
图4描述了客户层和中间层中Statement和ResultSet类之间的关系。
图4 Statement和ResultSet类之间的关系
下面我们来看看JWResultSet类。
JWResultSet类
com.jw.clent. JWResultSet类实现了JDBC ResultSet接口,同时引用了远程服务器的ResultSet接口。这个类封装了远程ResultSet存根,它提供了JDBC ResultSet接口。JWStatement.executeQuery()方法创建想客户端返回一个JWResult对象,它包含了Result接口的引用。因此客户端程序可以使用JDBC Result提供的任何方法,JWResult对象内部会调用远程服务器的ResultSet。比如当客户端调用rs.next(rs一个一个JWResultSet对象的引用),next()方法内部会调用RemoteResultSet.getNextRow(),这个方法会返回一个包含行数据的数组。当客户端调用rs.getString(1)方法时会返回保存在JWResultSet中行数据的当前值。
public class JWResultSet implements
java.sql.ResultSet
{
// The current ResultSet data row
private Object[] row;//自己替换[]
...
public boolean next() throws SQLException
{
try
{
// Get the current data row from remote
ResultSet
// All the getXXX methods will get data from
local 'row'
row = remoteResultSet.getNextRow();
}
catch(Exception ex)
{
return false;
}
if(row == null)
{
return false;
}
return true;
}
}
中间层类
中间层类位于com.jw.server包中,包括下面几个类:
? com.jw.server.RemoteDriverImpl类:这是一个驱动类它负责获得数据库连接,同时作为RMI服务器。
? com.jw.server.RemoteConnectionImpl类:是一个Connection 类,并且它负责获得JDBC Statement。
? com.jw.server.RemoteStatementImpl类:是一个Statement类,并且它负责获得JDBC ResultSet。
? com.jw.server.RemoteResultSetImpl类:是一个ResultSet类。
下面我们详细分析每个类:
RemoteDriverImpl
RemoteDriverImpl扮演一个RMI服务器(它继承了UnicastRemoteObject)并向JDBC驱动客户端提供了getConnection()方法。图2显示了RemoteDriverImpl的其他类与接口之间的关系。
com.jw.server.IremoteDriver是一个扩展了Remote的接口,它提供了getConnection()方法。这个方法创建JDBC Connection并作为一个IremoteConnection的引用返回给客户端。
由于RemoteDriverImpl同时还是JDBC驱动服务器的RMI服务器,它会读取ODBC的数据源名称(DSN),从部署在服务层中的DriverSettings.properties
文件读取用户名和密码。这些值可以创建一个JDBC Connection,此外,RemoteDriverImpl可以通过名称服务“RemoteDriver”注册自身,所有JDBC 的客户层可以通过Naming.lookup()方法与服务端连接。然后它会加载JDBC-ODBC桥来创建数据库连接:
public static void main(String args[])//自己替换[]
{
System.setSecurityManager(new RMISecurityManager());
try
{
// Get the data source name, data source user, data
source
// Password and log level
ResourceBundle settingsBundle =
ResourceBundle.getBundle(
"DriverSettings");
DSN = settingsBundle.getString("DSN");
dsUser = settingsBundle.getString("User");
dsPassword = settingsBundle.getString
("Password");
// Create a RemoteDriverImpl instance to register
with naming service
RemoteDriverImpl serverInstance = new
RemoteDriverImpl();
Naming.rebind
("RemoteDriver",serverInstance);
// Load the JDBC-ODBC Bridge driver
Class.forName("sun.jdbc.odbc.JdbcOdbcDriver");
}
...
}
getConnection()方法创建了JDBC-ODBC连接,然后返回RemoteConnectionImpl对象远程接口的引用来控制JDBC-ODBC连接:
public IRemoteConnection getConnection()
throws RemoteException,SQLException
{
String URL="jdbc:odbc:"+DSN;
Connection sqlCon =
DriverManager.getConnection(URL,dsUser,dsPassword);
RemoteConnectionImpl ConnectionInstance =
new RemoteConnectionImpl(sqlCon);
return (IRemoteConnection)ConnectionInstance;
}
下面让我们来看看RemoteConnectionImpl。
RemoteConnectionImpl类
RemoteConnectionImpl继承了UnicastRemoteObject类并实现了IremoteConnection接口(图2、3显示了RemoteConnectionImpl类的其他类和接口的关系)。IremoteConnection接口提供了创建JDBC Statement和关闭数据库连接的方法,RemoteConnectionImpl同时封装了传入构造中的JDBC连接。
createStatement()创建了JDBC-ODBC Statement并返回RemoteConnectionImpl对象的引用来控制JDBC-ODBC Statement:
public IRemoteStatement createStatement() throws
RemoteException,SQLException
{
RemoteStatementImpl StmtImplInstance = new
RemoteStatementImpl(sqlConnection.createStatement());
return (IRemoteStatement)StmtImplInstance;
}
接下来closeConnection()方法通过内部封装的close()方法关闭JDBC Connection:
public void closeConnection() throws
RemoteException,SQLException
{
sqlConnection.close();
}
下面再看RemoteStatementImpl。
RemoteStatementImpl类
如图3、4中显示的RemoteStatementImpl类,它继承了UnicastRemoteObject类并实现了IremoteStatement接口。这个接口提供了创建JDBC ResultSet和关闭Statement对象的方法。RemoteStatementImpl同样封装了闯入构造的Statement对象。
executeQuery()方法创建了JDBC-ODBC的ResultSet并返回一个RemoteStatementImpl对象的引用来控制JDBC-ODBC的ResultSet:
public IRemoteResultSet executeQuery(String Query)
throws RemoteException,SQLException
{
ResultSet rs = sqlStatment.executeQuery(Query);
RemoteResultSetImpl remoteRs = new
RemoteResultSetImpl(rs);
return (IRemoteResultSet)remoteRs;
}
close()方法简单的关闭了JDBC Statement对象:
public void close() throws RemoteException,
SQLException
{
sqlStatment.close();
}
最后再来看看RemoteResultSetImpl类。
RemoteResultSetImpl类
RemoteResultSetImpl继承了UnicastRemoteObject类并实现了IremoteResult接口(图4 显示了RemoteResultSetImpl类的其他类和接口的关系)。IremoteResult接口获得JDBC ResultSet行数据并关闭ResultSet对象。RemoteResultSetImpl同样封装了构造中的ResultSet对象。
getNextRow()方法返回一个包含对象数组的JDBC ResultSet给客户端JWResultSet,如果ResultSet不包含任何数据会返回null:
public Object[] getNextRow() throws
RemoteException,SQLException//自己替换[]
{
// Return null if all data has already been
iterated
if(sqlRs.next() == false)
return null;
// Prepare the data row in an array of Objects
Object []row = new Object[colNum];//自己替换[]
for(int i = 1; i <= colNum; i++)
{
row[i-1] = sqlRs.getString(i);//自己替换[]
}
return row;
}
close()方法会关闭ResultSet对象:
public void close() throws
RemoteException,SQLException
{
sqlRs.close();
}
以上就是中间层中的所有类。
Type3 总结
在这篇文章中,我向你展示了Type3 JDBC驱动的结构。驱动创建了客户层和服务层组件,并使用RMI实现它们之间的通信。我们还演示了客户层的不同对象如何通过引用与服务层中的远程对象交互。
在第2部分我会向你展示如何使用和部署这里创建的Type3驱动。我们将提供一个简单的应用程序,并通过序列图的方式向你展示驱动是如何加载的,如何创建一个JDBC Connection,如何创建一个 JDBC Statement。