JDBC,JdbcTemplate,JPA,Hibernate之间的关系概述

什么是JDBC?

Java语言访问数据库的一种规范,是一套API。JDBC (Java Database Connectivity) API,即Java数据库编程接口,是一组标准的Java语言中的接口和类,使用这些接口和类,Java客户端程序可以访问各种不同类型的数据库。

JDBC规范采用接口和实现分离的思想设计了Java数据库编程的框架。接口包含在java.sql及javax.sql包中,其中java.sql属于JavaSE,javax.sql属于JavaEE。

为了使客户端程序独立于特定的数据库驱动程序,JDBC规范建议开发者使用基于接口的编程方式,即尽量使应用仅依赖java.sql及javax.sql中的接口和类。

JAVA使用JDBC访问数据库的步骤:

1.得到数据库驱动程序

2.创建数据库连接

3.执行SQL语句

4.得到结果集

5.对结果集做相应的处理(增,删,改,查)

6.关闭资源:这里释放的是DB中的资源

简单的例子:

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. Connection con= null;  
  2. PreparedStatement pStmt=null;  
  3. ResultSet rs = null;  
  4. try{  
  5. con = ods.getConnection();  
  6. String sql = "select * from admin";  
  7. pStmt=con.prepareStatement(sql);  
  8. rs=pStmt.executeQuery();  
  9. while(rs.next())  
  10. { }  
  11. }  
  12. catch(Exception ex) {  
  13. try{  
  14. con.rollback();  
  15. }catch(SQLException sqlex){  
  16. sqlex.printStackTrace(System.out);  
  17. }  
  18. ex.printStackTrace();  
  19. }finally{  
  20. try{  
  21. rs.close();  
  22. pStmt.close();  
  23. con.close();  
  24. }catch(Exception e){  
  25. e.printStackTrace();  
  26. }  
  27. }  


JdbcTemplate的作用
常见的JDBC代码,简单的select语句也需要冗长的出错处理,并且每个函数都不断地重复同样的代码。
JdbcTemplate正是为了减少上述繁琐的代码而设计出来的。它是对JDBC的一种封装,抽象我们常用的一些方法。Simple and Stupid就是它的目标。下面是完成了刚才JDBC代码同样功能的JdbcTemplate的代码:

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. String sql = "select * from admin";  
  2. jdbcTemplate.query(sql,new RowCallbackHandler() {  
  3. public void processRow(ResultSet rs) throws SQLException {  
  4. }  
  5. } );  
使用模板方式封装 jdbc数据库操作-固定流程的动作,提供丰富callback回调接口功能,方便用户自定义加工细节,更好模块化jdbc操作,简化传统的JDBC操作的复杂和繁琐过程。
1) 使用JdbcTemplate 更新(insert /update /delete)

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. int k = jdbcTemplate.update("UPDATE tblname SET prop1=?,prop2=?..."new Object[]{...});  

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. jdbcTemplate.update("INSERT INTO tblname VALUES(?,?,..)"new Object[]{...},  
  2.      new int[]{Types.VARCHAR,Types.NUMERIC});  

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. jdbcTemplate.update("INSERT INTO tblname VALUES(?,?,..)",                       
  2.         new PreparedStatementSetter(){                            
  3.                public void setValues(PreparedStatement ps) throws SQLException{        
  4.                     ps.setLong(1, user.getId(1));  
  5.                     ps.setString(2, user.getName(2));     
  6.                     ps.setDate(3new java.sql.Date(new Date().getTime());    
  7.                     ps.setTimestamp(4new Timestamp(new Date().getTime());  
  8.                }                       
  9.         }  
  10. );  

2) 使用JdbcTemplate 查询 (select)

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. final User user = new User();  
  2. jdbcTemplate.query("SELECT id,name,.. FROM tblname WHERE id=1",  
  3.        new RowCallbackHandler(){  
  4.               public void processRow(ResultSet rs) throws SQLException{  
  5.                     user.setId(rs.getLong(1));  
  6.                     user.setName(rs.getString(2));  
  7.               }  
  8.       }  
  9. );    

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. List uGroup = jdbcTemplate.query("SELECT id,name,.. FROM tblname WHERE igroup=1",  
  2.      new RowMapper(){  
  3.             public Object mapRow(ResultSet rs,int no) throws SQLException{  
  4.                      User user = new User();  
  5.                      user.setId(rs.getLong(1));  
  6.                      user.setName(rs.getString(2));  
  7.                      return user ;  
  8.             }  
  9.      }  
  10. };   

3)使用JdbcTemplate 便捷方法

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. List uNames = jdbcTemplate.queryForList("SELECT name FROM tblname WHERE id>?",  
  2.     new Integer []{5}, String.class);   

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. List<Map> uMapList = (List<Map>) jdbcTemplate.queryForList( "SELECT id, name FROM tblname WHERE id>?",  
  2.              new Integer []{5});  
  3. for(Map<String,Object> uMap :uMapList){  
  4.       Integer id = uMap.get("id");  
  5.       String name = uMap.get("name");  
  6. };   

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. String user = jdbcTemplate.queryForObject("SELECT name FROM tblname WHERE id=?",  
  2.      new Integer []{5}, String.class );  

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. int uNum = jdbcTemplate.queryForInt("SELECT count(*) FROM tblname WHERE id>?",  
  2.     new Integer []{5});   

