在一个应用里面,可能涉及到连接多个不同数据库进行操作,而每次连接写不同的实现会很麻烦。前面已经会了用JDBC连接数据库,那么利用反射和工厂模式,可以实现连接不同的数据库,这样处理起来将会很方便。同时建造数据库连接池,处理多个业务数据处理。
 
 

 

那么具体怎么实现呢,下面一起来看一下:
整体结构如下:

 

第一步,先处理连接不同数据库
1、首先,将数据库配置信息创建一个公用类:JdbcUrl.java
主数据库可以用默认的构造方法,如果是连接其他库,则通过传递参数的方式来处理。
数据库参数有如下几个:
 
复制代码
  1 /**
  2  * 数据库连接配置信息类
  3  * @author Damon
  4  */
  5 public class JdbcUrl
  6 {
  7 
  8     /** 定义数据库参数 */
  9 
 10     // 数据库类型
 11     private String DBType;
 12     // 数据库服务器IP
 13     private String IP;
 14     // 数据库服务器端口
 15     private String Port;
 16     // 数据库名称
 17     private String DBName;
 18     // 用户名
 19     private String UserName;
 20     // 密码
 21     private String PassWord;
 22 
 23 
 24     /**
 25      * 默认构造方法,连接默认数据库
 26      */
 27     public JdbcUrl()
 28     {
 29         // TODO Auto-generated constructor stub
 30         DBType = SysCon.DATABASE_TYPE_MYSQL;
 31         IP = "127.0.0.1";
 32         DBName = "mysql";
 33         Port = "3306";
 34         UserName = "damon";
 35         PassWord = "damon";
 36     }
 37 
 38     /**
 39      * 连接指定数据库
 40      * @param urlType 传入连接类型标识
 41      */
 42     public JdbcUrl(String urlType)
 43     {
 44         if ("mysql".equals(urlType))
 45         {
 46             DBType = SysCon.DATABASE_TYPE_MYSQL;
 47             IP = "127.0.0.1";
 48             DBName = "mysql";
 49             Port = "3306";
 50             UserName = "damon";
 51             PassWord = "damon";
 52         }
 53     }
 54 
 55     /**
 56      * 获取连接句柄
 57      * @return String
 58      */
 59     public String getJdbcUrl()
 60     {
 61         String sUrl = "";
 62 
 63         if (DBType.trim().toUpperCase().equals("MYSQL"))
 64         {
 65             sUrl = "jdbc:mysql://" + IP + ":" + Port + "/" + DBName;
 66         }
 67         else if (DBType.trim().toUpperCase().equals("DB2"))
 68         {
 69             sUrl = "jdbc:db2://" + IP + ":" + Port + "/" + DBName;
 70         }
 71 
 72         else if (DBType.trim().toUpperCase().equals("ORACLE"))
 73         {
 74             sUrl = "jdbc:oracle:thin:@" + IP + ":" + Port + ":" + DBName;
 75         }
 76 
 77         else if (DBType.trim().toUpperCase().equals("SQLSERVER"))
 78         {
 79             sUrl = "jdbc:microsoft:sqlserver://" + IP + ":" + Port + ";databaseName=" + DBName + ";selectMethod=cursor";
 80         }
 81         else if (DBType.trim().toUpperCase().equals("WEBLOGICPOOL"))
 82         {
 83             sUrl = "jdbc:weblogic:pool:" + DBName;
 84         }
 85         else
 86         {
 87             System.out.println("暂无对应数据库驱动");
 88         }
 89         return sUrl;
 90     }
 91 
 92     // getters and setters
 93 
 94     public String getDBType()
 95     {
 96         return DBType;
 97     }
 98 
 99     public void setDBType(String dBType)
100     {
101         DBType = dBType;
102     }
103 
104     public String getIP()
105     {
106         return IP;
107     }
108 
109     public void setIP(String iP)
110     {
111         IP = iP;
112     }
113 
114     public String getPort()
115     {
116         return Port;
117     }
118 
119     public void setPort(String port)
120     {
121         Port = port;
122     }
123 
124     public String getDBName()
125     {
126         return DBName;
127     }
128 
129     public void setDBName(String dBName)
130     {
131         DBName = dBName;
132     }
133 
134     public String getUserName()
135     {
136         return UserName;
137     }
138 
139     public void setUserName(String userName)
140     {
141         UserName = userName;
142     }
143 
144     public String getPassWord()
145     {
146         return PassWord;
147     }
148 
149     public void setPassWord(String passWord)
150     {
151         PassWord = passWord;
152     }
153 
154 }
View Code
复制代码

 

