Hibernate学习笔记(2):Hibernate中子查询(subselect)的使用

 有些数据库不支持视图的创建,而实际业务中我们需要通过视图才能实现我们的需求,这时就可以使用Hibernate中子查询(subselect)

一,官方文档说明

There is no difference between a view and a base table for a Hibernate mapping. This is transparent at the database level, although some DBMS do not support views properly, especially with updates. Sometimes you want to use a view, but you cannot create one in the database (i.e. with a legacy schema). In this case, you can map an immutable and read-only entity to a given SQL subselect expression using @org.hibernate.annotations.Subselect:

@Entity
@Subselect("select item.name, max(bid.amount), count(*) "
        + "from item "
        + "join bid on bid.item_id = item.id "
        + "group by item.name")
@Synchronize( {"item", "bid"} ) //tables impacted
public class Summary {
    @Id
    public String getId() { return id; }
    ...
}

 

定义这个实体用到的表为同步(synchronize),确保自动刷新(auto-flush)正确执行,并且依赖原实体的查询不会返回过期数据。在属性元素和嵌套映射元素中都可使用 <subselect>

We will now explore the same options using the hbm.xml structure. You can declare a persistent class using the class element. For example:

<class
        name="ClassName"
        table="tableName"
        discriminator-value="discriminator_value"
        mutable="true|false"
        schema="owner"
        catalog="catalog"
        proxy="ProxyInterface"
        dynamic-update="true|false"
        dynamic-insert="true|false"
        select-before-update="true|false"
        polymorphism="implicit|explicit"
        where="arbitrary sql where condition"
        persister="PersisterClass"
        batch-size="N"
        optimistic-lock="none|version|dirty|all"
        lazy="(16)true|false"
        entity(17)-name="EntityName"
        check=(18)"arbitrary sql check condition"
        rowid=(19)"rowid"
        subsel(20)ect="SQL expression"
        abstra(21)ct="true|false"
        node="element-name"
/>

 

(1)name(可选):持久化类(或者接口)的 Java 全限定名。 如果这个属性不存在,Hibernate 将假定这是一个非 POJO 的实体映射。 
(2)table(可选 — 默认是类的非全限定名):对应的数据库表名。 
(3)discriminator-value(可选 — 默认和类名一样):一个用于区分不同的子类的值,在多态行为时使用。它可以接受的值包括 null 和 not null。 
(4)mutable(可选,默认值为 true):表明该类的实例是可变的或者不可变的。 
(5)schema(可选):覆盖在根 <hibernate-mapping> 元素中指定的 schema 名字。 
(6)catalog(可选):覆盖在根 <hibernate-mapping> 元素中指定的 catalog 名字。 
(7)proxy(可选):指定一个接口,在延迟装载时作为代理使用。你可以在这里使用该类自己的名字。 
(8)dynamic-update(可选,默认为 false):指定用于 UPDATE 的 SQL 将会在运行时动态生成,并且只更新那些改变过的字段。 
(9)dynamic-insert(可选,默认为 false):指定用于 INSERT 的 SQL 将会在运行时动态生成,并且只包含那些非空值字段。 
(10)select-before-update(可选,默认为 false):指定 Hibernate 除非确定对象真正被修改了(如果该值为 true — 译注),否则不会执行 SQL UPDATE 操作。在特定场合(实际上,它只在一个瞬时对象(transient object)关联到一个新的 session 中时执行的 update() 中生效),这说明 Hibernate 会在 UPDATE 之前执行一次额外的 SQL SELECT 操作来决定是否确实需要执行 UPDATE。
(11)polymorphisms (optional - defaults to implicit): determines whether implicit or explicit query polymorphisms is used.
(12)where(可选)指定一个附加的 SQL WHERE 条件,在抓取这个类的对象时会一直增加这个条件。 
(13)persister(可选):指定一个定制的 ClassPersister。 
(14)batch-size(可选,默认是 1)指定一个用于 根据标识符(identifier)抓取实例时使用的 "batch size"(批次抓取数量)。 
(15)optimistic-lock(乐观锁定)(可选,默认是 version):决定乐观锁定的策略。 
(16)lazy(可选):通过设置 lazy="false",所有的延迟加载(Lazy fetching)功能将被全部禁用(disabled)。
(17)entity-name (optional - defaults to the class name): Hibernate3 allows a class to be mapped multiple times, potentially to different tables. It also allows entity mappings that are represented by Maps or XML at the Java level. In these cases, you should provide an explicit arbitrary name for the entity. See 第 4.4 节 “动态模型(Dynamic models)” and 第 20 章 XML 映射 for more information.
(18)check(可选):这是一个 SQL 表达式, 用于为自动生成的 schema 添加多行(multi-row)约束检查。 
(19)rowid(可选):Hibernate 可以使用数据库支持的所谓的 ROWIDs,例如:Oracle 数据库,如果你设置这个可选的 rowid,Hibernate 可以使用额外的字段 rowid 实现快速更新。ROWID 是这个功能实现的重点,它代表了一个存储元组(tuple)的物理位置。 
(20)subselect(可选):它将一个不可变(immutable)并且只读的实体映射到一个数据库的子查询中。当你想用视图代替一张基本表的时候,这是有用的,但最好不要这样做。更多的介绍请看下面内容。 
(21)abstract(可选):用于在 <union-subclass> 的层次结构(hierarchies)中标识抽象超类。  

 若指明的持久化类实际上是一个接口,这也是完全可以接受的。之后你可以用元素 <subclass> 来指定该接口的实际实现类。你可以持久化任何 static(静态的)内部类。你应该使用标准的类名格式来指定类名,比如:Foo$Bar

