15.10. 表达式

优质
小牛编辑
129浏览
2023-12-01

where 子句中允许使用的表达式包括 大多数你可以在 SQL 使用的表达式种类:

  • 数学运算符 +,-,*,/

  • 二进制比较运算符 =, >=, <=, <>, !=, like

  • 逻辑运算符 and,or,not

  • 括号 ( ),表示分组

  • in, not in, between, is null, is not null, is empty, is not empty, member of and not member of

  • "Simple" case, case ... when ... then ... else ... end, and "searched" case, case when ... then ... else ... end

  • 字符串连接符 ...||... or concat(...,...)

  • current_date(), current_time(), and current_timestamp()

  • second(...)minute(...)hour(...)day(...)month(...)year(...)

  • EJB-QL 3.0 定义的任何功能或操作符:substring(), trim(), lower(), upper(), length(), locate(), abs(), sqrt(), bit_length(), mod()

  • coalesce()nullif()

  • str() 把数字或者时间值转换为可读的字符串

  • cast(... as ...),其第二个参数是某 Hibernate 类型的名字,以及 extract(... from ...),只要 ANSI cast()extract() 被底层数据库支持

  • HQL index() 函数,作用于 join 的有序集合的别名。

  • HQL 函数,把集合作为参数:size(), minelement(), maxelement(), minindex(), maxindex(),还有特别的 elements()indices 函数,可以与数量词加以限定:some, all, exists, any, in

  • 任何数据库支持的 SQL 标量函数,比如 sign(), trunc(), rtrim(), sin()

  • JDBC 风格的参数传入 ?

  • 命名参数 :name:start_date:x1

  • SQL 直接常量 'foo', 69, 6.66E+2, '1970-01-01 10:00:01.0'

  • Java public static final 类型的常量 eg.Color.TABBY

关键字 inbetween 可按如下方法使用:

from DomesticCat cat where cat.name between 'A' and 'B'
from DomesticCat cat where cat.name in ( 'Foo', 'Bar', 'Baz' )

而且否定的格式也可以如下书写:

from DomesticCat cat where cat.name not between 'A' and 'B'
from DomesticCat cat where cat.name not in ( 'Foo', 'Bar', 'Baz' )

同样,子句 is nullis not null 可以被用来测试空值(null)。

在 Hibernate 配置文件中声明 HQL“查询替代(query substitutions)”之后,布尔表达式(Booleans)可以在其他表达式中轻松的使用:

<property name="hibernate.query.substitutions"
>true 1, false 0</property
>

系统将该 HQL 转换为 SQL 语句时,该设置表明将用字符 10 来取代关键字 truefalse

from Cat cat where cat.alive = true

你可以用特殊属性 size,或是特殊函数 size() 测试一个集合的大小。

from Cat cat where cat.kittens.size 
> 0
from Cat cat where size(cat.kittens) 
> 0

对于索引了(有序)的集合,你可以使用 minindexmaxindex 函数来引用到最小与最大的索引序数。同理,你可以使用 minelementmaxelement 函数来引用到一个基本数据类型的集合中最小与最大的元素。例如:

from Calendar cal where maxelement(cal.holidays) 
> current_date
from Order order where maxindex(order.items) 
> 100
from Order order where minelement(order.items) 
> 10000

在传递一个集合的索引集或者是元素集(elementsindices 函数)或者传递一个子查询的结果的时候,可以使用 SQL 函数 any, some,all, exists, in

select mother from Cat as mother, Cat as kit
where kit in elements(foo.kittens)
select p from NameList list, Person p
where p.name = some elements(list.names)
from Cat cat where exists elements(cat.kittens)
from Player p where 3 
> all elements(p.scores)
from Show show where 'fizard' in indices(show.acts)

注意,在 Hibernate3 中,这些结构变量 — sizeelementsindicesminindexmaxindexminelementmaxelement — 只能在 where 子句中使用。

一个被索引过的(有序的)集合的元素(arrays,lists,maps)可以在其他索引中被引用(只能在 where 子句中):

from Order order where order.items[0].id = 1234
select person from Person person, Calendar calendar
where calendar.holidays['national day'] = person.birthDay
    and person.nationality.calendar = calendar
select item from Item item, Order order
where order.items[ order.deliveredItemIndices[0] ] = item and order.id = 11
select item from Item item, Order order
where order.items[ maxindex(order.items) ] = item and order.id = 11

[] 中的表达式甚至可以是一个算数表达式:

select item from Item item, Order order
where order.items[ size(order.items) - 1 ] = item

对于一个一对多的关联(one-to-many association)或是值的集合中的元素,HQL 也提供内建的 index() 函数。

select item, index(item) from Order order
    join order.items item
where index(item) < 5

如果底层数据库支持标量的 SQL 函数,它们也可以被使用:

from DomesticCat cat where upper(cat.name) like 'FRI%'

如果你还不能对所有的这些深信不疑,想想下面的查询。如果使用 SQL,语句长度会增长多少,可读性会下降多少:

select cust
from Product prod,
    Store store
    inner join store.customers cust
where prod.name = 'widget'
    and store.location.name in ( 'Melbourne', 'Sydney' )
    and prod = all elements(cust.currentOrder.lineItems)

提示: 会像如下的语句

SELECT cust.name, cust.address, cust.phone, cust.id, cust.current_order
FROM customers cust,
    stores store,
    locations loc,
    store_customers sc,
    product prod
WHERE prod.name = 'widget'
    AND store.loc_id = loc.id
    AND loc.name IN ( 'Melbourne', 'Sydney' )
    AND sc.store_id = store.id
    AND sc.cust_id = cust.id
    AND prod.id = ALL(
        SELECT item.prod_id
        FROM line_items item, orders o
        WHERE item.order_id = o.id
            AND cust.current_order = o.id
    )