Blob(二进制大对象)

Blob(二进制大对象)

Oracle LOB:

  • LOB,即Large Objects(大对象),是用来存储大量的二进制和文本数据的一种数据类型(一个LOB字段可存储多达4GB的数据)。
  • LOB分为两种类型:内部LOB和外部LOB。
        -内部LOB将数据以字节流的形式存储在数据库的内部。因而,内部LOB的许多操作都可以参与事务,也可以像处理普通数据一样对其进行备份和恢复操作。
         Oracle支持三种类型的内部LOB:
             ·BLOB(二进制数据)
             ·CLOB(单字节字符数据)
             ·NCLOB(多字节字符数据)
        -CLOB和NCLOB类型适用于存储超长的文本数据,BLOB字段适用于存储大量的二进制数据,如图像、视频、音频,文件等。
        -目前只支持一种外部LOB类型,即BFILE类型。在数据库内,该类型仅存储数据在操作系统中的位置信息,而数据的实体以外部文件的形式存在于操作系统的文件系统中。因而,该类型所表示的数据是只读的,不参与事务。该类型可帮助用户管理大量的由外部程序访问的文件。
MySQL BLOB类型介绍:
  • MySQL中,BLOB是一个二进制大型对象,是一个可以存储大量数据的容器,它能容纳不同大小的数据。
  • MySQL的四种BLOB类型(除了在存储的最大信息量上不同外,他们是等同的)

           

类型大小(单位:字节)
TinyBlob最大 255
Blob最大 65 K 或 65535
MediumBlob最大 16 M 或 16777215
LongBlob最大 4 G 或 4294967295
  • 实际使用中根据需要存入的数据大小定义不同的BLOB类型。
  • 需要注意的是:如果存储的文件过大,数据库的性能会下降。
使用JDBC来写入BLOB类型数据到Oracle中:
  • oracle的blob字段比long字段的性能要好,可以用来保存如图片之类的二进制数据。
  • oracle的blob字段由两部分组成:数据(值)和指向数据的指针(定位器)。
  • 尽管值与表自身一起存储,但是一个Blob列并不包含值,仅有它的定位指针。为了使用大对象,程序必须声明定位器类型的本地变量。
  • 当Oracle内部LOB被创建时,定位器被存放在列中,值被存放在LOB段中,LOB端是在数据库内部表的一部分。
  • 因为Blob自身有一个cursor,当写入Blob字段必须使用指针(定位器)对Blob进行操作,因而在写入Blob之前,必须获得指针(定位器)才能进行写入
  • 如何获得Blob的指针(定位器):需要先插入一个empty的blob,这将创建一个blob的指针,然后再把这个empty的blob的指针查询出来,这样通过两步操作,就获得了blob的指针,可以真正的写入blob数据了。
使用BLOB的步骤:
    1.插入空blob:
        insert into javatest(name,content) values (?,empty_blob());
    2.获得blob的cursor
        select content from javatest where name = ? for update;
        ·注意:必须加 for update,锁定该行,直至该行被修改完毕,保证不产生并发冲突。

    3.利用 io ,和获取到的 cursor 往数据库写数据流。

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.sql.Blob;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;

public class BlobExam {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Connection con = null;
		PreparedStatement ps = null;
		PreparedStatement ps2 = null;
		ResultSet rs = null;
		InputStream is = null;
		OutputStream os = null;
		
		try {
			//加载驱动
			Class.forName("com.mysql.jdbc.Driver");
			//获得连接
			con = DriverManager.getConnection("jdbc:mysql://localhost:3306/test","root","123456");
			//预编译语句
			ps = con.prepareStatement("insert into blob_test values (?,?);");
			//设置值
			ps.setString(1, "小明");
			ps.setBlob(2, new FileInputStream("testA.jpg"));
			//执行,自动提交
			ps.execute();
			
			//预编译语句
			ps2 = con.prepareStatement("select * from blob_test where name=?;");
			//设置值
			ps2.setString(1, "小明");
			//获得查询结果集
			rs = ps2.executeQuery();
			while(rs.next()){
				Blob b = rs.getBlob("path");
				is = b.getBinaryStream();
				os = new FileOutputStream("testB.jpg");
				//写入文件
				int temp = 0;
				while((temp = is.read()) != -1){
					os.write(temp);
				}
			}
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		} catch (SQLException e) {
			e.printStackTrace();
		} catch (FileNotFoundException e) {
			e.printStackTrace();
		} catch (IOException e) {
			e.printStackTrace();
		}
	}
}

