测试驱动开发 测试前移_为什么测试驱动的开发有用?

测试驱动开发 测试前移

有关如何更有效地应用TDD的技巧,以及为什么它是一种有价值的技术 (Tips on how to apply TDD more efficiently, and why it's a valuable technique)

There's a common pattern we follow when we start a project using TDD. We describe the specifications of what we expect the system to do in the form of a special test. This "special test" can be an end-to-end with the front-end or an integration test that executes an HTTP request to test the back-end.

当使用TDD启动项目时,我们遵循一种常见的模式。 我们以特殊测试的形式描述了我们期望系统执行的操作的规范。 此“特殊测试”可以是前端的端到端,也可以是执行HTTP请求以测试后端的集成测试。

It's the first test we write. We do it before a single line of code is written. That special test will serve as a guideline to make sure we don't break anything that prevents the regular flow from working. If we don't do that and rely solely on unit tests, there's a chance that, eventually, we will have all tests passing but the server will not be starting or the user won’t be able to do anything on the screen.

这是我们编写的第一个测试。 我们在编写一行代码之前就完成了。 该特殊测试将作为准则,以确保我们不会破坏任何阻止常规流程正常运行的内容。 如果我们不这样做,而仅依靠单元测试,则最终有机会通过所有测试,但服务器将无法启动,否则用户将无法在屏幕上执行任何操作。

When starting a project using TDD, there's a common pattern to create a special test to make sure we don’t break anything that prevents the regular flow from working.
使用TDD启动项目时,通常会使用一种模式来创建特殊测试,以确保我们不会破坏任何妨碍正常流程正常工作的内容。

After we make that special test pass with a naive implementation (or we can keep it failing if we are using ATDD to drive the application internals), we start building the units of the system using a similar pattern on a micro level, never breaking any test we created earlier. We describe each unit of the system through a failing test and make it pass with a naive implementation first. Then, we identify smells and refactor it if necessary so that we can keep the cycle going over and over again.

在通过幼稚的实现使特殊测试通过之后(或者如果我们使用ATDD来驱动应用程序内部,我们可以使其保持失败),我们开始在微观级别上使用类似的模式来构建系统的单元,而不会破坏任何单元我们之前创建的测试。 我们通过失败的测试描述系统的每个单元,并使其首先通过幼稚的实现。 然后,我们识别气味并在必要时对其进行重构 ,以便使循环不断进行。

That’s called the Red/Green/Refactor cycle of TDD.

这称为TDD的红色/绿色/重构周期 。

This cycle will drive us to build all the pieces of our application with enough confidence that it will be robust and maintainable. It will also expose problems early if we were to get stuck due to the wrong assumption of how the API is supposed to behave.

这个周期将驱使我们以足够的信心来构建应用程序的所有部分,以确保其健壮和可维护。 如果由于错误地假设API的行为而使我们陷入困境,它也会尽早暴露出问题。

There's one important thing we should be careful about: we should avoid refactoring code or adding a new test while another test is failing. If we do that, there's a high chance we will get stuck because of the unnecessary cognitive load of worrying about another rule we have already covered. To prevent that, we need to fix the failing test before starting anything else.

我们应该注意一件事: 在另一个测试失败时,我们应该避免重构代码或添加新测试。 如果这样做,很有可能由于担心我们已经涵盖的另一条规则而不必要的认知负担而陷入困境。 为防止这种情况,我们需要在开始任何其他操作之前先修复失败的测试。

In TDD, we should avoid refactoring code or adding a new test while another test is failing.

在TDD中,我们应该避免在另一个测试失败时重构代码或添加新测试。

There are circumstances where one would prefer writing tests after writing the code. However, there are some negative effects that come with that approach:

在某些情况下,人们更愿意在编写代码后编写测试。 但是,这种方法会带来一些负面影响:

  • We can miss important functionality because it’s harder to know if the coverage matches our expectation.

    我们可能会错过重要的功能,因为很难知道覆盖范围是否符合我们的期望。
  • It can create false positives because we won’t see a failing test first.

    它会产生误报,因为我们不会首先看到失败的测试。

  • It can make us over-engineer the architecture because we won’t have any guidelines to force us to write the minimum amount of code that fits in our most basic requirements.

    它可能使我们对架构进行过度设计 ,因为我们没有任何准则可以迫使我们编写满足我们最基本要求的最少代码量。

  • It's harder to validate if the message for the failing test is clear and pointing to the cause of that failure or not.

    很难验证失败测试的消息是否清晰并指出失败的原因。

One thing to keep in mind is that TDD can be posed as a discipline, but there's no way to create a discipline for writing tests after the production code.

要记住的一件事是,可以将TDD视为一门学科,但是无法创建一种在生产代码之后编写测试的学科 。

There are cases when there's no value in applying TDD or automated testing at all. It's when we're testing some IO layers, support functions for the tests, or things built using a declarative language like HTML or CSS (we can test the visual in CSS, but not the CSS code). However, testing is a fundamental part of the process that ensures a complex piece of functionality satisfies a set of expectations. That alone allows us to be confident enough that each part of the system works as expected.

在某些情况下,应用TDD或自动测试根本没有价值 。 是在测试某些IO层,测试的支持功能或使用声明性语言(如HTML或CSS)构建的东西的时候(我们可以在CSS中测试外观,但不能测试CSS代码)。 但是,测试是该过程的基本部分,可确保复杂的功能满足一系列期望。 仅凭这一点,我们就足以确信系统的每个部分都能按预期工作。

There are cases when there's no value in applying TDD or automated testing at all, like when testing IO layers, support functions for the tests, or code written with a declarative language.
在某些情况下,应用TDD或自动测试根本没有价值,例如在测试IO层,测试的支持功能或使用声明性语言编写的代码时。

There's a concept called The Transformation Priority Premise. The TL;DR is that there are some transformations we can apply when making the code more generic in the "green" phase of the TDD cycle.

有一个概念称为“转换优先级前提” 。 TL; DR是在使代码在TDD周期的“绿色”阶段更加通用时可以应用一些转换。

"Refactor" is when we change the structure of the code without changing its behavior. The Transformations are not called "refactoring" because they change the structure and the behavior of the code to make it more generic.

“ 重构 ”是指我们在不更改代码结构的情况下更改其结构。 转换不称为“ 重构 ”,因为它们会更改代码的结构和行为以使其更通用。

An example of using the Transformation Priority is when we make a test that forces us from returning a single constant to returning an argument that will contain more than one value. In this case, it's the constant->scalar priority transformation.

使用转换优先级的一个示例是,当我们进行测试以迫使我们从返回单个常量到返回将包含多个值的参数时。 在这种情况下,这是常量->标量优先级转换。

So what are these transformations? Perhaps we can make a list of them:

那么这些转换是什么? 也许我们可以列出它们:

So what are these transformations? Perhaps we can make a list of them:

那么这些转换是什么? 也许我们可以列出它们:

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)将常量替换为变量或参数 *(statement-> statements)添加更多无条件语句。

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)根本没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量 *(statement-> statements)添加更多无条件语句。

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)将常量替换为变量或参数 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)将常量替换为变量或参数 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)将常量替换为变量或参数 *(statement-> statements)添加更多无条件语句。 *(无条件->如果)分割执行路径 *(标量->数组) *(数组->容器)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)将常量替换为变量或参数 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(array->容器)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)将常量替换为变量或参数 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(数组->容器) *(语句->递归)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)将常量替换为变量或参数 *(statement-> statements)添加更多无条件语句。 *(无条件->如果)分割执行路径 *(标量->数组) *(数组->容器) *(语句->递归)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)* (if->while)

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(array->容器) *(语句->递归) *(if-> while)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)* (if->while)

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)将常量替换为变量或参数 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(array->容器) *(语句->递归) *(if-> while)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)* (if->while)* (expression->function) replacing an expression with a function or algorithm

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(array->容器) *(语句->递归) *(if-> while) *(表达式->函数)用以下表达式替换表达式函数或算法

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)* (if->while)* (expression->function) replacing an expression with a function or algorithm

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(array->容器) *(语句->递归) *(if-> while) *(表达式->函数)用函数或算法

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)* (if->while)* (expression->function) replacing an expression with a function or algorithm* (variable->assignment) replacing the value of a variable.

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(array->容器) *(语句->递归) *(if-> while) *(表达式->函数) 替换变量值的 函数或算法 *(变量->赋值)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)* (if->while)* (expression->function) replacing an expression with a function or algorithm* (variable->assignment) replacing the value of a variable.

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(array->容器) *(语句->递归) *(if-> while) *(表达式->函数)用以下表达式替换表达式 替换变量值的 函数或算法 *(变量->赋值)

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)* (if->while)* (expression->function) replacing an expression with a function or algorithm* (variable->assignment) replacing the value of a variable.There are likely others.

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(array->容器) *(语句->递归) *(if-> while) *(表达式->函数) 替换变量值的 函数或算法 *(变量->赋值) 可能还有其他人。

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)* (if->while)* (expression->function) replacing an expression with a function or algorithm* (variable->assignment) replacing the value of a variable.There are likely others.

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)根本没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)用变量或参数替换常量 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(array->容器) *(语句->递归) *(if-> while) *(表达式->函数)用以下表达式替换表达式 替换变量值的 函数或算法 *(变量->赋值) 可能还有其他人。

