大致结构:

Person(人): id,name,age,bookId

Book(书):id,bookName

Author(作者):id,authorName,bookId

一个人 只有 一本书,一本书 有多个 作者,一个作者 只出 一本书;(可能举例不好,明白就行)

Person ----> Book : OneToOne

Book ----> Author: OneToMany

期望达到的效果:

sql: select p.id id,p.name,b.bookName bookName from person p left join book b on p.bookId=b.id ;

不管这个person有没有book都要列出这个person

如果是规范的hibernate实体对象(不知道怎么描述规范,大致是符合hibernate面向对象的表设计的实体类)

sql相应的hql: select p.id id,p.name,b.bookName bookName from person p left join p.book b with p.bookId=b.id ; (有很多写法,但如果hibernate映射,我查到的说法是无法用hql写left join)

@Entity
@Table
public class Person{
private String id;
private String name;
private String age;
@OneToOne
private Book book; //...
}
@Entity
@Table
public class Book {
private String id;
private String bookName;
@OneToOne
private Person person;
@OneToMany
private List<Author> authors;
//...
}
@Entity
@Table
public class Author{
private String id;
private String authorName;
@ManyToOne
private Book book; //...
}

如果是这样,其实得到person就可以得到person的所有关联对象。但,如果考虑极致的查询效率、内存占用。那么(个人)感觉这做法不好。

较好的做法是,你要什么sql就返回什么指定的列映射到dto(所以,个人还是喜欢用ibatis/mybatis);

而且,现在遇到的问题是。实体对象Perosn中 private Book book; 写成了 private String bookId;导致hql无法达到left的效果(在没映射的关系下,反正没找到能达到left join效果的hql)。

所以,就用hibernate的sql来实现。(也可以用jdbcTmplate.query(…),只是分页是自己sql写好的;返回到自定义对象只要实现RowMapper就可以)

Demo:

String sql = "select p.id id ,p.name name ,b.bookName bookName from person p left join book b on p.bookId=b.id";
SQLQuery q = ....;
//q.addScalar("id",StringType.INSTANCE);
//q.addScalar("name",StringType.INSTANCE);
//q.addScalar("bookName",StringType.INSTANCE);
q.setResultTransformer(Transformers.aliasToBean(Dto.class));

问题:

用的数据库是oracle,所以导致默认获得到的别名alias都转成了大写。但是,在dto中又是驼峰式的。

所以,在Transformers中会报exception:can not find setter;

因为,Transformers是根据alias反射找到setter。但alias都是大写ID、NAME、BOOKNAME,但setter其实是setId、setName、setBookName。

解决:

1、sql中可以写成  select p.id “id” ,p.name “name” ,b.bookName “bookName” from person p left join book b on p.bookId=b.id ; 即sql中alias都加上双引号(oracle的规定),那么Transformers得到的alias不会被转换成大写。

2、设置addScalar(…),但个人觉得写的太多而且不通用,每个都要写。(最好加上类型)

3、重写/扩展oracle的dialect;

4、我也在博问发求助了:Hibernate原生sql查询多表返回自定义对象问题? ,又大致去查了下,在stackoverflow找到一个:mapping Hibernate query results to custom class? 重点看2L v.ladynev的回答,并且ta给了ta自己重写的transformers fluent-hibernate

其实ta的做法和我在看transformers源码时想到的一样(本来以为代码量很少),既然是alias反射dto是找不到setter,那么就想办法找到setter;

