24

インデクサーの拡張メソッド、それらは良いでしょうか?

私はPOCOを再水和するいくつかのコードで遊んでいました。

このコードは、SqlDataReaderから返された行を繰り返し処理し、リフレクションを使用して列の値からプロパティを割り当てます。コールスタックの下に、次のようなコードがありました:-

poco.Set("Surname", "Smith"); // uses extension method ...

Setメソッドは拡張メソッドとして作成されました。

このようなコードを書くことができたら素晴らしいと思います

poco["Surname"] = "Smith";  // extension methods for indexers ?

つまり、インデクサーの拡張メソッドを書きたかったのです。

.Netにインデクサーの拡張メソッドがない理由はありますか?他の人は拡張メソッドインデクサーの他の良い使い方を持っていますか?

余談ですが...インデクサーの拡張メソッドを記述できれば、次のようなコードを記述できます…</ p>

var poco = PocoFactory();  
    poco.Surname = “Smith”; // is this JavaScript ...
    poco[Surname] = “Smith” ; // … or is this c# or both

私のコードからのいくつかのスニペット

/////////////////////////////////////////////
// Client calling code
IDab dab = DabFactory.Create( "Northwind" );
string sql = @"select * from Customers ";
var persons = dab.ExecuteReader<NorthwindCustomer>(sql);
if (dab != null{
   Assert.That(persons[0].CustomerID , Is.EqualTo("ALFKI"));}
/////////////////////////////////////////////
List<T> IDab.ExecuteReader<T>(string commandText) 
{
    List<T> pocos = new List<T>();
    // setup connection
    SqlDataReader reader = command.ExecuteReader(CommandBehavior.CloseConnection);
    while (reader.Read())
    {
            Dictionary<string, int> colMappings = null ;
            if (colMappings == null){
                colMappings = reader.GetSqlDataReaderColumnMappings();}
            T poco = new T();
            poco.DbToMem<T>(reader, colMappings);
            pocos.Add(poco);
        }
    }
    // connection cleanup ...
    return pocos ;
}

// the set extension method signature
public static void Set<T>(this T thisClientObject, string thisPropertyName, object newValue) where T : class
4

2 に答える 2

12

インデクサーはプロパティと多くの共通点を共有しており(内部的には、インデクサーインデックスを持つプロパティです)、拡張プロパティは存在しません。同意すると、それらが便利なシナリオがあります。

提示されたシナリオに関して-いくつかの点で、これは非常に似ていdynamicます。もちろん、文字列インデクサーを持つインターフェイスを宣言すると、リーダーコードで直接使用できますが、このインターフェイスを繰り返し実装するのは多くの不要な作業になります。

拡張メソッドについて; これは鏡面反射を使用しますか?のようなトリックを検討することをお勧めしますHyperDescriptor。これを大量に実行している場合は、CPU時間を大幅に節約できる可能性があります。その場合の一般的な使用法は次のとおりです。

PropertyDescriptorCollection props = TypeDescriptor.GetProperties(typeof(T));
while (reader.Read())
{
     T poco = new T();
     // abbreviated...
     (per prop)
        props[propName].SetValue(poco, cellValue);
}

最初に返された列を確認し(行ごとではなくグリッドごとに1回)、一致した列にのみアクセスすることで、これをさらに最適化できます...

または、ORMツールを見てください。Expressionデータの読み取りにも使用できます(DbLinqの場合、usenetのどこかにこの完全な例があります)

于 2009-03-01T15:50:44.610 に答える