针对 C++20 契约(断言)进行测试
Testing against the C++20 contracts (assertions)
Herb Sutter 发表了 talk at ACUU Conference regarding the future of exceptions in C++ and upcoming contracts to replace or enhance existing assert。
他假定以下规则来处理错误:
系统损坏(例如堆栈溢出):终止
编程错误(例如违反前提条件):断言、契约
可恢复错误(例如,网络中断):异常,错误代码
虽然我同意,但我并没有使用这种方法,而是将 2 和 3 合并为一个。我这样做是因为我不知道如何测试断言或即将到来的合同。两者都使用类似的机制,它们的违规行为会导致程序终止并在此之前调用可选的处理程序。
来自 documentation(重点是我的):
A program may be translated with one of two violation continuation modes:
- off (default if no continuation mode is selected): after the execution
of the violation handler completes, std::terminate is called;
- on: after the execution of the violation handler completes, execution
continues normally.
Implementations are encouraged to not provide any
programmatic way to query, set, or modify the build level or to set or
modify the violation handler.
这意味着将根据合同进行测试的测试程序将需要额外的编译开关。虽然我不喜欢这样,但我理解为什么要这样做。更令人担心的是最后一部分,以及关于它的问题has been raised。如果设置我们自己的合同违规处理程序的能力是实现定义的,那么跨工具链将不一致。因此,如果可能的话,合同的任何测试都将不可移植。将它设置为链接器的另一个参数也会很尴尬(而且绝对不可移植)。
目前,无法针对 assert
进行测试,因为它只会中止程序,据我所知,自定义处理程序无法阻止这种情况。对于合同,这应该是可以使用自定义处理程序和构建开关实现的,但事实证明,这可能也不可能(或者由实现定义)。
或者是否有任何其他方法可以针对我遗漏的断言和合同进行测试?
在违反合同方面没有太大的实施差异。 “build level”确定合同检查是否完全发生。违规处理程序将在任何检查失败的合同上调用。
处理程序的原型由规范定义。关于违规处理程序的唯一实现差异是您如何设置它。请注意 "implementation-defined" 并不意味着 "does not allow this to happen"。也就是说,the standard says that there will be a mechanism to establish the handler,并且这种机制 将 与实现一起记录(这就是 "implementation-defined" 的意思,而不仅仅是 "unspecified")。该机制究竟是什么将取决于实施,但不提供一个不是一个选项。
唯一可能出现的真正问题是,如果多个实现决定允许用户通过祝福特定的全局名称来指定处理程序,并且这些实现使用不同的全局名称。然而,由于多种原因(与现有代码、宏等冲突),鉴于选择全局名称不是一个好主意,这在现实中不太可能成为问题。所以更有可能的是,他们会选择明显的机制:一个编译器开关,指定要用作处理程序的函数的名称。
请注意,将处理程序定义为静态而不是运行时定义的原因之一是确保开关是 编译器 开关而不是链接器开关.毕竟,编译器会发出任何概念检查代码。所以编译器可以在所述检查代码中使用有问题的函数的名称。
哦,当然,不同的编译器 may/will 有不同的编译器开关来确定调用哪个函数。但他们已经有了不同的开关来生成调试信息、优化级别和基本上所有其他内容。因此,无论您使用什么跨平台测试系统,都已经需要能够处理这些区别。违规处理者只是一个。
Herb Sutter 发表了 talk at ACUU Conference regarding the future of exceptions in C++ and upcoming contracts to replace or enhance existing assert。
他假定以下规则来处理错误:
系统损坏(例如堆栈溢出):终止
编程错误(例如违反前提条件):断言、契约
可恢复错误(例如,网络中断):异常,错误代码
虽然我同意,但我并没有使用这种方法,而是将 2 和 3 合并为一个。我这样做是因为我不知道如何测试断言或即将到来的合同。两者都使用类似的机制,它们的违规行为会导致程序终止并在此之前调用可选的处理程序。
来自 documentation(重点是我的):
A program may be translated with one of two violation continuation modes:
- off (default if no continuation mode is selected): after the execution of the violation handler completes, std::terminate is called;
- on: after the execution of the violation handler completes, execution continues normally.
Implementations are encouraged to not provide any programmatic way to query, set, or modify the build level or to set or modify the violation handler.
这意味着将根据合同进行测试的测试程序将需要额外的编译开关。虽然我不喜欢这样,但我理解为什么要这样做。更令人担心的是最后一部分,以及关于它的问题has been raised。如果设置我们自己的合同违规处理程序的能力是实现定义的,那么跨工具链将不一致。因此,如果可能的话,合同的任何测试都将不可移植。将它设置为链接器的另一个参数也会很尴尬(而且绝对不可移植)。
目前,无法针对 assert
进行测试,因为它只会中止程序,据我所知,自定义处理程序无法阻止这种情况。对于合同,这应该是可以使用自定义处理程序和构建开关实现的,但事实证明,这可能也不可能(或者由实现定义)。
或者是否有任何其他方法可以针对我遗漏的断言和合同进行测试?
在违反合同方面没有太大的实施差异。 “build level”确定合同检查是否完全发生。违规处理程序将在任何检查失败的合同上调用。
处理程序的原型由规范定义。关于违规处理程序的唯一实现差异是您如何设置它。请注意 "implementation-defined" 并不意味着 "does not allow this to happen"。也就是说,the standard says that there will be a mechanism to establish the handler,并且这种机制 将 与实现一起记录(这就是 "implementation-defined" 的意思,而不仅仅是 "unspecified")。该机制究竟是什么将取决于实施,但不提供一个不是一个选项。
唯一可能出现的真正问题是,如果多个实现决定允许用户通过祝福特定的全局名称来指定处理程序,并且这些实现使用不同的全局名称。然而,由于多种原因(与现有代码、宏等冲突),鉴于选择全局名称不是一个好主意,这在现实中不太可能成为问题。所以更有可能的是,他们会选择明显的机制:一个编译器开关,指定要用作处理程序的函数的名称。
请注意,将处理程序定义为静态而不是运行时定义的原因之一是确保开关是 编译器 开关而不是链接器开关.毕竟,编译器会发出任何概念检查代码。所以编译器可以在所述检查代码中使用有问题的函数的名称。
哦,当然,不同的编译器 may/will 有不同的编译器开关来确定调用哪个函数。但他们已经有了不同的开关来生成调试信息、优化级别和基本上所有其他内容。因此,无论您使用什么跨平台测试系统,都已经需要能够处理这些区别。违规处理者只是一个。