编程式事务
一、getCurrentSession()与openSession()的区别?
* 采用getCurrentSession()创建的session会绑定到当前线程中,而采用openSession()创建的session则不会
* 采用getCurrentSession()创建的session在commit或rollback时会自动关闭,而采用openSession()创建的session必须手动关闭
二、使用getCurrentSession()需要在hibernate.cfg.xml文件中加入如下配置:
* 如果使用的是本地事务(jdbc事务)
<property name="hibernate.current_session_context_class">thread</property>
* 如果使用的是全局事务(jta事务)
<property name="hibernate.current_session_context_class">jta</property>
三、例子:模拟日志管理(添加用户时,写入日志)
因为事务是session.beginTransaction();由Session开启的,所有添加用户和相应的日志,必须在一个事务中,那么就必须使用同一个Seesion,传递太麻烦。采用getCurrentSession()创建的session会绑定到当前线程中变量ThreadLocal中。
User.java
package com.ncepu.usermgr.model; public class User { private int id; private String name; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } }
User.hbm.xml
<?xml version="1.0"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> <hibernate-mapping package="com.ncepu.usermgr.model"> <class name="User" table="t_user"> <id name="id"> <generator class="native"/> </id> <property name="name"/> </class> </hibernate-mapping>
Log.java
package com.ncepu.usermgr.model; import java.util.Date; public class Log { private int id; //操作日志、安全日志、事件日志 private String type; private String detail; private Date time; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getType() { return type; } public void setType(String type) { this.type = type; } public String getDetail() { return detail; } public void setDetail(String detail) { this.detail = detail; } public Date getTime() { return time; } public void setTime(Date time) { this.time = time; } }
Log.hbm.xml
<hibernate-mapping package="com.ncepu.usermgr.model"> <class name="Log" table="t_log"> <id name="id"> <generator class="native"/> </id> <property name="type"/> <property name="detail"/> <property name="time"/> </class> </hibernate-mapping>
UserManager接口
package com.ncepu.usermgr.manager; import com.ncepu.usermgr.model.User; public interface UserManager { public void addUser(User user); }
UserManagerImpl实现类
package com.ncepu.usermgr.manager; import java.util.Date; import org.hibernate.Session; import com.ncepu.usermgr.model.Log; import com.ncepu.usermgr.model.User; import com.ncepu.usermgr.util.HibernateUtils; public class UserManagerImpl implements UserManager { public void addUser(User user) { Session session = null; try { //session = HibernateUtils.getSession(); session = HibernateUtils.getSessionFactory().getCurrentSession(); session.beginTransaction(); session.save(user); Integer.parseInt("asdfsdfsfsd"); Log log = new Log(); log.setType("安全日志"); log.setDetail("xxx进入系统"); log.setTime(new Date()); LogManager logManager = new LogManagerImpl(); logManager.addLog(log); session.getTransaction().commit(); }catch(Exception e) { e.printStackTrace(); session.getTransaction().rollback(); // }finally { // HibernateUtils.closeSession(session); } } }
LogManager接口
package com.ncepu.usermgr.manager; import com.ncepu.usermgr.model.Log; public interface LogManager { public void addLog(Log log); }
LogManagerImpl实现类
package com.ncepu.usermgr.manager; import com.ncepu.usermgr.model.Log; import com.ncepu.usermgr.util.HibernateUtils; public class LogManagerImpl implements LogManager { public void addLog(Log log) { HibernateUtils.getSessionFactory().getCurrentSession().save(log); } }
Client.java
package com.ncepu.usermgr.client; import com.ncepu.usermgr.manager.UserManager; import com.ncepu.usermgr.manager.UserManagerImpl; import com.ncepu.usermgr.model.User; public class Client { public static void main(String[] args) { User user = new User(); user.setName("张三"); UserManager userManager = new UserManagerImpl(); userManager.addUser(user); } }
工具类SessionFactory
package com.ncepu.usermgr.util; import org.hibernate.Session; import org.hibernate.SessionFactory; import org.hibernate.cfg.Configuration; public class HibernateUtils { private static SessionFactory factory; static { try { Configuration cfg = new Configuration().configure(); factory = cfg.buildSessionFactory(); }catch(Exception e) { e.printStackTrace(); } } public static SessionFactory getSessionFactory() { return factory; } public static Session getSession() { return factory.openSession(); } public static void closeSession(Session session) { if (session != null) { if (session.isOpen()) { session.close(); } } } }
导入数据库工具类
package com.ncepu.usermgr.util; import org.hibernate.cfg.Configuration; import org.hibernate.tool.hbm2ddl.SchemaExport; public class ExportDB { public static void main(String[] args) { //读取hibernate.cfg.xml文件 Configuration cfg = new Configuration().configure(); SchemaExport export = new SchemaExport(cfg); export.create(true, true); } }
hibernate.cfg.xml配置文件
<!DOCTYPE hibernate-configuration PUBLIC "-//Hibernate/Hibernate Configuration DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd"> <hibernate-configuration> <session-factory> <property name="hibernate.connection.url">jdbc:mysql://localhost/spring_hibernate_1</property> <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property> <property name="hibernate.connection.username">root</property> <property name="hibernate.connection.password">admin</property> <property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property> <property name="hibernate.show_sql">true</property> <property name="hibernate.current_session_context_class">thread</property> <!-- <property name="hibernate.current_session_context_class">jta</property> --> <mapping resource="com/ncepu/usermgr/model/User.hbm.xml"/> <mapping resource="com/ncepu/usermgr/model/Log.hbm.xml"/> </session-factory> </hibernate-configuration>