【问题标题】:How to get the full value from an excel cell, not the displayed (rounded) value?如何从 excel 单元格中获取完整值,而不是显示(四舍五入)值?
【发布时间】:2016-03-08 07:53:27
【问题描述】:

我在从工作表的单元格中检索准确值时遇到问题。如果我打开文件,单元格有一个十进制数字,它只显示 4 位小数,但如果我单击特定单元格,则值不同,有 6 位小数。我知道这是一个应用于单元格的设置,以便仅显示 4 位小数。

现在我正在尝试在 C# 中检索单元格的数据,使用 ClosedXML.Excel,而不是 Microsoft.Office.Interop.Excel,但我唯一能得到的是 4 位小数的值,而不是“整个" 一个,这是一个很大的问题,因为我稍后会计算,由于缺少 2 个小数,我有很大的差异。

我尝试过使用inputSheet.Worksheet.Cell(row,col).GetDouble(),或Convert.ToDouble(inputSheet.Worksheet.Cell(row, col).Value),甚至引用“RichText”属性,或...ToString(“0.000000”),但无论我使用什么,我都只能检索到4 位十进制值而不是完整的 6 位小数之一。

【问题讨论】:

  • 您是否尝试过使用 Oledb 导入 Excel 数据?
  • 不,但我不能使用它。我目前的项目很大程度上依赖于 CloseXLM
  • @Eris:我没有价值2。是否仅在 Microsoft.Office.Interop.Excel 中可用?
  • 访问富文本时似乎是库的错误,请检查我的答案

标签: c# excel closedxml


【解决方案1】:

我查看了该库的源代码,这是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 是假的,因为该类仍未分配它的正确值。 当它(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));

【讨论】:

  • 非常感谢 codroipo,它正在工作。我永远也想不通,远远超出了我目前的技能。
猜你喜欢
  • 2014-07-26
  • 2020-12-27
  • 2017-12-07
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-07-02
相关资源
最近更新 更多