使用Jdbc4操作Blob,Clob

      今天放假前最后一天上班整理代码时候,发现了使用jdbc4操作blob的一段代码,而是把电脑里面所有操作blob的代码片段找出来测试下,就有了上面的博文题目,特此说明,我的代码不知道是从那些博文里攫取的,所以本篇博客也算不上原创,如果无意中摘取了您博文中的片段,请留言,我会把原链接加上去。谢谢。
     下面开始介绍下如何使用Jdbc4,Hibernate 4操作blob,clob。
      环境:我使用的是oracle数据库。
     Jdbc4是什么东西我就不介绍了,有兴趣的请自行谷歌,如何看使用的架包是否是jdbc4呢?打开odbc.jar,可以看到一个Manifest.MF文件,打开可以看到里面有一行Specification-Version: 4.0,有着一行就说明你可以使用jdbc4的特性了。
     首先在oracle数据库中新建一个表,如下:
    

create table T_BLOB_TEST
(
  ID      NUMBER(4) not null,
  IMAGE   BLOB,
  CONTENT CLOB
);

alter table T_BLOB_TEST
  add constraint PK_T_TEST_BLOB_ID primary key (ID);

 
    在建一个序列,这个不是必选项,可以不建。
   

create sequence STUDENT_ID_SEQUENCE
minvalue 1
maxvalue 999
start with 206
increment by 1
cache 20;

   
   (一)使用Jdbc4操作Blob,Clob
   下面开始使用Jdbc4往表里面插数据了。
    先写一个简单的获取连接的静态方法.
  

public static Connection getConnection() throws Exception {
        Connection con = null;
        // 注册JDBC驱动类
        Class.forName("oracle.jdbc.driver.OracleDriver");
        con = DriverManager.getConnection(
                "jdbc:oracle:thin:@localhost:1521:xe", "tmd", "tmd");
        return con;
    }

 
    先写插入数据的方法,Blob类型如图片的可以直接读取到byte数组中,Clob类型的如文本文件需读取到String变量中,所以要有一个读取文本文件为String的方法,如下:
      

public static String readContent(File file) throws Exception {
        InputStreamReader read = new InputStreamReader(
                new FileInputStream(file), "utf-8");// 考虑到编码格式
        StringBuffer result = new StringBuffer((int) file.length());
        BufferedReader bufferedReader = new BufferedReader(read);
        String lineTxt = null;
        while ((lineTxt = bufferedReader.readLine()) != null) {
            result.append(lineTxt);
        }
        return result.toString();
    }

 
    重点来了,插入方法如下所示:
   

public static void jdbcInsertBlobTest(Connection conn, long id,
            String imgPath, String filePath) throws Exception {
        PreparedStatement pstmt = null;
        try {
            pstmt = conn
                    .prepareStatement("insert into t_blob_test(id,image,content) values (?,?,?)");
            pstmt.setLong(1, id);
            File f = new File(imgPath);
            Blob blob = conn.createBlob();
            InputStream in = new FileInputStream(f);
            //这个值应该至少是1
            OutputStream out = blob.setBinaryStream(1);
            byte[] temp = new byte[(int) f.length()];
            int length;
            while ((length = in.read(temp)) != -1) {
                out.write(temp, 0, length);
            }
            pstmt.setBlob(2, blob);
            f = new File(filePath);
            Clob clob = conn.createClob();
            String content = readContent(f);
            clob.setString(1, content);
            pstmt.setClob(3, clob);
            pstmt.executeUpdate();
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            if (pstmt != null) {
                pstmt.close();
            }
        }
    }
 测试方法如下:
   
Connection conn = null;
        PreparedStatement pstmt = null;
        try {
            conn = getConnection();
            jdbcInsertBlobTest(conn, 4L, "F:/saveFile/pic/test4.jpg","f:/saveFile/pic/system.log");
        } finally {
            if (pstmt != null) {
                pstmt.close();
            }
            if (conn != null) {
                conn.close();
            }
        }

     插入了怎么把数据库里的数据拉到本地来呢?可以这样做。
    