2、重写一个Connection类,实现Connection接口的方法,同时连接数据库。

参数有已实现的JdbrUrl类,主要新增方法为:createConnection()
根据DBType来对不同数据库进行处理:加载对应的数据库,然后获取数据库连接。
复制代码
  1 **
  2  * 数据库连接类,连接数据库
  3  * @author Damon
  4  */
  5 public class DBConn implements Connection
  6 {
  7 
  8     // 获取JdbcUrl信息
  9     private JdbcUrl JUrl;
 10 
 11     // 数据库连接
 12     private Connection con = null;
 13 
 14     // 连接是否已使用
 15     private boolean bNotInUse;
 16 
 17     private CharArrayWriter m_buf = new CharArrayWriter();
 18 
 19     private PrintWriter m_pw = new PrintWriter(m_buf, true);
 20 
 21     // 默认连接
 22     public DBConn()
 23     {
 24         // TODO Auto-generated constructor stub
 25         this.JUrl = new JdbcUrl();
 26     }
 27 
 28     // 指定数据库连接
 29     public DBConn(String urlType)
 30     {
 31         this.JUrl = new JdbcUrl(urlType);
 32     }
 33 
 34     // 创建连接
 35     public boolean createConnection()
 36     {
 37 
 38         // 根据数据库类型加载驱动及连接
 39         try
 40         {
 41             // 连接MySQL数据库
 42             if (SysCon.DATABASE_TYPE_MYSQL.equals(JUrl.getDBType()))
 43             {
 44                 // 加载数据库驱动
 45                 Class.forName("com.mysql.jdbc.Driver");
 46 
 47                 // 尝试连接数据库
 48                 con = DriverManager.getConnection(JUrl.getJdbcUrl(), JUrl.getUserName(), JUrl.getPassWord());
 49             }
 50             // 其他数据库类型判断及处理
 51             // SQLSERVER
 52             else if (SysCon.DATABASE_TYPE_SQLSERVER.equals(JUrl.getDBType()))
 53             {
 54                 Class.forName("com.microsoft.jdbc.sqlserver.SQLServerDriver");
 55                 con = DriverManager.getConnection(JUrl.getJdbcUrl(), JUrl.getUserName(), JUrl.getPassWord());
 56             }
 57             // DB2
 58             else if (SysCon.DATABASE_TYPE_DB2.equals(JUrl.getDBType()))
 59             {
 60                 Class.forName("com.ibm.db2.jcc.DB2Driver");
 61                 con = DriverManager.getConnection(JUrl.getJdbcUrl(), JUrl.getUserName(), JUrl.getPassWord());
 62             }
 63             // ORACLE
 64             else if (SysCon.DATABASE_TYPE_ORACLE.equals(JUrl.getDBType()))
 65             {
 66                 Class.forName("oracle.jdbc.driver.OracleDriver");
 67                 // 一个是缓存取到的记录数,一个是设置默认的批量提交数
 68                 Properties props = new Properties();
 69                 props.setProperty("user", JUrl.getUserName());
 70                 props.setProperty("password", JUrl.getPassWord());
 71                 props.setProperty("defaultRowPrefetch", "50");
 72                 props.setProperty("defaultExecuteBatch", "50");
 73                 con = DriverManager.getConnection(JUrl.getJdbcUrl(), props);
 74             }
 75             else
 76             {
 77                 System.out.println("未匹配到数据库类型!");
 78                 return false;
 79             }
 80 
 81         }
 82         catch (ClassNotFoundException e)
 83         {
 84             // TODO Auto-generated catch block
 85             System.out.println("加载驱动失败!");
 86             e.printStackTrace();
 87             return false;
 88         }
 89         catch (SQLException e)
 90         {
 91             // TODO Auto-generated catch block
 92             System.out.println("创建连接失败..." + e.getMessage());
 93             e.printStackTrace();
 94             return false;
 95         }
 96         return true;
 97     }
 98 
 99     protected void setInUse()
100     {
101         /**
102          * Record stack information when each connection is get We reassian
103          * System.err, so Thread.currentThread().dumpStack() can dump stack info
104          * into our class FilterPrintStream.
105          */
106         new Throwable().printStackTrace(m_pw);
107 
108         bNotInUse = false;
109 
110         /**
111          * record lastest access time
112          */
113     }
114 
115     /* 下面都是 实现Connection的方法,返回conn的实现 */
116     public <T> T unwrap(Class<T> iface) throws SQLException
117     {
118         // TODO Auto-generated method stub
119         return con.unwrap(null);
120     }
121 
122     public boolean isWrapperFor(Class<?> iface) throws SQLException
123     {
124         // TODO Auto-generated method stub
125         return false;
126     }
127 
128     public Statement createStatement() throws SQLException
129     {
130         // TODO Auto-generated method stub
131         return con.createStatement();
132     }
133 
134     public PreparedStatement prepareStatement(String sql) throws SQLException
135     {
136         // TODO Auto-generated method stub
137         return con.prepareStatement(sql);
138     }
139 
140     public CallableStatement prepareCall(String sql) throws SQLException
141     {
142         // TODO Auto-generated method stub
143         return con.prepareCall(sql);
144     }
145 
146     public String nativeSQL(String sql) throws SQLException
147     {
148         // TODO Auto-generated method stub
149         return con.nativeSQL(sql);
150     }
151 
152     public void setAutoCommit(boolean autoCommit) throws SQLException
153     {
154         // TODO Auto-generated method stub
155         con.setAutoCommit(autoCommit);
156     }
157 
158     public boolean getAutoCommit() throws SQLException
159     {
160         // TODO Auto-generated method stub
161         return con.getAutoCommit();
162     }
163 
164     public void commit() throws SQLException
165     {
166         // TODO Auto-generated method stub
167         con.commit();
168     }
169 
170     public void rollback() throws SQLException
171     {
172         // TODO Auto-generated method stub
173         con.rollback();
174     }
175 
176     public void close() throws SQLException
177     {
178         // TODO Auto-generated method stub
179         con.close();
180     }
181 
182     public boolean isClosed() throws SQLException
183     {
184         // TODO Auto-generated method stub
185 
186         return con.isClosed();
187     }
188 
189     public DatabaseMetaData getMetaData() throws SQLException
190     {
191         // TODO Auto-generated method stub
192         return con.getMetaData();
193     }
194 
195     public void setReadOnly(boolean readOnly) throws SQLException
196     {
197         // TODO Auto-generated method stub
198         con.setReadOnly(readOnly);
199     }
200 
201     public boolean isReadOnly() throws SQLException
202     {
203         // TODO Auto-generated method stub
204         return con.isReadOnly();
205     }
206 
207     public void setCatalog(String catalog) throws SQLException
208     {
209         // TODO Auto-generated method stub
210         con.setCatalog(catalog);
211     }
212 
213     public String getCatalog() throws SQLException
214     {
215         // TODO Auto-generated method stub
216         return con.getCatalog();
217     }
218 
219     public void setTransactionIsolation(int level) throws SQLException
220     {
221         // TODO Auto-generated method stub
222         con.setTransactionIsolation(level);
223     }
224 
225     public int getTransactionIsolation() throws SQLException
226     {
227         // TODO Auto-generated method stub
228         return con.getTransactionIsolation();
229     }
230 
231     public SQLWarning getWarnings() throws SQLException
232     {
233         // TODO Auto-generated method stub
234         return con.getWarnings();
235     }
236 
237     public void clearWarnings() throws SQLException
238     {
239         // TODO Auto-generated method stub
240         con.clearWarnings();
241     }
242 
243     public Statement createStatement(int resultSetType, int resultSetConcurrency) throws SQLException
244     {
245         // TODO Auto-generated method stub
246         return con.createStatement(resultSetType, resultSetConcurrency);
247     }
248 
249     public PreparedStatement prepareStatement(String sql, int resultSetType, int resultSetConcurrency)
250             throws SQLException
251     {
252         // TODO Auto-generated method stub
253         return con.prepareStatement(sql, resultSetType, resultSetConcurrency);
254     }
255 
256     public CallableStatement prepareCall(String sql, int resultSetType, int resultSetConcurrency) throws SQLException
257     {
258         // TODO Auto-generated method stub
259         return con.prepareCall(sql, resultSetType, resultSetConcurrency);
260     }
261 
262     public Map<String, Class<?>> getTypeMap() throws SQLException
263     {
264         // TODO Auto-generated method stub
265         return con.getTypeMap();
266     }
267 
268     public void setTypeMap(Map<String, Class<?>> map) throws SQLException
269     {
270         // TODO Auto-generated method stub
271         con.setTypeMap(map);
272     }
273 
274     public void setHoldability(int holdability) throws SQLException
275     {
276         // TODO Auto-generated method stub
277         con.setHoldability(holdability);
278     }
279 
280     public int getHoldability() throws SQLException
281     {
282         // TODO Auto-generated method stub
283         return con.getHoldability();
284     }
285 
286     public Savepoint setSavepoint() throws SQLException
287     {
288         // TODO Auto-generated method stub
289         return con.setSavepoint();
290     }
291 
292     public Savepoint setSavepoint(String name) throws SQLException
293     {
294         // TODO Auto-generated method stub
295         return con.setSavepoint(name);
296     }
297 
298     public void rollback(Savepoint savepoint) throws SQLException
299     {
300         // TODO Auto-generated method stub
301         con.rollback(savepoint);
302     }
303 
304     public void releaseSavepoint(Savepoint savepoint) throws SQLException
305     {
306         // TODO Auto-generated method stub
307         con.releaseSavepoint(savepoint);
308     }
309 
310     public Statement createStatement(int resultSetType, int resultSetConcurrency, int resultSetHoldability)
311             throws SQLException
312     {
313         // TODO Auto-generated method stub
314         return con.createStatement(resultSetType, resultSetConcurrency, resultSetHoldability);
315     }
316 
317     public PreparedStatement prepareStatement(String sql, int resultSetType, int resultSetConcurrency,
318             int resultSetHoldability) throws SQLException
319     {
320         // TODO Auto-generated method stub
321         return con.prepareStatement(sql, resultSetType, resultSetConcurrency, resultSetHoldability);
322     }
323 
324     public CallableStatement prepareCall(String sql, int resultSetType, int resultSetConcurrency,
325             int resultSetHoldability) throws SQLException
326     {
327         // TODO Auto-generated method stub
328         return null;
329     }
330 
331     public PreparedStatement prepareStatement(String sql, int autoGeneratedKeys) throws SQLException
332     {
333         // TODO Auto-generated method stub
334         return con.prepareStatement(sql, autoGeneratedKeys);
335     }
336 
337     public PreparedStatement prepareStatement(String sql, int[] columnIndexes) throws SQLException
338     {
339         // TODO Auto-generated method stub
340         return con.prepareStatement(sql, columnIndexes);
341     }
342 
343     public PreparedStatement prepareStatement(String sql, String[] columnNames) throws SQLException
344     {
345         // TODO Auto-generated method stub
346         return con.prepareStatement(sql, columnNames);
347     }
348 
349     public Clob createClob() throws SQLException
350     {
351         // TODO Auto-generated method stub
352         return con.createClob();
353     }
354 
355     public Blob createBlob() throws SQLException
356     {
357         // TODO Auto-generated method stub
358         return con.createBlob();
359     }
360 
361     public NClob createNClob() throws SQLException
362     {
363         // TODO Auto-generated method stub
364         return con.createNClob();
365     }
366 
367     public SQLXML createSQLXML() throws SQLException
368     {
369         // TODO Auto-generated method stub
370         return con.createSQLXML();
371     }
372 
373     public boolean isValid(int timeout) throws SQLException
374     {
375         // TODO Auto-generated method stub
376         return con.isValid(timeout);
377     }
378 
379     public void setClientInfo(String name, String value) throws SQLClientInfoException
380     {
381         // TODO Auto-generated method stub
382         con.setClientInfo(name, value);
383     }
384 
385     public void setClientInfo(Properties properties) throws SQLClientInfoException
386     {
387         // TODO Auto-generated method stub
388         con.setClientInfo(properties);
389     }
390 
391     public String getClientInfo(String name) throws SQLException
392     {
393         // TODO Auto-generated method stub
394         return con.getClientInfo(name);
395     }
396 
397     public Properties getClientInfo() throws SQLException
398     {
399         // TODO Auto-generated method stub
400         return con.getClientInfo();
401     }
402 
403     public Array createArrayOf(String typeName, Object[] elements) throws SQLException
404     {
405         // TODO Auto-generated method stub
406         return con.createArrayOf(typeName, elements);
407     }
408 
409     public Struct createStruct(String typeName, Object[] attributes) throws SQLException
410     {
411         // TODO Auto-generated method stub
412         return con.createStruct(typeName, attributes);
413     }
414 
415 }
View Code
复制代码

 

