通用 DAO,Spring,休眠
Generic DAO, Spring, Hibernate
我想了解如何在我的数据库上实现添加、编辑、删除和搜索等通用方法,我已经建立连接(休眠)并且工作正常
我有这个方法,很管用
Class: GenericDAO
public <T> T save(final T o){
Session session=HibernateUtil.getSessionFactory().openSession();
Transaction trans=session.beginTransaction();
Object object = (T) session.save(o);
trans.commit();
return (T) object;
}
并在主要
GenericDAO gen = new GenericDAO();
gen.save(object);
我还有其他不知道如何使用的方法
Class: GenericDAO
public void delete(final Object object){
Session session=HibernateUtil.getSessionFactory().openSession();
Transaction trans=session.beginTransaction();
session.delete(object);
trans.commit();
}
/***/
public <T> T get(final Class<T> type, final int id){
Session session=HibernateUtil.getSessionFactory().openSession();
Transaction trans=session.beginTransaction();
Object object = (T) session.get(type, id);
trans.commit();
return (T) object;
}
public <T> List<T> getAll(final Class<T> type) {
Session session=HibernateUtil.getSessionFactory().openSession();
Transaction trans=session.beginTransaction();
final Criteria crit = session.createCriteria(type);
List<T> list = crit.list();
trans.commit();
return list;
}
谢谢
我认为GenericDAO
class是基础class。它不是直接使用的。你看过这篇文章了吗?我检查了这篇文章并创建了一个示例项目。
例子
GitHub - generic-dao-hibernate sample
例如,您可能希望根据 MySQL 第一步示例创建一个 API 来检索所有员工列表。
员工 table 架构如下:
基础SQL
CREATE TABLE employees (
emp_no INT NOT NULL, -- UNSIGNED AUTO_INCREMENT??
birth_date DATE NOT NULL,
first_name VARCHAR(14) NOT NULL,
last_name VARCHAR(16) NOT NULL,
gender ENUM ('M','F') NOT NULL, -- Enumeration of either 'M' or 'F'
hire_date DATE NOT NULL,
PRIMARY KEY (emp_no) -- Index built automatically on primary-key column
-- INDEX (first_name)
-- INDEX (last_name)
);
O/R映射
Hibernate 要求您配置映射对象关系设置。之后,您将享受将对象转换为 sql 和将 sql 转换为对象的乐趣。
实体 class 基于 SQL
@Entity, @Table, @Id, @Column, @GeneratedValue
来自 Hibernate
@Data, @NoArgsConstructor
来自lombok,它减少了getter/setter代码
@XmlRootElement, @XmlAccessorType
来自 jaxb,你可能不需要使用它
@Entity
@Data
@NoArgsConstructor
@Table(name = "employees")
@XmlAccessorType(XmlAccessType.FIELD)
@XmlRootElement
public class Employees implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@Column(name = "emp_no", unique = true)
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Integer empNo;
@Column(name = "birth_date")
private Date birthDate;
@Column(name = "first_name")
private String firstName;
@Column(name = "last_name")
private String lastName;
@Column(name = "gender")
@Enumerated(EnumType.STRING)
private Gender gender;
@Column(name = "hire_date")
private Date hireDate;
}
前端资源 Class
你总是需要编写DAO(Data Access Object)来访问数据库。 GenericDAO
是一种减少样板源代码的方法。
员工资源class
- WEB上的CRUD操作API
#create
、#read
、#update
或 #delete
应该等同于
- SQL
INSERT
、SELECT
、UPDATE
和 DELETE
您需要用键标识一条或多条记录。在这种情况下,id
是示例主键。
@Path("/employee")
public class EmployeesResource {
static Logger log = LoggerFactory.getLogger(EmployeesResource.class);
@GET
@Produces(MediaType.APPLICATION_JSON)
public List<Employees> index(@BeanParam Employees paramBean) {
EmployeesDao dao = (EmployeesDao) SpringApplicationContext.getBean("employeesDao");
List<Employees> result = dao.read();
System.out.println("Get all employees: size = " + result.size());
return result;
}
@GET
@Path("{id}")
@Produces(MediaType.APPLICATION_JSON)
public Employees show(@PathParam("id") Integer id) {
EmployeesDao dao = (EmployeesDao) SpringApplicationContext.getBean("employeesDao");
System.out.println("Get employees -> id = " + id);
return dao.read(id);
}
@POST
@Consumes(MediaType.APPLICATION_JSON)
public Integer create(Employees obj) {
EmployeesDao dao = (EmployeesDao) SpringApplicationContext.getBean("employeesDao");
return dao.create(obj);
}
@PUT
@Path("{id}")
@Consumes(MediaType.APPLICATION_JSON)
public void update(Employees obj, @PathParam("id") String id) {
EmployeesDao dao = (EmployeesDao) SpringApplicationContext.getBean("employeesDao");
dao.update(obj);
}
@DELETE
@Path("{id}")
public void destroy(@PathParam("id") Integer id) throws Exception {
EmployeesDao dao = (EmployeesDao) SpringApplicationContext.getBean("EmployeesDao");
dao.delete(id);
}
}
GenericDao 接口和实现
接口(来自 ibm 的 post)
根据post,我们可以声明dao接口。然后我们应该实现该接口的方法。
public interface GenericDao<T, PK extends Serializable> {
/** Persist the newInstance object into database */
PK create(T newInstance);
/**
* Retrieve an object that was previously persisted to the database using
* the indicated id as primary key
*/
T read(PK id);
List<T> read();
/** Save changes made to a persistent object. */
void update(T transientObject);
/** Remove an object from persistent storage in the database */
void delete(PK id) throws Exception;
void delete(T persistentObject) throws Exception;
}
实施
public class GenericDaoHibernateImpl<T, PK extends Serializable> implements GenericDao<T, PK> {
private Class<T> type;
@Autowired
private SessionFactory sessionFactory;
public SessionFactory getSessionFactory() {
return sessionFactory;
}
public void setSessionFactory(SessionFactory sessionFactory) {
this.sessionFactory = sessionFactory;
}
public GenericDaoHibernateImpl(Class<T> type) {
this.type = type;
}
// Not showing implementations of getSession() and setSessionFactory()
private Session getSession() {
Session session = sessionFactory.getCurrentSession();
return session;
}
@Transactional(readOnly = false, rollbackFor = RuntimeException.class)
public PK create(T o) {
return (PK) getSession().save(o);
}
@Transactional(readOnly = false, rollbackFor = RuntimeException.class)
public void update(T o) {
getSession().update(o);
}
@Transactional(readOnly = true)
public T read(PK id) {
return (T) getSession().get(type, id);
}
@SuppressWarnings("unchecked")
@Transactional(readOnly = true)
public List<T> read() {
return (List<T>) getSession().createCriteria(type).list();
}
@Transactional(readOnly = false, rollbackFor = RuntimeException.class)
public void delete(PK id) {
T o = getSession().load(type, id);
getSession().delete(o);
}
@Transactional(readOnly = false, rollbackFor = RuntimeException.class)
public void delete(T o) {
getSession().delete(o);
}
如果您在项目中仅使用简单的增删改查操作,则无需为SQL操作追加任何代码。例如,您可以使用 extends GenericDao<Division, Integer>
或 extends GenericDao<Personnel, Integer>
.
创建另一个简单的 SQL table,如 divisions_table
或 personnel_table
编辑
要实例化与每个table相关的真实dao class,您需要配置applicationContext.xml
和bean。
例子
<bean id="employeesDao" parent="abstractDao">
<!-- You need to configure the interface for Dao -->
<property name="proxyInterfaces">
<value>jp.gr.java_conf.hangedman.dao.EmployeesDao</value>
</property>
<property name="target">
<bean parent="abstractDaoTarget">
<constructor-arg>
<value>jp.gr.java_conf.hangedman.models.Employees</value>
</constructor-arg>
</bean>
</property>
</bean>
P.S.
你要记得这篇文章是十年前写的。而且,你应该认真考虑哪个 O/R 映射器真的好,哪个不好。我认为 O/R 映射器现在略有下降。你可以找到 MyBatis , JOOQ
而不是 Hibernate
这是实现以休眠为中心的通用 DAO 的一种方法。它提供基本的 CRUD 操作和简单的搜索,但可以扩展以包含其他通用功能。
IGenericDAO 接口
public interface IGenericDAO<T extends Serializable> {
T findOne(long id);
List<T> findAll();
void create(T entity);
void update(T entity);
void delete(T entity);
void deleteById(long entityId);
public void setClazz(Class<T> clazzToSet);
}
AbstractTemplateDAO
import java.io.Serializable;
import java.util.List;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.springframework.beans.factory.annotation.Autowired;
public abstract class AbstractHibernateDAO<T extends Serializable> implements IGenericDAO<T> {
private Class<T> clazz;
@Autowired
SessionFactory sessionFactory;
public final void setClazz(Class<T> clazzToSet) {
this.clazz = clazzToSet;
}
@Override
public T findOne(long id) {
return (T) getCurrentSession().get(clazz, id);
}
@Override
public List<T> findAll() {
return getCurrentSession().createQuery("from " + clazz.getName(),clazz).getResultList();
}
@Override
public void create(T entity) {
getCurrentSession().persist(entity);
}
@Override
public void update(T entity) {
getCurrentSession().merge(entity);
}
@Override
public void delete(T entity) {
getCurrentSession().delete(entity);
}
@Override
public void deleteById(long entityId) {
T entity = findOne(entityId);
delete(entity);
}
protected final Session getCurrentSession() {
return sessionFactory.getCurrentSession();
}
}
GenericHiberateDAO
注意:这里使用了作用域原型。 spring 容器在每次请求时创建一个新的 dao 实例。
@Repository
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
public class GenericHibernateDAO<T extends Serializable> extends AbstractHibernateDAO<T>
implements IGenericDAO<T> {
//
}
服务class
展示如何在服务中使用自动装配通用 dao class 并向模型 class 传递参数。另外,请注意此实现使用 @Transactional 注释进行 spring 事务管理。
@Service
public class TestService implements ITestService {
private IGenericDAO<TestModel> dao;
@Autowired
public void setDao(IGenericDAO<TestModel> daoToSet) {
dao = daoToSet;
dao.setClazz(TestModel.class);
}
@Override
@Transactional
public List<TestModel> findAll() {
return dao.findAll();
}
}
应用程序配置
展示如何使用@EnableTransactionManagement
设置spring自动事务管理
@Configuration
@ComponentScan("com.base-package")
@EnableTransactionManagement
public class AppConfig {
// add hibernate configuration
// add beans
}
我想了解如何在我的数据库上实现添加、编辑、删除和搜索等通用方法,我已经建立连接(休眠)并且工作正常
我有这个方法,很管用
Class: GenericDAO
public <T> T save(final T o){
Session session=HibernateUtil.getSessionFactory().openSession();
Transaction trans=session.beginTransaction();
Object object = (T) session.save(o);
trans.commit();
return (T) object;
}
并在主要
GenericDAO gen = new GenericDAO();
gen.save(object);
我还有其他不知道如何使用的方法
Class: GenericDAO
public void delete(final Object object){
Session session=HibernateUtil.getSessionFactory().openSession();
Transaction trans=session.beginTransaction();
session.delete(object);
trans.commit();
}
/***/
public <T> T get(final Class<T> type, final int id){
Session session=HibernateUtil.getSessionFactory().openSession();
Transaction trans=session.beginTransaction();
Object object = (T) session.get(type, id);
trans.commit();
return (T) object;
}
public <T> List<T> getAll(final Class<T> type) {
Session session=HibernateUtil.getSessionFactory().openSession();
Transaction trans=session.beginTransaction();
final Criteria crit = session.createCriteria(type);
List<T> list = crit.list();
trans.commit();
return list;
}
谢谢
我认为GenericDAO
class是基础class。它不是直接使用的。你看过这篇文章了吗?我检查了这篇文章并创建了一个示例项目。
例子
GitHub - generic-dao-hibernate sample
例如,您可能希望根据 MySQL 第一步示例创建一个 API 来检索所有员工列表。
员工 table 架构如下:
基础SQL
CREATE TABLE employees (
emp_no INT NOT NULL, -- UNSIGNED AUTO_INCREMENT??
birth_date DATE NOT NULL,
first_name VARCHAR(14) NOT NULL,
last_name VARCHAR(16) NOT NULL,
gender ENUM ('M','F') NOT NULL, -- Enumeration of either 'M' or 'F'
hire_date DATE NOT NULL,
PRIMARY KEY (emp_no) -- Index built automatically on primary-key column
-- INDEX (first_name)
-- INDEX (last_name)
);
O/R映射
Hibernate 要求您配置映射对象关系设置。之后,您将享受将对象转换为 sql 和将 sql 转换为对象的乐趣。
实体 class 基于 SQL
@Entity, @Table, @Id, @Column, @GeneratedValue
来自 Hibernate@Data, @NoArgsConstructor
来自lombok,它减少了getter/setter代码@XmlRootElement, @XmlAccessorType
来自 jaxb,你可能不需要使用它@Entity @Data @NoArgsConstructor @Table(name = "employees") @XmlAccessorType(XmlAccessType.FIELD) @XmlRootElement public class Employees implements Serializable { private static final long serialVersionUID = 1L; @Id @Column(name = "emp_no", unique = true) @GeneratedValue(strategy = GenerationType.IDENTITY) private Integer empNo; @Column(name = "birth_date") private Date birthDate; @Column(name = "first_name") private String firstName; @Column(name = "last_name") private String lastName; @Column(name = "gender") @Enumerated(EnumType.STRING) private Gender gender; @Column(name = "hire_date") private Date hireDate; }
前端资源 Class
你总是需要编写DAO(Data Access Object)来访问数据库。 GenericDAO
是一种减少样板源代码的方法。
员工资源class
- WEB上的CRUD操作API
#create
、#read
、#update
或#delete
应该等同于
- SQL
INSERT
、SELECT
、UPDATE
和DELETE
您需要用键标识一条或多条记录。在这种情况下,id
是示例主键。
@Path("/employee")
public class EmployeesResource {
static Logger log = LoggerFactory.getLogger(EmployeesResource.class);
@GET
@Produces(MediaType.APPLICATION_JSON)
public List<Employees> index(@BeanParam Employees paramBean) {
EmployeesDao dao = (EmployeesDao) SpringApplicationContext.getBean("employeesDao");
List<Employees> result = dao.read();
System.out.println("Get all employees: size = " + result.size());
return result;
}
@GET
@Path("{id}")
@Produces(MediaType.APPLICATION_JSON)
public Employees show(@PathParam("id") Integer id) {
EmployeesDao dao = (EmployeesDao) SpringApplicationContext.getBean("employeesDao");
System.out.println("Get employees -> id = " + id);
return dao.read(id);
}
@POST
@Consumes(MediaType.APPLICATION_JSON)
public Integer create(Employees obj) {
EmployeesDao dao = (EmployeesDao) SpringApplicationContext.getBean("employeesDao");
return dao.create(obj);
}
@PUT
@Path("{id}")
@Consumes(MediaType.APPLICATION_JSON)
public void update(Employees obj, @PathParam("id") String id) {
EmployeesDao dao = (EmployeesDao) SpringApplicationContext.getBean("employeesDao");
dao.update(obj);
}
@DELETE
@Path("{id}")
public void destroy(@PathParam("id") Integer id) throws Exception {
EmployeesDao dao = (EmployeesDao) SpringApplicationContext.getBean("EmployeesDao");
dao.delete(id);
}
}
GenericDao 接口和实现
接口(来自 ibm 的 post)
根据post,我们可以声明dao接口。然后我们应该实现该接口的方法。
public interface GenericDao<T, PK extends Serializable> {
/** Persist the newInstance object into database */
PK create(T newInstance);
/**
* Retrieve an object that was previously persisted to the database using
* the indicated id as primary key
*/
T read(PK id);
List<T> read();
/** Save changes made to a persistent object. */
void update(T transientObject);
/** Remove an object from persistent storage in the database */
void delete(PK id) throws Exception;
void delete(T persistentObject) throws Exception;
}
实施
public class GenericDaoHibernateImpl<T, PK extends Serializable> implements GenericDao<T, PK> {
private Class<T> type;
@Autowired
private SessionFactory sessionFactory;
public SessionFactory getSessionFactory() {
return sessionFactory;
}
public void setSessionFactory(SessionFactory sessionFactory) {
this.sessionFactory = sessionFactory;
}
public GenericDaoHibernateImpl(Class<T> type) {
this.type = type;
}
// Not showing implementations of getSession() and setSessionFactory()
private Session getSession() {
Session session = sessionFactory.getCurrentSession();
return session;
}
@Transactional(readOnly = false, rollbackFor = RuntimeException.class)
public PK create(T o) {
return (PK) getSession().save(o);
}
@Transactional(readOnly = false, rollbackFor = RuntimeException.class)
public void update(T o) {
getSession().update(o);
}
@Transactional(readOnly = true)
public T read(PK id) {
return (T) getSession().get(type, id);
}
@SuppressWarnings("unchecked")
@Transactional(readOnly = true)
public List<T> read() {
return (List<T>) getSession().createCriteria(type).list();
}
@Transactional(readOnly = false, rollbackFor = RuntimeException.class)
public void delete(PK id) {
T o = getSession().load(type, id);
getSession().delete(o);
}
@Transactional(readOnly = false, rollbackFor = RuntimeException.class)
public void delete(T o) {
getSession().delete(o);
}
如果您在项目中仅使用简单的增删改查操作,则无需为SQL操作追加任何代码。例如,您可以使用 extends GenericDao<Division, Integer>
或 extends GenericDao<Personnel, Integer>
.
divisions_table
或 personnel_table
编辑
要实例化与每个table相关的真实dao class,您需要配置applicationContext.xml
和bean。
例子
<bean id="employeesDao" parent="abstractDao">
<!-- You need to configure the interface for Dao -->
<property name="proxyInterfaces">
<value>jp.gr.java_conf.hangedman.dao.EmployeesDao</value>
</property>
<property name="target">
<bean parent="abstractDaoTarget">
<constructor-arg>
<value>jp.gr.java_conf.hangedman.models.Employees</value>
</constructor-arg>
</bean>
</property>
</bean>
P.S.
你要记得这篇文章是十年前写的。而且,你应该认真考虑哪个 O/R 映射器真的好,哪个不好。我认为 O/R 映射器现在略有下降。你可以找到 MyBatis , JOOQ
而不是 Hibernate这是实现以休眠为中心的通用 DAO 的一种方法。它提供基本的 CRUD 操作和简单的搜索,但可以扩展以包含其他通用功能。
IGenericDAO 接口
public interface IGenericDAO<T extends Serializable> {
T findOne(long id);
List<T> findAll();
void create(T entity);
void update(T entity);
void delete(T entity);
void deleteById(long entityId);
public void setClazz(Class<T> clazzToSet);
}
AbstractTemplateDAO
import java.io.Serializable;
import java.util.List;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.springframework.beans.factory.annotation.Autowired;
public abstract class AbstractHibernateDAO<T extends Serializable> implements IGenericDAO<T> {
private Class<T> clazz;
@Autowired
SessionFactory sessionFactory;
public final void setClazz(Class<T> clazzToSet) {
this.clazz = clazzToSet;
}
@Override
public T findOne(long id) {
return (T) getCurrentSession().get(clazz, id);
}
@Override
public List<T> findAll() {
return getCurrentSession().createQuery("from " + clazz.getName(),clazz).getResultList();
}
@Override
public void create(T entity) {
getCurrentSession().persist(entity);
}
@Override
public void update(T entity) {
getCurrentSession().merge(entity);
}
@Override
public void delete(T entity) {
getCurrentSession().delete(entity);
}
@Override
public void deleteById(long entityId) {
T entity = findOne(entityId);
delete(entity);
}
protected final Session getCurrentSession() {
return sessionFactory.getCurrentSession();
}
}
GenericHiberateDAO
注意:这里使用了作用域原型。 spring 容器在每次请求时创建一个新的 dao 实例。
@Repository
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
public class GenericHibernateDAO<T extends Serializable> extends AbstractHibernateDAO<T>
implements IGenericDAO<T> {
//
}
服务class
展示如何在服务中使用自动装配通用 dao class 并向模型 class 传递参数。另外,请注意此实现使用 @Transactional 注释进行 spring 事务管理。
@Service
public class TestService implements ITestService {
private IGenericDAO<TestModel> dao;
@Autowired
public void setDao(IGenericDAO<TestModel> daoToSet) {
dao = daoToSet;
dao.setClazz(TestModel.class);
}
@Override
@Transactional
public List<TestModel> findAll() {
return dao.findAll();
}
}
应用程序配置
展示如何使用@EnableTransactionManagement
设置spring自动事务管理@Configuration
@ComponentScan("com.base-package")
@EnableTransactionManagement
public class AppConfig {
// add hibernate configuration
// add beans
}