So what are these transformations? Perhaps we can make a list of them:* ({}–>nil) no code at all -> code that employs nil* (nil->constant)* (constant->constant+) a simple constant to a more complex constant* (constant->scalar) replacing a constant with a variable or an argument* (statement->statements) adding more unconditional statements.* (unconditional->if) splitting the execution path* (scalar->array)* (array->container)* (statement->recursion)* (if->while)* (expression->function) replacing an expression with a function or algorithm* (variable->assignment) replacing the value of a variable.There are likely others.— Excerpt from The Transformation Priority Premise article

那么这些转换是什么? 也许我们可以列出它们: *({} –> nil)完全没有代码->使用nil的代码 *(nil-> constant) *(constant-> constant +)从简单常量到更复杂的常量 * (constant-> scalar)将常量替换为变量或参数 *(statement-> statements)添加更多无条件语句。 *(无条件-> if)分割执行路径 *(标量->数组) *(array->容器) *(语句->递归) *(if-> while) *(表达式->函数)用以下表达式替换表达式 替换变量值的 函数或算法 *(变量->赋值) 可能还有其他人。 —摘自《转型优先前提》一文

In TDD, The Transformation Priority Premise can give us a guideline for the "green" phase.
在TDD中,“转换优先级前提”可以为我们提供“绿色”阶段的指导。