<think>好的,我现在要解决用户的问题:如何在Java中将BLOB二进制数据转换为图片。用户需要示例代码,所以我要先回忆一下相关的Java处理方法。 首先,BLOB是数据库中的二进制大对象,存储图片数据。转换的关键步骤是从BLOB获取二进制数据,然后转换成图片格式,比如BufferedImage。可能需要使用ImageIO类,或者处理字节流。 想到用户可能使用JDBC从数据库读取BLOB。所以步骤可能是:从ResultSet获取Blob对象,读取其二进制流,然后用ImageIO.read()转换为BufferedImage。这部分代码需要处理IO异常,可能还要关闭资源。 另外,用户提供的引用中有提到OpenCV相关的转换器,比如OpenCVFrameConverter.ToMat。如果用户需要处理OpenCV的Mat对象,可能需要额外的库,比如JavaCV。不过用户的问题没有明确提到OpenCV,所以可能需要提供一个基础示例,再提到可能的扩展。 还要注意Blob的获取方式,不同的数据库驱动可能有差异,比如MySQL的BLOB类型处理。示例代码需要通用一些,使用标准的JDBC接口。 然后,检查是否有引用中的内容相关。比如引用[2]提到将二进制数据作为图像Content-Type提供,这可能涉及MIME类型,但在转换过程中可能不需要,除非要处理HTTP响应。用户的问题集中在Java代码转换,所以可能不需要这部分,但可以提一下确保数据确实是有效的图片格式。 引用[3]中的OpenCVFrameConverter可能用于将帧转换为Mat对象,如果用户需要结合OpenCV处理,可以补充相关示例,但可能需要作为额外信息。 现在整理步骤: 1. 从数据库获取Blob对象。 2. 将Blob转换为InputStream。 3. 使用ImageIO读取InputStream生成BufferedImage。 4. 处理异常,关闭资源。 示例代码大致结构: Blob blob = resultSet.getBlob("image_column"); InputStream is = blob.getBinaryStream(); BufferedImage image = ImageIO.read(is); 需要处理可能出现的异常,比如IOException、SQLException。同时,确保ImageIO支持图片格式,比如PNG、JPEG。 另外,如果用户需要保存到文件,可以用ImageIO.write()。或者,如果使用OpenCV,可能需要将字节数组转换为Mat对象,这时候需要用到JavaCV的库,比如通过OpenCVFrameConverter。 可能的问题:用户是否有第三方库的使用限制?基础示例应使用标准库,高级示例可能需要引入依赖。 总结回答结构: 1. 基本方法使用JDBC和ImageIO。 2. 提供代码示例。 3. 提到可能的异常处理。 4. 可选的高级方法,如结合OpenCV。 5. 相关问题,比如处理不同格式、性能优化等。 还要检查引用中的内容是否相关。引用[1]、[3]、[4]涉及OpenCV,如果用户可能用到,可以添加一个使用JavaCV的示例,但需要说明需要依赖库。 例如,使用JavaCV将字节数组转换为Mat: byte[] bytes = ...; Mat mat = new Mat(bytes); 或者通过转换器,但具体代码可能需要更多步骤。 最后,生成相关问题,如处理不同图片格式、保存到文件、OpenCV处理等。</think>在Java中将BLOB二进制数据转换为图片的核心步骤是通过JDBC获取二进制流,并使用`ImageIO`类实现解码。以下是具体实现方法: ### 基础方法(标准JDBC + ImageIO) ```java try (Connection conn = DriverManager.getConnection(dbUrl, user, password); PreparedStatement stmt = conn.prepareStatement("SELECT image_blob FROM images WHERE id = ?")) { stmt.setInt(1, imageId); ResultSet rs = stmt.executeQuery(); if (rs.next()) { Blob blob = rs.getBlob("image_blob"); // 从数据库获取BLOB对象[^1] try (InputStream is = blob.getBinaryStream()) { BufferedImage image = ImageIO.read(is); // 关键解码步骤 if (image != null) { ImageIO.write(image, "PNG", new File("output.png")); // 保存为文件 } } } } catch (SQLException | IOException e) { e.printStackTrace(); } ``` ### 高级方法(OpenCV集成) 若需进一步处理图像特征(如边缘检测),可通过JavaCV调用OpenCV: ```java byte[] blobBytes = blob.getBytes(1, (int) blob.length()); // 提取字节数组 Mat mat = new Mat(blobBytes); // 转换为OpenCV矩阵对象[^3] // 此处可添加OpenCV图像处理操作 ``` ### 注意事项 1. **格式兼容性**:`ImageIO`支持的格式取决于已注册的插件,可通过`ImageIO.getReaderFormatNames()`查看 2. **异常处理**:需捕获`SQLException`(数据库操作错误)和`IOException`(流处理错误) 3. **资源释放**:使用try-with-resources确保Blob/Stream正确关闭
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值