1.使用jdk中的方法进行传输。在ResultSet 中有getBlob()方法,在PreparedStatement中有setBlob()方法,所以大多数人都会尝试setBlob
(),getBlob() 进行读写,或者两个数据库之间BLOB的传输。这种方法实际上是行不通的,据网上的一些资料介绍,说sun官方的文档有些方法
都是错误的。
2.使用ResultSet.getBinaryStream 和PreparedStatement.setBinaryStream对BLOB进行读写或两个数据库间的传输。这种方法我自己尝试过,
发现,如果BLOB中存储的是文本文件的话,就没问题,如果是二进制文件,传输就会有问题。
根据自己的经验,以及查阅了Oracle的官方文档,都是使用如下处理方法:
1.新建记录,插入BLOB数据
1.1首先新建记录的时候,使用oracle的函数插入一个空的BLOB,假设字段A是BLOB类型的:
insert xxxtable(A,B,C) values(empty_blob(),'xxx','yyyy')
1.2后面再查询刚才插入的记录,然后更新BLOB,在查询前,注意设置Connection的一个属性:
conn.setAutoCommit(false)如果缺少这一步,可能导致fetch out of sequence等异常.
1.3 查询刚才插入的记录,后面要加“ for update ”,如下:
select A from xxxtable where xxx=999 for update ,如果缺少for update,可能出现row containing the LOB value is not locked
的异常
1.4 从查询到的 BLOB字段中,获取blob并进行更新,代码如下:
BLOB blob = (BLOB) rs.getBlob("A")
OutputStream os = blob.getBinaryOutputStream()
BufferedOutputStream output = new BufferedOutputStream(os)
后面再使用output.write方法将需要写入的内容写到output中就可以了。例如我们将一个文件写入这个字段中:
BufferedInputStream input = new BufferedInputStream(new File("c://hpWave.log").toURL().openStream())
byte[] buff = new byte[2048] //用做文件写入的缓冲
int bytesRead
while(-1 != (bytesRead = input.read(buff, 0, buff.length))) {
output.write(buff, 0, bytesRead)
System.out.println(bytesRead)
}
上面的代码就是从input里2k地读取,然后写入到output中。
1.5上面执行完毕后,记得关闭output,input,以及关闭查询到的ResultSet
1.6最后执行conn.commit()将更新的内容提交,以及执行conn.setAutoCommit(true)改回Connction的属性
2.修改记录,方法与上面的方法类似,
2.1首先更新BLOB以外的其他字段
2.2 使用1.3中类似的方法获取记录
2.3 修改的过程中,注意以下:a 需要更新的记录中,BLOB有可能为NULL,这样在执行blob.getBinaryOutputStream()获取的值可能为
null,那么就关闭刚才select的记录,再执行一次update xxxtable set A = empty_blob() where xxx, 这样就先写入了一个空的BLOB(不是null),然后再
使用1.3,1.4中的方法执行更新记录.b 注意别忘了先执行setAutoCommit(false),以及"for update",以及后面的conn.commit()等。
3.读取BLOB字段中的数据.
3.1 读取记录不需要setAutoCommit(),以及 select ....for update.
3.2 使用普通的select 方法查询出记录
3.3 从ResultSet中获取BLOB并读取,如下:
BLOB b_to = (BLOB) rs.getBlob("A")
InputStream is = b_from.getBinaryStream()
BufferedInputStream input = new BufferedInputStream(is)
byte[] buff = new byte[2048]
while(-1 != (bytesRead = input.read(buff, 0, buff.length))) {
//在这里执行写入,如写入到文件的BufferedOutputStream里
System.out.println(bytesRead)
}
通过循环取出blob中的数据,写到buff里,再将buff的内容写入到需要的地方
4.两个数据库间blob字段的传输
类似上面1和3的方法,一边获取BufferedOutputStream,另外一边获取BufferedInputStream,然后读出写入,需要注意的是写入所用的
Connection要执行conn.setAutoCommit(false)以及获取记录时添加“ for update ”以及最后的commit()
总结以上方法,其根本就是先创建空的BLOB,再获取其BufferedOutputStream进行写入,或获取BufferedInputStream进行读取
(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)
}
}
BLOB类型的字段用于存储二进制数据MySQL中,BLOB是个类型系列,包括:TinyBlob、Blob、MediumBlob、LongBlob,这几个类型之间的唯一区别是在存储文件的最大大小上不同。
MySQL的四种BLOB类型
类型 大小(单位:字节)
TinyBlob 最大 255
Blob 最大 65K
MediumBlob 最大 16M
LongBlob 最大 4G
这个是mysql下存取blob字段的一个很简单的类,跟据自己的需要改改就行了/**
* Title: BlobPros.java
* Project: test
* Description: 把图片存入mysql中的blob字段,并取出
* Call Module: mtools数据库中的tmp表
* File: C:\downloads\luozsh.jpg
* Copyright:Copyright (c) 2003-2003
* Company: uniware
* Create Date: 2002.12.5
* @Author: FeiFan
* @version 1.0 版本*
*
* Revision history
* Name Date Description
* ---- ---- -----------
*Chenqh 2003.12.5对图片进行存取
*
* note: 要把数据库中的Blob字段设为longblob
*
*/
//package com.uniware
import java.io.*
import java.util.*
import java.sql.*
public class BlobPros
{
private static final String URL = "jdbc:mysql://10.144.123.63:3306/mtools?user=wind&password=123&useUnicode=true"
private Connection conn = null
private PreparedStatement pstmt = null
private ResultSet rs = null
private File file = null
public BlobPros()
{
}
/**
* 向数据库中插入一个新的BLOB对象(图片)
*
* @param infile - 要输入的数据文件
* @throws java.lang.Exception
*
*/
public void blobInsert(String infile) throws Exception
{
FileInputStream fis = null
try
{
Class.forName("org.gjt.mm.mysql.Driver").newInstance()
conn = DriverManager.getConnection(URL)
file = new File(infile)
fis = new FileInputStream(file)
//InputStream fis = new FileInputStream(infile)
pstmt = conn.prepareStatement("insert into tmp(descs,pic) values(?,?)")
pstmt.setString(1,file.getName()) //把传过来的第一个参数设为文件名
//pstmt.setBinaryStream(2,fis,(int)file.length()) //这种方法原理上会丢数据,因为file.length()返回的是long型
pstmt.setBinaryStream(2,fis,fis.available()) //第二个参数为文件的内容
pstmt.executeUpdate()
}
catch(Exception ex)
{
System.out.println("[blobInsert error : ]" + ex.toString())
}
finally
{
//关闭所打开的对像//
pstmt.close()
fis.close()
conn.close()
}
}
/**
* 从数据库中读出BLOB对象
*
* @param outfile - 输出的数据文件
* @param picID - 要取的图片在数据库中的ID
* @throws java.lang.Exception
*
*/
public void blobRead(String outfile,int picID) throws Exception
{
FileOutputStream fos = null
InputStream is = null
byte[] Buffer = new byte[4096]
try
{
Class.forName("org.gjt.mm.mysql.Driver").newInstance()
conn = DriverManager.getConnection(URL)
pstmt = conn.prepareStatement("select pic from tmp where id=?")
pstmt.setInt(1,picID)//传入要取的图片的ID
rs = pstmt.executeQuery()
rs.next()
file = new File(outfile)
if(!file.exists())
{
file.createNewFile()//如果文件不存在,则创建
}
fos = new FileOutputStream(file)
is = rs.getBinaryStream("pic")
int size = 0
/* while(size != -1)
{
size = is.read(Buffer) //从数据库中一段一段的读出数据
//System.out.println(size)
if(size != -1)//-1表示读到了文件末
fos.write(Buffer,0,size)
} */
while((size = is.read(Buffer)) != -1)
{
//System.out.println(size)
fos.write(Buffer,0,size)
}
}
catch(Exception e)
{
System.out.println("[OutPutFile error : ]" + e.getMessage())
}
finally
{
//关闭用到的资源
fos.close()
rs.close()
pstmt.close()
conn.close()
}
}
public static void main(String[] args)
{
try
{
BlobPros blob = new BlobPros()
//blob.blobInsert("C:\\Downloads\\luozsh1.jpg")
blob.blobRead("c:/downloads/luozishang.jpg",47)
}
catch(Exception e)
{
System.out.println("[Main func error: ]" + e.getMessage())
}
}
}