我正在使用 NPOI v1.2.3 导出网格,但在使单元格格式正常工作时遇到问题。
我有一个类将对象列表导出到 XLS 文件。为每个对象创建一行,并为每个配置的属性添加一个单元格。可以在每个属性级别设置单元格数据格式。
我读过您不应该为每个单元格创建新样式 https://stackoverflow.com/questions/3425931/npoi-excel-number-format-not-showing-in-excel-sheet-in-asp-net。我无法对我的样式进行硬编码,因为我的导出器需要支持任何类。相反,我编写了一个小型缓存系统,如果尚未为当前单元格的格式创建新的 CellStyle,则该系统仅创建新的 CellStyle。
不幸的是,这仍然没有解决问题。最终 XLS 文件中的格式未正确应用。在我的测试用例中,XLS 中的大多数单元格都使用“日期”格式,即使只有几列是日期。但是,第一列正确地使用了自定义格式。没有单元格被设置为文本,尽管这应该是大多数单元格。
我究竟做错了什么?
Code
下面的“AddRecords”方法用于添加数据行(页眉和页脚行分别添加)。最后一段代码是延迟加载 CellStyles 的方法。
private void AddRecords( Sheet sheet, IList<T> records )
{
foreach( var record in records )
{
// append row
var row = sheet.CreateRow ( sheet.LastRowNum + 1 );
// iterate through all configured columns
foreach ( var column in GetColumns() )
{
// append cell
Cell cell = row.CreateCell ( row.LastCellNum == -1 ? 0 : row.LastCellNum );
// get the property value of the column from the record
object value = GetCellValue ( column, record );
// extension method that takes an object value and calls the appropriate type-specific SetCellValue overload
cell.SetCellValue ( value );
// get format from the column definition ("m/d", "##.###", etc.), or use the default
string dataFormat = column.DataFormat ?? GetDefaultDataFormat ( value );
// find/create cell style
cell.CellStyle = GetCellStyleForFormat( sheet.Workbook, dataFormat );
}
}
}
/// <summary>
/// Returns a default format string based on the object type of value.
///
/// http://poi.apache.org/apidocs/org/apache/poi/ss/usermodel/BuiltinFormats.html
/// </summary>
/// <param name="value"></param>
/// <returns></returns>
private string GetDefaultDataFormat( object value )
{
if( value == null )
{
return "General";
}
if( value is DateTime )
{
return "m/d";
}
if( value is bool )
{
return "[=0]\"Yes\";[=1]\"No\"";
}
if( value is byte || value is ushort || value is short ||
value is uint || value is int || value is ulong || value is long )
{
return "0";
}
if( value is float || value is double )
{
return "0.00";
}
// strings and anything else should be text
return "text";
}
private readonly Dictionary<string, CellStyle> _cellStyleCache = new Dictionary < string, CellStyle > ();
private CellStyle GetCellStyleForFormat( Workbook workbook, string dataFormat )
{
if( !_cellStyleCache.ContainsKey ( dataFormat ) )
{
var newDataFormat = workbook.CreateDataFormat ();
var style = workbook.CreateCellStyle ();
style.DataFormat = newDataFormat.GetFormat ( dataFormat );
_cellStyleCache[dataFormat] = style;
}
return _cellStyleCache[dataFormat];
}