3

次のような入力があります。"Steve Brian McFistycuffs Johnson"

データベースを調べて、それらの単語を含むすべてのレコードを任意の順序で返したいと思います。

私がすることは、次のようなものです

var matches = myContext.MyTable.Where( c => c.Name.Contains("Steve") && c.Name.Contains("Brian") && c.Name.Contains("McFistycuffs") && c.Name.Conains("Johnson"));

ただし、これを動的に行う必要があり、単語数が変わる可能性があります。

また、何を選択しても、エンティティ フレームワークによる SQL ステートメントの作成に影響を与えないようにしたいと考えています。可能な限り最適化するようにしたいと思います。これは、ラムダ内である種の foreach を実行しても、エンティティ フレームワークが SQL に簡単にマップできないため、おそらく機能しないことを意味します。

助言がありますか?

4

6 に答える 6

5

何かのようなもの:

static void Main()
{
    string[] words = {"abc", "def"};
    var filter = ContainsAllWords<Person>(x => x.Name, words);
    // same as: x => x.Name.Contains("abc") && x.Name.Contains("def")

    // then:
    var rows = myContext.MyTable.Where(filter);
}
static Expression<Func<T,bool>> ContainsAllWords<T>(
    Expression<Func<T, string>> selector, string[] words)
{
    if (words == null || words.Length == 0) return x => true;

    Expression body = Expression.Call(selector.Body, "Contains", null,
        Expression.Constant(words[0], typeof(string)));
    for (int i = 1; i < words.Length; i++)
    {
        body = Expression.AndAlso(body,
            Expression.Call(selector.Body, "Contains", null,
                Expression.Constant(words[i], typeof(string))));
    }
    return Expression.Lambda<Func<T, bool>>(body, selector.Parameters);
}
于 2013-05-29T09:41:40.490 に答える
3
var list = new[] { "Brian", "Steve", "Johnson" };
var x = myContext.MyTable.Where(c => list.All(p => c.Name.Contains(p)))
于 2013-05-29T09:39:53.387 に答える
1
var arr = new string[] {"Steve","Brian","McFistycuffs","Johnson"}; 
var matches = myContext.MyTable.Where(c =>arr.All(ar => ar==c.Name));
于 2013-05-29T09:41:25.610 に答える
0

勝手な推測ですが、正規表現を使ってみてはどうでしょうか?

(コードはまだテストされていません)

var input = "Steve Brian McFistycuffs Johnson";
var re = new Regex(input.Replace(" ","|"));

var matches = myContext.MyTable.Where( c => re.IsMatch(c.Name));
于 2013-05-29T09:41:07.347 に答える