Groovy 1.8 :: LINQ应用

时间:2011-06-06 13:53:16

标签: linq groovy dsl

更新8/31/2011
Guillaume Laforge几乎完成了它:
    http://gaelyk.appspot.com/tutorial/app-engine-shortcuts#query

看起来他正在进行AST转换以实现:

alias as Entity

位。很酷的东西,Groovy 1.8 + AST转换= JVM上的LINQ-esque查询。 GL的解决方案需要更多的工作,因为我可以看到完全查询功能(如子查询,使用(字段)语法等连接),但他的Gaelyk项目显然没有必要。

修改

作为实现纯LINQ语法的一种解决方法,我决定使用别名来解决这些问题。这不是一个大问题,并且消除了可能需要复杂的AST转换才能实现的主要障碍。

所以,而不是:

from   c as Composite
join   t as Teams
...

我现在定义了别名(注意:需要强制转换才能在字段上自动完成):

def(Teams t,Composite c,Schools s) = [Teams.new(),Composite.new(),Schools.new()]

并使用map,join等的map语法

from    c:Composite
join    t:Teams
...

要解决问题#2(请参阅下面的原文),请将实例级别的getProperty方法添加到每个pogo别名(其范围仅限于调用它的ORM闭包,不错)。我们只是在构建一个sql语句时返回字符串属性名称。

[t,c,s].each{Object o-> o.metaClass.getProperty = { String k-> k } }

取得“好”进展; - )

现在要弄清楚该怎么做“=”,这很棘手,因为set属性是无效的。可能必须使用eq,neq,gt等,但是真的更喜欢文字符号,这使得更接近sql的可读性。

如果有兴趣,LINQ在幕后做了很多工作。 Jon Skeet(赞美他的名字)有一个很好的回复: How LINQ works internally?

ORIGINAL

一直在检查LINQ,给人留下了深刻的印象。

// LINQ example
var games =
    from t in Teams
    from g in t.Games
    where g.gameID = 212
    select new { g.gameDate,g.gameTime };


// Seeking Groovy Nirvana
latest { Integer teamID->
    from   c as Composite
    join   t as Teams
    join   s as Schools on ( schoolID = {
                     from   Teams
                     where  t.schoolID = s.schoolID } )

    where   t.teamID = "$teamID"
    select  c.location, c.gameType, s.schoolName
    group   c.gameID
    order   c.gameDate, c.gameTime
}

建议的Groovy版本编译得很好,如果我使用相应的POGO定义别名c,t,s,我会在字段上获得强类型的IDE auocomplete,很好。但是,在LINQ附近,除了查询本身之外没有(可见的)变量定义,完全自包含且强类型,哇。

好的,可以在Groovy中完成吗?我想(希望)是的,但我挂了2个问题:

1)如何在没有def'ing的情况下隐式填充别名变量?目前我重写了String上的asType()所以在“from c as Composite”中,c被强制转换为Composite。很好,但IDE“认为”在闭包范围内未定义c是一个字符串,因此在POGO字段上没有自动完成; - (

2)由于#1没有解决,我按照上面的方法定义了别名,所以我可以获得自动完成功能。很好,被黑客攻击(与LINQ相比),但确实如此。这里的问题是在“select c.location,c.gameType ...”中,我希望不对字段进行求值,而只是将“c.location”返回到ORM select方法,而不是null(这是它的默认值)。 getProperty()应该在这里工作,但我需要它只在从ORM范围调用时应用于pogo字段(例如orm字段特定的方法,如select,order,group等)。位丢失,也许有一种方法来注释orm方法,或者只通过orm方法调用调用“特殊”pogo getProperty(这是上面的nirvana查询中的闭包代理)。

应该指出我不打算为Groovy创建一个全面的LINQ,但是这个LINQ的一个特定子集我很乐意看到。

1 个答案:

答案 0 :(得分:0)

Guillaume使用AST变换的最大原因之一是“=”的问题。即使您在Groovy中使用==进行比较,也不是从为其调用的compareTo方法中使用==,您无法区分==,!=,< =,> =,<,> ;。在讨论的Groovy的更高版本中有两种可能的路径。一种是用于每种方法比较一种不同的方法,另一种方法是存储一个可以在运行时访问的最小AST。这是C#的方向,是一个非常强大的工具。问题更多的是如何有效地做到这一点。