目录
读取
读取所有的记录
手动读取记录
读取单独的字段
解析
写入
写入所有记录
映射
自动映射
手动映射
引用映射
下标指定
列名指定
同名处理
默认值
类型转换
可选类型转换
ConvertUsing
运行时映射
配置
允许注释
缓存
注释
Byte 计数
Culture 信息
分割符
列数变化
编码
是否有头记录
忽略列名空格
忽略私有访问
忽略读取异常
忽略引号
列名是否大小写敏感
映射访问
属性绑定标记
Quote
所有字段加引号
所有字段不加引号
读取异常的回调
注册类映射
跳过空白记录
Trim 字段
Trim 列名
解绑类映射
空字段是否抛出异常
其他
查看异常信息
DataReader 与 DataTable
首页 后端开发 C#.Net教程 技术解答CSV 文件的一个 .NET 库:CsvHelper 中文文档

技术解答CSV 文件的一个 .NET 库:CsvHelper 中文文档

Jul 27, 2018 am 11:32 AM
c# csv

CsvHelper 是读写 CSV 文件的一个 .NET 库。可以通过 Visual Studio 的包管理器下载 CsvHelper。自动映射定义:没有提供映射文件的情况下,默认为自动映射,自动映射会按顺序依次映射到类的属性中去。

GitHub 地址

读取

读取所有的记录

var csv = new CsvReader( textReader );
var records = csv.GetRecords<MyClass>(); // 把 CSV 记录映射到 MyClass,返回的 records 是个 IEnumerable<T> 对象
登录后复制

如果想要自定义映射关系,可以看下面映射一节。
由于 records 是个 IEnumerable 对象,因此仅当访问的时候才会返回一个记录,访问一次就返回一个记录。如果想要想列表那样访问,可以作如下处理:

var csv = new CsvReader( textReader );
var records = csv.GetRecords<MyClass>().ToList();
登录后复制

手动读取记录

可以按行循环读取每一行的数据

var csv = new CsvReader( textReader );
while( csv.Read() )
{
    var record = csv.GetRecord<MyClass>();
}
登录后复制

读取单独的字段

var csv = new CsvReader( textReader );
while( csv.Read() )
{
    var intField = csv.GetField<int>( 0 );
    var stringField = csv.GetField<string>( 1 );
    var boolField = csv.GetField<bool>( "HeaderName" );
}
登录后复制

如果读取的类型跟预期的可能会不同,那么可以用 TryGetField

var csv = new CsvReader( textReader );
while( csv.Read() )
{
    int intField;
    if( !csv.TryGetField( 0, out intField ) )
    {
        // Do something when it can&#39;t convert.
    }
}
登录后复制

解析

想要每一行作为一个字符串返回可以使用 CsvParser 。

var parser = new CsvParser( textReader );
while( true )
{
    var row = parser.Read(); // row 是个字符串
    if( row == null )
    {
        break;
    }
}
登录后复制

写入

写入所有记录

var csv = new CsvWriter( textWriter );
csv.WriteRecords( records );
登录后复制
var csv = new CsvWriter( textWriter );
foreach( var item in list )
{
    csv.WriteRecord( item );
}
登录后复制
var csv = new CsvWriter( textWriter );
foreach( var item in list )
{
    csv.WriteField( "a" );
    csv.WriteField( 2 );
    csv.WriteField( true );
    csv.NextRecord();
}
登录后复制

映射

自动映射

没有提供映射文件的情况下,默认为自动映射,自动映射会按顺序依次映射到类的属性中去。如果属性是一个自定义类,那么会继续依次按照这个自定义类的属性进行填入。如果出现了循环引用,那么自动映射会停止。

手动映射

如果 CSV 文件和自定义的类并不是完全的匹配,那么可以定义一个匹配类来处理。

public sealed class MyClassMap : CsvClassMap<MyClass>
{
    public MyClassMap()
    {
        Map( m => m.Id );
        Map( m = > m.Name );
    }
}
登录后复制
本文由 tangyikejun 翻译

引用映射

如果属性是一个自定义类,对应于CSV文件的多个列,那么可以使用引用映射。