3、公共的数据库连接池

数据库配置和数据库连接已经搞定,那么可以建一个数据库连接池来进行数据库的连接及处理。
主要有2个方法,连接默认数据库和连接指定的数据库。
复制代码
 1 /**
 2      * 获取默认数据库连接
 3      * @param uri
 4      * @return
 5      */
 6     public static DBConn getConnection()
 7     {
 8         DBConn dbConn = new DBConn();
 9         if (!dbConn.createConnection())
10         {
11             // 如果创建连接失败
12             DBSemaphore.unLock();
13             return null;
14         }
15 
16         // 连接成功,设置该连接属性
17         try
18         {
19             // 特殊处理连接的AutoCommit是否已经被设置
20             dbConn.setAutoCommit(true);
21             dbConn.setInUse();
22             DBSemaphore.unLock();
23             return dbConn;
24         }
25         catch (Exception ex)
26         {
27             ex.printStackTrace();
28             DBSemaphore.unLock();
29             return null;
30         }
31 
32     }
33 
34     /**
35      * 通过URI地址获取指定数据库连接
36      * @param uri
37      * @return
38      */
39     public static DBConn getConnection(String uri)
40     {
41         DBConn dbConn = new DBConn(uri);
42         if (!dbConn.createConnection())
43         {
44             // 如果创建连接失败
45             // DBSemaphore.UnLock();
46             return null;
47         }
48         try
49         {
50             // 特殊处理连接的AutoCommit是否已经被设置
51             dbConn.setAutoCommit(true);
52             // dbConn.setInUse();
53             // DBSemaphore.UnLock();
54             return dbConn;
55         }
56         catch (Exception ex)
57         {
58             ex.printStackTrace();
59             // DBSemaphore.UnLock();
60             return null;
61         }
62 
63     }
View Code
复制代码

 