4)使用jdbc 操作类

a)扩展 MappingSqlQuery类

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. class JdbcQueryObject extends MappingSqlQuery { // extends SqlQuery  
  2.       public JdbcQueryObject (DataSource ds,String sql){  
  3.             this.setDataSource( ds );  
  4.             this.setSql( sql );  
  5.             this.declareParameter(new Sqlparameter("propName",  
  6.                 Types.VARCHAR);// propName 提示作用  
  7.         this.compile();  
  8.       }  
  9.       public Object mapRow(ResultSet rs,int p) throws SQLException{  
  10.                  // ...  
  11.      }  
  12. }  
  13. JdbcQueryObject queryObj = new JdbcQueryObject( ds,  
  14.       "SELECT .. FROM tblName WHERE param=?");  
  15. List list = queryObj.execute(new Object[]{...});  

b)使用 SqlFunction 类 查询单条结果

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. SqlFunction queryFun = new SqlFunction( ds,  
  2.       "select count(*) from tblName where ..." ,new int[]{Types.CHAR,...} );  
  3. queryFun.compile();  
  4. queryFun.run(new Object[]{p1,p2,..});  

c)使用 SqlUpdate 类 更新

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. SqlUpdate updateFunc = new SqlUpdate(ds ,"INSERT tblName ...");  
  2. updateFunc.declareParameter( new SqlParameter("prop",Types.CHAR) );  
  3. updateFunc.compile();  
  4. updateFunc.update(new String[]{s1,s1});  

5)支持jdbc 事务

spring的事务管理有两种方式:编程式事务、声明式事务

这里谈一下 基于数据库单一资源的编程式事务:

spring用实现TransactionDefinition接口的类定义事务的属性:传播行为;隔离级别;超时值;只读标志

默认实现为:DefaultTransactionDefinition类

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. PlatformTransactionManager tm =    
  2. new DataSourceTransactionManager(  
  3.             jdbcTemplate.getDataSource() );  
  4. TransactionStatus status = null;  
  5. try{  
  6.     //null 默认事务属性配置DefaultTransactionDefinition  
  7.     status = tm.getTransaction(null);              
  8.   for(final String wd: words){              
  9.    try {  
  10.      jdbcTemplate.update( insertWordSql,  
  11.           new PreparedStatementSetter(){  
  12.   
  13.     public void setValues(PreparedStatement pstate)  
  14.                      throws SQLException {  
  15.                 pstate.setString(1, wd) ;  
  16.         pstate.setTimestamp(2,  
  17.         new Timestamp( new Date().getTime() ));                                  
  18.     }                      
  19.         }  
  20.      );                      
  21.               
  22.    } catch (DataAccessException e) {  
  23.        e.printStackTrace();  
  24.        //tm.rollback(status);  
  25.     }  
  26.     } // end for  
  27. finally {  
  28.      tm.commit(status);  
  29. }     


JPA的作用

JPA全称为Java Persistence API ,Java持久化API是Sun公司在Java EE 5规范中提出的Java持久化接口。JPA吸取了目前Java持久化技术的优点,旨在规范、简化Java对象的持久化工作。使用JPA持久化对象,并不是依赖于某一个ORM框架。 

在说为什么要使用JPA之前,我们有必要了解为什么要使用ORM技术。

ORM 是Object-Relation-Mapping,即对象关系影射技术,是对象持久化的核心。ORM是对JDBC的封装,从而解决了JDBC的各种存在问题:

a) 繁琐的代码问题

