主頁 > .NET開發 > C#中的CSV檔案讀寫

C#中的CSV檔案讀寫

2022-06-15 06:26:18 .NET開發

目錄
  • CSV檔案標準
    • 檔案示例
    • RFC 4180
    • 簡化標準
  • 讀寫CSV檔案
    • 使用CsvHelper
    • 使用自定義方法
      • 基于簡化標準的寫CSV檔案
      • 使用TextFieldParser決議CSV檔案
      • 使用正則運算式決議CSV檔案
      • 完整的CsvFile工具類
  • 總結
  • 附錄

專案中經常遇到CSV檔案的讀寫需求,其中的難點主要是CSV檔案的決議,本文會介紹CsvHelperTextFieldParser正則運算式三種決議CSV檔案的方法,順帶也會介紹一下CSV檔案的寫方法,

CSV檔案標準

在介紹CSV檔案的讀寫方法前,我們需要了解一下CSV檔案的格式,

檔案示例

一個簡單的CSV檔案:

Test1,Test2,Test3,Test4,Test5,Test6
str1,str2,str3,str4,str5,str6
str1,str2,str3,str4,str5,str6

一個不簡單的CSV檔案:

"Test1
"",""","Test2
"",""","Test3
"",""","Test4
"",""","Test5
"",""","Test6
"","""
" 中文,D23 ","3DFD4234""""""1232""1S2","ASD1"",""23,,,,213
23F32","
",,asd
" 中文,D23 ","3DFD4234""""""1232""1S2","ASD1"",""23,,,,213
23F32","
",,asd

你沒看錯,上面兩個都是CSV檔案,都只有3行CSV資料,第二個檔案多看一眼都是精神污染,但專案中無法避免會出現這種檔案,

RFC 4180

CSV檔案沒有官方的標準,但一般專案都會遵守 RFC 4180 標準,這是一個非官方的標準,內容如下:

  1. Each record is located on a separate line, delimited by a line break (CRLF).
  2. The last record in the file may or may not have an ending line break.
  3. There maybe an optional header line appearing as the first line of the file with the same format as normal record lines. This header will contain names corresponding to the fields in the file and should contain the same number of fields as the records in the rest of the file (the presence or absence of the header line should be indicated via the optional "header" parameter of this MIME type).
  4. Within the header and each record, there may be one or more fields, separated by commas. Each line should contain the same number of fields throughout the file. Spaces are considered part of a field and should not be ignored. The last field in the record must not be followed by a comma.
  5. Each field may or may not be enclosed in double quotes (however some programs, such as Microsoft Excel, do not use double quotes at all). If fields are not enclosed with double quotes, then double quotes may not appear inside the fields.
  6. Fields containing line breaks (CRLF), double quotes, and commas should be enclosed in double-quotes.
  7. If double-quotes are used to enclose fields, then a double-quote appearing inside a field must be escaped by preceding it with another double quote.

翻譯一下:

  1. 每條記錄位于單獨的行上,由換行符 (CRLF) 分隔,
  2. 檔案中的最后一條記錄可能有也可能沒有結束換行符,
  3. 可能有一個可選的標題行出現在檔案的第一行,格式與普通記錄行相同,此標題將包含與檔案中的欄位對應的名稱,并且應包含與檔案其余部分中的記錄相同數量的欄位(標題行的存在或不存在應通過此 MIME 型別的可選“標頭”引數指示),
  4. 在標題和每條記錄中,可能有一個或多個欄位,以逗號分隔,在整個檔案中,每行應包含相同數量的欄位,空格被視為欄位的一部分,不應忽略,記錄中的最后一個欄位后面不能有逗號,
  5. 每個欄位可以用雙引號括起來,也可以不用雙引號(但是某些程式,例如 Microsoft Excel,根本不使用雙引號),如果欄位沒有用雙引號括起來,那么雙引號可能不會出現在欄位內,
  6. 包含換行符 (CRLF)、雙引號和逗號的欄位應該用雙引號括起來,
  7. 如果使用雙引號將欄位括起來,則出現在欄位中的雙引號必須在其前面加上另一個雙引號,

簡化標準

上面的標準可能比較拗口,我們對它進行一些簡化,要注意一下,簡化不是簡單的刪減規則,而是將類似的類似進行合并便于理解,
后面的代碼也會使用簡化標準,簡化標準如下:

  1. 每條記錄位于單獨的行上,由換行符 (CRLF) 分隔,
    注:此處的行不是普通文本意義上的行,是指符合CSV檔案格式的一條記錄(后面簡稱為CSV行),在文本上可能占據多行,

  2. 檔案中的最后一條記錄需有結束換行符,檔案的第一行為標題行(標題行包含欄位對應的名稱,標題數與記錄的欄位數相同),
    注:原標準中可有可無的選項統一規定為必須有,方便后期的決議,而且沒有標題行讓別人怎么看資料,

  3. 在標題和每條記錄中,可能有一個或多個欄位,以逗號分隔,在整個檔案中,每行應包含相同數量的欄位空格被視為欄位的一部分,不應忽略,記錄中的最后一個欄位后面不能有逗號
    注:此標準未做簡化,雖然也有其它標準使用空格、制表符等做分割的,但不使用逗號分割的檔案還叫逗號分隔值檔案嗎,

  4. 每個欄位都用雙引號括起來,出現在欄位中的雙引號必須在其前面加上另一個雙引號
    注:原標準有必須使用雙引號和可選雙引號的情況,那全部使用雙引號肯定不會出錯,*

讀寫CSV檔案

在正式讀寫CSV檔案前,我們需要先定義一個用于測驗的Test類,代碼如下:

class Test
{
    public string Test1{get;set;}
    public string Test2 { get; set; }
    public string Test3 { get; set; }
    public string Test4 { get; set; }
    public string Test5 { get; set; }
    public string Test6 { get; set; }

    //Parse方法會在自定義讀寫CSV檔案時用到
    public static Test Parse (string[]fields )
    {
        try
        {
            Test ret = new Test();
            ret.Test1 = fields[0];
            ret.Test2 = fields[1];
            ret.Test3 = fields[2];
            ret.Test4 = fields[3];
            ret.Test5 = fields[4];
            ret.Test6 = fields[5];
            return ret;
        }
        catch (Exception)
        {
            //做一些例外處理,寫日志之類的
            return null;
        }
    }
}

生成一些測驗資料,代碼如下:

static void Main(string[] args)
{
    //檔案保存路徑
    string path = "tset.csv";
    //清理之前的測驗檔案
    File.Delete("tset.csv");
      
    Test test = new Test();
    test.Test1 = " 中文,D23 ";
    test.Test2 = "3DFD4234\"\"\"1232\"1S2";
    test.Test3 = "ASD1\",\"23,,,,213\r23F32";
    test.Test4 = "\r";
    test.Test5 = string.Empty;
    test.Test6 = "asd";

    //測驗資料
    var records = new List<Test> { test, test };

    //寫CSV檔案
    /*
    *直接把后面的寫CSV檔案代碼復制到此處
    */

    //讀CSV檔案
     /*
    *直接把后面的讀CSV檔案代碼復制到此處
    */
   
    Console.ReadLine();
}

使用CsvHelper

CsvHelper 是用于讀取和寫入 CSV 檔案的庫,支持自定義類物件的讀寫,
github上標星最高的CSV檔案讀寫C#庫,使用MS-PL、Apache 2.0開源協議,
使用NuGet下載CsvHelper,讀寫CSV檔案的代碼如下:

 //寫CSV檔案
using (var writer = new StreamWriter(path))
using (var csv = new CsvWriter(writer, CultureInfo.InvariantCulture))
{
    csv.WriteRecords(records);
}

using (var writer = new StreamWriter(path,true))
using (var csv = new CsvWriter(writer, CultureInfo.InvariantCulture))
{
    //追加
    foreach (var record in records)
    {
        csv.WriteRecord(record);
    }
}

//讀CSV檔案
using (var reader = new StreamReader(path))
using (var csv = new CsvReader(reader, CultureInfo.InvariantCulture))
{
    records = csv.GetRecords<Test>().ToList();
    //逐行讀取
    //records.Add(csv.GetRecord<Test>());
}

如果你只想要拿來就能用的庫,那文章基本上到這里就結束了,

使用自定義方法

為了與CsvHelper區分,新建一個CsvFile類存放自定義讀寫CSV檔案的代碼,最后會提供類的完整原始碼,CsvFile類定義如下:

/// <summary>
/// CSV檔案讀寫工具類
/// </summary>
public class CsvFile
{
    #region 寫CSV檔案
    //具體代碼...
    #endregion

    #region 讀CSV檔案(使用TextFieldParser)
    //具體代碼...
    #endregion

    #region 讀CSV檔案(使用正則運算式)
    //具體代碼...
    #endregion

}

基于簡化標準的寫CSV檔案

根據簡化標準(具體標準內容見前文),寫CSV檔案代碼如下:

#region 寫CSV檔案
//欄位陣列轉為CSV記錄行
private static string FieldsToLine(IEnumerable<string> fields)
{
    if (fields == null) return string.Empty;
    fields = fields.Select(field =>
    {
        if (field == null) field = string.Empty;
        //簡化標準,所有欄位都加雙引號
        field = string.Format("\"{0}\"", field.Replace("\"", "\"\""));

        //不簡化標準
        //field = field.Replace("\"", "\"\"");
        //if (field.IndexOfAny(new char[] { ',', '"', ' ', '\r' }) != -1)
        //{
        //    field = string.Format("\"{0}\"", field);
        //}
        return field;
    });
    string line = string.Format("{0}{1}", string.Join(",", fields), Environment.NewLine);
    return line;
}

//默認的欄位轉換方法
private static IEnumerable<string> GetObjFields<T>(T obj, bool isTitle) where T : class
{
    IEnumerable<string> fields;
    if (isTitle)
    {
        fields = obj.GetType().GetProperties().Select(pro => pro.Name);
    }
    else
    {
        fields = obj.GetType().GetProperties().Select(pro => pro.GetValue(obj)?.ToString());
    }
    return fields;
}

/// <summary>
/// 寫CSV檔案,默認第一行為標題
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="list">資料串列</param>
/// <param name="path">檔案路徑</param>
/// <param name="append">追加記錄</param>
/// <param name="func">欄位轉換方法</param>
/// <param name="defaultEncoding"></param>
public static void Write<T>(List<T> list, string path,bool append=true, Func<T, bool, IEnumerable<string>> func = null, Encoding defaultEncoding = null) where T : class
{
    if (list == null || list.Count == 0) return;
    if (defaultEncoding == null)
    {
        defaultEncoding = Encoding.UTF8;
    }
    if (func == null)
    {
        func = GetObjFields;
    }
    if (!File.Exists(path)|| !append)
    {
        var fields = func(list[0], true);
        string title = FieldsToLine(fields);
        File.WriteAllText(path, title, defaultEncoding);
    }
    using (StreamWriter sw = new StreamWriter(path, true, defaultEncoding))
    {
        list.ForEach(obj =>
        {
            var fields = func(obj, false);
            string line = FieldsToLine(fields);
            sw.Write(line);
        });
    }
}
#endregion

使用時,代碼如下:

//寫CSV檔案
//使用自定義的欄位轉換方法,也是文章開頭復雜CSV檔案使用欄位轉換方法
CsvFile.Write(records, path, true, new Func<Test, bool, IEnumerable<string>>((obj, isTitle) =>
{
    IEnumerable<string> fields;
    if (isTitle)
    {
        fields = obj.GetType().GetProperties().Select(pro => pro.Name + Environment.NewLine + "\",\"");
    }
    else
    {
        fields = obj.GetType().GetProperties().Select(pro => pro.GetValue(obj)?.ToString());
    }
    return fields;
}));

//使用默認的欄位轉換方法
//CsvFile.Write(records, path);

你也可以使用默認的欄位轉換方法,代碼如下:

CsvFile.Save(records, path);

使用TextFieldParser決議CSV檔案

TextFieldParser是VB中決議CSV檔案的類,C#雖然沒有類似功能的類,不過可以呼叫VB的TextFieldParser來實作功能,
TextFieldParser決議CSV檔案的代碼如下:

#region 讀CSV檔案(使用TextFieldParser)
/// <summary>
/// 讀CSV檔案,默認第一行為標題
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="path">檔案路徑</param>
/// <param name="func">欄位決議規則</param>
/// <param name="defaultEncoding">檔案編碼</param>
/// <returns></returns>
public static List<T> Read<T>(string path, Func<string[], T> func, Encoding defaultEncoding = null) where T : class
{
    if (defaultEncoding == null)
    {
        defaultEncoding = Encoding.UTF8;
    }
    List<T> list = new List<T>();
    using (TextFieldParser parser = new TextFieldParser(path, defaultEncoding))
    {
        parser.TextFieldType = FieldType.Delimited;
        //設定逗號分隔符
        parser.SetDelimiters(",");
        //設定不忽略欄位前后的空格
        parser.TrimWhiteSpace = false;
        bool isLine = false;
        while (!parser.EndOfData)
        {
            string[] fields = parser.ReadFields();
            if (isLine)
            {
                var obj = func(fields);
                if (obj != null) list.Add(obj);
            }
            else
            {
                //忽略標題行業
                isLine = true;
            }
        }
    }
    return list;
}
#endregion

使用時,代碼如下:

//讀CSV檔案
records = CsvFile.Read(path, Test.Parse);

使用正則運算式決議CSV檔案

如果你有一個問題,想用正則運算式來解決,那么你就有兩個問題了,

正則運算式有一定的學習門檻,而且學習后不經常使用就會忘記,正則運算式解決的大多數是一些不易變更需求的問題,這就導致一個穩定可用的正則運算式可以傳好幾代,
本節的正則運算式來自 《精通正則運算式(第3版)》 第6章 打造高效正則運算式——簡單的消除回圈的例子,有興趣的可以去了解一下,運算式說明如下:

注:這本書最終版的決議CSV檔案的正則運算式是Jave版的使用占有優先量詞取代固化分組的版本,也是百度上經常見到的版本,不過占有優先量詞在C#中有點問題,本人能力有限解決不了,所以使用了上圖的版本,不過,這兩版正則運算式性能上沒有差異,

正則運算式決議CSV檔案代碼如下:

#region 讀CSV檔案(使用正則運算式)
/// <summary>
/// 讀CSV檔案,默認第一行為標題
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="path">檔案路徑</param>
/// <param name="func">欄位決議規則</param>
/// <param name="defaultEncoding">檔案編碼</param>
/// <returns></returns>
public static List<T> Read_Regex<T>(string path, Func<string[], T> func, Encoding defaultEncoding = null) where T : class
{
    List<T> list = new List<T>();
    StringBuilder sbr = new StringBuilder(100);
    Regex lineReg = new Regex("\"");
    Regex fieldReg = new Regex("\\G(?:^|,)(?:\"((?>[^\"]*)(?>\"\"[^\"]*)*)\"|([^\",]*))");
    Regex quotesReg = new Regex("\"\"");

    bool isLine = false;
    string line = string.Empty;
    using (StreamReader sr = new StreamReader(path))
    {
        while (null != (line = ReadLine(sr)))
        {
            sbr.Append(line);
            string str = sbr.ToString();
            //一個完整的CSV記錄行,它的雙引號一定是偶數
            if (lineReg.Matches(sbr.ToString()).Count % 2 == 0)
            {
                if (isLine)
                {
                    var fields = ParseCsvLine(sbr.ToString(), fieldReg, quotesReg).ToArray();
                    var obj = func(fields.ToArray());
                    if (obj != null) list.Add(obj);
                }
                else
                {
                    //忽略標題行業
                    isLine = true;
                }
                sbr.Clear();
            }
            else
            {
                sbr.Append(Environment.NewLine);
            }                   
        }
    }
    if (sbr.Length > 0)
    {
        //有決議失敗的字串,報錯或忽略
    }
    return list;
}

//重寫ReadLine方法,只有\r\n才是正確的一行
private static string ReadLine(StreamReader sr) 
{
    StringBuilder sbr = new StringBuilder();
    char c;
    int cInt;
    while (-1 != (cInt =sr.Read()))
    {
        c = (char)cInt;
        if (c == '\n' && sbr.Length > 0 && sbr[sbr.Length - 1] == '\r')
        {
            sbr.Remove(sbr.Length - 1, 1);
            return sbr.ToString();
        }
        else 
        {
            sbr.Append(c);
        }
    }
    return sbr.Length>0?sbr.ToString():null;
}

private static List<string> ParseCsvLine(string line, Regex fieldReg, Regex quotesReg)
{
    var fieldMath = fieldReg.Match(line);
    List<string> fields = new List<string>();
    while (fieldMath.Success)
    {
        string field;
        if (fieldMath.Groups[1].Success)
        {
            field = quotesReg.Replace(fieldMath.Groups[1].Value, "\"");
        }
        else
        {
            field = fieldMath.Groups[2].Value;
        }
        fields.Add(field);
        fieldMath = fieldMath.NextMatch();
    }
    return fields;
}
#endregion

使用時代碼如下:

//讀CSV檔案
records = CsvFile.Read_Regex(path, Test.Parse);

目前還未發現正則運算式決議有什么bug,不過還是不建議使用,

完整的CsvFile工具類

完整的CsvFile類代碼如下:

using Microsoft.VisualBasic.FileIO;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Text.RegularExpressions;


namespace ConsoleApp4
{
    /// <summary>
    /// CSV檔案讀寫工具類
    /// </summary>
    public class CsvFile
    {
        #region 寫CSV檔案
        //欄位陣列轉為CSV記錄行
        private static string FieldsToLine(IEnumerable<string> fields)
        {
            if (fields == null) return string.Empty;
            fields = fields.Select(field =>
            {
                if (field == null) field = string.Empty;
                //所有欄位都加雙引號
                field = string.Format("\"{0}\"", field.Replace("\"", "\"\""));

                //不簡化
                //field = field.Replace("\"", "\"\"");
                //if (field.IndexOfAny(new char[] { ',', '"', ' ', '\r' }) != -1)
                //{
                //    field = string.Format("\"{0}\"", field);
                //}
                return field;
            });
            string line = string.Format("{0}{1}", string.Join(",", fields), Environment.NewLine);
            return line;
        }

        //默認的欄位轉換方法
        private static IEnumerable<string> GetObjFields<T>(T obj, bool isTitle) where T : class
        {
            IEnumerable<string> fields;
            if (isTitle)
            {
                fields = obj.GetType().GetProperties().Select(pro => pro.Name);
            }
            else
            {
                fields = obj.GetType().GetProperties().Select(pro => pro.GetValue(obj)?.ToString());
            }
            return fields;
        }

        /// <summary>
        /// 寫CSV檔案,默認第一行為標題
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="list">資料串列</param>
        /// <param name="path">檔案路徑</param>
        /// <param name="append">追加記錄</param>
        /// <param name="func">欄位轉換方法</param>
        /// <param name="defaultEncoding"></param>
        public static void Write<T>(List<T> list, string path,bool append=true, Func<T, bool, IEnumerable<string>> func = null, Encoding defaultEncoding = null) where T : class
        {
            if (list == null || list.Count == 0) return;
            if (defaultEncoding == null)
            {
                defaultEncoding = Encoding.UTF8;
            }
            if (func == null)
            {
                func = GetObjFields;
            }
            if (!File.Exists(path)|| !append)
            {
                var fields = func(list[0], true);
                string title = FieldsToLine(fields);
                File.WriteAllText(path, title, defaultEncoding);
            }
            using (StreamWriter sw = new StreamWriter(path, true, defaultEncoding))
            {
                list.ForEach(obj =>
                {
                    var fields = func(obj, false);
                    string line = FieldsToLine(fields);
                    sw.Write(line);
                });
            }
        }
        #endregion

        #region 讀CSV檔案(使用TextFieldParser)
        /// <summary>
        /// 讀CSV檔案,默認第一行為標題
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="path">檔案路徑</param>
        /// <param name="func">欄位決議規則</param>
        /// <param name="defaultEncoding">檔案編碼</param>
        /// <returns></returns>
        public static List<T> Read<T>(string path, Func<string[], T> func, Encoding defaultEncoding = null) where T : class
        {
            if (defaultEncoding == null)
            {
                defaultEncoding = Encoding.UTF8;
            }
            List<T> list = new List<T>();
            using (TextFieldParser parser = new TextFieldParser(path, defaultEncoding))
            {
                parser.TextFieldType = FieldType.Delimited;
                //設定逗號分隔符
                parser.SetDelimiters(",");
                //設定不忽略欄位前后的空格
                parser.TrimWhiteSpace = false;
                bool isLine = false;
                while (!parser.EndOfData)
                {
                    string[] fields = parser.ReadFields();
                    if (isLine)
                    {
                        var obj = func(fields);
                        if (obj != null) list.Add(obj);
                    }
                    else
                    {
                        //忽略標題行業
                        isLine = true;
                    }
                }
            }
            return list;
        }
        #endregion

        #region 讀CSV檔案(使用正則運算式)
        /// <summary>
        /// 讀CSV檔案,默認第一行為標題
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="path">檔案路徑</param>
        /// <param name="func">欄位決議規則</param>
        /// <param name="defaultEncoding">檔案編碼</param>
        /// <returns></returns>
        public static List<T> Read_Regex<T>(string path, Func<string[], T> func, Encoding defaultEncoding = null) where T : class
        {
            List<T> list = new List<T>();
            StringBuilder sbr = new StringBuilder(100);
            Regex lineReg = new Regex("\"");
            Regex fieldReg = new Regex("\\G(?:^|,)(?:\"((?>[^\"]*)(?>\"\"[^\"]*)*)\"|([^\",]*))");
            Regex quotesReg = new Regex("\"\"");

            bool isLine = false;
            string line = string.Empty;
            using (StreamReader sr = new StreamReader(path))
            {
                while (null != (line = ReadLine(sr)))
                {
                    sbr.Append(line);
                    string str = sbr.ToString();
                    //一個完整的CSV記錄行,它的雙引號一定是偶數
                    if (lineReg.Matches(sbr.ToString()).Count % 2 == 0)
                    {
                        if (isLine)
                        {
                            var fields = ParseCsvLine(sbr.ToString(), fieldReg, quotesReg).ToArray();
                            var obj = func(fields.ToArray());
                            if (obj != null) list.Add(obj);
                        }
                        else
                        {
                            //忽略標題行業
                            isLine = true;
                        }
                        sbr.Clear();
                    }
                    else
                    {
                        sbr.Append(Environment.NewLine);
                    }                   
                }
            }
            if (sbr.Length > 0)
            {
                //有決議失敗的字串,報錯或忽略
            }
            return list;
        }

        //重寫ReadLine方法,只有\r\n才是正確的一行
        private static string ReadLine(StreamReader sr) 
        {
            StringBuilder sbr = new StringBuilder();
            char c;
            int cInt;
            while (-1 != (cInt =sr.Read()))
            {
                c = (char)cInt;
                if (c == '\n' && sbr.Length > 0 && sbr[sbr.Length - 1] == '\r')
                {
                    sbr.Remove(sbr.Length - 1, 1);
                    return sbr.ToString();
                }
                else 
                {
                    sbr.Append(c);
                }
            }
            return sbr.Length>0?sbr.ToString():null;
        }
       
        private static List<string> ParseCsvLine(string line, Regex fieldReg, Regex quotesReg)
        {
            var fieldMath = fieldReg.Match(line);
            List<string> fields = new List<string>();
            while (fieldMath.Success)
            {
                string field;
                if (fieldMath.Groups[1].Success)
                {
                    field = quotesReg.Replace(fieldMath.Groups[1].Value, "\"");
                }
                else
                {
                    field = fieldMath.Groups[2].Value;
                }
                fields.Add(field);
                fieldMath = fieldMath.NextMatch();
            }
            return fields;
        }
        #endregion

    }
}

使用方法如下:

//寫CSV檔案
CsvFile.Write(records, path, true, new Func<Test, bool, IEnumerable<string>>((obj, isTitle) =>
{
    IEnumerable<string> fields;
    if (isTitle)
    {
        fields = obj.GetType().GetProperties().Select(pro => pro.Name + Environment.NewLine + "\",\"");
    }
    else
    {
        fields = obj.GetType().GetProperties().Select(pro => pro.GetValue(obj)?.ToString());
    }
    return fields;
}));

//讀CSV檔案
records = CsvFile.Read(path, Test.Parse);

//讀CSV檔案
records = CsvFile.Read_Regex(path, Test.Parse);

總結

  • 介紹了CSV檔案的 RFC 4180 標準及其簡化理解版本
  • 介紹了CsvHelperTextFieldParser正則運算式三種決議CSV檔案的方法
  • 專案中推薦使用CsvHelper,如果不想引入太多開源組件可以使用TextFieldParser,不建議使用正則運算式

附錄

  • CsvHelper github鏈接
  • CsvHelper專案備份 提取碼:33j7
  • RFC 4180標準

轉載請註明出處,本文鏈接:https://www.uj5u.com/net/490877.html

標籤:C#

上一篇:使用外鍵時在null上呼叫成員函式comments()

下一篇:【學習筆記】WPF-01:前言

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • WebAPI簡介

    Web體系結構: 有三個核心:資源(resource),URL(統一資源識別符號)和表示 他們的關系是這樣的:一個資源由一個URL進行標識,HTTP客戶端使用URL定位資源,表示是從資源回傳資料,媒體型別是資源回傳的資料格式。 接下來我們說下HTTP. HTTP協議的系統是一種無狀態的方式,使用請求/ ......

    uj5u.com 2020-09-09 22:07:47 more
  • asp.net core 3.1 入口:Program.cs中的Main函式

    本文分析Program.cs 中Main()函式中代碼的運行順序分析asp.net core程式的啟動,重點不是剖析原始碼,而是理清程式開始時執行的順序。到呼叫了哪些實體,哪些法方。asp.net core 3.1 的程式入口在專案Program.cs檔案里,如下。ususing System; us ......

    uj5u.com 2020-09-09 22:07:49 more
  • asp.net網站作為websocket服務端的應用該如何寫

    最近被websocket的一個問題困擾了很久,有一個需求是在web網站中搭建websocket服務。客戶端通過網頁與服務器建立連接,然后服務器根據ip給客戶端網頁發送資訊。 其實,這個需求并不難,只是剛開始對websocket的內容不太了解。上網搜索了一下,有通過asp.net core 實作的、有 ......

    uj5u.com 2020-09-09 22:08:02 more
  • ASP.NET 開源匯入匯出庫Magicodes.IE Docker中使用

    Magicodes.IE在Docker中使用 更新歷史 2019.02.13 【Nuget】版本更新到2.0.2 【匯入】修復單列匯入的Bug,單元測驗“OneColumnImporter_Test”。問題見(https://github.com/dotnetcore/Magicodes.IE/is ......

    uj5u.com 2020-09-09 22:08:05 more
  • 在webform中使用ajax

    如果你用過Asp.net webform, 說明你也算是.NET 開發的老兵了。WEBform應該是2011 2013左右,當時還用visual studio 2005、 visual studio 2008。后來基本都用的是MVC。 如果是新開發的專案,估計沒人會用webform技術。但是有些舊版 ......

    uj5u.com 2020-09-09 22:08:50 more
  • iis添加asp.net網站,訪問提示:由于擴展配置問題而無法提供您請求的

    今天在iis服務器配置asp.net網站,遇到一個問題,記錄一下: 問題:由于擴展配置問題而無法提供您請求的頁面。如果該頁面是腳本,請添加處理程式。如果應下載檔案,請添加 MIME 映射。 WindowServer2012服務器,添加角色安裝完.netframework和iis之后,運行aspx頁面 ......

    uj5u.com 2020-09-09 22:10:00 more
  • WebAPI-處理架構

    帶著問題去思考,大家好! 問題1:HTTP請求和回傳相應的HTTP回應資訊之間發生了什么? 1:首先是最底層,托管層,位于WebAPI和底層HTTP堆疊之間 2:其次是 訊息處理程式管道層,這里比如日志和快取。OWIN的參考是將訊息處理程式管道的一些功能下移到堆疊下端的OWIN中間件了。 3:控制器處理 ......

    uj5u.com 2020-09-09 22:11:13 more
  • 微信門戶開發框架-使用指導說明書

    微信門戶應用管理系統,采用基于 MVC + Bootstrap + Ajax + Enterprise Library的技術路線,界面層采用Boostrap + Metronic組合的前端框架,資料訪問層支持Oracle、SQLServer、MySQL、PostgreSQL等資料庫。框架以MVC5,... ......

    uj5u.com 2020-09-09 22:15:18 more
  • WebAPI-HTTP編程模型

    帶著問題去思考,大家好!它是什么?它包含什么?它能干什么? 訊息 HTTP編程模型的核心就是訊息抽象,表示為:HttPRequestMessage,HttpResponseMessage.用于客戶端和服務端之間交換請求和回應訊息。 HttpMethod類包含了一組靜態屬性: private stat ......

    uj5u.com 2020-09-09 22:15:23 more
  • 部署WebApi隨筆

    一、跨域 NuGet參考Microsoft.AspNet.WebApi.Cors WebApiConfig.cs中配置: // Web API 配置和服務 config.EnableCors(new EnableCorsAttribute("*", "*", "*")); 二、清除默認回傳XML格式 ......

    uj5u.com 2020-09-09 22:15:48 more
最新发布
  • C#多執行緒學習(二) 如何操縱一個執行緒

    <a href="https://www.cnblogs.com/x-zhi/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2943582/20220801082530.png" alt="" /></...

    uj5u.com 2023-04-19 09:17:20 more
  • C#多執行緒學習(二) 如何操縱一個執行緒

    C#多執行緒學習(二) 如何操縱一個執行緒 執行緒學習第一篇:C#多執行緒學習(一) 多執行緒的相關概念 下面我們就動手來創建一個執行緒,使用Thread類創建執行緒時,只需提供執行緒入口即可。(執行緒入口使程式知道該讓這個執行緒干什么事) 在C#中,執行緒入口是通過ThreadStart代理(delegate)來提供的 ......

    uj5u.com 2023-04-19 09:16:49 more
  • 記一次 .NET某醫療器械清洗系統 卡死分析

    <a href="https://www.cnblogs.com/huangxincheng/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/214741/20200614104537.png" alt="" /&g...

    uj5u.com 2023-04-18 08:39:04 more
  • 記一次 .NET某醫療器械清洗系統 卡死分析

    一:背景 1. 講故事 前段時間協助訓練營里的一位朋友分析了一個程式卡死的問題,回過頭來看這個案例比較經典,這篇稍微整理一下供后來者少踩坑吧。 二:WinDbg 分析 1. 為什么會卡死 因為是表單程式,理所當然就是看主執行緒此時正在做什么? 可以用 ~0s ; k 看一下便知。 0:000> k # ......

    uj5u.com 2023-04-18 08:33:10 more
  • SignalR, No Connection with that ID,IIS

    <a href="https://www.cnblogs.com/smartstar/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/u36196.jpg" alt="" /></a>...

    uj5u.com 2023-03-30 17:21:52 more
  • 一次對pool的誤用導致的.net頻繁gc的診斷分析

    <a href="https://www.cnblogs.com/dotnet-diagnostic/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/3115652/20230225090434.png" alt=""...

    uj5u.com 2023-03-28 10:15:33 more
  • 一次對pool的誤用導致的.net頻繁gc的診斷分析

    <a href="https://www.cnblogs.com/dotnet-diagnostic/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/3115652/20230225090434.png" alt=""...

    uj5u.com 2023-03-28 10:13:31 more
  • C#遍歷指定檔案夾中所有檔案的3種方法

    <a href="https://www.cnblogs.com/xbhp/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/957602/20230310105611.png" alt="" /></a&...

    uj5u.com 2023-03-27 14:46:55 more
  • C#/VB.NET:如何將PDF轉為PDF/A

    <a href="https://www.cnblogs.com/Carina-baby/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2859233/20220427162558.png" alt="" />...

    uj5u.com 2023-03-27 14:46:35 more
  • 武裝你的WEBAPI-OData聚合查詢

    <a href="https://www.cnblogs.com/podolski/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/616093/20140323000327.png" alt="" /><...

    uj5u.com 2023-03-27 14:46:16 more