Here is how to do a virtual view (subselect) in XML:

<class name="Summary">
    <subselect>
        select item.name, max(bid.amount), count(*)
        from item
        join bid on bid.item_id = item.id
        group by item.name
    </subselect>
    <synchronize table="item"/>
    <synchronize table="bid"/>
    <id name="name"/>
    ...
</class>

The <subselect> is available both as an attribute and a nested mapping element. 

二,个人对hibernate子查询的理解举例

1,基于视图的查询举例

   a)创建实体类(介于文章的长度,get和set方法的代码就不贴出来了)

   

@Entity
@Table(name="T_Order")
public class Order {
	@Id
	@GeneratedValue
	private int id;
	private String name;

 

@Entity
public class OrderItem {
	@Id
	@GeneratedValue
	private int id;
	private String name;
	private Double price;
	private int count;
	@ManyToOne
	private Order order;

  b)创建子查询视图(视图的字段是子查询的子集,即视图的字段要在子查询中可以找到对应的列)

@Entity
@Subselect("select o.id, o.name,sum(oi.price * oi.count) as count "
		+ "from T_Order o , OrderItem oi WHERE o.id = oi.order_id "
		+ "group by o.id , o.name ")
@Synchronize({ "T_Order", "OrderItem" })
public class OrderView {
	@Id
	private int id;
	private String name;
	private double count;

 c)创建测试类

public class OrderTest {
static SessionFactory factory = null;
	@BeforeClass
	public static void beforeClass(){
		factory = new AnnotationConfiguration().configure().buildSessionFactory();
	}
	@Test
	public void testGet(){
		Session session = factory.getCurrentSession();
		session.beginTransaction();
		Order o = new Order();
		o.setName("order1");
		session.save(o);
		for(int i = 0 ; i< 5 ; i++){
			OrderItem oi = new OrderItem();
			oi.setName("oi"+i);
			oi.setPrice(2.0);
			oi.setCount(4);
			oi.setOrder(o);
			session.save(oi);
		}
		session.getTransaction().commit();
		Session session2 = factory.getCurrentSession();
		session2.beginTransaction();
		List<OrderView> list = session2.createQuery("From OrderView ").list();
		for(OrderView ov : list){
			System.out.println(ov.getId());
			System.out.println(ov.getName());
			System.out.println(ov.getCount());
		}
		session2.getTransaction().commit();
	}
}

 

d)测试类输出结果