Writing correct software is hard. TDD is a common pattern where we use the tests to help driving the implementation of our system while retaining a huge percentage of test coverage. However, it's not a Silver Bullet.

编写正确的软件非常困难 。 TDD是一种常见的模式,在这种模式下,我们使用测试来帮助推动系统的实施,同时保留很大比例的测试覆盖率。 但是,它不是Silver Bullet 。

If we are using TDD, we should avoid refactoring the code when the tests are failing. To make it pass in the "green" phase, we use the Transformation Priority Premise to guide us in the most naive implementation approach we can take before refactoring.

如果使用的是TDD,则应在测试失败时避免重构代码。 为了使其通过“绿色”阶段,我们使用“转换优先级前提”来指导我们采用重构之前可以采取的最幼稚的实施方法。

In comparison with other ways of writing tests, TDD can take more time in the beginning. However, as with every new skill, with enough practice we will reach a plateau, and the time it takes to apply TDD will be no different than the time it would take to write tests in a traditional way.

与其他编写测试的方式相比,TDD在开始时可能会花费更多时间。 但是,与每一项新技能一样,如果有足够的实践,我们将达到一个平稳阶段,并且应用TDD所花费的时间与以传统方式编写测试所花费的时间没有什么不同。

The difference now is that your software will be less likely to behave in a way you didn't expect.

