PHP/Laravel -贷款计算逻辑设计理念

kiz8lqtg  于 2023-10-22  发布在  PHP
关注(0)|答案(1)|浏览(108)

背景

我正在开发一个应用程序模块,根据当前配置的国家/地区,该模块具有用于贷款计算的不同域类实现。
我正在努力解决一个类,它具有所有国家共享的主逻辑(一种父类)和一些派生类,它们在行为和计算逻辑上有一些细微的差异。

  • 主课程 *
  1. <?php
  2. namespace We\Domain\Loans;
  3. use Carbon\Carbon;
  4. use Illuminate\Support\Collection;
  5. use We\Domain\Constants\CreditStructureConstants;
  6. use We\Domain\Constants\RoundingPrecisionConstants;
  7. class LoanOptionPlan
  8. {
  9. use LoanCalculator;
  10. protected $integerAmountPrecision = RoundingPrecisionConstants::INTEGER_AMOUNT;
  11. protected $ratePrecision = RoundingPrecisionConstants::RATE;
  12. protected $principalAmount;
  13. protected $negativePrincipalWithFeesAmount;
  14. protected $repaymentAmount;
  15. protected $lastDueDate;
  16. protected $termTotalDays;
  17. protected $grossInterestRate;
  18. protected $netInterestRate;
  19. protected $annualAmortizationRate;
  20. protected $termAmortizationRate;
  21. protected $totalFinanceCharge;
  22. protected $annualPercentageRate;
  23. protected $installmentsQuantity;
  24. protected $installmentAmount;
  25. protected $grossInterestAmount;
  26. protected $interestTaxesAmount;
  27. protected $insuranceAmount;
  28. protected $cashOutFeeAmount;
  29. protected $cashInFeeAmount;
  30. protected $feeTotalAmount;
  31. protected $installments;
  32. protected $id;
  33. public function __construct(
  34. LoanOption $option, int $installmentsQuantity, ?int $planId = NULL
  35. )
  36. {
  37. $firstDueDate = $option->getFirstDueDate();
  38. $principalAmount = $option->getPrincipalAmount();
  39. $grossInterestRate = $option->getGrossInterestRate();
  40. $taxesCoefficient = $option->getTaxesCoefficient();
  41. $termTotalDays = $this->calculateTermTotalDays(
  42. $firstDueDate, $option->getRequestDate(), $installmentsQuantity
  43. );
  44. $cashOutFeeAmount = $this->calculateCashoutFeeAmount(
  45. $option->getCashOutMaxRate(), $option->getCashOutMinRate(), $termTotalDays, $installmentsQuantity,
  46. $principalAmount
  47. );
  48. $netInterestRate = $this->calculateNetInterestRate($grossInterestRate, $taxesCoefficient);
  49. $annualAmortizationRate = $this->calculateAnnualAmortizationRate(
  50. $netInterestRate, $option->getAnnualNetInsuranceRate()
  51. );
  52. $installmentTermDurationAverage = $this->calculateInstallmentTermDurationAverage(
  53. $termTotalDays, $installmentsQuantity
  54. );
  55. $termAmortizationRate = $this->calculateTermAmortizationRate(
  56. $annualAmortizationRate, $installmentTermDurationAverage
  57. );
  58. $principalWithFeesAmount = $this->calculatePrincipalWithFeesAmount(
  59. $principalAmount, $cashOutFeeAmount
  60. );
  61. $negativePrincipalWithFeesAmount = $this->calculateNegativePrincipalWithFeesAmount(
  62. $principalWithFeesAmount
  63. );
  64. $installmentAmount = $this->calculateInstallmentAmount(
  65. $termAmortizationRate, $installmentsQuantity, $negativePrincipalWithFeesAmount
  66. );
  67. $cashFlowValues = $this->calculateCashFlowValues($negativePrincipalWithFeesAmount);
  68. $this->principalAmount = $principalAmount;
  69. $this->negativePrincipalWithFeesAmount = $negativePrincipalWithFeesAmount;
  70. $this->repaymentAmount = $this->calculateRepaymentAmount($installmentAmount, $installmentsQuantity);
  71. $this->lastDueDate = $firstDueDate->copy()->addMonthsNoOverflow($installmentsQuantity);
  72. $this->termTotalDays = $termTotalDays;
  73. $this->grossInterestRate = $grossInterestRate;
  74. $this->netInterestRate = $netInterestRate;
  75. $this->annualAmortizationRate = $annualAmortizationRate;
  76. $this->termAmortizationRate = $termAmortizationRate;
  77. $this->totalFinanceCharge = $this->calculateTotalFinanceCharge(
  78. $cashFlowValues, $termTotalDays, $installmentsQuantity
  79. );
  80. $this->annualPercentageRate = $this->calculateAnnualPercentageRate($cashFlowValues);
  81. $this->installmentsQuantity = $installmentsQuantity;
  82. $this->installmentAmount = $installmentAmount;
  83. $this->cashOutFeeAmount = $cashOutFeeAmount;
  84. $this->feeTotalAmount = $cashOutFeeAmount;
  85. $this->addInstallments($option, $installmentsQuantity);
  86. $this->id = $planId;
  87. }
  88. protected function convertToArray() : array
  89. {
  90. foreach ($this->installments as $installment) {
  91. $installments[] = $installment->convertToArray();
  92. }
  93. return [
  94. 'plan_id' => $this->id,
  95. 'installmentsQuantity' => $this->installmentsQuantity,
  96. 'original_amount' => $this->principalAmount,
  97. 'amount' => $this->principalAmount,
  98. 'final_amount' => $this->repaymentAmount,
  99. 'days' => $this->termTotalDays,
  100. 'payment' => $this->installmentAmount,
  101. 'interestRate' => $this->grossInterestRate,
  102. 'cft' => $this->totalFinanceCharge,
  103. 'apr' => $this->annualPercentageRate,
  104. 'cashout_fee' => $this->cashOutFeeAmount,
  105. 'cashin_fee' => $this->cashInFeeAmount,
  106. 'administrative_fee' => $this->feeTotalAmount,
  107. 'interest_amount' => $this->grossInterestAmount,
  108. 'tax_amount' => $this->interestTaxesAmount,
  109. 'insurance_amount' => $this->insuranceAmount,
  110. 'installmentsOptions' => !empty($installments) ? $installments : []
  111. ];
  112. }
  113. public function getPrincipalAmount(): float
  114. {
  115. return $this->principalAmount;
  116. }
  117. public function getNegativePrincipalWithFeesAmount() : float
  118. {
  119. return $this->negativePrincipalWithFeesAmount;
  120. }
  121. public function getRepaymentAmount(): float
  122. {
  123. return $this->repaymentAmount;
  124. }
  125. public function getLastDueDate() : Carbon
  126. {
  127. return $this->lastDueDate;
  128. }
  129. public function getTermTotalDays() : int
  130. {
  131. return $this->termTotalDays;
  132. }
  133. public function getGrossInterestRate() : float
  134. {
  135. return round($this->grossInterestRate, $this->ratePrecision);
  136. }
  137. public function getNetInterestRate() : float
  138. {
  139. return round($this->netInterestRate, $this->ratePrecision);
  140. }
  141. public function getAnnualAmortizationRate() : float
  142. {
  143. return $this->annualAmortizationRate;
  144. }
  145. public function getTermAmortizationRate() : float
  146. {
  147. return $this->termAmortizationRate;
  148. }
  149. public function getTotalFinanceCharge() : float
  150. {
  151. return round($this->totalFinanceCharge, $this->ratePrecision);
  152. }
  153. public function getAnnualPercentageRate() : float
  154. {
  155. return round($this->annualPercentageRate, $this->ratePrecision);
  156. }
  157. public function getInstallmentsQuantity() : int
  158. {
  159. return $this->installmentsQuantity;
  160. }
  161. public function getInstallmentAmount() : float
  162. {
  163. return $this->installmentAmount;
  164. }
  165. public function getGrossInterestAmount() : float
  166. {
  167. return $this->grossInterestAmount;
  168. }
  169. public function getInterestTaxesAmount() : float
  170. {
  171. return $this->interestTaxesAmount;
  172. }
  173. public function getInsuranceAmount() : float
  174. {
  175. return $this->insuranceAmount;
  176. }
  177. public function getCashOutFeeAmount() : float
  178. {
  179. return $this->cashOutFeeAmount;
  180. }
  181. public function getCashInFeeAmount() : float
  182. {
  183. return $this->cashInFeeAmount;
  184. }
  185. public function getFeeTotalAmount() : float
  186. {
  187. return round($this->feeTotalAmount, $this->integerAmountPrecision);
  188. }
  189. public function getInstallments() : Collection
  190. {
  191. return $this->installments;
  192. }
  193. public function getId() : ?int
  194. {
  195. return $this->id;
  196. }
  197. protected function createInstallment(LoanOption $option, int $installmentNumber) : void
  198. {
  199. $this->installments->push(new LoanOptionPlanInstallment($option, $this, $installmentNumber));
  200. }
  201. private function addInstallments(LoanOption $option, int $installmentsQuantity)
  202. {
  203. for (
  204. $installmentNumber = CreditStructureConstants::FIRST_INSTALLMENT_NUMBER;
  205. $installmentNumber <= $installmentsQuantity; $installmentNumber++
  206. ) {
  207. $this->createInstallment($option, $installmentNumber);
  208. $this->sumInstallmentComponents($this->installments->last());
  209. }
  210. }
  211. private function sumInstallmentComponents(ArLoanOptionPlanInstallment $installment)
  212. {
  213. $cashInFeeAmount = $installment->getCashInFeeAmount();
  214. $this->grossInterestAmount += $installment->getGrossInterestAmount();
  215. $this->interestTaxesAmount += $installment->getInterestTaxesAmount();
  216. $this->insuranceAmount += $installment->getInsuranceAmount();
  217. $this->cashInFeeAmount += $cashInFeeAmount;
  218. $this->feeTotalAmount += $cashInFeeAmount;
  219. }
  220. }
  • 派生类 *
  1. <?php
  2. namespace We\Domain\Loans;
  3. use Carbon\Carbon;
  4. use Illuminate\Support\Collection;
  5. use We\Domain\Constants\CreditStructureConstants;
  6. use We\Domain\Constants\RoundingPrecisionConstants;
  7. class UyLoanOptionPlan extends LoanOptionPlan
  8. {
  9. private $indexedUnitExchangeRate;
  10. public function __construct(
  11. LoanOption $option, int $installmentsQuantity, float $indexedUnitExchangeRate, ?int $planId = NULL
  12. )
  13. {
  14. $this->indexedUnitExchangeRate = $indexedUnitExchangeRate;
  15. parent::__construct($option, $installmentsQuantity, $planId);
  16. }
  17. protected function createInstallment(LoanOption $option, int $installmentNumber) : void
  18. {
  19. $this->installments->push(new UyLoanOptionPlanInstallment($option, $this, $installmentNumber));
  20. }
  21. }

