PreparedStatement 、CallableStatement和ResultSet

14 篇文章 1 订阅

PreparedStatement 对象

在上一篇介绍中,SQL语句的执行是通过 Statement 对象实现的。Statement对象每次执行SQL语句时,都会对其进行编译。当相同的SQL语句执行多次是,Statement对象就会使数据库频繁编译相同的SQL语句,从而降低数据库的访问效率。 为了解决上述问题,Statement提供了一个子类 PreparedStatement 对象可以对SQL语句进行预编译。也就是说,当相同的SQL语句在此执行时,数据库只需要使用缓冲区中的数据,而不需要对SQL语句再次编译,从而有效提高数据的访问效率。我们可以通过下面这个程序,对此作进一步的了解。

 package com.rocky.test01;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
public class Demo1 {
    public static void main(String[] args) throws Exception {
        Connection conn=null;
        PreparedStatement ps=null;
        try {
            //加载数据库驱动
            Class.forName("com.mysql.jdbc.Driver");
            String url="jdbc:mysql://localhost:3306/test";
            String username="root";
            String password="root";
            //创建应用程序与数据库连接的Connection对象
            conn=DriverManager.getConnection(url, username, password);
            //执行的SQL语句
            String sql="insert into users(name,password,email,birthday) values(?,?,?,?)";
            //创建执行SQL语句的PreparedStatement对象
            ps =conn.prepareStatement(sql);
            ps.setString(1, "zhangsan");
            ps.setString(2, "123456");
            ps.setString(3, "zhangsan@sina.com");
            ps.setString(4, "2017-09-09");
            ps.executeUpdate();
        } catch (ClassNotFoundException e) {

            e.printStackTrace();
        }finally{
            //释放资源
            if(ps !=null){
                try {
                    ps.close();
                } catch (Exception e2) {
                    // TODO: handle exception
                }
                ps=null;
            }
            if(conn !=null){
                try {
                    conn.close();
                } catch (Exception e2) {
                    // TODO: handle exception
                }
                conn =null;
            }
        }   
    }
}

使用了 PreparedStatement 对象执行SQL语句的步骤。首先通过 Connection 对象的 PreparedStatement() 方法生成 PreparedStatement 对象然后调用 PreparedStatement 对象的 setXxx() 方法,给SQL语句中的参数赋值,最后通过调用 executeUpdate() 方法执行SQL语句。代码运行成功后,会在数据库test的users表中插入数据,进入MySQL,使用select语句查询users表,查询结果如下(下表中的数据,是将代码运行了三次产生的结果)
这里写图片描述

从上述结果中可以看出,users表中多了几条数据,说明PreparedStatement 对象可以成功执行对数据库的操作。

CallableStatement对象

CallableStatement接口是用于执行SQL存储过程的接口,它继承自 PreparedStatement接口,JDBC API提供了一个存储过程SQL转义语法,该语法允许对所有关系型数据库管理系统使用标准方式调用存储过程。此语法有一个包含结果参数的形式和一个不包含结果参数的形式,具体如下所示:

{?=call<procedure-name>[(<arg1>,<arg2>,....)]}
{call<procedure-name>[(<arg1>,<arg2>,...)]}

(1) IN类型:此类型是用于参数从外部传递给存储过程使用。
(2) OUT类型:此类型是存储过程执行过程中的返回值。
(3) IN、OUT混合类型:此类型是参数传入,然后返回。

如果使用结果参数,则必须将其注册为OUT参数。其他参数可用于输入、输出或同时用于二者。参数是根据编号按顺序引用的,第一个参数的编号是1。IN参数值是使用继承自 PreparedStatement 的setXxx() 方法设置的。在执行存储过程之前,必须注册所有的OUT参数的类型;它们的值是在执行后通过此类提供的 getXxx() 方法获取的。 CallableStatement可以返回一个 ResultSet 对象或者多个 ResultSet 对象,多个ResultSet对象是通过继承Statement 来处理的。 接下来,我们可以通过一个简单的代码,来创建一个存储过程。

delimiter //
create procedure add_pro(a int ,b int , out sum int)
begin
set sum=a+b;
end//
delimiter;

上面的程序创建了名为add_pro的存储过程,该存储过程包含三个参数:a、b是默认的参数,即传入参数,而sum使用out修饰,是传出参数。 调用存储过程使用CallableStatement,可以通过Connection的 prepareCall( ) 方法来创建 CallableStatement对象,创建该对象时需要传入调用存储过程的SQL语句,如下:

//使用Connection 来创建一个 CallableStatement对象
CallableStatement cs=conn.prepareCall("{call add_pro(?,?,?)}");

在上述代码中,?占位符可以是IN、OUT或者是INOUT参数,这取决于存储过程中的add_pro。存储过程的参数既有传入参数,也有传出参数。所谓传入参数就是Java程序必须为这些参数传入值,那么可以通过 CallableStatement 的setXxx( )方法为传入的参数设置值;所谓传出参数就是Java程序中可以通过该参数获取存储过程里的值,那么CallableStatement需要调用 registerOutParameter( ) 方法来注册该参数,示例如下:

//注册 CallableStatement 的第三个参数 int类型
cs.regiserOutParameter(3,Types.INTEGER);

经过上面的步骤之后,就可以调用 CallableStatement的 execute( )方法来执行存储过程,执行结束后通过 CallableStatement 对象的 getXxx( int index)方法来获取指定的传出参数的值。我们可以通过下面的这个例子,来进一步的了解:

package com.rocky.test01;
import java.sql.CallableStatement;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.Types;
public class Demo3 {
    public static void main(String[] args) throws Exception {
    Connection conn=null;
    CallableStatement cs=null;
    try {
        //加载数据库驱动
        Class.forName("com.mysql.jdbc.Driver");
        String url="jdbc:mysql://localhost:3306/test";
        String username="root";
        String password="root"; 
        //创建应用程序与数据库连接的Connection对象
        conn=DriverManager.getConnection(url, username, password);
        //执行的SQL语句
        String sql="insert into users(name,password,email,birthday) values(?,?,?,?)";
        //创建执行SQL语句的CallableStatement对象
        cs=conn.prepareCall("call add_pro(?,?,?)");
        cs.setInt(1, 4);
        cs.setInt(2, 5);
        //注册 CallableStatement的第三个参数为int 类型
        cs.registerOutParameter(3, Types.INTEGER);
        //执行存储过程
        cs.execute();
        System.out.println("执行的结果是:"+cs.getInt(3));
    } catch (ClassNotFoundException e) {e.printStackTrace();
    }finally{
        //释放资源
        if(cs !=null){
            try {
                cs.close();
            } catch (Exception e2) {
                // TODO: handle exception
            } cs=null;
        }
        if(conn !=null){
            try {
                conn.close();
            } catch (Exception e2) {
                // TODO: handle exception
            }conn =null;
        }
    }   
}
}

程序执行成功后,会将4和5两个数字相加的结果打印到控制台上,具体的结果如图所示:
这里写图片描述

ResultSet 对象

ResultSet主要用于存储结果集,并且只能通过next( )方法由前向后逐个获取结果集中的数据。但是,如果想获取结果集中任意位置的数据,则需要在创建 Statement 对象时,设置两个 ResultSet定义的常量,具体的设置方式如下:

Statement st =conn.createStatement(ResultSet.TYPE_SCROLL_INSENITVE,ResultSet.CONCUR_READ_ONLY); ResultSet rs=st.executeQuery(sql); 

在上述的方法中,常量ResultSet.TYPE_SCROLL_INSENSITIVE表示结果集可以滚动,常量ResultSet.CONCUR_READ_ONLY表示以只读的形式打开结果集。接下来,我们通过下列代码来对此做进一步的了解,通过示例来看出如何使用ResultSet对象滚动读取结果集中的数据。

package com.rocky.test01;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
public class Demo4 {
public static void main(String[] args) {
Connection conn=null;
Statement stmt=null;
try {
//注册数据库的驱动
Class.forName("com.mysql.jdbc.Driver");
String url ="jdbc:mysql://localhost:3306/test";
String username="root";
String password="root";
conn =DriverManager.getConnection(url, username, password);
String sql="select * from users";
Statement st=conn.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY);
ResultSet rs=st.executeQuery(sql);
System.out.println("第2条数据的name的值:");
//将指针定位到结果集中第2行的数据
rs.absolute(2);
System.out.println(rs.getString("name"));
System.out.println("第1条的数据name的值:");
//将指针定位到结果集中第一行数据之前
rs.beforeFirst();
//将指针向后滚动
rs.next();
System.out.println(rs.getString("name"));
System.out.println("第4条数据的name值为:");
//将指针定位到结果集中最后一条数据之后
rs.afterLast();
//将指针向前滚动
rs.previous();
System.out.println(rs.getString("name"));
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}finally{
if (stmt !=null){
try {
stmt.close();
} catch (Exception e2) {
e2.printStackTrace();
}
stmt =null;
}
if(conn !=null){
try {
conn.close();
} catch (Exception e2) {
// TODO: handle exception
}
conn =null;
}
}
}
}