现在的区别是您的软件将不太可能以您未曾期望的方式运行。

And for all practical means, that's no different than 100% test coverage.

对于所有实际方法,这与100%的测试覆盖率没有什么不同。

Thanks for reading. If you have some feedback, reach out to me on Twitter, Facebook or Github.

谢谢阅读。 如果您有任何反馈意见,请通过Twitter , Facebook或Github与我联系。

翻译自: https://www.freecodecamp.org/news/why-test-driven-development-4fb92d56487c/

测试驱动开发 测试前移

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/396027.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Anaconda管理多版本的python环境

通过Conda的环境管理功能,我们能同时安装多个不同版本的Python,并能根据需要自由切换。下面我将给大家分享一下,新增Python版本,切换,再切回主版本的详细过程。 方法/步骤 1首先确保你的系统里已经安装了Conda&#xf…

父子沪c转大牌过户_机动车异地过户(转籍)

最近我家换了一辆车,导航后台数据统计是去足浴城最多的车主,尬!从想起这个品牌到付定金,也就半天时间,买之前没了解这么透彻。不过,到手驾驶,还是比之前的车舒适很多的,就是容易在不…

android安卓系统2.3 使用说明书,Android2.3操作界面

Android2.3操作界面摩托罗拉XT882的界面相对于原生的Gingerbread还是有了不小的变化,首先最大的感觉就是主色调亮了很多。默认背景在qHD分辨率下非常的清晰,同时整个界面仍然采用了多分屏界面。下方由中国电信定制,状态栏加入了全新的单个状态…

《运营力——微信公众号 设计 策划 客服 管理 一册通》一一1.2 团队岗位介绍...

本节书摘来自异步社区出版社《运营力——微信公众号 设计 策划 客服 管理 一册通》一书中的第1章,第1.2节,作者: 杭州创博通信技术有限公司 , 施瑶君,更多章节内容可以访问云栖社区“异步社区”公众号查看。 1.2 团队岗位介绍 创…

一切都是关于“ –ilities”的

by George Stepanek通过乔治斯蒂芬内克 都是关于“邪恶”的 (It’s all about the “-ilities”) We were “feature complete.”我们“功能齐全”。 Four weeks into a 10-week Free Code Camp project to build an environmental pledge web application, we had gotten al…

1,滑动验证,前后台接口

http://www.geetest.com/install/sections/idx-client-sdk.html 转载于:https://www.cnblogs.com/yexiangwang/p/5481153.html

Linux 下 nginx反向代理与负载均衡

前面几篇记录下nginx的基本运功,代理服务器的访问,这里来试验下nginx的反向代理。 反向代理(Reverse Proxy)方式是指以代理服务器来接受internet上的连接请求,然后将请求转发给内部网络上的服务器,并将从服…

android 8.1没声音,Android 8.1重大改变!耳机孔不见了

原标题:Android 8.1重大改变!耳机孔不见了今天上午,Android Police爆料称,下一代的Pixel 2将首发Android 8.1。更重要的是,在这个新系统中,谷歌已经做好了放弃3.5mm耳机插口的准备,并将在底层优…

php变量前下滑_PHP变量

