oracle字段类型blob,clob,nclob

高手znp的面试必问问题其中之一就是这个,另一个是JAVA的8种基本数据类型。
今天咱也好好学学这个东西。
说明:三种大型对象(LOB),用来保存较大的图形文件或带格式的文本文件,如Miceosoft Word文档,以及音频、视频等非文本文件,最大长度是4GB。LOB有几种类型,取决于你使用的字节的类型,Oracle 8i实实在在地将这些数据存储在数据库内部保存。可以执行读取、存储、写入等特殊操作。

我们所操作的clobtest_table中属性是(字符型id,CLOB型picstr),目前我们假设一个大的字符对象str已经包含了我们需要存入picstr字段的数据。而且connection对象conn已经建立。以下的例子程序也因为不想占用太多的空间,所以对抛出异常没有写。大家参考一下api doc。就可以知道该抛出什么异常了,此处仅仅告诉大家如何去写。

代码:


(1)对数据库clob型执行插入操作
*************************************************

java.sql.PreparedStatement pstmt = null;
ResultSet rs = null;
String query = "";

conn.setAutoCommit(false);
query = "insert into clobtest_table(id,picstr) values(?,empty_clob())";
java.sql.PreparedStatement pstmt = conn.prepareStatement(query);
pstmt.setString(1,"001");
pstmt.executeUpdate();
pstmt = null
query = "select picstr from clobtest_table where id = '001' for update";
pstmt = con.prepareStatement(query)
rs= pstmt.executeQuery();

oracle.sql.CLOB clobtt = null;
if(rs.next()){
clobtt = (oracle.sql.CLOB)rs.getClob(1);
}
Writer wr = clobtt.getCharacterOutputStream();
wr.write(strtmp);
wr.flush();
wr.close();
rs.close();
con.commit();



(2)通过sql/plus查询是否已经成功插入数据库
*************************************************

PL/SQL的包DBMS_LOB来处理LOB数据。察看刚才的插入是否成功。使用DBMS_LOB包的getlength这个procedure来检测是否已经将str存入到picstr字段中了。如:

SQL> select dbms_lob.getlength(picstr) from clobtest_table;


(3)对数据库clob型执行读取操作
*************************************************

读取相对插入就很简单了。基本步骤和一半的取数据库数据没有太大的差别。
String description = ""
query = "select picstr from clobtest_table where id = '001'";
pstmt = con.prepareStatement(query);
ResultSet result = pstmt.executeQuery();
if(result.next()){
oracle.jdbc.driver.OracleResultSet ors =
(oracle.jdbc.driver.OracleResultSet)result;
oracle.sql.CLOB clobtmp = (oracle.sql.CLOB) ors.getClob(1);

if(clobtmp==null || clobtmp.length()==0){
System.out.println("======CLOB对象为空 ");
description = "";
}else{
description=clobtmp.getSubString((long)1,(int)clobtmp.length());
System.out.println("======字符串形式 "+description);
}
}

以下摘自http://spaces.msn.com/sailing8036/blog/cns!69D05DFD4294F487!147.trak

这种大容量字段有何用,也许大部分人马上联想到用来存储图片、文章,不过俺确是用它来存储Java串行化对象,当时用Swing做了个绘图工具,为了数据集中管理决定把图纸内容存到数据库,大家可以想象如果不用blob类型的字段,那表该如何设计呢?图元的种类N多,每个种类的属性N多,而且种类和属性都会变化,再加上还有图层的关系,最重要的一点就是存储速度(这个问题稍候我会解释)问题,要想设计出能完美解决以上问题的表结构将是想当困难的事,在项目进度的压力下对于两年前刚刚摆脱强大然而设计模式恶心的MFC,正投奔Java正营,边学语言边“设计”绘图工具的我,唯一能想到的而且最终证明在项目中成功实施的就是blob字段了,当然blob字段不是万能药,所有信息都存在blob中将导致你的数据无法利用SQL的查询、过滤、统计等功能,所以对应重要的字段还是有必要把他们“拉”出来的,看看以下的表结构,你就明白了:

/
// 图纸
/
create table EMHOOKUP
(
PICID NUMBER(8) not null,
PICNAME VARCHAR2(100) not null,
PICTUREDATA BLOB,
PICTYPE NUMBER(8),
PICMODIFYID NUMBER(8)
)
alter table EMHOOKUP add primary key (PICID)



下面让我们来研究一下,如果在Oracle数据库中进行实际的存储

DBTemplate db = new DBTemplate() {
public void dbProcess(Connection con) throws Exception {
StringBuffer sql = new StringBuffer();
sql.append(" insert into emhookup ");
sql.append(" (picid, picname, picmodifyid, picturedata, pictype)values ");
sql.append(" (?, ?, picmodifyidseq.nextval, EMPTY_BLOB(), ?)");

PreparedStatement ps = con.prepareStatement(sql.toString());
ps.setInt(1, picID);
ps.setString(2, picName);
ps.setInt(3, data.picType);
ps.executeUpdate();
ps.close();

sql.setLength(0);
sql.append(" select picturedata from emhookup ");
sql.append(" where picid=? for update ");
ps = con.prepareStatement(sql.toString());
ps.setInt(1, picID);
ResultSet rs = ps.executeQuery();
if (rs.next()) {
oracle.sql.BLOB blob = (oracle.sql.BLOB) rs.getBlob("picturedata");
ObjectOutputStream out = new ObjectOutputStream(blob.getBinaryOutputStream());
out.writeObject(DataProcess.toByteByGZIP(data));
out.flush();
out.close();
}
rs.close();
ps.close();
}
};
db.runWithTransaction();



从以上代码可知oracle中的blob/clob都需要先插入空数据empty_blob()/empty_clob()然后再锁定该行 for update 进行更新,并且必须在事务中运行runWithTransaction 。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值