次の構造の CSV ファイルがあるとします。
LINE1: ID,Description,Value
LINE2: 1,Product1,2
LINE3: ,,3
LINE4: ,,4
LINE5: 2,Product2,2
LINE6: ,,3
LINE7: ,,5
対応するFileHelpers
定義クラスで
[DelimitedRecord(",") ]
[IgnoreFirst(1)]
public class Product
{
public int ID { get; set; }
public string Description { get; set; }
[FieldConverter(ConverterKind.Decimal)]
public decimal Val{ get; set; }
}
FileHelpersを使用して、各オブジェクトが完全に読み込まれるようにオブジェクトを返すにはどうすればよいですか?
はっきりさせておきますが、現在、使用の結果はengine.ReadFile
行 2 と行 5 から完全に入力されたオブジェクトのみを生成します。これは、フィールドが空白であるため意味があります。ただし、3 行目と 4 行目のオブジェクトも1 行目と同じID
とを持つ必要があります。各行のフィールドはそのままにしておく必要があります。データの正しい用語はわかりませんが、「部分的に正規化」されているようです。Description
Value
これを解決するための現在のアプローチは、AfterRead
イベントを使用して次のとおりです。イベント、イベント ハンドラー、デリゲートなどについてよく理解していないため、FileHelpers を使用してこれを実現する別の方法を探しています。また、非常に不格好に見えます。
// call the csv repo using the ffg lines within a console app
var repo2 = new CSVRepositoryBase<Product>();
repo2.AfterRead +=new AfterReadHandler<object>(repo2_AfterRead);
var products = repo2.Read("some file path");
public static Product PreviousRecord { get; set; }
private static void repo2_AfterRead
(EngineBase engine, FileHelpers.Events.AfterReadEventArgs<object> e)
{
var record = (Product)e.Record;
if (PreviousRecord == null)
{
PreviousRecord = new Product();
PreviousRecord.ID = record.ID;
}
if (String.IsNullOrEmpty(record.ID)) // newline record = blank row
{
record.ID = PreviousRecord.ID;
}
PreviousRecord.ID = record.ID;
}
public class CSVRepositoryBase<T> where T : class
{
// shorten for brevity
public IEnumerable<T> Read(string fileName){/*snip*/ }
#region Events
public event AfterReadHandler<object> AfterRead
{
add { engine.AfterReadRecord += value; }
remove { engine.AfterReadRecord -= value; }
}
#endregion
}