0

我有一个用管道(|)分隔的 csv 文件。我正在使用以下代码行阅读它:

IEnumerable<string[]> lineFields = File.ReadAllLines(FilePath).Select(line => line.Split('|'));

现在,我需要将它绑定到一个GridView. 所以我正在创建一个动态DataTable如下:

DataTable dt = new DataTable();
int i = 0;
foreach (string[] order in lineFields)
{
    if (i == 0)
    {                        
        foreach (string column in order)
        {
            DataColumn _Column = new DataColumn();
            _Column.ColumnName = column;
            dt.Columns.Add(_Column);
            i++;
            //Response.Write(column);
            //Response.Write("\t");
        }
    }
    else
    {
        int j = 0;
        DataRow row = dt.NewRow();
        foreach (string value in order)
        {
            row[j] = value;                            
            j++;

            //Response.Write(column);
            //Response.Write("\t");
        }
        dt.Rows.Add(row);
    }
    //Response.Write("\n");
}

这工作正常。但我想知道是否有更好的方法来转换IEnumerable<string[]>DataTable. 我需要阅读很多这样的 CSV,所以我认为上面的代码可能存在性能问题。

4

2 回答 2

2

从.Net 4开始:

使用ReadLines.

DataTable FileToDataTable(string FilePath)
{

    var dt = new DataTable();

    IEnumerable<string[]> lineFields = File.ReadLines(FilePath).Select(line => line.Split('|'));
    dt.Columns.AddRange(lineFields.First().Select(i => new DataColumn(i)).ToArray());

    foreach (var order in lineFields.Skip(1))
        dt.Rows.Add(order);

    return dt;
}

编辑:而不是这个代码,使用@Jodrell 答案的代码,这可以防止枚举器的双重收费)。

在.Net 4 之前:

使用流式传输:

DataTable FileToDataTable1(string FilePath)
{
    var dt = new DataTable();

    using (var st = new StreamReader(FilePath))
    {
        // first line procces
        if (st.Peek() >= 0)
        {
            var order = st.ReadLine().Split('|');
            dt.Columns.AddRange(order.Select(i => new DataColumn(i)).ToArray());
        }

        while (st.Peek() >= 0)
            dt.Rows.Add(st.ReadLine().Split('|'));
    }
    return dt;
}
于 2014-11-13T10:03:28.040 回答
1

因为,在您的链接示例中,该文件有一个标题行。

const char Delimiter = '|';

var dt = new DataTable;
using (var m = File.ReadLines(filePath).GetEnumerator())
{
    m.MoveNext();
    foreach (var name in m.Current.Split(Delimiter))
    {
        dt.Columns.Add(name);
    }

    while (m.MoveNext())
    {
        dt.Rows.Add(m.Current.Split(Delimiter));
    }
}

这将一次性读取文件。

于 2014-11-13T10:25:25.797 回答