在实际的业务开发中,访问数据库的操作肯定是很频繁的,意味我们可以将数据库访问的相关代码进行封装调用,减少代码冗余。
使用ThreadLocal存储SqlSession
使用MyBatis访问数据库,生成SqlSession的部分是通用的流程,我们可以对其进行封装,将其封装成MyBatisUtil类,对外提供获取SqlSession和关闭SqlSession的方法。
package com.test.util;
import java.io.IOException;
import java.io.InputStream;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
public class MyBatisUtil {
//factory实例化的过程是一个比较耗费性能的过程.
//保证有且只有一个factory
private static SqlSessionFactory factory;
private static ThreadLocal<SqlSession> tl = new ThreadLocal<>();
static{
try {
InputStream is = Resources.getResourceAsStream("mybatis.xml");
factory = new SqlSessionFactoryBuilder().build(is);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
/**
* 获取SqlSession的方法
*/
public static SqlSession getSession(){
SqlSession session = tl.get();
if(session==null){
tl.set(factory.openSession());
}
return tl.get();
}
public static void closeSession(){
SqlSession session = tl.get();
if(session!=null){
session.close();
}
tl.set(null);
}
}
SqlSessionFactory的实例化过程是一个非常耗费性能的过程,我们应该在程序中保证只有一个SqlSessionFactory,所以我们需要将其声明为“private static”,并且将SqlSessionFactory的实例化过程放置在类内的静态代码块中,它将会在类加载的时候执行。
然后我们需要将SqlSession存储在ThreadLocal中,保证每个线程都独立拥有自己的SqlSession对象。
OpenSessionInView
OpenSessionInView即在视图(view)中打开SqlSession并持有,具体是通过一个Filter来实现的。
package com.test.filter;
import java.io.IOException;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.annotation.WebFilter;
import org.apache.ibatis.session.SqlSession;
import com.test.util.MyBatisUtil;
/**
* 最开始是由Spring框架提出的.整合Hibernate框架是使用的是OpenSessionInView
*
*
* @author Administrator
*
*/
@WebFilter("/*")
public class OpenSessionInView implements Filter{
@Override
public void init(FilterConfig filterconfig) throws ServletException {
// TODO Auto-generated method stub
}
@Override
public void doFilter(ServletRequest servletrequest, ServletResponse servletresponse, FilterChain filterchain)
throws IOException, ServletException {
// InputStream is = Resources.getResourceAsStream("mybatis.xml");
// SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(is);
// SqlSession session = factory.openSession();
SqlSession session = MyBatisUtil.getSession();
try {
filterchain.doFilter(servletrequest, servletresponse);
session.commit();
} catch (Exception e) {
session.rollback();
e.printStackTrace();
}finally{
MyBatisUtil.closeSession();
}
// session.commit();
// session.close();
}
@Override
public void destroy() {
// TODO Auto-generated method stub
}
}
本文介绍了一种在Java应用中使用MyBatis框架高效访问数据库的方法。通过封装SqlSession的创建与关闭过程,利用ThreadLocal确保每个线程拥有独立的SqlSession实例,并通过Filter实现了OpenSessionInView模式,确保了事务的一致性和线程安全性。
2220

被折叠的 条评论
为什么被折叠?