可以写一个测试方式,这也数据库连接就完成了,通过传入不同的参数就可以获取到不同数据库的连接了。
 
复制代码
 1     public static void main(String[] args)
 2     {
 3         // 测试连接池
 4 
 5         // 1、连接mysql 数据库
 6         Connection conn = DBConnPool.getConnection();
 7 
 8         if (conn == null)
 9         {
10             System.out.println("获取连接失败!");
11         }
12         else
13         {
14             System.out.println("获取连接成功");
15         }
16 
17     }
View Code
复制代码

 

第二步,构建数据库连接池
前面已经实现连接不同数据库,那么怎么处理为连接池呢?
数据库连接池负责分配、管理和释放数据库连接,它允许应用程序重复使用一个现有的数据库连接,而不是再重新建立一个;释放空闲时间超过最大空闲时间的数据库连接来避免因为没有释放数据库连接而引起的数据库连接遗漏。这项技术能明显提高对数据库操作的性能。下面我们就来实现数据库连接池的功能:

 

数据库连接池的实现思想主要有如下几个方面:
1、可定义最大和最小的连接数:如果请求小于最小连接数,则直接分配连接,达到最大连接数则新请求等待;
2、用户在连接数据库时,不用新建连接,而是直接从连接池中获取连接,使用完毕后也不用关闭,而是释放给连接池,供下一个用户使用;
下面来进行实现:
其实就是实现了一个数据连接数的判断,最大连接数进行限制,实际的数据库连接池需结合对应的数据库连接池组件(比如 WebSphere等中间件),在判断是否有连接在使用的时候,会涉及到并发,这里需要用到volatile关键字。
 