public static void jdbcJqueryBlobTest(PreparedStatement pstmt, long id,
            String filePath) throws Exception {
        ResultSet rs = null;
        OutputStream out = null;
        InputStream in = null;
        try {
            pstmt.setLong(1, id);
            rs = pstmt.executeQuery();
            /*
             * if(rs.next()){ Blob blob=rs.getBlob(1); blob.getBinaryStream();
             * out=new FileOutputStream(filePath); out.write(blob.getBytes(1l,
             * (int) blob.length())); out.close(); }
             */
            if (rs.next()) {
                in = rs.getBinaryStream("image");
                out = new FileOutputStream(filePath + "result_img.jpg");
                byte[] buffer = new byte[1024];// 每次读取1k
                for (int len = 0; (len = in.read(buffer)) > 0;) {
                    out.write(buffer, 0, len);
                }

                Clob clob = rs.getClob("content");
                getClobToFile(filePath + "result_clob.log", clob);
            }
        } finally {
            if (out != null) {
                out.close();
            }
            if (rs != null) {
                rs.close();
            }
        }
    }

       注意上面注释的一段可是可以运行的,是另一种更简单的方法。
       删除方法很简单,如下所示:
    

pstmt = conn.prepareStatement("delete from t_blob_test where id=?");
                    
public static void jdbcDeleteBlobBeanById(PreparedStatement pstmt,Long id)throws Exception{
        pstmt.setLong(1, id);
        pstmt.executeUpdate();
    }

 
     使用Jdbc操作blob,clob已经写完了,请自行忽略上面有些单词写错了,今天有点不在状态,不想改了。
  
    -------------------------------------------我是分割线-----------------------------------------------------------------
    (二)使用Hibernate4操作Blob,Clob
    为什么要使用Hibernate4呢?因为我什么都想用最新的,但是今天在使用Hibernate4的时候就掉坑里去了,后面会说到,说明,我使用的Hibernate4版本是Hibernate 4.1.3的,在最新的Hibernate 4.3下面没测试过,我不保证下面的代码能在Hibernate 4.3下运行良好。
     先建JavaBean,这是我的习惯。
   

package com.bean;

import java.io.Serializable;
import java.sql.Blob;
import java.sql.Clob;

import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.SequenceGenerator;
import javax.persistence.Table;

import org.hibernate.annotations.GenericGenerator;
@Entity
@Table (name= "t_blob_test")
public class BlobBean implements Serializable{
    private static final long serialVersionUID = 1L;
    @GenericGenerator(name = "generator", strategy = "increment")
    @Id
    @GeneratedValue(generator = "generator")
    
    /*@Id
    @GeneratedValue(strategy = GenerationType.AUTO, generator = "IdSeq")
    @SequenceGenerator(name="IdSeq", sequenceName="STUDENT_ID_SEQUENCE")*/
    private long id;
    private Blob image;
    private Clob content;
    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public Blob getImage() {
        return image;
    }

    public void setImage(Blob image) {
        this.image = image;
    }

    public Clob getContent() {
        return content;
    }

    public void setContent(Clob content) {
        this.content = content;
    }

}

 
    如果你建立过序列,可以把Id上面的注释换成我注释掉的,否则就用我上面的,看个人爱好了。
    先写个获取Session的工具类,如下:
  

package com.hibernate.util;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;
import org.hibernate.service.ServiceRegistry;
import org.hibernate.service.ServiceRegistryBuilder;

public class HibernateUtil {
    private static final SessionFactory sessionFactory;
    static {
        try {
            Configuration cfg = new Configuration()
                    .configure("resources/hibernate.cfg.xml");
            ServiceRegistry serviceRegistry = new ServiceRegistryBuilder()
                    .applySettings(cfg.getProperties()).buildServiceRegistry();

            sessionFactory = cfg.buildSessionFactory(serviceRegistry);
        } catch (Throwable e) {
            throw new ExceptionInInitializerError(e);
        }
    }

    private HibernateUtil() {
    }

    public static SessionFactory getSessionFactory() {
        return sessionFactory;
    }

