其他分享
首页 > 其他分享> > CodeGo.net>如何从Excel单元格获得完整的价值,而不是显示(四舍五入)的价值?

CodeGo.net>如何从Excel单元格获得完整的价值,而不是显示(四舍五入)的价值?

作者:互联网

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

现在,我尝试使用ClosedXML.Excel(而不是Microsoft.Office.Interop.Excel)在C#中检索单元格的数据,但是我唯一能得到的是4个小数点的值,而不是“整个”值,这是一个很大的问题,因为我稍后会进行计算,并且由于缺少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;
}

我注意到,当到达字符串s = this.HasRichText吗? this._richText.ToString():this._cellValue ;,如果您从未调用过cell.RichText(因此也无法使用调试器进行查看),则会获得正确的值,因为this.HasRichText为false,因为该类仍未分配正确的价值.
当它(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));

标签:closedxml,c,excel
来源: https://codeday.me/bug/20191119/2032249.html