«За деньги всегда десятичные»? - PullRequest
10 голосов
/ 09 апреля 2010

Ну, правило " Для денег, всегда десятичное " не применяется внутри команды разработчиков Microsoft, потому что если бы это было:

Namespace: Microsoft.VisualBasic
Assembly:  Microsoft.VisualBasic (in Microsoft.VisualBasic.dll)

Financial.IPmt и все остальные методы будут получать / возвращать decimal, а не double как есть.

Теперь мне интересно, смогу ли я использовать эти методы, не беспокоясь с круглыми ошибками?

Стоит ли использовать другие библиотеки для работы с финансами? Если да, не могли бы вы указать мне несколько хороших (для C# использования)?

Ответы [ 3 ]

10 голосов
/ 09 апреля 2010

Правило использовать decimal для денег полезно, потому что большинство валют имеют десятичные единицы. Используя десятичную арифметику, вы избегаете введения и накопления ошибки округления.

Функции финансового класса использование плавающей запятой по нескольким причинам:

  • Они не накапливаются внутренне - они основаны на экспоненциальных / логарифмических вычислениях в замкнутой форме, а не на итерации и суммировании по периодам.
  • Они, как правило, не используют или не генерируют точные десятичные значения. Например, точная десятичная годовая процентная ставка, деленная на 12 ежемесячных платежей, становится повторяющейся десятичной дробью.
  • Они предназначены в первую очередь для поддержки принятия решений и, в конце концов, мало применимы к фактической бухгалтерии.

Pmt и округление может определить номинальный ежемесячный платеж, но как только эта сумма будет определена, накопление баланса - выполненные платежи, начисленные проценты и т. Д. - происходит в decimal. Кроме того, несвоевременные или авансовые платежи, платежные выходные и другие подобные неоднородности могут сделать недействительной прогнозируемую амортизацию, предусмотренную финансовыми функциями.

10 голосов
/ 09 апреля 2010

Вот интересное обсуждение именно этой темы: http://www.vbforums.com/showthread.php?t=524101

Примерно на 1/3 пути вниз кто-то объясняет, что он использует Double, потому что функции VB.NET были реализованы так же, как и VB6. VB6 не имеет десятичного типа, поэтому он использует double.

Итак, похоже, что если важна точность, вы не должны использовать эти функции.

Ответы на на этот вопрос имеют несколько многообещающих альтернатив - просто игнорируйте принятый ответ, предлагающий использовать библиотеку VB.

Ранее связанный вопрос был удален, поэтому вот некоторые из предложений, на которые я ссылался (примечание: я не пробовал их, YMMV) * ​​1018 *

4 голосов
/ 20 июля 2010

Вы можете использовать этот класс:

public class Financial
{
    #region Methods

    public static decimal IPmt(decimal Rate, decimal Per, decimal NPer, decimal PV, decimal FV, FinancialEnumDueDate Due)
    {
        decimal num;
        if (Due != FinancialEnumDueDate.EndOfPeriod)
        {
            num = 2;
        }
        else
        {
            num = 1;
        }
        if ((Per <= 0) || (Per >= (NPer + 1)))
        {
            //Argument_InvalidValue1=

            throw new ArgumentException("Argument 'Per' is not a valid value.");
        }
        if ((Due != FinancialEnumDueDate.EndOfPeriod) && (Per == 1))
        {
            return 0;
        }
        decimal pmt = Pmt(Rate, NPer, PV, FV, Due);
        if (Due != FinancialEnumDueDate.EndOfPeriod)
        {
            PV += pmt;
        }
        return (FV_Internal(Rate, Per - num, pmt, PV, FinancialEnumDueDate.EndOfPeriod) * Rate);
    }

    public static decimal PPmt(decimal Rate, decimal Per, decimal NPer, decimal PV, decimal FV, FinancialEnumDueDate Due)
    {
        if ((Per <= 0) || (Per >= (NPer + 1)))
        {
            throw new ArgumentException("Argument 'Per' is not valid.");
        }
        decimal num2 = Pmt(Rate, NPer, PV, FV, Due);
        decimal num = IPmt(Rate, Per, NPer, PV, FV, Due);
        return (num2 - num);
    }

    static decimal FV_Internal(decimal Rate, decimal NPer, decimal Pmt, decimal PV, FinancialEnumDueDate Due)
    {
        decimal num;
        if (Rate == 0)
        {
            return (-PV - (Pmt * NPer));
        }
        if (Due != FinancialEnumDueDate.EndOfPeriod)
        {
            num = 1 + Rate;
        }
        else
        {
            num = 1;
        }
        decimal x = 1 + Rate;
        decimal num2 = (decimal)Math.Pow((double)x, (double)NPer);
        return ((-PV * num2) - (((Pmt / Rate) * num) * (num2 - 1)));
    }

    static decimal Pmt(decimal Rate, decimal NPer, decimal PV, decimal FV, FinancialEnumDueDate Due)
    {
        decimal num;
        if (NPer == 0)
        {
            throw new ArgumentException("Argument NPer is not a valid value.");
        }
        if (Rate == 0)
        {
            return ((-FV - PV) / NPer);
        }
        if (Due != FinancialEnumDueDate.EndOfPeriod)
        {
            num = 1 + Rate;
        }
        else
        {
            num = 1;
        }
        decimal x = Rate + 1;
        decimal num2 = (decimal)Math.Pow((double)x, (double)NPer);
        return (((-FV - (PV * num2)) / (num * (num2 - 1))) * Rate);
    }

    #endregion Methods
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...