public sealed class PersonMap : CsvClassMap<Person>
{
    public PersonMap()
    {
        Map( m => m.Id );
        Map( m => m.Name );
        References<AddressMap>( m => m.Address );
    }
}

public sealed class AddressMap : CsvClassMap<Address>
{
    public AddressMap()
    {
        Map( m => m.Street );
        Map( m => m.City );
        Map( m => m.State );
        Map( m => m.Zip );
    }
}
登录后复制

下标指定

可以通过列下标指定映射

public sealed class MyClassMap : CsvClassMap<MyClass>
{
    public MyClassMap()
    {
        Map( m => m.Id ).Index( 0 );
        Map( m => m.Name ).Index( 1 );
    }
}
登录后复制

列名指定

也可以通过列名指定映射,这要求csv文件有一个头记录,也就是说第一行记录列名

public sealed class MyClassMap : CsvClassMap<MyClass>
{
    public MyClassMap()
    {
        Map( m => m.Id ).Name( "The Id Column" );
        Map( m => m.Name ).Name( "The Name Column" );
    }
}
登录后复制

同名处理

public sealed class MyClassMap : CsvClassMap<MyClass>
{
    public MyClassMap()
    {
        Map( m => m.FirstName ).Name( "Name" ).NameIndex( 0 );
        Map( m => m.LastName ).Name( "Name" ).NameIndex( 1 );
    }
}
登录后复制

默认值

public sealed class MyClassMap : CsvClassMap<MyClass>
{
    public override void MyClassMap()
    {
        Map( m => m.Id ).Index( 0 ).Default( -1 );
        Map( m => m.Name ).Index( 1 ).Default( "Unknown" );
    }
}
登录后复制

类型转换

public sealed class MyClassMap : CsvClassMap<MyClass>
{
    public MyClassMap()
    {
        Map( m => m.Id ).Index( 0 ).TypeConverter<MyIdConverter>();
    }
}
登录后复制

可选类型转换

默认的转换器会处理大部分的类型转换,但是有时候我们可能需要做一些小的改变,这个时候可以尝试是用可选类型转换。

public sealed class MyClassMap : CsvClassMap<MyClass>
{
    public MyClassMap()
    {
        Map( m => m.Description ).Index( 0 ).TypeConverterOption( CultureInfo.InvariantCulture ); // 
        Map( m => m.TimeStamp ).Index( 1 ).TypeConverterOption( DateTimeStyles.AdjustToUniversal ); // 时间格式转换
        Map( m => m.Cost ).Index( 2 ).TypeConverterOption( NumberStyles.Currency ); // 数值类型转换
        Map( m => m.CurrencyFormat ).Index( 3 ).TypeConverterOption( "C" );
        Map( m => m.BooleanValue ).Index( 4 ).TypeConverterOption( true, "sure" ).TypeConverterOption( false, "nope" ); // 内容转换
    }
}
登录后复制

ConvertUsing

public sealed class MyClassMap : CsvClassMap<MyClass>
{
    public MyClassMap()
    {
        // 常数
        Map( m => m.Constant ).ConvertUsing( row => 3 );
        // 把两列聚合在一起
        Map( m => m.Aggregate ).ConvertUsing( row => row.GetField<int>( 0 ) + row.GetField<int>( 1 ) );
        // Collection with a single value.
        Map( m => m.Names ).ConvertUsing( row => new List<string>{ row.GetField<string>( "Name" ) } );
        // Just about anything.
        Map( m => m.Anything ).ConvertUsing( row =>
        {
            // You can do anything you want in a block.
            // Just make sure to return the same type as the property.
        } );
    }
}
登录后复制

运行时映射

可以在运行时创建映射。

var customerMap = new DefaultCsvClassMap();

// mapping holds the Property - csv column mapping 
foreach( string key in mapping.Keys )
{
    var columnName = mapping[key].ToString();

    if( !String.IsNullOrEmpty( columnName ) )
    {
        var propertyInfo = typeof( Customer ).GetType().GetProperty( key );
        var newMap = new CsvPropertyMap( propertyInfo );
        newMap.Name( columnName );
        customerMap.PropertyMaps.Add( newMap );
    }
}

