Общий DAO, Spring, Hibernate

Я хочу понять, как я могу реализовать общие методы, такие как добавление, редактирование, удаление и поиск в моей базе данных, я уже установил соединение (спящий режим) и отлично работает

У меня есть этот метод, который работает

Класс: 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); 

также у меня есть другие методы, которые я не знаю, как их использовать

Класс: 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;
}

Спасибо


person NiB    schedule 03.08.2017    source источник
comment
Вам нужно быть немного более точным в отношении того, что не работает, если вы ожидаете какой-либо полезной помощи. Какие именно проблемы у вас есть?   -  person Kevin Anderson    schedule 03.08.2017
comment
Что вы имеете ввиду под "не работает"? Пожалуйста, поделитесь исключением и трассировкой стека.   -  person zuckermanori    schedule 03.08.2017
comment
Извините, я имел в виду, что не знаю, как их использовать, не то чтобы они не работают   -  person NiB    schedule 04.08.2017


Ответы (2)


Я думаю, что класс GenericDAO является базовым классом. Это не для использования напрямую. Вы проверяли эту статью? Я проверил эту статью и создал пример проекта.

Пример

GitHub — образец generic-dao-hibernate

Например, вы можете создать API для получения списка всех сотрудников в соответствии с примером первого шага MySQL.

Схема таблицы сотрудников выглядит следующим образом:

Базовый 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)
    );

Отображение О/П

Hibernate требует, чтобы вы настроили сопоставление параметров отношения объекта. После этого вам понравится преобразовывать объект в sql и sql в объект.

Класс сущностей на основе SQL

  • @Entity, @Table, @Id, @Column, @GeneratedValue из Hibernate
  • @Data, @NoArgsConstructor взяты из ломбока, это сокращает код геттера/сеттера
  • @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;
    }
    

Класс ресурсов для внешнего интерфейса

Вам всегда нужно писать DAO (объект доступа к данным) для доступа к базе данных. GenericDAO — это метод сокращения стандартного исходного кода.

Класс EmployeesResource

  • CRUD operations on WEB API
    • #create, #read, #update or #delete

должно быть эквивалентно

  • SQL
    • INSERT, SELECT, UPDATE and 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);
        }
    }

Универсальный интерфейс и реализация

Интерфейс (как в сообщении IBM)

Согласно сообщению, мы можем объявить интерфейс 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);
        }

Если вы используете в проекте только простые операции CRUD, вам не нужно добавлять какой-либо код для операций SQL. Например, вы можете создать другие простые таблицы SQL, такие как divisions_table или personnel_table, используя extends GenericDao<Division, Integer> или extends GenericDao<Personnel, Integer>.

ИЗМЕНИТЬ

Чтобы создать экземпляр реального класса dao, связанного с каждой таблицей, вам необходимо настроить 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 mapper действительно хорош, а какой нет. Я думаю, что O/R mapper сейчас немного снижается. Вместо Hibernate вы можете найти MyBatis , JOOQ

person hiropon    schedule 04.08.2017
comment
Да, я читал ту статью. пока не совсем понимаю как это реализовать - person NiB; 04.08.2017
comment
Если бы у меня был пользователь класса, и я хочу добавить, найти, перечислить по разным параметрам или удалить один, как я мог бы это реализовать? - person NiB; 04.08.2017
comment
@NicolasB Я сообщу о вставке/обновлении позже. Но это не так сложно. - person hiropon; 06.08.2017
comment
Еще раз спасибо! Было очень полезно - person NiB; 07.08.2017

Это один из способов реализации универсального 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 создает новый экземпляр дао при каждом запросе.

@Repository
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
public class GenericHibernateDAO<T extends Serializable> extends AbstractHibernateDAO<T> 
implements IGenericDAO<T> {
   //
}

Класс обслуживания

Показывает, как использовать автоподключение универсального дао в классе обслуживания и передать классу модели параметр. Также обратите внимание, что эта реализация использует аннотацию @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();
}
}

Конфигурация приложения

Показывает, как настроить Spring для автоматического управления транзакциями с помощью @EnableTransactionManagement.

@Configuration
@ComponentScan("com.base-package")
@EnableTransactionManagement
public class AppConfig {

       // add hibernate configuration

      // add beans


}
person Durja    schedule 06.01.2019