如何从 excel 单元格中获取完整值,而不是显示的(四舍五入的)值?
How to get the full value from an excel cell, not the displayed (rounded) value?
我在从工作表的单元格中检索准确值时遇到问题。如果我打开文件,单元格有一个十进制数,它只显示 4 位小数,但如果我单击特定的单元格,值是不同的,有 6 位小数。我知道这是应用于单元格的设置,以便仅显示 4 位小数。
现在我正在尝试使用 ClosedXML.Excel 而不是 Microsoft.Office.Interop.Excel 在 C# 中检索单元格的数据,但我唯一能得到的是 4 位小数的值,而不是"whole" 一个,这是一个很大的问题,因为我稍后要计算,由于缺少 2 个小数点,我有很大的差异。
我试过使用 inputSheet.Worksheet.Cell(row,col).GetDouble()
,或 Convert.ToDouble(inputSheet.Worksheet.Cell(row, col).Value)
,甚至参考 "RichText" 属性,或 ...ToString("0.000000"),但无论我使用什么,我只能检索 4 位十进制值而不是完整的 6 位小数。
我查看了库的源代码,下面是 Value
getter 的作用:
get
{
string str = this.FormulaA1;
if (!XLHelper.IsNullOrWhiteSpace(str))
{
string str2;
string sName;
if (str[0] == '{')
{
str = str.Substring(1, str.Length - 2);
}
if (str.Contains<char>('!'))
{
sName = str.Substring(0, str.IndexOf('!'));
if (sName[0] == '\'')
{
sName = sName.Substring(1, sName.Length - 2);
}
str2 = str.Substring(str.IndexOf('!') + 1);
}
else
{
sName = this.Worksheet.Name;
str2 = str;
}
if (this._worksheet.Workbook.WorksheetsInternal.Any<XLWorksheet>(w => (string.Compare(w.Name, sName, true) == 0)) && XLHelper.IsValidA1Address(str2))
{
return this._worksheet.Workbook.Worksheet(sName).Cell(str2).Value;
}
object obj2 = this.Worksheet.Evaluate(str);
IEnumerable enumerable = obj2 as IEnumerable;
if ((enumerable != null) && !(obj2 is string))
{
using (IEnumerator enumerator = enumerable.GetEnumerator())
{
while (enumerator.MoveNext())
{
return enumerator.Current;
}
}
}
return obj2;
}
string s = this.HasRichText ? this._richText.ToString() : this._cellValue;
if (this._dataType == XLCellValues.Boolean)
{
return (s != "0");
}
if (this._dataType == XLCellValues.DateTime)
{
return DateTime.FromOADate(double.Parse(s));
}
if (this._dataType == XLCellValues.Number)
{
return double.Parse(s);
}
if (this._dataType == XLCellValues.TimeSpan)
{
return TimeSpan.Parse(s);
}
return s;
}
我注意到,当它到达 string s = this.HasRichText ? this._richText.ToString() : this._cellValue;
时,如果您从未调用过 cell.RichText
(因此也没有使用调试器查看它),您会得到正确的值,因为 this.HasRichText
是false 因为 class 仍未为其分配正确的值。
当它(this.HasRichText
) 为true
时,您将得到this._richText.ToString()
,即格式化后的数字。
所以,如果你在访问 RichText
之前访问 Value
属性 你应该得到正确的值,无论如何你可以使用反射获得 _cellValue
,然后将它转换为 double
像这样:
var theRealDoubleValue = Convert.ToDouble(yourCell.GetType().GetField("_cellValue", System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance).GetValue(yourCell));
我在从工作表的单元格中检索准确值时遇到问题。如果我打开文件,单元格有一个十进制数,它只显示 4 位小数,但如果我单击特定的单元格,值是不同的,有 6 位小数。我知道这是应用于单元格的设置,以便仅显示 4 位小数。
现在我正在尝试使用 ClosedXML.Excel 而不是 Microsoft.Office.Interop.Excel 在 C# 中检索单元格的数据,但我唯一能得到的是 4 位小数的值,而不是"whole" 一个,这是一个很大的问题,因为我稍后要计算,由于缺少 2 个小数点,我有很大的差异。
我试过使用 inputSheet.Worksheet.Cell(row,col).GetDouble()
,或 Convert.ToDouble(inputSheet.Worksheet.Cell(row, col).Value)
,甚至参考 "RichText" 属性,或 ...ToString("0.000000"),但无论我使用什么,我只能检索 4 位十进制值而不是完整的 6 位小数。
我查看了库的源代码,下面是 Value
getter 的作用:
get
{
string str = this.FormulaA1;
if (!XLHelper.IsNullOrWhiteSpace(str))
{
string str2;
string sName;
if (str[0] == '{')
{
str = str.Substring(1, str.Length - 2);
}
if (str.Contains<char>('!'))
{
sName = str.Substring(0, str.IndexOf('!'));
if (sName[0] == '\'')
{
sName = sName.Substring(1, sName.Length - 2);
}
str2 = str.Substring(str.IndexOf('!') + 1);
}
else
{
sName = this.Worksheet.Name;
str2 = str;
}
if (this._worksheet.Workbook.WorksheetsInternal.Any<XLWorksheet>(w => (string.Compare(w.Name, sName, true) == 0)) && XLHelper.IsValidA1Address(str2))
{
return this._worksheet.Workbook.Worksheet(sName).Cell(str2).Value;
}
object obj2 = this.Worksheet.Evaluate(str);
IEnumerable enumerable = obj2 as IEnumerable;
if ((enumerable != null) && !(obj2 is string))
{
using (IEnumerator enumerator = enumerable.GetEnumerator())
{
while (enumerator.MoveNext())
{
return enumerator.Current;
}
}
}
return obj2;
}
string s = this.HasRichText ? this._richText.ToString() : this._cellValue;
if (this._dataType == XLCellValues.Boolean)
{
return (s != "0");
}
if (this._dataType == XLCellValues.DateTime)
{
return DateTime.FromOADate(double.Parse(s));
}
if (this._dataType == XLCellValues.Number)
{
return double.Parse(s);
}
if (this._dataType == XLCellValues.TimeSpan)
{
return TimeSpan.Parse(s);
}
return s;
}
我注意到,当它到达 string s = this.HasRichText ? this._richText.ToString() : this._cellValue;
时,如果您从未调用过 cell.RichText
(因此也没有使用调试器查看它),您会得到正确的值,因为 this.HasRichText
是false 因为 class 仍未为其分配正确的值。
当它(this.HasRichText
) 为true
时,您将得到this._richText.ToString()
,即格式化后的数字。
所以,如果你在访问 RichText
之前访问 Value
属性 你应该得到正确的值,无论如何你可以使用反射获得 _cellValue
,然后将它转换为 double
像这样:
var theRealDoubleValue = Convert.ToDouble(yourCell.GetType().GetField("_cellValue", System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance).GetValue(yourCell));