1
order1
40.0

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Hibernate的检索方式:(查询/加载) 1. 通过OID加载 session.get(Users.class, 1); * 2. 通过HQL/SQL 检索 hibernate query language (面向对象的查询语言) * a) 不再操纵表,它操纵的是持久化类的对象 b) 面向对象的 3. QBC ( query by criteria ) 更加面向对象 4. QBE ( query by Example ) 5. SQL Hibernate的检索策略: 1. 延迟检索(加载)映射文件改变lazy a) Lazy的取值: i. Many-to-one 1. false 2. proxy 3. no-proxy ii. set 一对多 1. true   2. false   3. extra 根据对set容器的不同,可以产生高效的sql访问数据库 2. 批量检索:batch-size=3 a) 可以使用批量检索: b) 在内存,如果有多个set(代理)容器需要初始化, 则当访问任何一个代理set容器时,一次初始化n个set容器,减少sql语句; c) 产生的语句是:select * from ….. where FK in (?,?,?... …n); 3. 迫切(fetch)左外连接检索: a) 在映射文件设置:fetch(抓取) i. Select 什么都不做 ii. Subselect 当访问内存的一个代理容器时,一次将所有的set容器用一个子查询,全部初始化 iii. Join 1. 对OID检索方式有用 2. 对hql检索方式没有用 b) 在hql使用迫切左外连接时,必须加 left join fetch 对象.关系属性 i. 如果不加fetch关键字,则hibernate不会抓取关系属性,但会遍历关系属性所对应的表 ii. 不加fetch关键字时,select 要指定返回的对象,否则它要返回数组 iii. 条件:持久化类之间有关系属性映射 Hibernate级联操作对象的关系属性: 映射文件设置: 1. Cascade a) none b) Save-update 当使用session的api对当前对象进行save,update操作时,对它的关系属性也进行save或者update c) Delete d) All = delete + save-update e) Delete-orphan 只是删除父子关系的子对象 f) All-delete-orphan 父子关系的表设为此值,表示,当删除一个父对象时,将这个set容器的子对象全部删除 2. Inverse 只能在set映射时使用;它的作用是:是否根据set容器存放的对象,产生update语句,维护关系(子)表的外键属性;取值为true时,将不产生update语句。当inverse设为true时,通过父亲增儿子时,必须建立双向关系. Session的一级缓存:(相当于一系列的map容器,它是需要维护的) 1. 提高效率 2. 维护缓存的对象和数据库对应表的记录之间进行同步 3. 当一级缓存的对象状态(属性)发展生改变时,session在特定的时刻清理缓存: a) 清理缓存的时间点: i. Transaction.commit(); 事务提交时 ii. Session.flush(); 4. 管理session一级缓存的方法 a) Session.evict(obj)从session的一级缓存移出一个对象 b) Session.clear() 将一级缓存的所有对象全部清空 c) Session.close() 关闭一个session 对象的状态和对象的生命周期: 持久化类的对象,在hibernate应用可以处于三种状态(根据对象和session之间的关系进行划分): 1. 临时态,瞬态:特点: a) 在数据库没有记录和它对应 b) 和session没有任何关系 c) New 出来的对象,都处于临时态 2. 持久态:特点: a) 处于session的一级缓存 b) 数据库有一条记录和它对应 c) Session会在特定的时刻(清理缓存时)维护这个对象和数据库的记录进行同步 d) 在同一个session的缓存,具有相同OID的持久态对象,只有一个(同一个session的一缓存,不可能同时有两个OID相同的同一个持久化类的对象存在) 3. 游离态,脱管 a) 不在一级缓存之 b) 数据库可能有记录和它对应,也可能没有记录和它对应 c) 从一个持久态对象转化过来的,从session的一级缓存出来的,因为调用了session的一些方法,产生了这种对象(session.close()) SessionFctory: Hibernate映射一对多关系: public class Dept implements java.io.Serializable { // Fields private Integer deptid; private String deptname; private Integer deptnum; private Integer actNum; private Date cdate; private Set emps = new HashSet(0) ; //getter/setter方法 略… } 映射文件: <hibernate-mapping> <class name="org.wllt.www.po.Dept " table="dept " catalog="hibernate"> <id name="deptid" type="java.lang.Integer" > <column name="deptid" /> <generator class="native" /> </id> <property name="deptname" type="java.lang.String"> <column name="deptname" length="20" /> </property> <property name="deptnum" type="java.lang.Integer"> <column name="deptnum" /> </property> <property name="actNum" type="java.lang.Integer"> <column name="actNum" /> </property> <property name="cdate" type="java.util.Date"> <column name="cdate" length="19" /> </property> <set name="emps" inverse ="true" lazy="true" cascade="none" batch-size="2" fetch="join" > <key> <column name="deptid" /> </key> <one-to-many class="org.wllt.www.po.Emp" /> </set> </class> </hibernate-mapping> Hibernate映射多对一: public class Emp implements java.io.Serializable { private Integer empid; private Dept dept; private String empname; //getter/setter方法略 } <hibernate-mapping> <class name="org.wllt.www.po.Emp" table="emp" catalog="hibernate"> <id name="empid" type="integer"> <column name="empid" /> <generator class="native" /> </id> <many-to-one name="dept" class="org.wllt.www.po.Dept" fetch="select" cascade ="none" lazy ="proxy"> <column name="deptid" /> </many-to-one> <property name="empname" type="string"> <column name="empname" length="30" /> </property> </class> </hibernate-mapping> Hibernate映射一对一(通过主键实现一对一的关系) //主PO对象(一个员工对个应一个身份证) public class Employees implements Serializable{ private Integer empid; private String name; private String sex; private Date birthday; private Double salary; //关系属性: private Idcard idcard; } <hibernate-mapping package ="org.wllt.www.po"> <class name="Employees" table="emp"> <id name="empid" type="java.lang.Integer"> <generator class="increment"/> </id> … 属性映射略 <one-to-one name="idcard" class="Idcard" fetch="select" lazy="proxy" cascade ="save-update"> </one-to-one> </class> </hibernate-mapping> //子PO对象 public class Idcard { private Integer empid; private String cardno; private String addr; private String fzjg; private Date enddate; //关系属性 private Employees emp; } <hibernate-mapping package="org.wllt.www.po3"> <class name="Idcard"> <id name="empid" type="java.lang.Integer"> <generator class="foreign "> <param name="property ">emp</param> </generator> </id> <one-to-one name="emp" class="Employees" fetch="join" constrained="true" cascade="save-update" > </one-to-one> <property name="cardno" type="java.lang.String" /> <property name="addr" type="java.lang.String" /> <property name="fzjg" type="java.lang.String" /> <property name="enddate" type="java.util.Date"/> </class> </hibernate-mapping> Hibernate映射一对一(通过外键实现一对一的关系) 子表引用主表的主键做外键,这个外键建立了unique约束、not-null约束 //主对象: public class Dept implements java.io.Serializable { private Integer deptid; private String deptname; private Integer deptnum; private Integer actNum; private Date cdate; //关系属性 private Phonenote phonenote; } <hibernate-mapping> <class name="org.wllt.www.po.Dept" table="dept" catalog="hibernate"> <id name="deptid" type="java.lang.Integer"> <column name="deptid" /> <generator class="native" /> </id> <-- …property 映射略 --> <one-to-one name="phonenote" class="org.wllt.www.po.Phonenote" property-ref="dept" cascade ="all" fetch="join" > </one-to-one> </class> </hibernate-mapping> 子对象 public class Phonenote implements java.io.Serializable { // Fields private Integer phonid; private Dept dept; private String phonecode; } <hibernate-mapping> <class name="org.wllt.www.po.Phonenote" table="phonenote" catalog="hibernate"> <id name="phonid" type="java.lang.Integer"> <column name="phonid" /> <generator class="native" /> </id> <many-to-one name="dept" class="org.wllt.www.po.Dept" fetch="select" cascade ="all"> <column name="deptid" unique="true" /> </many-to-one> <property name="phonecode" type="java.lang.String"> <column name="phonecode" length="20" not-null="true" /> </property> </class> </hibernate-mapping> Hibernate映射多对多(两个一对多实现多对多) 表:材料表和产品表多对多,在数据库间表即产品材料表用来存放两个表之间的关系 Java类:材料PO,产品PO,间PO,间PO的复合主键类(由于是两个一对多形成的多对多,所以,这里只讲一个一对多,另一个是相同的映射方法) //产品类: public class Product { private Integer cpbh; private String cpmc; private String gg; private String sh; private String nbxh; private String txm; private String bzfs; private Set proMals = new HashSet(0); } <hibernate-mapping package="org.wllt.www.po"> <class name="Product" table="prod"> <id name="cpbh" type="java.lang.Integer"> <column name="pro_id"></column> <generator class="increment"/> </id> <-- property 映射略 --> <set name="proMals" inverse="true" cascade="all-delete-orphan"> <key> <column name="pro_id"></column> </key> <one-to-many class="ProMal"/> </set> </class> </hibernate-mapping> 间PO public class ProMal { private ProMalId id; } 复合主键类: public class ProMalId implements Serializable{ private Product product; private Clzb clzb; } <hibernate-mapping package="org.wllt.www.po"> <class name="ProMal" table="pro_mat"> <composite-id name="id" class ="ProMalId"> <key-many-to-one name ="product" class ="Product" lazy="proxy" column="pro_id" > </key-many-to-one> <key-many-to-one name="clzb" class="Clzb"column="mat_id"> </key-many-to-one> </composite-id> </class> </hibernate-mapping> Hibernate映射多对多(两个PO类直接实现多对多) 表:材料表和产品表多对多,在数据库间表即产品材料表用来存放两个表之间的关系 Java类:材料PO,产品PO,这两个PO分别包含一个set容器, 相互放对方的对象,没有间PO, 程序员因为得不到间PO对象,所以不能直接操作数据库的间表。间表的记录的维护工作,只能交给hibernate来处理,这时,set容器的inverse属性必须设为false 主PO: public class Users implements java.io.Serializable { private Integer uerid; private String name; private String passwd; private Set roles = new HashSet(0); } <hibernate-mapping> <class name="org.wllt.www.po2.Users" table="users" atalog="hibernate"> <id name="uerid" type="java.lang.Integer"> <column name="UERID" /> <generator class="increment" /> </id> <-- property的映射略 --> <set name="roles" inverse ="false" table ="USERANDROLE"> <key> <column name="UERID" not-null="true" /> </key> <many-to-many class="org.wllt.www.po2.Roles " column ="ROLESID"/> </set> </class> </hibernate-mapping>

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值