Расчет налога с использованием moneyphp - PullRequest
0 голосов
/ 07 октября 2018

Я использую класс moneyphp / money для хранения денежных значений.Однако при расчете причитающегося налога у меня возникает проблема, когда рассчитанный налог является десятичным, а библиотека ищет целочисленное значение.

Пример:

$invoiceTotal = new Money("155" new Currency("USD")); //$1.55
$taxRate= 0.065;
$invoiceTotalWithTax = $invoiceTotal->multiply($taxRate);
echo $invoiceTotalWithTax; //0.10 whereas actual value is 1.55*0.065 = 0.10075
$formatter = new DecimalMoneyFormatter();
$formatter->format($invoiceTotalWithTax); //will return $0.10

Из приведенного выше примера теряется некоторое значение дробного процента.По отдельности это не много, однако, если мы обработаем несколько тысяч счетов за налоговый период, общий собранный налог в итоге превысит 1 цент.

  • Есть ли способ справиться с этими ситуациями с помощью пакета Money?
  • Если нет, то есть ли другой пакет, который может справиться с этим?

1 Ответ

0 голосов
/ 08 октября 2018

Бесстыдный плагин: я не знаю, есть ли способ сделать это с пакетом moneyphp/money, но вот как вы можете справиться с этой ситуацией с пакетом brick / money (заявление об отказе: я написалэто).

Выбор, который вы выберете, будет зависеть от того, чего вы пытаетесь достичь.

Вариант 1: используйте деньги со шкалой по умолчанию, округляя вверх или вниз

Используйте этот метод, если вам нужен результат в шкале по умолчанию для валюты (2 десятичных знака для USD), и вы знаете, какое округление применить:

use Brick\Money\Money;
use Brick\Math\RoundingMode;

$invoiceTotal = Money::ofMinor('155', 'USD'); // USD 1.55
// or
$invoiceTotal = Money::of('1.55', 'USD');

$taxRate = '0.065'; // prefer strings over floats!

$totalWithTax = $invoiceTotal->multipliedBy($taxRate, RoundingMode::DOWN); // USD 0.10
$totalWithTax = $invoiceTotal->multipliedBy($taxRate, RoundingMode::UP); // USD 0.11

У вас есть еще округлениеРежимы на выбор.Если вы не предоставляете режим округления, а результат не помещается в 2 десятичных знака, вы получите исключение.

Вариант 2: используйте деньги с пользовательской шкалой

Если вам нужно работать с заданной точностью, скажем, 5 десятичных знаков, вы можете указать это при создании Money:

use Brick\Money\Money;
use Brick\Money\Context\CustomContext;
use Brick\Math\RoundingMode;

$invoiceTotal = Money::of('1.55', 'USD', new CustomContext(5)); // USD 1.55000
$taxRate = '0.065';

$totalWithTax = $invoiceTotal->multipliedBy($taxRate); // USD 0.10075

Если результат не умещается в 5 десятичных знаков, вам необходимоукажите RoundingMode, или вы получите исключение.

Вариант 3: используйте деньги с автоматическим масштабированием

Используйте этот метод для автоматической корректировки масштаба результата на правильное числодесятичных разрядов:

use Brick\Money\Money;
use Brick\Money\Context\AutoContext;
use Brick\Math\RoundingMode;

$invoiceTotal = Money::of('1.55', 'USD', new AutoContext()); // USD 1.55
$taxRate = '0.065';

$totalWithTax = $invoiceTotal->multipliedBy($taxRate); // USD 0.10075

Режим округления не задействован, но если при делении получится десятичное число с бесконечным числом цифр, вы получите исключение.

Вариант 4:используйте RationalMoney

A RationalMoney - денежный объект, который представляет его сумму в виде рационального числа (дроби).Это особенно полезно, когда вам нужно объединить несколько операций без округления :

use Brick\Money\Money;
use Brick\Math\RoundingMode;

$amount = Money::of('1.55', 'USD'); // USD 1.55
$amount = $amount->toRational(); // USD 155/100

$amount = $amount->dividedBy(3); // USD 155/300
$amount = $amount->dividedBy(7); // USD 155/2100

После того, как вы выполнили все свои операции, вы можете преобразовать свое окончательное число в десятичное число, используярежим округления при необходимости:

use Brick\Money\Context\DefaultContext;
use Brick\Money\Context\CustomContext;

$amount->to(new DefaultContext(), RoundingMode::DOWN); // USD 0.07
$amount->to(new CustomContext(6), RoundingMode::DOWN); // USD 0.073809

Заключительные замечания

Пакет brick / money предлагает форматирование, округление наличных, распределение денег, конвертацию валюты и многое другое.Он основан на пакете brick / math , который выполняет вычисления для чисел любого масштаба.Не стесняйтесь попробовать!

...