public Object transformTuple(Object[] tuple, String[] aliases) {
Object result;
try {
if ( ! isInitialized ) {
initialize( aliases ); //exception
}
else {
check( aliases );
}
result = resultClass.newInstance();
for ( int i = 0; i < aliases.length; i++ ) {
if ( setters[i] != null ) {
setters[i].set( result, tuple[i], null );
}
}
}
catch ( InstantiationException e ) {
throw new HibernateException( "Could not instantiate resultclass: " + resultClass.getName() );
}
catch ( IllegalAccessException e ) {
throw new HibernateException( "Could not instantiate resultclass: " + resultClass.getName() );
}
return result;
} private void initialize(String[] aliases) {
PropertyAccessor propertyAccessor = new ChainedPropertyAccessor(
new PropertyAccessor[] {
PropertyAccessorFactory.getPropertyAccessor( resultClass, null ),
PropertyAccessorFactory.getPropertyAccessor( "field" )
}
);
this.aliases = new String[ aliases.length ];
setters = new Setter[ aliases.length ];
for ( int i = 0; i < aliases.length; i++ ) {
String alias = aliases[ i ];
if ( alias != null ) {
this.aliases[ i ] = alias;
setters[ i ] = propertyAccessor.getSetter( resultClass, alias ); //exception: alias都是大写,再深入看propertyAccessor.getSetter就知道是找不到setter
}
}
isInitialized = true;
}
// org.hibernate.property.BasicPropertyAccessor
private static Method setterMethod(Class theClass, String propertyName) { //propertyName就是alias,大写
BasicPropertyAccessor.BasicGetter getter = getGetterOrNull(theClass, propertyName);
Class returnType = getter == null?null:getter.getReturnType();
Method[] methods = theClass.getDeclaredMethods();
Method potentialSetter = null;
Method[] arr$ = methods;
int len$ = methods.length; for(int i$ = 0; i$ < len$; ++i$) {
Method method = arr$[i$];
String methodName = method.getName();
if(method.getParameterTypes().length == 1 && methodName.startsWith("set")) {
String testStdMethod = Introspector.decapitalize(methodName.substring(3));
String testOldMethod = methodName.substring(3); // 都是dto的setter值,Id、Name、BookName
if(testStdMethod.equals(propertyName) || testOldMethod.equals(propertyName)) { // 所以,此处if为false;method=null
potentialSetter = method;
if(returnType == null || method.getParameterTypes()[0].equals(returnType)) {
return method;
}
}
}
} return potentialSetter;
}
// 这是fluent-hibernate的源码 ;com.github.fluent.hibernate.internal.util.reflection.ReflectionUtils
/**
* Try to find a class getter method by a property name. Don't check parent classes or
* interfaces.
*
* @param classToCheck
* a class in which find a getter
* @param propertyName
* a property name
* @return the getter method or null, if such getter is not exist
*/
public static Method getClassGetter(Class<?> classToCheck, String propertyName) {
PropertyDescriptor[] descriptors = getPropertyDescriptors(classToCheck); for (PropertyDescriptor descriptor : descriptors) {
if (isGetter(descriptor, propertyName)) {
return descriptor.getReadMethod();
}
} return null;
} private static boolean isGetter(PropertyDescriptor descriptor, String propertyName) {
Method method = descriptor.getReadMethod();
return method != null && method.getParameterTypes().length == 0
&& descriptor.getName().equalsIgnoreCase(propertyName); //忽略大小写找到setter
} private static PropertyDescriptor[] getPropertyDescriptors(Class<?> beanClass) {
try {
return Introspector.getBeanInfo(beanClass).getPropertyDescriptors();
} catch (IntrospectionException ex) {
throw InternalUtils.toRuntimeException(ex);
}
}

我并没去完整认真的去看fluent-hibernate,可能是因为自己也是想通过找到setter的去解决 + 作者回答时候也是说主要就是怎么找到setter,把BookName当作BOOKNAME看待:

Using a custom result transformer

Another way to solve the problem — using a result transformer that ignores method names case (treat getFirstName() as getFIRSTNAME()). You can write your own or use FluentHibernateResultTransformer. You will not need to use quotes and aliases (if you have column names equal to DTO names)

ps: 其实想看认真看下的,本来我想的不用写多少代码量。但发现作者写了好多…

尚存疑问:

我记得我还看到的说法有说重写或扩展oralce的dialect的。记得之前公司的项目框架有重写oracle的dialect,貌似sql中的别名不会被转换成全大写。

具体不清楚,整天都是在写垃圾的业务实现代码,就知道叫加班,代码一点质量都没有,不好看懂、不好扩展、方法老旧,只是为了实现当前业务需求应付客户。搞的下班了没时间、也没经理去深入一些框架的东西,烦躁…><!