用JDBC的API编程访问数据库,代码量较大,特别是访问字段较多的表的时候,代码显得繁琐、累赘,容易出错。例如:PreparedStatement pstmt=con.prepareStatment("insert into account value(?,?,?,?,?,?,?,?,?)");

ORM则建立了Java对象与数据库对象之间的影射关系,程序员不需要编写复杂的SQL语句,直接操作Java对象即可,从而大大降低了代码量,也使程序员更加专注于业务逻辑的实现。

b) 数据库对象连接问题

关系数据对象之间,存在各种关系,包括1对1、1对多、多对1、多对多、级联等。在数据库对象更新的时候,采用JDBC编程,必须十分小心处理这些关系,以保证维持这些关系不会出现错误,而这个过程是一个很费时费力的过程。

ORM建立Java对象与数据库对象关系影射的同时,也自动根据数据库对象之间的关系创建Java对象的关系,并且提供了维持这些关系完整、有效的机制。

c) 系统架构问题

JDBC属于数据访问层,但是使用JDBC编程时,必须知道后台是用什么数据库、有哪些表、各个表有有哪些字段、各个字段的类型是什么、表与表之间什么关系、创建了什么索引等等与后台数据库相关的详细信息。

使用ORM技术,可以将数据库层完全隐蔽,呈献给程序员的只有Java的对象,程序员只需要根据业务逻辑的需要调用Java对象的Getter和 Setter方法,即可实现对后台数据库的操作,程序员不必知道后台采用什么数据库、有哪些表、有什么字段、表与表之间有什么关系。

d) 性能问题

采用JDBC编程,在很多时候存在效率低下的问题。

pstmt =conn.prepareStatement("insert into user_info values(?,?)");
       for (int i=0; i<1000; i++) {
          pstmt.setInt(1,i);
          pstmt.setString(2,"User"+i.toString());
          pstmt.executeUpdate();
       }

以上程序将向后台数据库发送1000次SQL语句执行请求,运行效率较低。

采用ORM技术,ORM框架将根据具体数据库操作需要,会自动延迟向后台数据库发送SQL请求,ORM也可以根据实际情况,将数据库访问操作合成,尽量减少不必要的数据库操作请求。


Hibernate与Jpa的关系

Jpa是一种规范,而Hibernate是它的一种实现。除了Hibernate,还有EclipseLink(曾经的toplink),OpenJPA等可供选择,所以使用Jpa的一个好处是,可以更换实现而不必改动太多代码。

在play中定义Model时,使用的是jpa的annotations,比如javax.persistence.Entity, Table, Column, OneToMany等等。但它们提供的功能基础,有时候想定义的更细一些,难免会用到hibernate本身的annotation。我当时想,jpa这么弱还要用它干什么,为什么不直接使用hibernate的?反正我又不会换成别的实现。

因为我很快决定不再使用hibernate,这个问题就一直放下了。直到我现在在新公司,做项目要用到Hibernate。

我想抛开jpa,直接使用hibernate的注解来定义Model,很快发现了几个问题:

  1. jpa中有Entity, Table,hibernate中也有,但是内容不同
  2. jpa中有Column,OneToMany等,Hibernate中没有,也没有替代品

我原以为hibernate对jpa的支持,是另提供了一套专用于jpa的注解,但现在看起来似乎不是。一些重要的注解如Column, OneToMany等,hibernate没有提供,这说明jpa的注解已经是hibernate的核心,hibernate只提供了一些补充,而不是两套注解。要是这样,hibernate对jpa的支持还真够足量,我们要使用hibernate注解就必定要使用jpa。

“如果hibernate认为jpa的注解够用,就直接用。否则会弄一个自己的出来作为补充”

jpa和hibernate都提供了Entity,我们应该用哪个,还是说可以两个一起用?网友回答说“Hibernate的Entity是继承了jpa的,所以如果觉得jpa的不够用,直接使用hibernate的即可”。

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值