JDBC的全称是Java DataBase Connection,也就是Java数据库链接,咱们能够用它来操做关系型数据库。JDBC接口及相关类在java.sql包和javax.sql包里。咱们能够用它来链接数据库,执行SQL查询,存储过程,并处理返回的结果。html
JDBC接口让Java程序和JDBC驱动实现了松耦合,使得切换不一样的数据库变得更加简单。java
有四类JDBC驱动。和数据库进行交互的Java程序分红两个部分,一部分是JDBC的API,实际工做的驱动则是另外一部分。mysql
A JDBC-ODBC Bridge plus ODBC Driver(类型1):它使用ODBC驱动链接数据库。须要安装ODBC以便链接数据库,正由于这样,这种方式如今已经基本淘汰了。sql
B Native API partly Java technology-enabled driver(类型2):这种驱动把JDBC调用适配成数据库的本地接口的调用。数据库
C Pure Java Driver for Database Middleware(类型3):这个驱动把JDBC调用转发给中间件服务器,由它去和不一样的数据库进行链接。用这种类型的驱动须要部署中间件服务器。这种方式增长了额外的网络调用,致使性能变差,所以不多使用。缓存
D Direct-to-Database Pure Java Driver(类型4):这个驱动把JDBC转化成数据库使用的网络协议。这种方案最简单,也适合经过网络链接数据库。不过使用这种方式的话,须要根据不一样数据库选用特定的驱动程序,好比OJDBC是Oracle开发的Oracle数据库的驱动,而MySQL Connector/J是MySQL数据库的驱动。服务器
JDBC API使用Java的反射机制来实现Java程序和JDBC驱动的松耦合。随便看一个简单的JDBC示例,你会发现全部操做都是经过JDBC接口完成的,而驱动只有在经过Class.forName反射机制来加载的时候才会出现。网络
我以为这是Java核心库里反射机制的最佳实践之一,它使得应用程序和驱动程序之间进行了隔离,让迁移数据库的工做变得更简单。在这里能够看到更多JDBC的使用示例。并发
JDBC链接是和数据库服务器创建的一个会话。你能够想像成是一个和数据库的Socket链接。oracle
建立JDBC链接很简单,只须要两步:
A. 注册并加载驱动:使用Class.forName(),驱动类就会注册到DriverManager里面并加载到内存里。 B. 用DriverManager获取链接对象:调用DriverManager.getConnnection()方法并传入数据库链接的URL,用户名及密码,就能获取到链接对象。
Connection con = null; try{ // load the Driver Class Class.forName("com.mysql.jdbc.Driver"); // create the connection now con = DriverManager.getConnection("jdbc:mysql://localhost:3306/UserDB", "pankaj", "pankaj123"); }catch (SQLException e) { System.out.println("Check database is UP and configs are correct"); e.printStackTrace(); }catch (ClassNotFoundException e) { System.out.println("Please include JDBC MySQL jar in classpath"); e.printStackTrace(); }
JDBC的DriverManager是一个工厂类,咱们经过它来建立数据库链接。当JDBC的Driver类被加载进来时,它会本身注册到DriverManager类里面,你能够看下JDBC Driver类的源码来了解一下。
而后咱们会把数据库配置信息传成DriverManager.getConnection()方法,DriverManager会使用注册到它里面的驱动来获取数据库链接,并返回给调用的程序。
使用DatabaseMetaData能够获取到服务器的信息。当和数据库的链接成功创建了以后,能够经过调用getMetaData()方法来获取数据库的元信息。DatabaseMetaData里面有不少方法,经过它们能够获取到数据库的产品名称,版本号,配置信息等。
DatabaseMetaData metaData = con.getMetaData(); String dbProduct = metaData.getDatabaseProductName();
Statement是JDBC中用来执行数据库SQL查询语句的接口。经过调用链接对象的getStatement()方法咱们能够生成一个Statement对象。咱们能够经过调用它的execute(),executeQuery(),executeUpdate()方法来执行静态SQL查询。
因为SQL语句是程序中传入的,若是没有对用户输入进行校验的话可能会引发SQL注入的问题,若是想了解更多关于SQL注入的,能够看下这里。
默认状况下,一个Statement同时只能打开一个ResultSet。若是想操做多个ResultSet对象的话,须要建立多个Statement。Statement接口的全部execute方法开始执行时都默认会关闭当前打开的ResultSet。
Statement的execute(String query)方法用来执行任意的SQL查询,若是查询的结果是一个ResultSet,这个方法就返回true。若是结果不是ResultSet,好比insert或者update查询,它就会返回false。咱们能够经过它的getResultSet方法来获取ResultSet,或者经过getUpdateCount()方法来获取更新的记录条数。
Statement的executeQuery(String query)接口用来执行select查询,而且返回ResultSet。即便查询不到记录返回的ResultSet也不会为null。咱们一般使用executeQuery来执行查询语句,这样的话若是传进来的是insert或者update语句的话,它会抛出错误信息为 “executeQuery method can not be used for update”的java.util.SQLException。
Statement的executeUpdate(String query)方法用来执行insert或者update/delete(DML)语句,或者 什么也不返回DDL语句。返回值是int类型,若是是DML语句的话,它就是更新的条数,若是是DDL的话,就返回0。
只有当你不肯定是什么语句的时候才应该使用execute()方法,不然应该使用executeQuery或者executeUpdate方法。
PreparedStatement对象表明的是一个预编译的SQL语句。用它提供的setter方法能够传入查询的变量。
因为PreparedStatement是预编译的,经过它能够将对应的SQL语句高效的执行屡次。因为PreparedStatement自动对特殊字符转义,避免了SQL注入攻击,所以应当尽可能的使用它。
可使用它的setNull方法来把null值绑定到指定的变量上。setNull方法须要传入参数的索引以及SQL字段的类型,像这样:
ps.setNull(10, java.sql.Types.INTEGER);.
有的时候表会生成主键,这时候就能够用Statement的getGeneratedKeys()方法来获取这个自动生成的主键的值了。
它和Statement相比优势在于:
PreparedStatement的一个缺点是,咱们不能直接用它来执行in条件语句;须要执行IN条件语句的话,下面有一些解决方案:
在查询数据库后会返回一个ResultSet,它就像是查询结果集的一张数据表。
ResultSet对象维护了一个游标,指向当前的数据行。开始的时候这个游标指向的是第一行。若是调用了ResultSet的next()方法游标会下移一行,若是没有更多的数据了,next()方法会返回false。能够在for循环中用它来遍历数据集。
默认的ResultSet是不能更新的,游标也只能往下移。也就是说你只能从第一行到最后一行遍历一遍。不过也能够建立能够回滚或者可更新的ResultSet,像下面这样。
Statement stmt = con.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_UPDATABLE);
当生成ResultSet的Statement对象要关闭或者从新执行或是获取下一个ResultSet的时候,ResultSet对象也会自动关闭。
能够经过ResultSet的getter方法,传入列名或者从1开始的序号来获取列数据。
根据建立Statement时输入参数的不一样,会对应不一样类型的ResultSet。若是你看下Connection的方法,你会发现createStatement和prepareStatement方法重载了,以支持不一样的ResultSet和并发类型。
一共有三种ResultSet对象。
ResultSet有两种并发类型。
setMaxRows能够用来限制返回的数据集的行数。固然经过SQL语句也能够实现这个功能。好比在MySQL中咱们能够用LIMIT条件来设置返回结果的最大行数。
setFetchSize理解起来就有点费劲了,由于你得知道Statement和ResultSet是怎么工做的。当数据库在执行一条查询语句时,查询到的数据是在数据库的缓存中维护的。ResultSet其实引用的是数据库中缓存的结果。
假设咱们有一条查询返回了100行数据,咱们把fetchSize设置成了10,那么数据库驱动每次只会取10条数据,也就是说得取10次。当每条数据须要处理的时间比较长的时候而且返回数据又很是多的时候,这个可选的参数就变得很是有用了。
咱们能够经过Statement来设置fetchSize参数,不过它会被ResultSet对象设置进来的值所覆盖掉。
存储过程就是数据库编译好的一组SQL语句,能够经过JDBC接口来进行调用。咱们能够经过JDBC的CallableStatement接口来在数据库中执行存储过程。初始化CallableStatement的语法是这样的:
CallableStatement stmt = con.prepareCall("{call insertEmployee(?,?,?,?,?,?)}"); stmt.setInt(1, id); stmt.setString(2, name); stmt.setString(3, role); stmt.setString(4, city); stmt.setString(5, country); //register the OUT parameter before calling the stored procedure stmt.registerOutParameter(6, java.sql.Types.VARCHAR); stmt.executeUpdate();
咱们得在执行CallableStatement以前注册OUT参数。关于这个更详细的资料能够看这里。
有时候相似的查询咱们须要执行不少遍,好比从CSV文件中加载数据到关系型数据库的表里。咱们也知道,执行查询能够用Statement或者PreparedStatement。除此以外,JDBC还提供了批处理的特性,有了它,咱们能够在一次数据库调用中执行多条查询语句。
JDBC经过Statement和PreparedStatement中的addBatch和executeBatch方法来支持批处理。
批处理比一条条语句执行的速度要快得多,由于它须要不多的数据库调用,想进一步了解请点这里。
默认状况下,咱们建立的数据库链接,是工做在自动提交的模式下的。这意味着只要咱们执行完一条查询语句,就会自动进行提交。所以咱们的每条查询,实际上都是一个事务,若是咱们执行的是DML或者DDL,每条语句完成的时候,数据库就已经完成修改了。
有的时候咱们但愿由一组SQL查询组成一个事务,若是它们都执行OK咱们再进行提交,若是中途出现异常了,咱们能够进行回滚。
JDBC接口提供了一个setAutoCommit(boolean flag)方法,咱们能够用它来关闭链接自动提交的特性。咱们应该在须要手动提交时才关闭这个特性,否则的话事务不会自动提交,每次都得手动提交。数据库经过表锁来管理事务,这个操做很是消耗资源。所以咱们应当完成操做后尽快的提交事务。在这里有更多关于事务的示例程序。
经过Connection对象的rollback方法能够回滚事务。它会回滚此次事务中的全部修改操做,并释放当前链接所持有的数据库锁。
转自:http://it.deepinmind.com/jdbc/2014/03/18/JDBC%E5%B8%B8%E8%A7%81%E9%9D%A2%E8%AF%95%E9%A2%98%E9%9B%86%E9%94%A6%28%E4%B8%80%29.html