Java Web(十) JDBC的增删改查,C3P0等连接池,dbutils框架的使用
前面做了一个非常垃圾的小demo,真的无法直面它,菜的抠脚啊,真的菜,好好努力把。菜鸡。
--WZY
一、JDBC是什么?
Java Data Base Connectivity,java数据库连接,在需要存储一些数据,或者拿到一些数据的时候,就需要往数据库里存取数据,那么java如何连接数据库呢?需要哪些步骤?
1、注册驱动
什么是驱动?
驱动就是JDBC实现类,通俗点讲,就是能够连接到数据库功能的东西就是驱动,由于市面上有很多数据库,Oracle、MySql等等,所以java就有一个连接数据库的实现规范接口,定义一系列的连接数据库接口(java.sql.Driver接口),但是不提供实现,而每个数据库厂家来提供这些接口的具体实现,这样一来,不管使用的是什么数据库,我们开发者写的代码都是相同的,就不必因为数据库的不同,而写法不同,唯一的不同就是数据库驱动不一样,使用mysql,那么就必须使用mysql的驱动,使用Oracle就必须使用oracle的驱动实现类。 看下面mysql连接数据的原理图,看看驱动是在哪里,起什么作用。就明白了什么是驱动了。
           
 
DriverManager,一个工具类,是用于操作管理JDBC实现类的,
原始写法:DriverManager.register(new Driver()); //因为使用的是MySql,所以在导包时就需要导入com.mysql.jdbc.Driver
现在写法:Class.forName("com.mysql.jdbc.Driver"); //不用导包,会执行com.mysql.jdbc.Driver类中的静态代码块,其静态代码块的内容为
              static {
                  try {
java.sql.DriverManager.registerDriver(new Driver());
                 } catch (SQLException E) {
                    throw new RuntimeException("Can't register driver!");
                             }
                        }  
会发现第二种加载驱动的方法的底层其实就是第一种加载驱动。为什么要这样呢?原因很简单, 第一种是硬编程,直接将数据库驱动给写死了,无法扩展,如果使用第一种,那么连接的数据库只能是mysql,因为导包导的是mysql的驱动包,如果换成Oracle,就会报错,需要在代码中将Oracle的驱动包导入,这样很麻烦,而第二种写法就不一样了,第二种是使用的字符串方法注册驱动的,我们只需要将该字符串提取到一个配置文件中,以后想换成oracle数据库,只需要将该字符串换成oracle驱动的类全名即可,而不需要到代码中去修改什么东西。
2、获得连接
使用DriverManage来获得连接,因为DriverManager是驱动实现类的管理者
Connection conn = DriverManager.getConnection(url,user,password);
url:确定数据库服务器的位置,端口号,数据库名
jdbc:mysql://localhost:3306/db
user:登录名称,默认root
password:密码,默认root
这里只是说mysql,别的数据库,url格式就不同了。
MySQL jdbc:mysql://localhost:3306/db 默认端口是3306,粗体为连接时使用的数据库名
Oracle jdbc:oracle:thin:@localhost:1521:db 默认端口号1521
DB2 jdbc:db2://localhost:6789/db 默认端口号6789
SQLServer jdbc:microsoft:sqlserver://localhost:1433;databaseName=db 默认端口号1433
SQLServer 2005 jdbc:sqlserver://localhost:1433;databaseName=db 默认端口号1433
3、获取执行sql语句对象,PraparedStament对象
通过Connection对象获取Statement或者PraparedStament对象(使用它)处理sql
Statement
Statement st = conn.createStatement(); //获取sql语句执行对象
st.excuteUpdate(sql); //执行增删改语句
st.excuteQuery(sql); //执行查询语句
sql语句必须是完整的。
PraparedStatment
sql语句可以不是完整的,可以将参数用?替代,然后在预编译后加入未知参数
PraparedStatment ps = conn.prapareStatement(sql); //获取sql语句执行对象praparedStatment
赋值
ps.setInt(Index,value); ps.setString(index,value); //可以设置很多中类型,index从1开始,代表sql语句中的第几个未知参数,
ps.excuteUpdate(); //执行增删改语句
ps.excuteQuery(sql); //执行查询语句
这两个的区别,常使用的是PraparedStatment对象,因为它可以预编译,效率高,可以设置参数等等优点
4、获得结果集对象
int count = ps.excuteUpdate(); //执行增删改的sql语句时,返回一个int类型的整数,代表数据库表影响的行数,
Result result = ps.excuteQuery(); //执行查询sql语句时,返回一个结果集对象,该对象装着所有查询到的数据信息,一行一行的存储数据库表信息。
5、处理结果
对查询到的Result结果进行处理,拿到所有数据,并封装成对象。
while(rs.next()){
获取行数据的第一种方式
rs.getString(index);//index代表第几列,从1开始
获取行数据的第二中方式
rs.getString(string); //string:代表字段名称。
}
        