其中,增加了DBsemaphor类,用以处理连接的使用和释放。
在原有DBConn基础上,加上是否已使用方法,在获取连接成功时调用,具体代码如下:
 
复制代码
 1 // 连接成功,设置该连接属性
 2         try
 3         {
 4             // 特殊处理连接的AutoCommit是否已经被设置
 5             dbConn.setAutoCommit(true);
 6             dbConn.setInUse();
 7             DBSemaphore.unLock();
 8             return dbConn;
 9         }
10         catch (Exception ex)
11         {
12             ex.printStackTrace();
13             DBSemaphore.unLock();
14             return null;
15         }
View Code
复制代码

新增DBSemaphore类属性及方法如下:

复制代码
 1 /**
 2  * 数据库同步对象
 3  * @author Damon
 4  */
 5 public class DBSemaphore
 6 {
 7     private static volatile boolean m_bInUse = false;
 8 
 9     public DBSemaphore()
10     {}
11 
12     /**
13      * 设置"使用标志"。 传入true表示请求“使用标志”,传入false表示释放“使用标志”。
14      * @param bNewValue boolean
15      * @return boolean
16      */
17     protected static synchronized boolean setInUseFlag(boolean bNewValue)
18     {
19         if (bNewValue == true)
20         {
21             // 请求“使用标志”
22             if (m_bInUse == true)
23             {
24                 // “使用标志”已经被占用
25                 return false;
26             }
27             else
28             {
29                 m_bInUse = true;
30                 return true;
31             }
32         }
33         else
34         {
35             // 释放“使用标志”
36             m_bInUse = false;
37             return true;
38         }
39     }
40 
41     protected static void lock() throws Exception
42     {
43         lock(0);
44     }
45 
46     protected static void lock(int nSeconds) throws Exception
47     {
48         if (nSeconds <= 0)
49         {
50             while (!setInUseFlag(true))
51             {
52                 Thread.sleep(100);
53             }
54         }
55         else
56         {
57             while (!setInUseFlag(true) && nSeconds-- > 0)
58             {
59                 Thread.sleep(100);
60             }
61 
62             if (nSeconds == 0)
63             {
64                 throw new Exception("Lock time out");
65             }
66         }
67     }
68 
69     protected static void unLock()
70     {
71         setInUseFlag(false);
72     }
73 }
View Code
复制代码

 

到这里,整个配置处理结束了,更多的就需要在实际项目中发挥了~