程序的运行结果如下:
这里写图片描述

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
jdbc: 1.数据库连接的方式: ODBC:开放式数据库连接。 C语言实现的,提供语言和(基于SQL的)数据库进行交互的“一致性”的接口 JDBC:java版本的ODBC JDBC连接数据库的步骤: 1.注册驱动(加载驱动): 注册的方式: 1.使用类加载器(使用反射的方式) Class.forName(driverName); 2.实例化Driver Driver driver = new oracle.jdbc.driver.OracleDriver(); DriverManager.registerDriver(driver); 3.加虚拟机参数jdbc.drivers -Djdbc.drivers=oracle.jdbc.driver.OracleDriver 4.从Jdk6.0以后要求,JDBC 4.0 Drivers 必须包括 META-INF/services/java.sql.Driver 文件,有了这个文件以后不需要在显示的使用Class.forName来进行驱动的注册 Oracle数据库进行连接的时候,使用的驱动类: 1.oracle.jdbc.driver.OracleDriver 2.oracle.jdbc.OracleDriver 2.建立连接 连接方式: 1.DriverManager(中的getConnection其实也是调用的Driver.connect方法) getConnection(url);//没有用户名密码 //将用户名密码存放在java.util.Properties对象中 getConnection(url,properties); getConnection(url,user,passwd); 2.直接调用Driver.connect方法执行 Driver d = new DriverClass(); d.connect(url,properties); 3.创建Statement: Statement: 1.创建时不需要传递sql语句,但是执行时需要传递sql语句 2.如果涉及到动态参数的传递,必须使用字符串拼接 PreparedStatement: 1.创建时就需要传递sql语句,执行的时候不需要传递sql语句 2.如果涉及到动态参数的传递,可以使用字符串拼接,也可以使用?占位的形式 给?号传值使用的是 pstmt.setType(index,value); index从1开始 3.提供预编译的功能,某种程度上可以避免sql注入的问题 4.提前做语法检查,在给?赋值的过程中要求数据类型一定要匹配,这样在某种程度上可以避免因为数据类型不匹配而发生的异常 CallableStatement:主要用来执行pl/sql的一些过程,函数等。 1.写一条恒成立的select语句,无论你输入的条件是什么,总是能讲中的所有数据输出 select id,last_name from s_emp where '1' ='1'; where 1=1; findByOption(Integer age,String province,String gender){ String sql = select * from s_emp where 1=1; if(age!=null){ sql + "and age < age"; } if(province!=null){ sql + "and province=province"; } if(gender!=null){ sql + "and gender = gender"; } } 4.执行sql语句: execute:返回boolean类型的值,代是否有结果集返回(如果执行select操作,是有ResultSet的,返回值为true) executeUpdate:返回int类型的值,代的是,操作执行完成后,受影响的数据库的行计数(针对于insert,update,delete) executeQuery:返回的是ResultSet ResultSet:类似于指针或者游标的东西,里边保存的不是所有的结果,而是指向结果集的正上方。所以如果一旦连接关闭那么ResultSet将取不到值 5.处理结果 如果有结果集,处理结果集 ResultSet next(),每执行一次,向下移动一次,如果有值,返回true,如果没值返回false while(rs.next()){ rs.getType(index/columnName); 如果传的是index,那么索引是从1开始的。 select id,last_name from s_emp; 那么1代的就是id,依次类推 } 6.关闭资源 先开的后关 D:\oraclexe\app\oracle\product\10.2.0\server\jdbc\lib\ojdbc14.jar linux下启动数据库监听的命令: lsnrctl start; "select id from s_emp where name = '"+name+"'"; table s_user( id , name, age) class User{ } //分别使用Statement对象和PreparedStatement对象实现 public class JDBCTest{ //查找s_user中所有的数据,并且返回他们的集合 public Collection<User> findAll(){ } //按照名字进行删除 public int deleteUser(String name){ } //将user对象的信息更新到数据库中 public int updateUser(User user){ } //讲User对象保存到数据库中 public void insertUser(User user){ } } JDBC:本身自动帮我做事务管理的操作 AutoCommit = true; Connection.setAutoCommit(false); 正常: conn.commit(); 异常: conn.rollback(); JDBC批处理: addBatch executeBatch Statement PreparedStatement Statement{ 1.获取连接 getConnection(); 2.创建Statement conn.createStatement(); 3.执行sql语句 String sql .... insert delete update executeUpdate select ----- Result executeQuery 4.如果有结果集处理结果集 5.资源关闭 } Statement: update/delete/insert: stmtExecute(String sql){ 1.获取连接 getConnection(); 2.创建Statement conn.createStatement(); 3.执行sql语句 stmt.execute(sql); 4.资源关闭 } Select: stmtQuery(String sql,ResultHander handler){ 1.获取连接 getConnection(); 2.创建Statement conn.createStatement(); 3.执行sql语句 ResultSet set = executeQuery 4.如果有结果集处理结果集 handler!=null handler.handler(set); 5.资源关闭 } 结果集处理的规则: ResultHandler{ handler(ResultSet rs); } PreparedStatement update/delete/insert: pstmtExecute(String sql,PstmtSetter setter){ 1.获取连接 getConnection(); 2.创建PreparedStatement conn.prepareStatement(sql); if(setter!=null){ setter.setter(pstmt); } 3.执行sql语句 pstmt.execute(sql); 4.资源关闭 } ?设值的标准: PstmtSetter{ setter(PreparedStatement pstmt); } Select: stmtQuery(String sql,PstmtSetter setter,ResultHander handler){ 1.获取连接 getConnection(); 2.创建PreparedStatement conn.prepareStatement(); setter!=null setter.setter(pstmt); 3.执行sql语句 ResultSet set = executeQuery 4.如果有结果集处理结果集 handler!=null handler.handler(set); 5.资源关闭 } 结果集处理的规则: ResultHandler{ handler(ResultSet rs); } 现层 业务逻辑层 持久层 java中的对象分为: 1.域对象:主要作为信息的载体 2.行为对象:注重操作 Register(User user){ } insert(User user){ String sql = "insert into s_user( user.getId()+user.getName().......)"; } insert(User user){ save(user); } ORM Object Relational Class-Object 名字 名 属性 字段(列) object 记录 User s_User id id name name passwd passwd insert into s_user(id,name,passwd) values(?,?,?); Hibernate:ORM的中间件,或者说是实现了ORM的一个框架,对JDBC做了轻量级的封装。 ORM:使用元数据信息来描述对象和数据库之间的关系,并且能够自动实现java中持久化对象到关系型数据库的映射 脏检查:自动对缓存中的数据进行检查,并且选择在合适的时机和数据库之间进行交互,以保持数据的一致性 延迟加载:从数据库中还原对象的时候,不会立即对对象进行初始化,而是等到用到的时候才会进行初始化 Core: POJO hibernate.cfg.xml .hbm.xml Session: 1.轻量级的,创建和销毁不需要消耗很大的资源 2.非线程安全的 3.hibernate的一级缓存 4.介于Connection和Transaction之间的一个对象 5.hibernate中用来做持久化操作的 SessionFactory 1.重量级的,创建和销毁需要消耗很大的资源,不建议频繁创建和销毁 2.线程安全的,一个数据库对应一个Sessionfactory(一般一个应用程序对应一个SessionFactory就够了) 3.是一个很大的缓存,本身维护了一个可配置的二级缓存 4.用来构建Session对象 Configuration 1.启动和配置Hibernate 2.读取hibernate的配置文件和映射文件 3.构建SessionFactory对象 Transaction 1.事务管理对象 Query 1.查询对象,HQL Criteria 1.hibernate提供的更面向对象的一种查询方式。 准备工作: 1.java中的POJO对象存在 2.数据库得存在 3.hibernate的配置文件(hibernate.cfg.xml)得存在 4.POJO.hbm.xml文件存在 5.hibernate的jar包以及数据库的驱动包存在 Hibernate的编程: 1.构建Configuration对象读取配置文件 2.读取映射文件 3.构建SessionFactory对象 4.构建Session对象 5.开启事务 6.持久化操作 7.事务的提交(正常提交,不正常回滚) 8.关闭资源 主键增长策略: 1.increment:自动增长 select max(id) from table; 找到最大值之后+1赋给要持久化的对象 2.assigned:指派 hibernate不在自动生成主键值,需要你在插入时自己指明 3.hilo:高低值算法,由数据库提供高值,程序提供低值 value = hi*(max_lo+1)+lo 4.sequences Cat: cid name color weight age --------------------------- 1.每个人都有一个地址 person{ id name } Address{ country province city street } table: id name country province city street Component:数据库中采用一张的设计,java中采用两个类的设计 ----------------------------------- Employee: id name salary award amount table id name salary award formula: ---------------------------- 关系模型: 多对一: 订单和客户之间的关系 Order{ id amount customer } Customer{ id name } many-to-one:标签中对于cascade的取值delete,delete-orphan,all-delete-orphan(只用unique属性值不为true不能出现)慎用 cascade:级联属性 none:不做任何级联操作 save-update:对当前对象执行save,update, saveOrupdate,会级联操作和它相关联的对象 delete:在删除当前对象的时候,级联删除和他相关联的对象 all: save-update+delete delete-orphan:解除关联关系时,删除和当前对象失去关联的对象 all-delete-orphan:all+delete-orphan 单向的一对多的关系,在进行关联关系的操作时,会执行不必要的update语句,所以,一般情况下,我们不会做单向一对多的映射。 inverse="true":让其中一方放弃对关联关系的维护 一般在做双向多对一(一对多)关联关系映射的时候,一般会设置让一的一方放弃对关联关系的维护,以减少不必要的更新语句 一对一: 基于外键的一对一 Wife Husband id id name name h_id references Husband(id) unique 基于主键的一对一 Wife Husband id references Husband(id) id name name create table Husband( id number primary key, name varchar2(15) ); create table Wife( id number primary key references husband(id), name varchar2(15) ); 多对多: Teacher Student id id name name 桥:s_t s_id t_id ------------------------------ 操作持久化对象: Hibernate中对象的三种状态: Transient(瞬态): 1.由new关键字创建 2.没有和Session进行关联的 3.数据库中没有对应的记录存在 4.操作不会影响数据库中的数据 Persistent(持久态): 1.和Session之间有关联 2.在数据库中有对应记录存在,并且有持久化标识 3.对持久对象的更动,会对数据库中的数据产生影响。(自动脏检查机制) Detached(托管状态): 1.和Session失去关联 2.数据库中有对应记录存在 3.对托管对象的更动,在托管期间不会影响数据库,但是将托管状态重新和数据库进行关联的时候会将托管对象重新变为持久态,那么在托管期间发生的更动也会被更新到数据库中 get()/load():从数据库中还原数据 get: 1.先从缓存中进行查找,如果找到就直接返回 2.如果找不到,get()会立即发送sql语句到数据库中查找数据,如果找到就返回,如果找不到返回null; load:(使用延迟加载策略) 1.load()方法默认要加载的对象一定存在,所以很放心的使用代理对象,等到用到的时候从缓存中查找数据,如果找到,就返回,找不到发送sql语句到数据库中查找,如果数据库中没有对应记录存在,抛ObjectNotFoundException Hibernate中的查询: 1.OID检索: get load 2.导航对象图查询: Customer ---- Order之间有对应关系 通过检索到Customer以后,想获取和Customer相关的对象,可以使用 cus.getOrders(); 3.HQL: Query sql: select name from t_cus; hql: select name from hiber.many2one.Customer query.list(); 直接发送一条select语句去中查找所有的属性 query.iterate(); 先发送一条select语句从中查找所有的id,然后根据id从缓存中进行查找,找到就返回,找不到再发送sql语句按id从数据库中进行查找,所以可能会产生N+1的问题 Customer order select from customer order where customer.id=order.c_id; 4.QBC(Query By Criteria) Criteria Restrictions Order Projections QBE(Query By Example) 5.sql A B 总额 《 4000 - 100 1900 2100 900 Session默认在遇到以下情况的时候会flush: 1.事务提交的时候 2.某些查询操作执行的时候(不是所有的查询) 3.当应用程序显示的调用session.flush操作的时候 悲观锁: LockMode LockMode.UPGRADE:借助于数据库的 select ... for update; 那么事务一定会等到上一个获取了锁的事务commit之后才执行,如果上一个事务一直不提交,那么它就一直等下去 LockMode.UPGRADE_NOWAIT: select ... for update nowait; nowait就是不等待,一旦在操作过程中发现要操作的数据被加了锁,那么直接抛 ORA-00054: 资源正忙, 但指定以 NOWAIT 方式获取资源 乐观锁: 版本控制(version) 时间戳(timestamp)

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值