总结:java的JDBC就分为5步,4个属性
属性:driver、url、user、password
五步:
注册驱动、获取连接、获取执行sql语句对象、获取结果集对象、处理结果。
二、JDBC的CURD操作
创建(Create)、更新(Update)、读取(Retrieve)和删除(Delete)操作
查询所有(读取Retrieve)
findAll()
@Test
public void findAll() throws Exception{
//1 注册驱动
Class.forName("com.mysql.jdbc.Driver");
//2 获得连接
Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test", "root", "root");
//3语句执行者,sql语句
Statement st = conn.praparedStatement("select * from t_user");
//4 执行查询语句
ResultSet rs = st.executeQuery();
//5处理数据
// * 如果查询多个使用,使用while循环进行所有数据获取
// * 技巧:如果查询结果最多1条,使用 if(rs.next()) { 查询到了 } else { 没有数据 }
while(rs.next()){
int id = rs.getInt(1);
String username = rs.getString(2);
String password =rs.getString(3);
System.out.print(id + ", ");
System.out.print(username + ", ");
System.out.println(password);
}
//6释放资源
rs.close();
st.close();
conn.close(); }
save(),增加操作(创建Create)
@Test
public void save() throws Exception{
//1 注册驱动
Class.forName("com.mysql.jdbc.Driver");
//2 获得连接
Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test", "root", "root");
//3语句执行者
Statement st = conn.praparedStatement("insert into t_user(username,password) values(?,?)");
//3.1赋值
st.setString(1,"xiaoming");
st.setString(2,"123");
//4 执行DML语句
int r = st.executeUpdate(); //5处理数据
System.out.println(r); //6释放资源
//rs.close();
st.close();
conn.close();
}
update(),更新
@Test
public void update() throws Exception{
//1 注册驱动
Class.forName("com.mysql.jdbc.Driver");
//2 获得连接
Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test", "root", "root");
//3语句执行者
Statement st = conn.praparedStatement("update t_user set username = ? where id = ? ");
//3.1赋值参数
st.setString(1,"xiaoye");
st.setInt(2,2);
//4 执行DML语句
int r = st.executeUpdate(); //5处理数据
System.out.println(r); //6释放资源
//rs.close();
st.close();
conn.close();
}
delete(),删除
@Test
public void delete() throws Exception{
//1 注册驱动
Class.forName("com.mysql.jdbc.Driver");
//2 获得连接
Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test", "root", "root");
//3语句执行者
Statement st = conn.praparedStatement("delete from t_user where id = ?");
//3.1赋值参数
st.setInt(1,2);
//4 执行DML语句
int r = st.executeUpdate(); //5处理数据
System.out.println(r); //6释放资源
//rs.close();
st.close();
conn.close();
}
上面重复代码过多,所以使用一个获得连接的工具类,来帮我们获得连接,并且把四个属性提取出来,放在配置文件中
使用jdbcInfo.properties(放在src下面即可)保存四个属性。以方便修改
jdbcInfo.properties
driver = com.mysql.jdbc.Driver
url = jdbc:mysql://localhost:3306/myums
user = root
password =root
写一个工具类,注册驱动,提供连接,就不必每次都重复写注册驱动,连接代码了
JdbcUtils.java
 public class JdbcUtils {
     private static String url;
     private static String user;
     private static String password;
     static{
         try {
             // 将具体参数存放配置文件中, xml,properties(key=value)
             // 1 加载 properties 文件 src (类路径) -->  WEB-INF/classes
             // 方式1: 使用类加载ClassLoader的方式加载资源
             InputStream is = JdbcUtils.class.getClassLoader().getResourceAsStream("jdbcInfo.properties");
             // 方式2:使用Class对象加载,必须添加/,表示src
             // InputStream is = JdbcUtils.class.getResourceAsStream("/jdbcInfo.properties");
             //  * 如果不使用/表示,从当前类所在的包下加载资源
             //InputStream is = JdbcUtils.class.getResourceAsStream("/com/itheima/d_utils/jdbcInfo2.properties");
             // 2 解析
             Properties props = new Properties();
             props.load(is);
             // 3 获得配置文件中数据
             String driver = props.getProperty("driver");
             url = props.getProperty("url");
             user = props.getProperty("user");
             password = props.getProperty("password");;
             // 4 注册驱动
             Class.forName(driver);
         } catch (Exception e) {
             throw new RuntimeException(e);
         }
     }
     /**
      * 获得连接
      * @return
      */
     public static Connection getConnection(){
         try {
             Connection conn = DriverManager.getConnection(url, user, password);
             return conn; //获得连接
         } catch (Exception e) {
             //将编译时异常 转换 运行时 , 以后开发中 运行时异常使用比较多的。
         }
     }
     /**
      * 释放资源
      * @param conn
      * @param st
      * @param rs
      */
     public static void closeResource(Connection conn,Statement st,ResultSet rs){
         try {
             if (rs != null) {
                 rs.close();
             }
         } catch (Exception e) {
             throw new RuntimeException(e);
         } finally{
             try {
                 if (st != null) {
                     st.close();
                 }
             } catch (Exception e) {
                 throw new RuntimeException(e);
             } finally{
                 try {
                     if (conn != null) {
                         conn.close();
                     }
                 } catch (Exception e) {
                     throw new RuntimeException(e);
                 }
             }
         }
     }
模版代码
//模板代码
public void demo01(){
// 0 提供变量
Connection conn = null;
Statement st = null;
ResultSet rs = null; try {
//1 获得连接
conn = JdbcUtils.getConnection(); //2 获得语句执行者
//3执行sql语句
//4处理结果 } catch (Exception e) {
throw new RuntimeException(e);
} finally{
//end 释放资源
JdbcUtils.closeResource(conn, st, rs);
}
}
三、连接池
在上面,我们在进行CRUD时,一直重复性的写一些代码,比如最开始的注册驱动,获取连接代码,一直重复写,通过编写一个获取连接的工具类后,解决了这个问题,但是又会出现新的问题,每进行一次操作,就会获取一个连接,用完之后,就销毁,就这样一直新建连接,销毁连接,新建,销毁,连接Connection 创建与销毁 比较耗时的。所以应该要想办法解决这个问题。
连接池就是为了解决这个问题而出现的一个方法,为了提高性能,开发连接池,连接池中一直保持有n个连接,供调用者使用,调用者用完返还给连接池,继续给别的调用者使用,比如连接池中一开始就有10个连接,当有5个用户拿走了5个连接后,池中还剩5个,当第6个用户在去池中拿连接而前面5个连接还没归还时,连接池就会新建一个连接给第六个用户,让池中一直能够保存最少5个连接,而当这样新建了很多连接后,用户归还连接回来时,会比原先连接池中的10个连接更多,连接池就会设置一个池中最大空闲的连接数,如果超过了这个数,就会将超过的连接给释放掉,连接池就是这样工作的。
现在介绍几款连接池,DBCP、C3P0、tomcat内置连接池(JNDI)(这个不讲)
DBCP连接池,
两种方式获得连接,使用配置文件,不使用配置文件
1、不使用配置文件,自己手动设置参数
导包
              
核心类BasicDataSource,通过new出BasicDataSource对象,设置参数 然后获得连接
//创建核心类
BasicDataSource bds = new BasicDataSource();
//配置4个基本参数
bds.setDriverClassName("com.mysql.jdbc.Driver");
bds.setUrl("jdbc:mysql:///myums");
bds.setUsername("root");
bds.setPassword("root"); //管理连接配置
bds.setMaxActive(50); //最大活动数
bds.setMaxIdle(20); //最大空闲数
bds.setMinIdle(5); //最小空闲数
bds.setInitialSize(10);//初始化个数 //获取连接
try {
Connection conn = bds.getConnection();
System.out.println(conn); } catch (SQLException e) {
throw new RuntimeException(e);
}
2、使用配置文件,参数写入配置文件中即可,也就是通过配置文件来配置驱动、用户名、密码、等信息
导包
              
导入配置文件dbcpconfig.properties
#连接设置
driverClassName=com.mysql.jdbc.Driver
url=jdbc:mysql://localhost:3306/test
username=root
password=root #<!-- 初始化连接 -->
initialSize=10 #最大连接数量
maxActive=50 #<!-- 最大空闲连接 -->
maxIdle=20 #<!-- 最小空闲连接 -->
minIdle=5 #<!-- 超时等待时间以毫秒为单位 6000毫秒/1000等于60秒 -->
maxWait=60000 #JDBC驱动建立连接时附带的连接属性属性的格式必须为这样:[属性名=property;]
#注意:"user" 与 "password" 两个属性会被明确地传递,因此这里不需要包含他们。
connectionProperties=useUnicode=true;characterEncoding=gbk #指定由连接池所创建的连接的自动提交(auto-commit)状态。
defaultAutoCommit=true #driver default 指定由连接池所创建的连接的只读(read-only)状态。
#如果没有设置该值,则“setReadOnly”方法将不被调用。(某些驱动并不支持只读模式,如:Informix)
defaultReadOnly= #driver default 指定由连接池所创建的连接的事务级别(TransactionIsolation)。
#可用值为下列之一:(详情可见javadoc。)NONE,READ_UNCOMMITTED, READ_COMMITTED, REPEATABLE_READ, SERIALIZABLE
defaultTransactionIsolation=READ_UNCOMMITTED
获取连接
//通过类加载器获取指定配置文件的输入流,Dbcp1是一个类名,
InputStream is = Dbcp1.class.getClassLoader().getResourceAsStream("dbcpconfig.properties");
Properties properties = new Properties();
properties.load(is);
//加载配置文件,获得配置信息
DataSource ds = BasicDataSourceFactory.createDataSource(properties);
Connection conn = ds.getConnection();
System.out.println(conn);
C3P0连接池
导包
            
                
从配置信息中获取 配置文件必须为xml
c3p0-config.xml
<c3p0-config>
<!-- 默认配置,如果没有指定则使用这个配置 -->
<default-config>
<property name="driverClass">com.mysql.jdbc.Driver</property>
<property name="jdbcUrl">jdbc:mysql://localhost:3306/myums</property>
<property name="user">root</property>
<property name="password">root</property> <property name="checkoutTimeout">30000</property>
<property name="idleConnectionTestPeriod">30</property>
<property name="initialPoolSize">10</property>
<property name="maxIdleTime">30</property>
<property name="maxPoolSize">100</property>
<property name="minPoolSize">10</property>
<property name="maxStatements">200</property>
<user-overrides user="test-user">
<property name="maxPoolSize">10</property>
<property name="minPoolSize">1</property>
<property name="maxStatements">0</property>
</user-overrides>
</default-config>
<!-- 命名的配置 -->
<named-config name="jxpx">
<property name="driverClass">com.mysql.jdbc.Driver</property>
<property name="jdbcUrl">jdbc:mysql://localhost:3306/myums</property>
<property name="user">root</property>
<property name="password">root</property>
<!-- 如果池中数据连接不够时一次增长多少个 -->
<property name="acquireIncrement">5</property>
<property name="initialPoolSize">20</property>
<property name="minPoolSize">10</property>
<property name="maxPoolSize">40</property>
<property name="maxStatements">0</property>
<property name="maxStatementsPerConnection">5</property>
</named-config>
</c3p0-config>
从配置文件中看,需要注意一个地方,一个是default-config,一个是name-config,两者都区别在于创建核心类对象时,如果将name-config作为参数传进去,那么将会调用name-config下的配置信息,否则将调用default-config下的配置信息,
两种方式使用c3p0,加参数,使用named-config 的配置信息,不加参数,自动加载配置信息,加载的是default-config中的信息
获得连接,使用核心类
//1 c3p0...jar 将自动加载配置文件。规定:WEB-INF/classes (src) c3p0-config.xml,也就是将配置文件放在src下就会自动加载。
//ComboPooledDataSource dataSource = new ComboPooledDataSource(); //自动从配置文件 <default-config>
ComboPooledDataSource dataSource = new ComboPooledDataSource(); //手动指定配置文件 <named-config name="jxpx">
Connection conn = dataSource.getConnection();
System.out.println(conn);
四、dbutils框架的使用
DBUtil是一个框架,用于简化JDBC开发, 像之前有连接池来优化获取连接操作,而DBUtils用来操作sql语句、将获取的数据封装到我们想要的结果,也就不需要在像之前用statement、预处理对象、ResultSet这些东西来处理sql语句了, DBUtils全部帮帮我们做好了,只需要两句代码就可以解决问题。
1、导包
          
2、核心类 QueryRunner
方式一,没有事务
new QueryRunner(dataSource);//将连接池传进去,因为不用管理事务,所以它将自动帮我们维护连接
增删改:update(sql,params...) 执行DML sql语句,并设置实际参数(可变参数,任意个参数,取决于有多少问号) 这里也就是用预处理了。
              
其中JdbcUtils是一个工具类,获取c3p0的数据源
               
 
查询:query(sql,handler,params...) 执行DDL sql:查询语句,handler:将我们查询到的数据封装到想要的结果。 params:设置实际参数,可变。
              
处理类:BeanListHandler,还有别的很多处理类
                    
BeanListHandler:将查询每一条数据封装到指定JavaBean,在将JavaBean封装到List集合中 最后返回集合 new List<User,User,...>
使用:BeanListHandler<User>(User.class)
BeanHandler: 将查询的一条数据封装到指定JavaBean,并返回javabean实例
使用:BeanHandler<User>(User.class)
                     
 
ScalarHandler:处理一行一列结果集,也就是一个单元格,单个数据(不是一条数据),(聚合)函数
                     
 
ArrayHandler:将查询一条记录所有数据封装到数组中, Object arr[] ={1,"jack","1234"}
使用:new ArrayHandler()
ArrayListHandler 将查询的所有记录每条记录分别封装到数组中,在将数组封装到list集合中,最后返回集合 new List() list.add(arr);
ColumnListHandler 将执行列封装到list集合中,返回list集合 List list= {"jack","rose","tom"}
KeyedHandler 将每一条记录封装到Map<String,Object>A中,在将mapA 封装到mapB中,mapB.value 就是mapA mapB.key 就是指定的key
                    
MapHandler 将一条记录封装到map 并返回map {id=2,username=jack,password=1234}
                    
MapListHandler 将每条记录都分别封装到Map中,然后将Map添加到List集合中,最后返回list集合 list<map,map>
方式二、使用事务,必须手动管理连接,且程序进行维护
构造方法:new QueryRunner() 这里不用参数,因为连接将手动获取
增删改:update(conn,sql,params...)
查询:query(conn,sql,handler,params...)
跟没有事务差不多,多了个conn
删除:
不使用dbutils来处理事务
                
使用dbutils框架中的工具类DbUtils来处理事务
                
  
五、总结
一篇很基础的对JDBC操作的文章,一步步从最基础最原生的JDBC代码讲起,一步步优化,优化连接,使用连接池,优化操作代码,使用第三方框架dbutils来操作。最终两句代码就搞定了对数据库的增删改查操作,其中要了解dbutils和连接池是如何实现的话,需要一些设计模式的知识,比如在dbutils中使用的策略模式等等,我感觉我暂时还不用去了解,还没到那种深度,等后面厉害了,再回过头来慢慢理解其中的精华。现在基本上会用就行了。其中所有用到的开发jar包,和配置文件我都会放在下面的链接中。
http://pan.baidu.com/s/1cIip8y 8he6
Java Web(十) JDBC的增删改查,C3P0等连接池,dbutils框架的使用的更多相关文章
- 纯Java JDBC连接数据库,且用JDBC实现增删改查的功能
		
Java JDBC连接数据库 package cn.cqvie.yjq; import java.sql.*; /** * 注册数据库的驱动程序,并得到数据库的连接对象 * @author yu * ...
 - 使用java对sql server进行增删改查
		
import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedStatement; import ...
 - JDBC基础学习(一)—JDBC的增删改查
		
一.数据的持久化 持久化(persistence): 把数据保存到可掉电式存储设备中以供之后使用.大多数情况下,数据持久化意味着将内存中的数据保存到硬盘上加以固化,而持久化的实现过程大多通过各 ...
 - Oracle使用JDBC进行增删改查 表是否存在
		
Oracle使用JDBC进行增删改查 数据库和表 table USERS ( USERNAME VARCHAR2(20) not null, PASSWORD VARCHAR2(20) ) a ...
 - java对xml文件做增删改查------摘录
		
java对xml文件做增删改查 package com.wss; import java.io.File;import java.util.ArrayList;import java.util.Lis ...
 - 通过flask实现web页面简单的增删改查bootstrap美化版
		
通过flask实现web页面简单的增删改查bootstrap美化版 项目目录结构 [root@node1 python]# tree -L 2 . ├── animate.css ├── fileut ...
 - 通过flask实现web页面简单的增删改查
		
通过flask实现web页面简单的增删改查 # 1.后台程序falsk_web01.py #coding:utf-8 from flask import Flask,render_template,r ...
 - Java API实现Hadoop文件系统增删改查
		
Java API实现Hadoop文件系统增删改查 Hadoop文件系统可以通过shell命令hadoop fs -xx进行操作,同时也提供了Java编程接口 maven配置 <project x ...
 - 前端使用AngularJS的$resource,后端ASP.NET Web API,实现增删改查
		
AngularJS中的$resource服务相比$http服务更适合与RESTful服务进行交互.本篇后端使用ASP.NET Web API, 前端使用$resource,实现增删改查. 本系列包括: ...
 
随机推荐
- UVa 10346 - Peter's Smokes
			
题目大意:Peter有n支烟,每k个剩下的烟头可以卷成一支新烟,问Peter能吸多少跟烟? 简单数学题. #include <cstdio> int main() { #ifdef LOC ...
 - App外包开发周期一般多长?
			
很多人问我,开发一个app要用多长时间.事实上开发一款app没有固定周期的,得因产品而论,你软件的功能需求决定了app外包开发的周期.但是除了app本身以外,人为因素往往对开发周期也有一定的影响.例如 ...
 - 字符集UTF-8MB4  MySQL utf8mb4 字符集,用于存储emoji表情
			
字符集UTF-8MB4 utf8mb4兼容utf8,且比utf8能表示更多的字符.看unicode编码区从1 - 126就属于传统utf8区,当然utf8mb4也兼容这个区,126行以下就是utf8m ...
 - dev中TreeList的应用(转)
			
如果需要在单元格添加时则用TreeList如果只是单纯读取数据或检索数据时则用GridControl 1.如果点击添加 时则添加TreeList的节点: protected internal void ...
 - Android实现版本更新
			
Android 实现从后台下载apk文件,保存到本地sd卡,使用系统安装apk,完成版本更新功能 LoadAppUtil.java import java.io.File; import java.i ...
 - Add Strings Leetcode
			
Given two non-negative integers num1 and num2 represented as string, return the sum of num1 and num2 ...
 - Java元注解
			
元注解是指注解的注解,包括@Retention @Target @Document @Inherited四种. 1.@Retention: 定义注解的保留策略@Retention(RetentionP ...
 - 建立、配置和使用Activity——启动其他Activity并返回结果
			
Activity还提供了一个startActivityForResult(Intent intent,int requestCode)方法来启动其他Activity.该方法用于启动指定Activity ...
 - 响应的系统设置的事件——重写onConfigurationChanged响应系统设置更改
			
如果程序需要监听系统设置的更改,则可以考虑重写Activity的onConfigurationChanged(Configuration newConfig)方法,该方法是一个基于回调的事件处理方法: ...
 - Scrum
			
Scrum[编辑] 维基百科,自由的百科全书 Scrum是一种敏捷软件开发的方法学,用于迭代式增量软件开发过程.Scrum在英语是橄榄球运动中争球的意思. 虽然Scrum是为管理软件开发项目而开发 ...