12

Entity Framework 4 を使用して、メンバー名のコレクションに基づいて動的な並べ替えを実装しようとしています。基本的に、ユーザーは並べ替えるフィールドと並べ替えの順序を選択できます。式ツリーの例を見てきましたが、これをつなぎ合わせることができません。詳細は次のとおりです。

列名のコレクション:

public List<string> sortColumns;
sortColumns = new List<string>();

/// Example subset of video fields.  The collection will vary.
sortColumns.Add("Width");
sortColumns.Add("Height");
sortColumns.Add("Duration");
sortColumns.Add("Title");

ビデオ クラスは次のように定義されます。

public class Video
{
    public string Title { get; set; }
    public int Width { get; set; }
    public int Height { get; set; }
    public float Duration { get; set; }
    public string Filename { get; set; }
    public DateTime DateCreated { get; set; }
    .
    .
    .
}
public List<Video> Videos;

私がやりたいことは、実行時に式ツリーを構築するために sortColumns コレクションを列挙することです。また、ユーザーは昇順または降順の並べ替えを指定でき、式ツリーはいずれかを処理する必要があります。

VS 2008 用の Dynamic LINQ ライブラリを試しましたが、VS 2010 では動作しないようです (何か間違っている可能性があります)。

要するに、ユーザー入力に基づいて Videos コレクションを動的に並べ替える式ツリーが必要です。どんな助けでも大歓迎です。

4

2 に答える 2

21

OrderByまず、@Slace がここに書いた拡張メソッドが必要です。すばらしいコードと、ソリューションの最も難しい部分を提供してくれた Slace のおかげですあなたの特定の状況で動作するように、わずかな変更を加えました。

public static class QueryableExtensions
{
    public static IQueryable<T> OrderBy<T>(this IQueryable<T> source, string sortProperty, ListSortDirection sortOrder)
    {
        var type = typeof(T);
        var property = type.GetProperty(sortProperty);
        var parameter = Expression.Parameter(type, "p");
        var propertyAccess = Expression.MakeMemberAccess(parameter, property);
        var orderByExp = Expression.Lambda(propertyAccess, parameter);
        var typeArguments = new Type[] { type, property.PropertyType };
        var methodName = sortOrder == ListSortDirection.Ascending ? "OrderBy" : "OrderByDescending";
        var resultExp = Expression.Call(typeof(Queryable), methodName, typeArguments, source.Expression, Expression.Quote(orderByExp));

        return source.Provider.CreateQuery<T>(resultExp);
    }
}

リストをソートするメソッドを作成します。以下の方法で注意すべき点がいくつかあります。

  1. 演算子は式ツリーを取らないため、はList<string>に変換されます。IQueryable<string>Enumerable
  2. このメソッドは、並べ替え列のリストを逆の順序で反復処理します (リストの最初の項目に最高の並べ替え優先度を与えると仮定します)。

.

private void PrintVideoList(IEnumerable<string> sortColumns, ListSortDirection sortOrder)
{
    var videos = this.GetVideos();
    var sortedVideos = videos.AsQueryable();

    foreach (var sortColumn in sortColumns.Reverse())
    {
        sortedVideos = sortedVideos.OrderBy(sortColumn, sortOrder);
    }

    // Test the results
    foreach (var video in sortedVideos)
    {
        Console.WriteLine(video.Title);
    }
}

その後、次のようにメソッドを使用できるはずです。

// These values are entered by the user
var sortColumns = new List<string> { "Width", "Title", "Height" };
var sortOrder = ListSortDirection.Ascending;

// Print the video list base on the user selection
this.PrintVideoList(sortColumns, sortOrder);
于 2012-07-05T03:17:22.763 に答える