    /**
     * 获取session对象
     * 
     * @return
     */
    public static Session openSession() {
        Session session = null;
        if (null == session || false == session.isOpen()) {
            session = sessionFactory.openSession();
        }
        return session;
    }

    public static void closeSession(Session session) {
        try {
            if (null != session && session.isOpen()) {
                session.close();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

 
    然后就是Hibernate的配置文件了,我的配置文件放在resources下面。
   

<?xml version='1.0' encoding='UTF-8'?>  
<!DOCTYPE hibernate-configuration PUBLIC  
          "-//Hibernate/Hibernate Configuration DTD 3.0//EN"  
          "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
    <session-factory>
        <!-- 数据库的驱动 -->
        <property name="connection.driver_class">
            oracle.jdbc.driver.OracleDriver
        </property>
        <!-- 数据库的URL -->
        <property name="connection.url">
            jdbc:oracle:thin:@localhost:1521:xe
        </property>
        <!-- 数据库的用户名 -->
        <property name="connection.username">tmd</property>
        <!-- 数据库的密码 -->
        <property name="connection.password">tmd</property>
        <!-- 数据库的方言 -->
        <property name="hibernate.dialect">
            org.hibernate.dialect.Oracle10gDialect
        </property>

        <!-- Enable Hibernate's automatic session context management -->
        <property name="current_session_context_class">thread</property>

        <!-- 显示操作的sql语句 -->
        <property name="hibernate.show_sql">true</property>
        <!-- 格式sql语句 -->
        <property name="hibernate.format_sql">false</property>
        <!-- 自动创建和更新表结构 -->
        <property name="hibernate.hbm2ddl.auto">update</property>
        <!-- 映射文件引入 -->
        <mapping class="com.bean.BlobBean" />
    </session-factory>
</hibernate-configuration>

 
     然后是接口:
   

import com.bean.BlobBean;

public interface HibernateDao {
    public void saveBlobBean(BlobBean bean);
    
    public void deleteBlobBean(Long id);
    
    public BlobBean getBlobBeanById(Long id);
}
 实现类:
   
import org.hibernate.Session;
import org.hibernate.Transaction;

import com.bean.BlobBean;
import com.hibernate.dao.HibernateDao;
import com.hibernate.util.HibernateUtil;

public class HibernateDaoImpl implements HibernateDao{

    public void saveBlobBean(BlobBean bean) {
        // 获取session对象
        Session session = HibernateUtil.openSession();
        // 打开事务
        Transaction ts = session.beginTransaction();
        try {
            // 保存
            session.save(bean);
            // 提交事务
            ts.commit();
        } catch (Exception e) {
            ts.rollback();
        }finally{
            HibernateUtil.closeSession(session);
        }
    }
    
    public void deleteBlobBean(Long id) {
        // 获取session对象
        Session session = HibernateUtil.openSession();
        // 打开事务
        Transaction ts = session.beginTransaction();
        try {
            // 保存
            session.delete(getBlobBeanById(id));
            // 提交事务
            ts.commit();
        } catch (Exception e) {
            ts.rollback();
        }finally{
            HibernateUtil.closeSession(session);
        }
    }

    
    public BlobBean getBlobBeanById(Long id) {
        Session session = HibernateUtil.openSession();
        BlobBean blobBean = null;
        Transaction ts = session.beginTransaction();
        try {
            blobBean = (BlobBean) session.get(BlobBean.class, id);
            ts.commit();
        } catch (Exception e) {
            ts.rollback();
        }
        return blobBean;
    }
}

     测试类,我就不分析了,全部贴出来
   

public class HibernateBlobBeanTest {
    private HibernateDao dao = new HibernateDaoImpl();
   
     public static void main(String[] args) throws Exception {
         HibernateBlobBeanTest t=new HibernateBlobBeanTest();
        // t.saveBlobBeanTest();
        // t.saveBlobBeanMethod2Test();
        // t.getBlobBeanTest(150L);
         //t.deleteBlobBeanTest(152L);
    }
    
    public void saveBlobBeanTest() throws Exception {
        BlobBean blobBean = new BlobBean();
        File file = new File("F:/saveFile/pic/test4.jpg");
        FileInputStream fis = new FileInputStream(file);
        File file1 = new File("f:/saveFile/pic/system.log");
        Reader reader = new FileReader(file1);
        Session session = HibernateUtil.openSession();
        blobBean.setImage(Hibernate.getLobCreator(session).createBlob(fis,
                file.length()));
        blobBean.setContent(Hibernate.getLobCreator(session).createClob(reader,
                file1.length()));
        dao.saveBlobBean(blobBean);
        HibernateUtil.closeSession(session);
    }

    public void saveBlobBeanMethod2Test() throws Exception {
        BlobBean blobBean = new BlobBean();
        File file = new File("F:/saveFile/pic/test4.jpg");
        FileInputStream fis = new FileInputStream(file);
        File file1 = new File("f:/saveFile/pic/system.log");
        Reader reader = new FileReader(file1);
        Session session = HibernateUtil.openSession();
        blobBean.setImage(session.getLobHelper().createBlob(fis, file.length()));
        blobBean.setContent(session.getLobHelper().createClob(reader,
                file1.length()));
        dao.saveBlobBean(blobBean);
        HibernateUtil.closeSession(session);
    }

    public void getBlobBeanTest(Long id) throws Exception {
        BlobBean blobBean = dao.getBlobBeanById(id);
        if (blobBean != null) {
            Blob blob = blobBean.getImage();
            // 根据blob对象的getBinaryStream()方法 获取输入流 对象
            InputStream is = blob.getBinaryStream();
            // 定义写出的文件
            File file = new File("f:/saveFile/pic/result_img.jpg");
            // 写出的输出流
            FileOutputStream fos = new FileOutputStream(file);
            // 缓冲区
            byte[] buffer = new byte[1024];
            // 读取的长度
            int len = 0;
            // 循环读取,直到文件结尾
            while ((len = is.read(buffer)) != -1) {
                // 写出
                fos.write(buffer, 0, len);
            }
            // 关闭流
            fos.close();
            is.close();

            // 获取Clob字段
            Clob clob = blobBean.getContent();
            // 根据clob对象的getCharacterStream() 获取字符输入流
            Reader r = clob.getCharacterStream();
            // 定义写出的文件
            File file1 = new File("f:/saveFile/pic/result_txt.log");
            // 创建输出流对象
            FileWriter fileWriter = new FileWriter(file1);
            // 缓冲区
            char[] cbuf = new char[1024];

            // 读取长度
            int len1 = 0;
            // 循环读取,直到文件结尾
            while ((len1 = r.read(cbuf)) != -1) {
                // 写出
                fileWriter.write(cbuf, 0, len1);
            }
            // 关闭流
            fileWriter.close();
            r.close();
        }
    }

    public void deleteBlobBeanTest(Long id) throws Exception {
        dao.deleteBlobBean(id);
    }

}

 
  使用Hibernate4操作Blob,Clob例子完。
 
    -------------------------------------我是分割线----------------------------------------------------------------------------
   (三)使用Spring3结合Hibernate4操作Blob,Clob
    也许你会问,前面不是介绍了使用Hibernate4操作Blob,Clob吗?结合Spring不就是依赖Spring注入下就完了吗,我个人认为,也行你单个框架使用的很好,但多个框架结合在一起的时候就不是1+1=2这么简单了。我来说下我今天遇到的坑。

    一开始我是打算使用最新版的架包的,Spring 4我不熟,所以没用,但Hibernate4我了解过,所以我一开始使用了Spring 3.2.4+Hibernate 4.3,配置好以后,已启动就报错了,错误信息如下:
     

Caused by: org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'txManager' defined in class path resource [resources/applicationContext.xml]: Invocation of init method failed; nested exception is java.lang.NoSuchMethodError: org.hibernate.engine.spi.SessionFactoryImplementor.getConnectionProvider()Lorg/hibernate/service/jdbc/connections/spi/ConnectionProvider;
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1482)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:521)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:458)
    at org.springframework.beans.factory.support.AbstractBeanFactory$1.getObject(AbstractBeanFactory.java:295)
    at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:223)
    at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:292)
    at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:194)
    at org.springframework.beans.factory.support.BeanDefinitionValueResolver.resolveReference(BeanDefinitionValueResolver.java:323)
    ... 59 more