csv.Configuration.RegisterClassMap(CustomerMap);
登录后复制
本文由 tangyikejun 翻译

配置

允许注释

// Default value
csv.Configuration.AllowComments = false;
登录后复制

自动映射

var generatedMap = csv.Configuration.AutoMap<MyClass>();
登录后复制

缓存

TextReader 或 TextWriter 中读写的缓存

// Default value
csv.Configuration.BufferSize = 2048;
登录后复制

注释

被注释掉的那行不会被加载进来

// Default value
csv.Configuration.Comment = &#39;#&#39;;
登录后复制

Byte 计数

记录当前读取了多少 Byte 了,需要设置 Configuration.Encoding 与 CSV 文件一致。这个设置会影响解析的速度。

// Default value
csv.Configuration.CountBytes = false;
登录后复制

Culture 信息

// Default value
csv.Configuration.CultureInfo = CultureInfo.CurrentCulture;
登录后复制

分割符

// Default value
csv.Configuration.Delimiter = ",";
登录后复制

列数变化

如果开启,发现列数变化会抛出 CsvBadDataException

// Default value
csv.Configuration.DetectColumnCountChanges = false;
登录后复制

编码

// Default value
csv.Configuration.Encoding = Encoding.UTF8;
登录后复制

是否有头记录

// Default value
csv.Configuration.HasHeaderRecord = true;
登录后复制

忽略列名空格

是否忽略列名中的空格

// Default value
csv.Configuration.IgnoreHeaderWhiteSpace = false;
登录后复制

忽略私有访问

读写的时候是否忽略私有访问器

// Default value
csv.Configuration.IgnorePrivateAccessor = false;
登录后复制

忽略读取异常

读取发生异常之后仍继续读取

// Default value
csv.Configuration.IgnoreReadingExceptions = false;
登录后复制

忽略引号

不把引号作为转义符

// Default value
csv.Configuration.IgnoreQuotes = false;
登录后复制

列名是否大小写敏感

// Default value
csv.Configuration.IsHeaderCaseSensitive = true;
登录后复制

映射访问

可以对自定义的类映射进行访问

var myMap = csv.Configuration.Maps[typeof( MyClass )];
登录后复制

属性绑定标记

用来寻找自定义类的属性

// Default value
csv.Configuration.PropertyBindingFlags = BindingFlags.Public | BindingFlags.Instance;
登录后复制
本文由 tang yi ke jun 翻译

Quote

定义用来转义包含分隔符,括号或者行尾的转义符

// Default value
csv.Configuration.Quote = &#39;"&#39;;
登录后复制

所有字段加引号

写入csv的时候是否对所有字段加引号。QuoteAllFields 和 QuoteNoFields 不能同时为 true 。

// Default value
csv.Configuration.QuoteAllFields = false;
登录后复制

所有字段不加引号

QuoteAllFields 和 QuoteNoFields 不能同时为 true 。

// Default value
csv.Configuration.QuoteNoFields = false;
登录后复制

读取异常的回调

csv.Configuration.ReadingExceptionCallback = ( ex, row ) =>
{
    // Log the exception and current row information.
};
登录后复制

注册类映射

使用了类映射的话,需要进行注册才会被实际使用到。

csv.Configuration.RegisterClassMap<MyClassMap>();
csv.Configuration.RegisterClassMap<AnotherClassMap>();
登录后复制

跳过空白记录

如果所有字段都是空的,就会被认为是空字段

// Default value
csv.Configuration.SkipEmptyRecords = false;
登录后复制

Trim 字段

把字段内容收尾的空白字符删去。

// Default value
csv.Configuration.TrimFields = false;
登录后复制

Trim 列名

// Default value
csv.Configuration.TrimHeaders = false;
登录后复制

解绑类映射

// Unregister single map.
csv.Configuration.UnregisterClassMap<MyClassMap>();
// Unregister all class maps.
csv.Configuration.UnregisterClassMap();
登录后复制

空字段是否抛出异常

// Default value
csv.Configuration.WillThrowOnMissingField = true;
登录后复制

类型转换

