web-dev-qa-db-ja.com

データリーダーを動的クエリ結果に変換する方法

通常、WebMatrixクエリ(IEnumerable<dynamic>データ型)からクエリ結果を取得し、その結果をテーブルに表示するビューがあります。

@model MySite.Models.Entity
@foreach(var row in Model.Data)
{
    <tr>
        @foreach (var column in row.Columns)
        {
            <td>@column<span>:</span> @row[column]</td>
        }
    </tr>
}

データベースにクエリを実行するモデルは次のとおりです。

public class Entity
{
    public dynamic Data {get; set; }
    public Entity(String table)
    {
        if (table == "User" || table == "Group)
        {
            WebMatrix.Data.Database db = new WebMatrix.Data.Database();
            db.Open(ConString);
            Data = db.Query("SELECT * FROM " + table);
        }
        else
        {
            using (OdbcConnection con = ne4w OdbcConnection(ConString))
            {
                OdbcCommand com = new OdbcCommand("Select * From " + table);
                command.CommandType = System.Data.CommandType.Text;
                connection.Open();
                OdbcDataReader reader = command.ExecuteReader();

他のさまざまな投稿を読んで私が試したさまざまなことは次のとおりです。

                // Atempt 1
                Data = reader;
                // Error in view, 'Invalid attempt to call FieldCount when reader is closed' (on 'var row `in` Model.Data')

                // Atempt 2
                Data = reader.Cast<dynamic>;
                // Error: 'Cannot convert method group "Cast" to non-delegate type "dynamic". Did you intend to invoke the method?

                // Atempt 3
                Data = reader.Cast<IEnumerable<dynamic>>;
                // Error same as Atempt 2

                // Atempt 4
                Data = reader.Cast<IEnumerable<string>>;
                // Error same as Atempt 2
            }
        }
    }
}

リーダーオブジェクトをIEnumerable<dynamic>オブジェクトに移動するための最良の方法を探しています。

これは単純化された例であり、2つのクエリタイプの理由は明らかではありませんが、私のコードでは必要であることに注意してください。

11
Travis Heeter

基本的なC#構文がありません。

Data = reader;
// You cant do this. You have to loop the reader to get the values from it.
// If you simply assign reader object itself as the data you wont be 
// able to get data once the reader or connection is closed. 
// The reader is typically closed in the method.

Data = reader.Cast<dynamic>;
// You should call the Cast method. And preferably execute the resulting query. 
// As of now you're merely assigning method reference to a variable
// which is not what you want. 
// Also bear in mind that, as I said before there's no real benefit in casting to dynamic

Data = reader.Cast<IEnumerable<dynamic>>;
// Cast method itself returns an IEnumerable. 
// You dont have to cast individual rows to IEnumerable

Data = reader.Cast<IEnumerable<string>>;
// Meaningless I believe. 
// The data you get from database is not always strings

あなたが犯す主な間違いはメソッドを呼び出さないことです。これはあなたが望むものです:

Data = reader.Cast<IDataRecord>().ToList();
                               ^^ // notice the opening and closing parentheses

これは、処理が簡単なもの(たとえば、フロントエンドで表示する方法)に応じて、さまざまな方法で実行できます。

  1. データレコードを返します。

    public IEnumerable<IDataRecord> SelectDataRecord()
    {
        ....
    
        using (var reader = cmd.ExecuteReader())
            foreach (IDataRecord record in reader as IEnumerable)
                yield return record; //yield return to keep the reader open
    }
    
  2. ExpandoObjectsを返します。 おそらくこれはあなたが望んでいたことですか?

    public IEnumerable<dynamic> SelectDynamic()
    {
        ....
    
        using (var reader = cmd.ExecuteReader())
        {
            var names = Enumerable.Range(0, reader.FieldCount).Select(reader.GetName).ToList();
            foreach (IDataRecord record in reader as IEnumerable)
            {
                var expando = new ExpandoObject() as IDictionary<string, object>;
                foreach (var name in names)
                    expando[name] = record[name];
    
                yield return expando;
            }
        }
    }
    
  3. プロパティバッグの返却順序

    public IEnumerable<Dictionary<string, object>> SelectDictionary()
    {
        ....
    
        using (var reader = cmd.ExecuteReader())
        {
            var names = Enumerable.Range(0, reader.FieldCount).Select(reader.GetName).ToList();
            foreach (IDataRecord record in reader as IEnumerable)
                yield return names.ToDictionary(n => n, n => record[n]);
        }
    }
    
  4. プレーンオブジェクト配列のシーケンスを返す

    public IEnumerable<List<object>> SelectObjectArray()
    {
        ....
    
        using (var reader = cmd.ExecuteReader())
        {
            var indices = Enumerable.Range(0, reader.FieldCount).ToList();
            foreach (IDataRecord record in reader as IEnumerable)
                yield return indices.Select(i => record[i]).ToList();
        }
    }
    
  5. データ行を返す

    public IEnumerable<DataRow> SelectDataRow()
    {
        ....
    
        using (var reader = cmd.ExecuteReader())
        {
            var table = new DataTable();
            table.BeginLoadData();
            table.Load(reader);
            table.EndLoadData();
            return table.AsEnumerable(); // in Assembly: System.Data.DataSetExtensions
        }
    }
    
  6. 大事なことを言い忘れましたが、それが役立つ場合は、手動で配管することなく、強く型付けされたシーケンスを返すことができます。式ツリーを使用して、実行時にコードをコンパイルできます。たとえば、 this を参照してください。

37
nawfal

リーダーの結果をループしてみてください。

OdbcDataReader reader = command.ExecuteReader();

while(reader.Read())
{
    var item = reader["yourField"].ToString();
}

Command.ExecuteReader();の実行されていないIQueriable結果にオブジェクトを設定しようとしているようです。

0