Caused by: java.lang.NoSuchMethodError: org.hibernate.engine.spi.SessionFactoryImplementor.getConnectionProvider()Lorg/hibernate/service/jdbc/connections/spi/ConnectionProvider;
    at org.springframework.orm.hibernate4.SessionFactoryUtils.getDataSource(SessionFactoryUtils.java:90)
    at org.springframework.orm.hibernate4.HibernateTransactionManager.afterPropertiesSet(HibernateTransactionManager.java:335)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.invokeInitMethods(AbstractAutowireCapableBeanFactory.java:1541)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1479)
    ... 66 more

   
  这是Hibernate的一个bug,解决方法是把hibernate版本降到4.1,详情请    http://stackoverflow.com/questions/16417217/transactionmanager-cannot-initialize
   下面简单的介绍下Spring结合Hibernate操作Blob,Clob
    先写个公共的Dao。
   

import java.io.Serializable;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.List;

import org.hibernate.Criteria;
import org.hibernate.Query;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.criterion.Projections;
import org.springframework.beans.factory.annotation.Autowired;

public abstract class AbstractBaseDao<Entity extends Serializable> {

    private Class<Entity> entityClass;

    @Autowired
    private SessionFactory sessionFactory;

    public AbstractBaseDao() {
        try {
            Type genType = getClass().getGenericSuperclass();
            Type[] params = ((ParameterizedType) genType)
                    .getActualTypeArguments();
            entityClass = (Class<Entity>) params[0];
        } catch (Exception e) {
            entityClass = null;
        }
    }