变量来源于数学,是计算机语言中能储存计算结果或能表示值抽象概念。变量可以通过变量名访问。变量是存储数据的“容器”。命名规则变量以 $ 符号开始,后面跟着变量的名称变量名必须以字母或者下划线字符开始变量名只能包含字母数字字符以及下划线(A-Z、a…

《计算机科学概论(第12版)》—第0章0.3节学习大纲

本节书摘来自异步社区《计算机科学概论(第12版)》一书中的第0章0.3节学习大纲,作者【美】J. 格伦•布鲁克希尔(J. Glenn Brookshear) , 丹尼斯•布里罗(Dennis Brylow),更多章节内容可以访问云栖…

blued停止邮箱注册_停止让我注册!

blued停止邮箱注册by Conor Sheehan由Conor Sheehan 停止让我注册! (Stop Making Me Sign Up!) Installing a new app can be exciting. When you’ve found one that may be just what you need, opening it is like unboxing a new toy. So why do so many apps …

Android Sutido 编译速度优化

虽然Android Studio 此时已经更新到了Android Studio 2.1版本,build 版本android-studio-bundle-143.2739321。但是在安装该版本都是根据自己的标准进行安装,所以需要在安装之后进行一系列的调整。下面文章根据3个方面进行讲解。分别为Android Studio本身…

卷积神经网络计算题试题_卷积神经网络的计算

转自:https://zhuanlan.zhihu.com/p/631747741. 卷积卷积神经网络中的卷积是指定义好卷积核(kernel),并对图像(或者特征图,feature map)进行滑动匹配,即对应位置相乘再相加。其特点就在于能够捕捉局部的空间特征。具体过程如下图所…

html字符串转换jsx,javascript – 将React.element转换为JSX字符串

我正在尝试构建一个组件,>带孩子和>渲染DOM中的子项,以及>出于文档的目的,在pre中显示子DOM一种解决方案是将JSX作为单独的prop传递.这使得它重复,因为我已经能够通过this.props.children访问它.理想情况下,我只需要以某种方式将子prop转换为字符串,以便我可以在pre中…

Leetcode:0002(两数之和)

LeetCode:0002(两数之和) 题目描述:给定两个非空链表来表示两个非负整数。位数按照逆序方式存储,它们的每个节点只存储单个数字。将两数相加返回一个新的链表。你可以假设除了数字 0 之外,这两个数字都不会…

《Excel 职场手册:260招菜鸟变达人》一第 13 招 利用数据验证给单元格添加注释,不用批注...

本节书摘来异步社区《Excel 职场手册:260招菜鸟变达人》一书中的第1章,第13节,作者: 聂春霞 , 佛山小老鼠 责编: 王峰松,更多章节内容可以访问云栖社区“异步社区”公众号查看。 第 13 招 利用数据验证给单…

招银网络笔试java_春招|招银网络Java软件开发 电话面试+一二三面面经

电话面试:1.自我介绍2.介绍项目经历(我介绍的时候说了用到spring,spring MVC和hibernate框架)3.为什么用spring框架,有什么优点4.详细说下aop5.解释下IOC,IOC有什么好处6.spring MVC的运行流程7.spring除了注解注入还有什么方式8.hibernate框…

Numpy and Pandas

安装 视频链接:https://morvanzhou.github.io/tutorials/data-manipulation/np-pd/ pip install numpy pip install pandas Numpy 学习 Numpy属性 import numpy as nparray np.array([[1,2,3],[2,3,4]]) print(array) print(number of dim:,array.ndim)//几维度 pr…

认证android retrofit,Retrofit之项目介绍

项目介绍官网对retrofit介绍是这是一个"类型安全(type-safe)"的Android/Java http客户端. 目前retrofit的最新正式版本为1.9.0. 2.0版本预计2015年底发布, 相较于之前版本, 2.0版本在架构上做了很大改变, 本文代码相关的内容都是基于retrofit2.0-beta2.注: 在编程语言…

层次聚类算法 算法_聚类算法简介

层次聚类算法 算法Take a look at the image below. It’s a collection of bugs and creepy-crawlies of different shapes and sizes. Take a moment to categorize them by similarity into a number of groups.看看下面的图片。 它是各种形状和大小的错误和令人毛骨悚然的爬…