2

我在其他答案中阅读了如何在运行时更改 linq 查询的参数。但是是否可以在创建查询后更改查询浏览的数据源?也许可以为查询提供一个空包装器,在该包装器中可以插入和拔出数据源,而查询不知道?

例如,让我们假设这种情况:

// d1 is a dictionary
var keys =  from entry in d1
            where entry.Value < 5
            select entry.Key;

现在,让我们假设我希望 myQuery 保持不变,而不是修改 d1,除了我希望它处理一个全新的字典 d2。

这样做的原因是我想将谁提供查询与谁提供数据源分离。即认为字典是与服务相关的元数据,而查询是服务消费者发现哪组服务符合他们的标准的方法。我需要对每个服务的元数据应用相同的查询。

我想我正在(可能是错误地)制作的一个平行是使用正则表达式:可以编译一个正则表达式,然后将其应用于任何字符串。我想对查询和字典做同样的事情。

4

1 回答 1

4

LINQ 查询实际上只是一个方法调用。因此,您不能在设置后“重新分配”已调用方法的对象。

您可以通过创建一个包装类来模拟这一点,该类允许您更改实际枚举的内容,即:

public class MutatingSource<T> : IEnumerable<T>
{
     public MutatingSource(IEnumerable<T> originalSource)
     {
         this.Source = originalSource;
     }

     public IEnumerable<T> Source { get; set; }

     IEnumerator IEnumerable.GetEnumerator()
     {
         return this.GetEnumerator();
     }

     public IEnumerator<T> GetEnumerator()
     {
        return Source.GetEnumerator();
     }
}

这将允许您创建查询,然后在事后“改变主意”,即:

var someStrings = new List<string> { "One", "Two", "Three" };

var source = new MutatingSource<string>(someStrings);

// Build the query
var query = source.Where(i => i.Length < 4);

source.Source = new[] {"Foo", "Bar", "Baz", "Uh oh"};

foreach(var item in query)
    Console.WriteLine(item);

这将打印Foo, Bar, 和Baz(来自“更改的”源项目)。


根据评论/编辑进行编辑:

我想我正在(可能是错误地)制作的一个平行是使用正则表达式:可以编译一个正则表达式,然后将其应用于任何字符串。我想对查询和字典做同样的事情。

在这种情况下,查询不像正则表达式。查询被直接翻译成一系列方法调用,这些方法调用只对底层源起作用。因此,您无法更改该源(没有“查询对象”,只有方法调用的返回值)。

更好的方法是将查询移动到一个方法中,即:

IEnumerable<TKey> QueryDictionary<TKey,TValue>(IDictionary<TKey,TValue> dictionary)
{
    var keys =  from entry in dictionary
        where entry.Value < 5
        select entry.Key;
    return keys;
}

然后,您可以根据需要使用它:

var query1 = QueryDictionary(d1);
var query2 = QueryDictionary(d2);
于 2013-10-10T22:27:38.883 回答