12

テーブルに多対多の関係を使用しています。

クエリがあります:

var query = from post in context.Posts
        from tag in post.Tags where tag.TagId == 10
        select post;

わかりました、うまくいきます。id で指定されたタグの付いた投稿を取得します。

タグ ID のコレクションがあります。そして、コレクション内のすべてのタグを持つ投稿を取得したいと考えています。

私は次の方法を試します:

var tagIds = new int[]{1, 3, 7, 23, 56};

var query = from post in context.Posts
        from tag in post.Tags where tagIds.Contains( tag.TagId )
        select post;

うまくいきません。クエリは、指定されたタグのいずれかを持つすべての投稿を返します。

このような節を取得したいのですが、コレクション内の任意の数のタグに対して動的に:

post.Tags.Whare(x => x.TagId = 1 && x.TagId = 3 && x.TagId = 7 && ... )
4

4 に答える 4

31

外側のクエリで各投稿のタグを射影するべきではありません。むしろ、外部フィルターのチェックを実行する内部クエリを使用する必要があります。(SQL では、これを相関サブクエリと呼んでいました。)

var query = 
    from post in context.Posts
    where post.Tags.All(tag => tagIds.Contains(tag.TagId))
    select post;

代替構文:

var query = 
    context.Posts.Where(post =>
        post.Tags.All(tag => 
            tagIds.Contains(tag.TagId)));

編集スラウマの説明に従って修正。以下のバージョンは、少なくともtagIdsコレクション内のすべてのタグを含む投稿を返します。

var query = 
    from post in context.Posts
    where tagIds.All(requiredId => post.Tags.Any(tag => tag.TagId == requiredId))
    select post;

代替構文:

var query = 
    context.Posts.Where(post => 
        tagIds.All(requiredId => 
            post.Tags.Any(tag =>
                tag.TagId == requiredId)));

編集2:Slaumaごとに上記を修正。また、以下のクエリ構文をフルに活用する別の代替手段も含まれています。

// Project posts from context for which
// no Ids from tagIds are not matched
// by any tags from post
var query =
    from post in context.Posts
    where
    ( 
        // Project Ids from tagIds that are
        // not matched by any tags from post
        from requiredId in tagIds
        where
        (
            // Project tags from post that match requiredId
            from tag in post.Tags
            where tag.TagId == requiredId
            select tag
        ).Any() == false
        select requiredId 
    ).Any() == false
    select post;

Transact-SQL で演算子.Any() == falseをシミュレートするために使用しました。NOT EXISTS

于 2012-05-08T20:02:00.267 に答える
4

これは実際には非常に簡単です。

var tags = context.Posts.Where(post => post.Tags.All(tag => tagIds.Contains(tag)));
于 2012-05-08T20:01:01.293 に答える
4

別のオプションは、タグのコレクションに指定したセットのみを含め、他を含めないようにする場合は、2 つのリストを交差させることです。

var query = from post in context.Posts
  let tags = post.Tags.Select(x => x.Id).ToList()
  where tags.Intersect(tagIds).Count() == tags.Length
  select post;
于 2012-05-08T20:04:52.247 に答える
0

で試してみてくださいAny

var query = from post in context.Posts
    from tag in post.Tags where tagIds.Any(t => t == tag.TagId )
    select post;
于 2012-05-08T20:01:04.197 に答える