Подтвердить что ты не робот

Как получить сложный класс и его членов с помощью Hibernate Projection?

У меня есть класс, следующий за тем, который нужно извлечь из БД с помощью Hibernate. Проблема заключается в том, что мой класс имеет несколько членов, и большинство из них являются классами, как их можно получить?

@Entity
public class Student {
  @Id
  long id;
  String name;
  String fname;
  @OneToMany
  List<Course> courses;
  @ManyToOne
  Dealer dealer;
  ...
}

@Entity
public class Dealer {
   @Id
   long id;
   String name; 
   @OneToMany(fetch = FetchType.LAZY, mappedBy = "cr.dealer", cascade = CascadeType.ALL)
   Set<Car> cars = new HashSet<Cars>(0);
   ..

}

Мне нужно получить идентификатор студента 1 и все его курсы, его дилер и список автомобилей дилеров.

Моя проекция следующая, но ничего не возвращает.

  ...
    .setProjection(Projections.projectionList()

    .add(Projections.property("friends.cars").as("cars")
    ...
4b9b3361

Ответ 1

Поскольку у вас есть список курсов и набор автомобилей, вы можете просто получить весь граф в одном запросе:

select s
from Student s
left join fetch s.courses
left join fetch s.dealer d
left join fetch d.cars
where s.id = :id

Поскольку вы извлекаете две коллекции, этот запрос будет генерировать декартовый продукт, поэтому вам нужно убедиться, что в выбранных дочерних коллекциях не слишком много записей.

Если вы не хотите работать с декартовым продуктом, вы можете просто запустить этот запрос:

select s
from Student s
left join fetch s.courses
left join fetch s.dealer d
where s.id = :id

а затем вы получите доступ к продавцу. Чтобы получить эту коллекцию с помощью отдельного запроса:

Student s = ...;
s.getDealer().getCars().size();

Ответ 2

    // Projection is not needed, Hibernate will load child values as shown below

    Student student = session.get(Student.class);
    List<Course> courses = student.getCourses();
    Dealer dealer = student.getDealer();

    //  If u want records only where child records are present, u can use LEFT_OUTER_JOIN

    Criteria criteria = getHibernateSession().createCriteria(Student.class);
    criteria.createAlias("Course", "Course", JoinType.LEFT_OUTER_JOIN);

    // If u want to use Projections for performance, u have to add each and every column in projection

    Criteria criteria = getHibernateSession().createCriteria(A.class);
    criteria.createAlias("b", "b", JoinType.INNER_JOIN);
    criteria.createAlias("b.r", "b.r", JoinType.INNER_JOIN);
    criteria.createAlias("b.c", "b.c", JoinType.LEFT_OUTER_JOIN);
    ProjectionList projectionList = Projections.projectionList();
    projectionList.add(Projections.groupProperty("column1"));
    projectionList.add(Projections.property("column2"));
    projectionList.add(Projections.property("column3"));
    criteria.setProjection(projectionList);
    criteria.setResultTransformer(Transformers.aliasToBean(Table.class));

Ответ 3

Если высокая производительность не вызывает беспокойства, вы должны позволить Hibernate выполнять свою работу. Просто используйте получателей ваших сущностей. Например:

Student student1 = session.get(Student.class, 1L);
List<Course> courses = student1.getCourses();
Dealer dealer = student1.getDealer();
Set<Car> cars = dealer.getCars();

Ответ 4

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

Student student = null;
Dealer dealer = null;
Course course = null;
Car car = null;

var myStudent = Session.QueryOver<Student>(() => student)
.Left.JoinQueryOver(() => student.courses, () => courses)
.Left.JoinQueryOver(() => student.dealer, () => dealer)
.Left.JoinQueryOver(() => dealer.cars, () => car)
.SelectList(list => list
  .Select(() => student.Name)
  .Select(() => student.Age)
  .Select(() => courses.Description)
  .Select(() => dealer.locaiton)
  .Select(() => car.Model))
  .TransformUsing(Transformers.AliasToBean<StudentModel>())
  .List<StudentModel>().AsQueryable();

Создайте DTO StudentModel для получения результатов. Это всего лишь подсказка для начала, вы можете изменить это в соответствии с вашим требованием. Надеюсь, это сработает.:)