第16章:Python TDD实现多币种货币运算

写在前面


这本书是我们老板推荐过的,我在《价值心法》的推荐书单里也看到了它。用了一段时间 Cursor 软件后,我突然思考,对于测试开发工程师来说,什么才更有价值呢?如何让 AI 工具更好地辅助自己写代码,或许优质的单元测试是一个切入点。 就我个人而言,这本书确实很有帮助。第一次读的时候,很多细节我都不太懂,但将书中内容应用到工作中后,我受益匪浅。比如面对一些让人抓狂的代码设计时,书里的方法能让我逐步深入理解代码的逻辑与设计。 作为一名测试开发工程师,我想把学习这本书的经验分享给大家,希望能给大家带来帮助。因为现在工作中大多使用 Python 代码,所以我把书中JAVA案例都用 Python 代码进行了改写 。


问题背景说明

在复杂的金融场景中,经常会遇到多种货币混合运算的情况。实现多币种货币运算能够进一步提升货币类的功能完整性,使其更贴近实际的业务需求。通过TDD实现这一功能,可以保证代码在复杂运算场景下的正确性和稳定性。

实现多币种货币运算,如$5 + 10 CHF = $10(当汇率为2:1时),是对前面功能的综合应用和拓展,也是Python TDD实践的深入。

16.1 思路说明

编写测试用例来验证多币种加法运算。在已有代码基础上,完善Sum类(用于表示货币相加的表达式)和相关类的方法,确保不同币种货币能根据汇率正确相加。

16.2 测试代码

python 复制代码
import unittest


class Bank:
    def __init__(self):
        self.rates = {}

    def addRate(self, from_currency, to_currency, rate):
        self.rates[(from_currency, to_currency)] = rate

    def rate(self, from_currency, to_currency):
        if from_currency == to_currency:
            return 1
        return self.rates.get((from_currency, to_currency), 1)

    def reduce(self, source, to):
        if isinstance(source, Sum):
            amount = source.augend.reduce(self, to).amount + source.addend.reduce(self, to).amount
            return Money(amount, to)
        elif isinstance(source, Money):
            rate = self.rate(source.currency, to)
            return Money(source.amount / rate, to)


class Expression:
    def reduce(self, bank, to):
        pass


class Money(Expression):
    def __init__(self, amount, currency):
        self.amount = amount
        self.currency = currency

    def plus(self, addend):
        return Sum(self, addend)

    def reduce(self, bank, to):
        rate = self.rate(bank, to)
        return Money(self.amount / rate, to)


class Sum(Expression):
    def __init__(self, augend, addend):
        self.augend = augend
        self.addend = addend

    def reduce(self, bank, to):
        amount = self.augend.reduce(bank, to).amount + self.addend.reduce(bank, to).amount
        return Money(amount, to)


class Dollar(Money):
    def __init__(self, amount):
        super().__init__(amount, "USD")


class Franc(Money):
    def __init__(self, amount):
        super().__init__(amount, "CHF")


class TestMixedAddition(unittest.TestCase):
    def test_mixed_addition(self):
        bank = Bank()
        bank.addRate("CHF", "USD", 2)
        five_dollars = Dollar(5)
        ten_francs = Franc(10)
        result = five_dollars.plus(ten_francs).reduce(bank, "USD")
        self.assertEqual(result.amount, 10)
        self.assertEqual(result.currency, "USD")


if __name__ == '__main__':
    unittest.main()

16.3 源码说明

新增Expression接口,Money类和Sum类都实现了该接口的reduce方法。Sum类用于表示货币相加的表达式,包含augendaddend两个操作数。Bank类的reduce方法可以处理Sum对象和Money对象的转换。在测试用例test_mixed_addition中,创建不同货币对象进行相加并转换,验证多币种加法运算的正确性。

相关推荐
一抓掉一大把2 分钟前
秒杀-StackExchangeRedisHelper连接单例
java·开发语言·jvm
星释16 分钟前
Rust 练习册 :Minesweeper与二维数组处理
开发语言·后端·rust
Q_Q196328847534 分钟前
python+django/flask基于深度学习的个性化携程美食数据推荐系统
spring boot·python·深度学习·django·flask·node.js·php
胡耀超37 分钟前
通往AGI的模块化路径:一个可能的技术架构(同时解答微调与RAG之争)
人工智能·python·ai·架构·大模型·微调·agi
清空mega1 小时前
从零开始搭建 flask 博客实验(常见疑问)
后端·python·flask
xier_ran1 小时前
关键词解释:DAG 系统(Directed Acyclic Graph,有向无环图)
python·算法
开发者小天1 小时前
React中的useRef的用法
开发语言·前端·javascript·react.js
xixixin_1 小时前
【React】检测元素是否出现在用户视窗内
开发语言·前端·javascript·react.js
Js_cold1 小时前
Verilog局部参数localparam
开发语言·fpga开发·verilog
Acrelhuang1 小时前
小小电能表,如何撬动家庭能源革命?
java·大数据·开发语言·人工智能·物联网