类型转换是 CsvHelper 把字符串转换为 .NET 类型(以及反过来)的方法。

其他

查看异常信息

Exception.Data["CsvHelper"]

// Row: &#39;3&#39; (1 based)
// Type: &#39;CsvHelper.Tests.CsvReaderTests+TestBoolean&#39;
// Field Index: &#39;0&#39; (0 based)
// Field Name: &#39;BoolColumn&#39;
// Field Value: &#39;two&#39;
登录后复制

DataReader 与 DataTable

DataReader 对象写入到 CSV

var hasHeaderBeenWritten = false;
while( dataReader.Read() )
{
    if( !hasHeaderBeenWritten )
    {
        for( var i = 0; i < dataReader.FieldCount; i++ )
        {
            csv.WriteField( dataReader.GetName( i ) );
        }
        csv.NextRecord();
        hasHeaderBeenWritten = true;
    }

    for( var i = 0; i < dataReader.FieldCount; i++ )
    {
        csv.WriteField( dataReader[i] );
    }
    csv.NextRecord();
}
登录后复制

DataTable 对象写入到 CSV

using( var dt = new DataTable() )
{
    dt.Load( dataReader );
    foreach( DataColumn column in dt.Columns )
    {
        csv.WriteField( column.ColumnName );
    }
    csv.NextRecord();

    foreach( DataRow row in dt.Rows )
    {
        for( var i = 0; i < dt.Columns.Count; i++ )
        {
            csv.WriteField( row[i] );
        }
        csv.NextRecord();
    }
}
登录后复制

CSV 转 DataTable

while( csv.Read() )
{
    var row = dt.NewRow();
    foreach( DataColumn column in dt.Columns )
    {
        row[column.ColumnName] = csv.GetField( column.DataType, column.ColumnName );
    }
    dt.Rows.Add( row );
}
登录后复制

相关文章:

.net CsvHelper 2.0

jQuery EasyUI API 中文文档 - Documentation 文档_jquery

相关视频:

Ruby中文文档

以上是技术解答CSV 文件的一个 .NET 库:CsvHelper 中文文档的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

使用 C# 的活动目录 使用 C# 的活动目录 Sep 03, 2024 pm 03:33 PM

使用 C# 的 Active Directory 指南。在这里,我们讨论 Active Directory 在 C# 中的介绍和工作原理以及语法和示例。

C# 序列化 C# 序列化 Sep 03, 2024 pm 03:30 PM

C# 序列化指南。这里我们分别讨论C#序列化对象的介绍、步骤、工作原理和示例。

C# 中的随机数生成器 C# 中的随机数生成器 Sep 03, 2024 pm 03:34 PM

C# 随机数生成器指南。在这里,我们讨论随机数生成器的工作原理、伪随机数和安全数的概念。

C# 数据网格视图 C# 数据网格视图 Sep 03, 2024 pm 03:32 PM

C# 数据网格视图指南。在这里,我们讨论如何从 SQL 数据库或 Excel 文件加载和导出数据网格视图的示例。

C# 中的模式 C# 中的模式 Sep 03, 2024 pm 03:33 PM

C# 模式指南。在这里,我们讨论 C# 中模式的介绍和前 3 种类型,以及其示例和代码实现。

C# 中的质数 C# 中的质数 Sep 03, 2024 pm 03:35 PM

C# 素数指南。这里我们讨论c#中素数的介绍和示例以及代码实现。

C# 中的阶乘 C# 中的阶乘 Sep 03, 2024 pm 03:34 PM

C# 阶乘指南。这里我们讨论 C# 中阶乘的介绍以及不同的示例和代码实现。

c#多线程和异步的区别 c#多线程和异步的区别 Apr 03, 2025 pm 02:57 PM

多线程和异步的区别在于,多线程同时执行多个线程,而异步在不阻塞当前线程的情况下执行操作。多线程用于计算密集型任务,而异步用于用户交互操作。多线程的优势是提高计算性能,异步的优势是不阻塞 UI 线程。选择多线程还是异步取决于任务性质:计算密集型任务使用多线程,与外部资源交互且需要保持 UI 响应的任务使用异步。

See all articles