我想知道是否有一种方法可以重新实现或更改主类中调用的“calculateCashoutFeeAmount”和“calculateInstallmentAmount”函数引用,并使用相应的“子”实现(将具有不同的签名),而无需复制构造函数逻辑。计算函数是从trait中使用的,所以派生类将在内部使用不同的trait。
我不知道我的设计是否可以接受,我已经想了一整天了,但我还没有得出一个结论。一个来自“父”类的抽象方法可能是一种可能性,但是我不能从那个类创建示例。我面临的另一个问题是,许多计算相互依赖(Matroska风格),因此可能会发生鸡生蛋还是蛋生蛋的情况。
我应该改变设计,还是可以在不重复代码的情况下解决?
任何建议或意见将不胜感激。

qnakjoqk

qnakjoqk1#

你可以通过使用接口或抽象类,或者使用依赖注入来解决这个问题。在这个场景中,LoanOptionPlan类和UyLoanOptionPlan类之间似乎存在关系。但是,当像calculateCashoutFeeAmountcalculateInstallmentAmount这样的方法的签名发生变化时,这种关系就会破裂。
此类包含太多属性,无法理解。我认为这是一个更大的问题。要管理类中过多的属性,您可以使用DTO对数据进行分组,将相关属性模块化为子类,并通过使用Facade设计模式隐藏不必要的属性来降低复杂性。
我可以根据你的评论给予一个例子,但你应该根据你的工作来更广泛地思考。这个例子可能给予不了你一个解决方案,我只是根据你的代码思考。首先,您可以创建一个接口来封装计算函数。这个接口将作为实现它的类必须遵守的契约。举例来说:

  1. interface LoanCalculatorInterface
  2. {
  3. public function calculateCashoutFeeAmount(LoanOption $option, int $termTotalDays, int $installmentsQuantity, float $principalAmount): float;
  4. public function calculateInstallmentAmount(float $termAmortizationRate, int $installmentsQuantity, float $negativePrincipalWithFeesAmount): float;
  5. // Other calculation methods...
  6. }