【Hibernate】hibernate原生sql利用transformers返回多表自定义类型对象的更多相关文章

  1. Hibernate SQLQuery 原生SQL 查询及返回结果集处理-1

    第一篇:官方文档的处理方法,摘自官方 在迁移原先用JDBC/SQL实现的系统,难免需要采用hibernat native sql支持. 1.使用SQLQuery hibernate对原生SQL查询执行 ...

  2. Hibernate运行原生sql并将查询的结果转化为对象

    原生SQL查询执行的控制是通过SQLQuery接口进行的,通过执行Session.createSQLQuery()获取这个接口.下面来描述如何使用这个API进行查询.标量查询(Scalar queri ...

  3. Hibernate SQLQuery 原生SQL 查询及返回结果集处理-2

    1. 返回List, .setResultTransformer(      Transformers.ALIAS_TO_ENTITY_MAP);将结果转为Map,存放到list中,即list中为若干 ...

  4. Hibernate执行原生SQL返回List<Map>类型结果集

    我是学java出身的,web是我主要一块: 在做项目的时候最让人别扭的就是hibernate查询大都是查询出List<T>(T指代对应实体类)类型 如果这时候我用的联合查询,那么返回都就是 ...

  5. hibernate使用原生SQL查询返回结果集的处理

    今天没事的时候,看到公司框架里有一个用原生SQL写的函数,说实在以前自己也干过这事,但好久都没有用,都忘得差不多了,现在基本都是用的hql语句来查询结果.hibernate中使用createSQLQu ...

  6. java:Hibernate框架3(使用Myeclipse逆向工程生成实体和配置信息,hql语句各种查询(使用hibernate执行原生SQL语句,占位符和命名参数,封装Vo查询多个属性,聚合函数,链接查询,命名查询),Criteria)

    1.使用Myeclipse逆向工程生成实体和配置信息: 步骤1:配置MyEclipse Database Explorer: 步骤2:为项目添加hibernate的依赖: 此处打开后,点击next进入 ...

  7. hibernate 5原生sql查询测试学习代码

    基本查询 import java.util.List; import org.hibernate.SQLQuery; import org.hibernate.Session; import org. ...

  8. hibernate在使用sql查询query自动转化成model类型数据,query.addEntity

    hibernate使用自动的hql查询或者其封装的查询方法都能字段转化成对象 而如果在hibernate中使用sql时大多返回为Object[]对象 那么如何将object[]转换成model呢,答案 ...

  9. hibernate使用原生SQL查询

    以下是Demo测试Hibernate 原生SQL查询: import java.util.Iterator; import java.util.List; import java.util.Map; ...

随机推荐

  1. ARTS Week 5

    Nov 25, 2019 ~ Dec 1, 2019 Algorithm 深度优先搜索--书籍分配 题目描述:有b1-b5五本书,要分配给五个学生,分别是a1-a5.但每个学生都有其喜欢的书,要检查是 ...

  2. BZOJ 4556(后缀数组+主席树求前驱后继+二分||后缀数组+二分+可持久化线段树)

    换markdown写了.. 题意: 给你一个1e5的字符串,1e5组询问,求\([l_1,r_1]\)的所有子串与\([l_2,r_2]\)的lcp 思路: 首先可以发现答案是具有单调性的,我们考虑二 ...

  3. React+wangeditor+node富文本处理带图片上传

    最近有个需求出现在我的视野中,因为我的另外的博客需要上传文章,但是我不想每次都在我的数据库中慢慢的修改格式,所以我另做了一个后台去编辑文本后发送给服务器,那么这里就涉及到两点,一个是富文本,一个是需要 ...

  4. C/C++中的排序和查找

    以下内容来自<C/C++程序设计实用案例教程> 1.排序 1.1使用qsort函数 C/C++库函数提供了快速排序函数qsort(q时quick的简写),需要引入头文件<stdlib ...

  5. Mysql设置创建时间字段和更新时间字段自动获取时间,填充时间

    1.引言在实际开发中,每条数据的创建时间和修改时间,尽量不需要应用程序去记录,而由数据库获取当前时间自动记录创建时间,获取当前时间自动记录修改时间. 2.创建语句(1)–添加CreateTime 设置 ...

  6. SSL公钥证书传递进行隐匿传输数据

    title: 使用X.509公钥证书传递进行隐匿传输数据 date: 2018-02-11 17:47:50 tags: --- 使用X.509公钥证书传递进行隐匿传输数据 看到国外一篇有关于在ssl ...

  7. 性能优化-CPU占用过高问题排查

    1. 性能优化是什么? 1.1 性能优化就是发挥机器本来的性能 1.2 性能瓶颈在哪里,木桶效应.   CPU占用过高 1.现象重现 CPU占用过高一般情况是代码中出现了循环调用,最容易出现的情况有几 ...

  8. Java开发最佳实践(二) ——《Java开发手册》之"异常处理、MySQL 数据库"

    二.异常日志 (一) 异常处理 (二) 日志规约 三.单元测试 四.安全规约 五.MySQL数据库 (一) 建表规约 (二) 索引规约 (三) SQL语句 (四) ORM映射 六.工程结构 七.设计规 ...

  9. 刷题85. Maximal Rectangle

    一.题目说明 题目,85. Maximal Rectangle,计算只包含1的最大矩阵的面积.难度是Hard! 二.我的解答 看到这个题目,我首先想到的是dp,用dp[i][j]表示第i行第j列元素向 ...

  10. python3-cookbook笔记:第十章 模块与包

    python3-cookbook中每个小节以问题.解决方案和讨论三个部分探讨了Python3在某类问题中的最优解决方式,或者说是探讨Python3本身的数据结构.函数.类等特性在某类问题上如何更好地使 ...