    public Session getSession() {
        return sessionFactory.getCurrentSession();
    }

    public Entity findById(Serializable id) {
        return (Entity) getSession().get(entityClass, id);
    }

    public void save(Entity e) {
        getSession().save(e);
    }

    public void saveOrUpdate(Entity e) {
        getSession().saveOrUpdate(e);
    }

    public void update(Entity e) {
        getSession().update(e);
    }

    public void delete(Entity e) {
        getSession().delete(e);
    }

    public void delete(Long id) {
        getSession().delete(get(id));
    }

    public Entity get(Long id) {
        return (Entity) getSession().get(entityClass, id);
    }

    public List<Entity> listAll() {
        Criteria criteria = getSession().createCriteria(entityClass);
        criteria.setProjection(Projections.rowCount());
        return criteria.list();
    }

    public List<Entity> listAll(int pageNum) {
        return listAll(pageNum, PageUtil.DEFAULT_PAGE_SIZE);
    }

    public List<Entity> listAll(int pageNum, int pageSize) {
        Criteria criteria = getSession().createCriteria(entityClass);
        criteria.setFirstResult(PageUtil.getPageStart(pageNum, pageSize));
        return criteria.list();
    }

    public List<Entity> query(String hql, Object[] args) {
        Query query = getSession().createQuery(hql);
        if (args != null) {
            for (int i = 0; i < args.length; i++) {
                query.setParameter(i, args[i]);
            }
        }
        return query.list();
    }
}

 
   具体的Dao如下,接口我就不写了。
   

import org.springframework.stereotype.Repository;

import com.bean.BlobBean;
import com.common.AbstractBaseDao;
import com.dao.BlobDao;
@Repository
public class BlobDaoImpl extends AbstractBaseDao<BlobBean> implements BlobDao {
    public void saveBlobBean(BlobBean bean) {
        super.save(bean);
    }

    public BlobBean getBlobBean(Long id) {
        return super.get(id);
    }

    public void deleteBlobBean(Long id) {
        super.delete(id);
    }
}

 
   如何操作Blob,Clob主要在Service层方法,如下:
   

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.InputStream;
import java.io.Reader;
import java.sql.Blob;
import java.sql.Clob;

import org.hibernate.Hibernate;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import com.bean.BlobBean;
import com.dao.BlobDao;
import com.service.BlobService;

