Это очень хороший пример использования для шаблона, называемого шаблонный метод .
Например, вы можете создать абстрактный класс, который обернет весь ваш шаблонный код в perform
метод:
abstract public class HibernateAction<T> {
private EntityManagerFactory entityManagerFactory;
//I'm passing EntityManagerFactory, because it should be singleton and you shouldn't
//probably create it from scratch everytime
public HibernateAction(EntityManagerFactory entityManagerFactory) {
this.entityManagerFactory = entityManagerFactory;
}
protected abstract T action(EntityManager entityManager, T entity);
public boolean perform(T entity) throws Exception {
try {
var entityManager = entityManagerFactory.createEntityManager();
entityManager.getTransaction().begin();
action(entityManager, entity); //call to action which need to be overriden
entityManager.getTransaction().commit();
entityManager.close();
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
}
тогда вы можете просто создать класс, который наследует HibernateAction
:
public class UpdateAction extends HibernateAction<Program> {
public UpdateAction(EntityManagerFactory entityManagerFactory) {
super(entityManagerFactory);
}
@Override
protected Program action(EntityManager entityManager, Program entity) {
return entityManager.merge(entity);
}
}
и, наконец, вы можете использовать его так:
public boolean update(Program program) throws Exception {
return updateAction.perform(program);
}
Но так как анонимные методы поддерживаются в Java (начиная с Java 8), вы также можете переписать его немного менее многословно, используя функции более высокого порядка:
public class HibernateAction2{ // no longer abstract
private EntityManagerFactory entityManagerFactory;
public HibernateAction2(EntityManagerFactory entityManagerFactory) {
this.entityManagerFactory = entityManagerFactory;
}
//we expect a user to pass lambda function, which would tell us what to do with an entity manager
public boolean perform (Consumer<EntityManager> action) throws Exception {
try {
var entityManager = entityManagerFactory.createEntityManager();
entityManager.getTransaction().begin();
action.accept(entityManager);
entityManager.getTransaction().commit();
entityManager.close();
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
}
и затем вы можете использовать его как:
hibernateAction2.perform(em -> em.merge(program)); //for merge
hibernateAction2.perform(em -> em.persist(program)); //for persist, etc.
Это называется шаблон ссуды или шаблон ссуды (или скобка на языках FP), потому что вы "одалживаете" менеджера сущностей из hibernateAction2, чтобы использовать его для выполнить какое-либо действие, но оно обрабатывает все остальные вещи, например создание объекта или закрытие соединения.