C# CSV文件解析

什么是CSV

逗号分隔值(Comma-Separated Values,CSV),其文件以纯文本形式存储表格数据(数字和文本),文件的每一行都是一个数据记录。每个记录由一个或多个字段组成,用逗号分隔。使用逗号作为字段分隔符是此文件格式的名称的来源,因为分隔字符也可以不是逗号,有时也称为字符分隔值。

CSV广泛用于不同体系结构的应用程序之间交换数据表格信息,解决不兼容数据格式的互通问题,一般按照传输双方既定标准进行格式定义,而其本身并无明确格式标准。

CSV用逗号分隔字段的基本思想是清楚的,但是当字段数据也可能包含逗号或者甚至嵌入换行符时,该想法变得复杂。 CSV实现可能无法处理这些字段数据,或者可能会使用引号来包围字段。引用并不能解决所有问题:有些字段可能需要嵌入引号,因此CSV实现可能包含转义字符或转义序列。

RFC 4180提出了MIME类型(”text/csv”)对于CSV格式的标准,可以作为一般使用的常用定义,满足大多数实现似乎遵循的格式。
CSV的格式规范

下面的格式规范定义来源于RFC 4180,附上原文供参考,一共也就七点。

1. 每一行记录位于一个单独的行上,用回车换行符CRLF(也就是\r\n)分割。

    Each record is located on a separate line, delimited by a line break (CRLF). For example:


2. 文件中的最后一行记录可以有结尾回车换行符,也可以没有。

    The last record in the file may or may not have an ending line break. For example:

 

 

 
3. 第一行可以存在一个可选的标题头,格式和普通记录行的格式一样。标题头要包含文件记录字段对应的名称,应该有和记录字段一样的数量。(在MIME类型中,标题头行的存在与否可以通过MIME type中的可选”header”参数指明)

    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). For example:

 

 

 
4. 在标题头行和普通行每行记录中,会存在一个或多个由半角逗号(,)分隔的字段。整个文件中每行应包含相同数量的字段,空格也是字段的一部分,不应被忽略。每一行记录最后一个字段后不能跟逗号。(通常用逗号分隔,也有其他字符分隔的CSV,需事先约定)

    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. For example:

 

 

 
5. 每个字段可用也可不用半角双引号(“)括起来(不过有些程序,如Microsoft的Excel就根本不用双引号)。如果字段没有用引号括起来,那么该字段内部不能出现双引号字符。

    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. For example:

 

 

 
6. 字段中若包含回车换行符、双引号或者逗号,该字段需要用双引号括起来。

    Fields containing line breaks (CRLF), double quotes, and commas should be enclosed in double-quotes. For example:(下面原文的例子可能有些问题)

 

 

 
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. For example:

 

 

 
关于CSV文件解析

上面说过,CSV并不是一种单一的、定义明确的格式(尽管RFC 4180有一个被通常使用的定义)。因此在实践中,术语“CSV”泛指具有以下特征的任何文件:
    纯文本,使用某个字符集,比如ASCII、Unicode、EBCDIC或GB2312;
    由记录组成(典型的是每行一条记录);
    每条记录被分隔符分隔为字段(典型分隔符有逗号、分号或制表符;有时分隔符可以包括可选的空格);
    每条记录都有同样的字段序列。
所以,在常规的约束条件下,存在着许多CSV变体,故CSV文件并不完全互通,如使用约定好的其他分隔符、转义规则等。因此,实际使用CSV需要数据交换双方约定规则(其实大体思路不变,一些细节),在进行CSV文件读写就免不了进行文件的解析。

正如CSV并不明确的格式,CSV文件的解析同样没有标准方法,一般可以自己实现读写,网上也有很多种不同语言的实现版本。例如opencsv、csvreader等。它们可能会与RFC中的规定有所出入,例如在csvreader中有要求:
    前缀和后缀的空格字符,逗号和制表符,与逗号或记录分隔符相邻的内容将被修剪。
    为了保证前导和后缀空白字符的保留,必须通过将字段嵌入到双引号集合中来限定字段。
使用时需要注意。

在ExcelReport支持csv的开发过程中,需要一个NETStandard的csv解析器。在nuget上找了几个试用,但都不太适合。

于是,便有了:AxinLib.IO.CSV

 

 

 

先看看怎么用:

 

读出的数据:

 

回到开始:

步骤一、了解CSV格式规范

 总结:

1)字段分隔符:

1
,

2)行结束符:

1
\r\n

3)转义符:

1
"

4)出现以上字符的字段需要用转义符前后括起来

5)转义符的原义需要两个转义符标识

6)文件中的最后一行记录可以有结尾回车换行符,也可以没有。

步骤二、定义状态

通过主流程识别状态:

 

1)新字段开始 
2)非转义状态
3)转义状态
4)转义符原义判定状态
5)字段结束
6)行结束
7)文档结束
8)解析失败

步骤三:通过状态迁移图查缺补漏

步骤四:根据状态迁移图实现代码

 详参见:

AxinLib.IO.CSV 源码:https://github.com/hanzhaoxin/AxinLib/tree/master/src/AxinLib.IO.CSV

 

posted @ 2023-02-12 08:50  funiyi816  阅读(1282)  评论(0编辑  收藏  举报