@Service
@Transactional
public class BlobServiceImpl implements BlobService {
    @Autowired
    private BlobDao dao;

    @Autowired
    SessionFactory sessionFactory;

    public boolean saveBlobBean(String imagePath, String contentPath) {
        boolean flag = false;
        BlobBean blobBean = new BlobBean();
        try {
            File file = new File(imagePath);
            FileInputStream fis = new FileInputStream(file);
            File file1 = new File(contentPath);
            Reader reader = new FileReader(file1);
            blobBean.setImage(Hibernate.getLobCreator(
                    sessionFactory.getCurrentSession()).createBlob(fis,
                    file.length()));
            blobBean.setContent(Hibernate.getLobCreator(
                    sessionFactory.getCurrentSession()).createClob(reader,
                    file1.length()));
            dao.saveBlobBean(blobBean);
            flag = true;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return flag;
    }

    public boolean saveBlobBeanWithTx(String imagePath, String contentPath) {
        boolean flag = false;
        BlobBean blobBean = new BlobBean();
        try {
            File file = new File(imagePath);
            FileInputStream fis = new FileInputStream(file);
            File file1 = new File(contentPath);
            Reader reader = new FileReader(file1);
            // 必须开启事务
            Session session = sessionFactory.getCurrentSession();
            blobBean.setImage(session.getLobHelper().createBlob(fis,
                    file.length()));
            blobBean.setContent(session.getLobHelper().createClob(reader,
                    file1.length()));
            dao.saveBlobBean(blobBean);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return flag;
    }

    public boolean getBlobBean(Long id, String savePath) {
        boolean flag = false;
        BlobBean blobBean = dao.getBlobBean(id);
        if (blobBean == null) {
            return flag;
        }
        try {
            if (blobBean != null) {
                // 获取Blob字段
                Blob blob = blobBean.getImage();
                // 根据blob对象的getBinaryStream()方法 获取输入流 对象
                InputStream is = blob.getBinaryStream();
                // 定义写出的文件
                File file = new File(savePath + "result_img.jpg");
                // 写出的输出流
                FileOutputStream fos = new FileOutputStream(file);
                // 缓冲区
                byte[] buffer = new byte[1024];
                // 读取的长度
                int len = 0;
                // 循环读取,直到文件结尾
                while ((len = is.read(buffer)) != -1) {
                    // 写出
                    fos.write(buffer, 0, len);
                }
                // 关闭流
                fos.close();
                is.close();
                // 获取Clob字段
                Clob clob = blobBean.getContent();
                // 根据clob对象的getCharacterStream() 获取字符输入流
                Reader r = clob.getCharacterStream();
                // 定义写出的文件
                File file1 = new File(savePath + "result_clob.log");
                // 创建输出流对象
                FileWriter fileWriter = new FileWriter(file1);
                // 缓冲区
                char[] cbuf = new char[1024];
                // 读取长度
                int len1 = 0;
                // 循环读取,直到文件结尾
                while ((len1 = r.read(cbuf)) != -1) {
                    // 写出
                    fileWriter.write(cbuf, 0, len1);
                }
                // 关闭流
                fileWriter.close();
                r.close();
                flag = true;
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        return flag;
    }

    public boolean deleteBlobBean(Long id) {
        boolean flag = false;
        try {
            dao.deleteBlobBean(id);
            flag = true;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return flag;
    }

}

 
   测试方法可以看我的附件。
   使用Spring结合Hibernate操作Blob,Clob例子完
 
   ------------------------------------------ 我是分割线----------------------------------------------------------------------
   (四)使用Mybatis操作Blob,Clob
    我个人习惯在使用Hibernate后喜欢看下Mybatis下怎么做,其实使用Mybatis操作Blob,Clob很简单,主要是2个转换器,官网介绍如下:

   http://mybatis.github.io/mybatis-3/apidocs/reference/org/apache/ibatis/type/BlobTypeHandler.html
   http://mybatis.github.io/mybatis-3/apidocs/reference/org/apache/ibatis/type/ClobTypeHandler.html
    也行是一段日子没用Mybatis了,今天一直在使用Blob,Clob类型测试,怎么都不成功,错误信息如下:
    

Caused by: org.apache.ibatis.reflection.ReflectionException: Could not set property 'image' of 'class com.bean.BlobBean' with value '[B@1ef3a22' Cause: java.lang.IllegalArgumentException: argument type mismatch
    at org.apache.ibatis.reflection.wrapper.BeanWrapper.setBeanProperty(BeanWrapper.java:172)
    at org.apache.ibatis.reflection.wrapper.BeanWrapper.set(BeanWrapper.java:54)
    at org.apache.ibatis.reflection.MetaObject.setValue(MetaObject.java:130)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.applyPropertyMappings(DefaultResultSetHandler.java:370)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.getRowValue(DefaultResultSetHandler.java:336)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleRowValuesForSimpleResultMap(DefaultResultSetHandler.java:289)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleRowValues(DefaultResultSetHandler.java:264)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleResultSet(DefaultResultSetHandler.java:234)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleResultSets(DefaultResultSetHandler.java:152)
    at org.apache.ibatis.executor.statement.PreparedStatementHandler.query(PreparedStatementHandler.java:57)
    at org.apache.ibatis.executor.statement.RoutingStatementHandler.query(RoutingStatementHandler.java:70)
    at org.apache.ibatis.executor.SimpleExecutor.doQuery(SimpleExecutor.java:57)
    at org.apache.ibatis.executor.BaseExecutor.queryFromDatabase(BaseExecutor.java:259)
    at org.apache.ibatis.executor.BaseExecutor.query(BaseExecutor.java:132)
    at org.apache.ibatis.executor.CachingExecutor.query(CachingExecutor.java:105)
    at org.apache.ibatis.executor.CachingExecutor.query(CachingExecutor.java:81)
    at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:104)

 
     后来想起来,Mybatis配置时候是使用byte[]代表Blob,String类型代表Clob,而是修改JavaBean为:
   

import java.io.Serializable;

public class MybatisBlobBean implements Serializable {
    private static final long serialVersionUID = 1L;
    private long id;
    private byte[] image;
    private String content;

    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public byte[] getImage() {
        return image;
    }

    public void setImage(byte[] image) {
        this.image = image;
    }

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
    }

    public static long getSerialversionuid() {
        return serialVersionUID;
    }

}

 
   主要配置如下:
   

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" 
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.mybatis.mapper.IBlobBeanMapper">

    <resultMap id="blobBeanResult" type="MybatisBlobBean">
        <result property="id" column="ID" />
        <result property="image" column="IMAGE" javaType="byte[]"
            jdbcType="BLOB" typeHandler="org.apache.ibatis.type.BlobTypeHandler" />
        <result property="content" column="CONTENT" javaType="java.lang.String"
            jdbcType="CLOB" typeHandler="org.apache.ibatis.type.ClobTypeHandler" />
    </resultMap>

    <select id="getMybatisBlobBeanById" parameterType="long" resultMap="blobBeanResult">
        select * from t_blob_test where id=#{id}
    </select>

    <select id="deleteMybatisBlobBeanById" parameterType="long">
        delete from
        t_blob_test where id=#{id}
    </select>

    <insert id="saveMybatisBlobBean" parameterType="java.util.Map">
        <selectKey resultType="long" keyProperty="id" order="BEFORE">   
            <![CDATA[SELECT STUDENT_ID_SEQUENCE.NEXTVAL AS ID FROM DUAL]]>
        </selectKey>
        insert into t_blob_test(id,image,content)
        values(#{id,jdbcType=NUMERIC},
        #{image,jdbcType=BLOB},
        #{content,jdbcType=CLOB}
        )
    </insert>
</mapper>

 
   测试方法如下:
   

import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.util.HashMap;
import java.util.Map;

import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.AbstractJUnit4SpringContextTests;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

import com.bean.MybatisBlobBean;
import com.mybatis.mapper.IBlobBeanMapper;

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:com/mybatis/conf/mybatis_applicationContext.xml")
public class MybatisBlobBeanTest extends AbstractJUnit4SpringContextTests {

    @Autowired
    SqlSessionFactory sqlSessionFactory;

    @Test
    public void saveBlobBeanTest() throws Exception {
        SqlSession sqlSession = sqlSessionFactory.openSession();
        IBlobBeanMapper mapper = sqlSession.getMapper(IBlobBeanMapper.class);
        Map<String, Object> param = new HashMap<String, Object>();
        File file = new File("F:/saveFile/pic/test4.jpg");
        byte[] image = getBytesFromFile(file);
        file = new File("f:/saveFile/pic/system.log");
        String context = readContent(file);
        param.put("image", image);
        param.put("content", context);
        mapper.saveMybatisBlobBean(param);
    }

    @Test
    public void testGetBlobBeanById() throws Exception {
        SqlSession sqlSession = sqlSessionFactory.openSession();
        IBlobBeanMapper mapper = sqlSession.getMapper(IBlobBeanMapper.class);
        MybatisBlobBean blobBean = mapper.getMybatisBlobBeanById(146L);
        if (blobBean != null) {
            byte[] image = blobBean.getImage();
            // 根据blob对象的getBinaryStream()方法 获取输入流 对象
            InputStream is = new ByteArrayInputStream(image);
            // 定义写出的文件
            File file = new File("f:/saveFile/pic/result_img.jpg");
            // 写出的输出流
            FileOutputStream fos = new FileOutputStream(file);
            // 缓冲区
            byte[] buffer = new byte[1024];
            // 读取的长度
            int len = 0;
            // 循环读取,直到文件结尾
            while ((len = is.read(buffer)) != -1) {
                // 写出
                fos.write(buffer, 0, len);
            }
            // 关闭流
            fos.close();
            is.close();

            // 获取Clob字段
            String content=blobBean.getContent();
            // 根据clob对象的getCharacterStream() 获取字符输入流
            // 定义写出的文件
            File file1 = new File("f:/saveFile/pic/result_txt.log");
            OutputStreamWriter out=new OutputStreamWriter(new FileOutputStream(file1),"utf-8");
            out.write(content);
            out.close();
        }
    }

    @Test
    public void testDeleteBlobBeanById() throws Exception{
        SqlSession sqlSession = sqlSessionFactory.openSession();
        IBlobBeanMapper mapper = sqlSession.getMapper(IBlobBeanMapper.class);
        mapper.deleteMybatisBlobBeanById(4L);
    }
    public static byte[] getBytesFromFile(File file) throws Exception {
        if (file.length() > Integer.MAX_VALUE) {
            throw new Exception("文件太大");
        }
        try {
            FileInputStream stream = new FileInputStream(file);
            ByteArrayOutputStream out = new ByteArrayOutputStream(
                    (int) file.length());
            byte[] b = new byte[(int) file.length()];
            for (int n; (n = stream.read(b)) != -1;) {
                out.write(b, 0, n);
            }
            stream.close();
            out.close();
            return out.toByteArray();
        } catch (IOException e) {
        }
        return null;
    }

    public static String readContent(File file) throws Exception {
        InputStreamReader read = new InputStreamReader(
                new FileInputStream(file), "utf-8");// 考虑到编码格式
        StringBuffer result = new StringBuffer((int) file.length());
        BufferedReader bufferedReader = new BufferedReader(read);
        String lineTxt = null;
        while ((lineTxt = bufferedReader.readLine()) != null) {
            result.append(lineTxt);
        }
        return result.toString();
    }
}

 
     其他的配置请见附件。附件没有上传Jar包,我用的是Hibernate 4.1.3+Spring .2.4+Junit 4.11+Mybatis 3.2,本来是打算上传了,Iteye上传附件上传了5分钟没反应,我的博文全丢了,只是我第二次编辑的结果,说多了都是泪,Jar请自己找吧,抱歉。
     全文全。本篇博文是我下午整理代码时候弄的,例子很简单,代码本人亲测通过,如果您觉得有什么不对的地方,欢迎指出,期待各位的留言。

你可能感兴趣的:(Jdbc4操作Blob,Mybatis操作Blob)