接下来,创建一个名为LoanCalculator的类来实现这个接口:

  1. class LoanCalculator implements LoanCalculatorInterface
  2. {
  3. public function calculateCashoutFeeAmount(LoanOption $option, int $termTotalDays, int $installmentsQuantity, float $principalAmount): float
  4. {
  5. // Cashout fee calculation logic
  6. // ...
  7. }
  8. public function calculateInstallmentAmount(float $termAmortizationRate, int $installmentsQuantity, float $negativePrincipalWithFeesAmount): float
  9. {
  10. // Installment amount calculation logic
  11. // ...
  12. }
  13. }

现在,在LoanOptionPlan类中,不再使用这些计算函数作为属性,而是将它们作为服务注入。这可以通过依赖注入来实现。举例来说:

  1. class LoanOptionPlan
  2. {
  3. private $loanCalculator;
  4. public function __construct(
  5. LoanOption $option,
  6. int $installmentsQuantity,
  7. LoanCalculatorInterface $loanCalculator,
  8. ?int $planId = NULL
  9. ) {
  10. $cashOutFeeAmount = $loanCalculator->calculateCashoutFeeAmount($option, $this->termTotalDays, $installmentsQuantity, $principalAmount);
  11. $installmentAmount = $loanCalculator->calculateInstallmentAmount($this->termAmortizationRate, $installmentsQuantity, $negativePrincipalWithFeesAmount);
  12. }
  13. }

UyLoanOptionPlan中,适当地实现calculateCashoutFeeAmountcalculateInstallmentAmount方法:

  1. class UyLoanCalculator implements LoanCalculatorInterface
  2. {
  3. public function calculateCashoutFeeAmount(LoanOption $option, int $termTotalDays, int $installmentsQuantity, float $principalAmount): float
  4. {
  5. // Specific cashout fee calculation logic for UyLoanPlan
  6. // ...
  7. }
  8. public function calculateInstallmentAmount(float $termAmortizationRate, int $installmentsQuantity, float $negativePrincipalWithFeesAmount): float
  9. {
  10. // Specific installment amount calculation logic for UyLoanPlan
  11. // ...
  12. }
  13. }

这样,当你想改变计算函数的逻辑时,你只需要创建一个实现LoanCalculatorInterface的新类。这种方法使您的代码更加灵活、可读和易于维护。

展开查看全部

相关问题