From 9741275c72d1872e8e306242b7c64d56368aa1f8 Mon Sep 17 00:00:00 2001
From: DeepMissea <398752853@qq.com>
Date: Sun, 19 Mar 2017 15:45:57 +0800
Subject: [PATCH 001/945] =?UTF-8?q?=E5=88=9D=E6=AD=A5=E7=BF=BB=E8=AF=91?=
=?UTF-8?q?=E5=AE=8C=E6=88=90=E3=80=82?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/improving-swift-compile-times.md | 63 ++++++++++++++-------------
1 file changed, 32 insertions(+), 31 deletions(-)
diff --git a/TODO/improving-swift-compile-times.md b/TODO/improving-swift-compile-times.md
index effe8999af1..889f74d6cba 100644
--- a/TODO/improving-swift-compile-times.md
+++ b/TODO/improving-swift-compile-times.md
@@ -1,27 +1,28 @@
-* 原文地址:[Improving Swift compile times](https://medium.com/@johnsundell/improving-swift-compile-times-ee1d52fb9bd#.hfqaeq76p)
+> * 原文地址:[Improving Swift compile times](https://medium.com/@johnsundell/improving-swift-compile-times-ee1d52fb9bd#.hfqaeq76p)
* 原文作者:[John Sundell](https://medium.com/@johnsundell?source=post_header_lockup)
* 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-* 译者:
+* 译者:[Deepmissea](http://deepmissea.blue)
* 校对者:
-# Improving Swift compile times #
-For all its awesomeness, one thing that can sometimes be quite cumbersome when working with Swift on a bit larger scale is how long it can *currently* take to compile. While it’s expected that compile times are going to be longer in Swift compared to, for instance, Objective-C — since the Swift compiler does so much more in terms of assuring runtime safety — I wanted to look into if we can somehow help the compiler out to make it able to work faster.
+# 提高 Swift 的编译时间
-So last week I dove into one of our larger Swift projects at [Hyper](http://www.hyper.no) . It has somewhere around 350 source files and 30,000 lines of code. In the end I managed to reduce the average time for a clean build on this project [by over 20%](https://twitter.com/johnsundell/status/837318595973611521) — so I thought I’d spend this week’s blog post detailing how I did it.
+在 Swift 所有的魅力中,有一件是可能会相当麻烦,那就是在用 Swift 编写更大规模的项目时,它**一般**会编译多久。尽管 Swift 编译器在保证运行时安全方面做的更多,但是它的编译时间要比 Objective-C 编译时间要长很多。我想研究一下,是否我们可以帮助编译器让他工作的更快。
-Now, before we begin, I just want to say that **I don’t intend this post to in any way be critisism towards Swift or the team working on it** — I know the developers working on the Swift compiler, both at Apple and in the open source community, are continuously making major improvements in both the speed, functionality and stability of the compiler. Hopefully this blog post will be rendered redundant over time, but until then I just want to provide some practical tips & tricks that I’ve found can make compile times faster.
+所以,上周我在 [Hyper](http://www.hyper.no) 投身于我们 Swift 的大项目之一。它大哥有 350 个源文件以及 30,000 行的代码。最后我设法将这个项目的平均构建时间减少了 [20%](https://twitter.com/johnsundell/status/837318595973611521)。所以我想在我这周的博客上详细的介绍我是怎么做的。
-#### Step 1: Gather data ####
+现在,在我们开始之前,我只想说**我不想这篇文章以任何形式的方式来批判 Swift 或它的团队工作**。我知道 Swift 编译器的开发者,包含 Apple 公司和开源社区,都在持续地对编译器速度、功能和稳定性做出重大改进。希望这篇博文能随着时间的流逝而感到多余,但在那之前,我只是想提供一些我发现可以提升编译速度的实用技巧。
-Before starting any optimization work, it’s always good to establish a baseline that you can measure your improvements against. For me, this was done through two simple scripts that I added as *Run script phases* for the app’s target in Xcode.
+#### Step 1: 采集数据
-Before *Compile Sources*, I added the following script:
+在开始优化工作之前,建立一个能衡量你改进的基准总是好的。我是通过在 Xcode 里添加两个简单的脚本到应用的 target 作为**运行脚本阶段**来实现的。
+
+在**编译源文件**之前,添加下面的脚本:
```
echo "$(date +%s)" > "buildtimes.log"
```
-and at the end, I added this script:
+在最后,添加这个脚本:
```
startime=$( "buildtimes.log"
```
-Now, this measures only the time it takes to compile the **app’s own source files** (in order to measure the compile time for the entire app, you could use Xcode behaviors to hook into the *Build Starts* and *Build Succeeds* events). Since compile times vary a lot depending on what machine the code is being compiled on — I also *git ignored buildtimes.log*.
+现在,这个脚本只会测算编译器编译**应用自己的源文件**的时间(为了测量出整个引用的编译时间,你可以使用 Xcode 的特性来挂载(hook)到 **Build Starts** 和 **Build Succeeds** 上)。由于编译时间非常依赖于编译它的设备,所以我也 **git ignored 了 buildtimes.log 文件**。
-Next, I wanted to highlight what specific code blocks that take extra long to compile, in order to identify bottlenecks that I could then fix. To do this, you can simply set a threshold by passing the following arguments to the Swift compiler under the *Other Swift Flags *build setting in Xcode:
+接下来,我想突出哪些个别代码块耗费了额外的长时间来编译,以便识别瓶颈,这样我就可以修复它。要做到这个,就需要通过向 Xcode 中 Build Setting 里的 **Other Swift Flags** 传递下面的参数给 Swift 编译器来设置一个临界值:
```
-Xfrontend -warn-long-function-bodies=500
```
-Using the above arguments you will get a warning if any function in your project takes more than **500 miliseconds** to compile. This is the threshold I started out with (and the continously lowered it as I fixed more and more bottlenecks).
+使用上面的参数后,在你的项目中,如果有任何函数耗费了超过 500 毫秒的编译时间,你就会得到一个警告。这是我开始设置的临界值(并且随着我对更多瓶颈的修复,这个标准在不断的降低)。
-#### Step 2: Fix all the warnings ####
+#### Step 2: 消除所有的警告
-When enabling warnings for long function compile times, you will probably start to see a few of them in your project. At first, it can look seemingly random that a function takes long to compile, but soon patterns start to emerge. Here are two common patterns that I’ve noticed take particularly long to compile using the Swift 3.0 compiler:
+在设置了函数编译时间过长的警告之后,你可能会在项目中开始发现一些。最开始,你会觉得编译时间过长的函数是随机的,但是很快模式(patterns)就开始出现了。这里我注意到了两个使用 Swift 3.0 编译器编译函数时间过长的常见模式:
-**Custom operators (especially overloaded ones with generic parameters)**
+**自定义运算符(特别是带有通用参数的重载)**
-One of the concepts that were new to many iOS & macOS developers when Swift came out is operator overloads, and like many new shiny things — we get excited about trying them out. Now, I’m not going to argue here whether custom operators & overloads are good or bad, but they can have a pretty big impact on compile times, especially if used it more complex expressions.
+当 Swift 出现时,对于大多数 iOS 和 macOS 开发者来说,运算符重载是全新的概念之一,但就像许多新鲜事物一样,我们很兴奋的使用它们。现在,我不打算在这讨论自定义或重载运算符是好是坏,但它们的确对编译时间有很大影响,尤其是如果使用更加负载的表达式。
-Consider the following operator, that adds two *IntegerConvertible* numbers to form a custom number type:
+思考下面的运算符,添加了两个 **IntegerConvertible** 类型的数字来形成自定义的数字类型:
```
func + CustomNumber {
@@ -71,7 +72,7 @@ func addNumbers() -> CustomNumber {
}
```
-Looks simple enough, but the above* addNumbers()* function takes quite a long time to compile (over *300 ms* on my late 2013 MBP). Compare that to if we implement the same logic but using a protocol extension instead:
+看上去很简单,但是上面的 **addNumbers()** 函数会花费很长一段时间来编译(在我 2013 年的 MBP 上超过 **300 ms**)。对比一下,如果我们用协议扩展来实现相同逻辑:
```
extension IntegerConvertible {
@@ -88,13 +89,13 @@ func addNumbers() -> CustomNumber {
}
```
-With this change, our *addNumbers()* function now takes **less than 1 ms to compile**. That’s **~300 times faster!**
+通过这个改变,我们的 **addNumbers()** 函数现在编译时间**不到 1 ms**。**这快了 300 倍!**
-So, if you are making heavy use of custom/overloaded operators, especially ones with generic parameters (or if you’re using 3rd party libraries that do so — like many Auto Layout libraries), consider rewriting the same logic but using normal functions, protocol extensions or some other technique instead.
+所以,如果你大量的使用了自定义/重载运算符,特别是带有通用参数的(或者如果你使用的第三方库来做这些,比如许多自动布局的库),考虑一下用普通函数、协议扩展或其他的技术来重写吧。
-**Collection literals**
+**集合字面量**
-Another pattern that I’ve found to often become a compile time bottleneck is the use of collection literals, especially when the compiler needs to do a lot of work to infer the type of those literals. Let’s say you have a method that converts a model into a JSON-like dictionary, like this:
+另一个我发现的编译时间瓶颈是使用集合字面量,特别是编译器需要做很多工作来推断那些字面量的类型。让我们假设你有一个函数,它要把模型转换成一个类似 JSON 的字典,像这样:
```
extension User {
@@ -116,7 +117,7 @@ extension User {
}
```
-The above* toJSON()* function takes my computer about *500 ms* to compile. Now let’s try to construct that very same dictionary line-by-line instead of using a literal:
+上面 **toJSON()** 函数在我的电脑上大概要 **500 ms** 的时间来编译。现在让我们试着逐行重构这个像字典的东西来代替字面量:
```
extension User {
@@ -138,14 +139,14 @@ extension User {
}
```
-It now compiles in around *5 ms* — **100 times faster!**
+它现在编译时间大概在 **5 ms** 左右,**提高了 100 倍!**
-#### Step 3: Conclusions ####
+#### Step 3: 结论 ####
-What both of the above examples make very clear is that some of the nice features of the Swift compiler, such as type inference and overloading, come at a time cost. This is, if we think about it, quite logical. Since the compiler has to do more work to perform inference, it will take longer. But as we can also see above, if we just slightly tweak our code to help the compiler resolve our expressions more easily — we can dramatically speed up our compile times.
+上面的两个例子非常清晰的说明了 Swift 编译器的一些新特性,比如类型推演和重载,都是付出了时间开销。如果我们仔细思考一下,也很符合逻辑。由于编译器不得不做更多的工作来执行推演,所以花费了更多的时间。但是我们也看到了,如果我们稍微调整一下我们的代码,帮助编译器更简单的解决表达式,我们就可以很大程度的加快编译时间。
-Now, I’m not saying that you should always let compile times guide your decisions on how to write code. Sometimes it may be worth having the compiler do more work, if it makes your code clearer and easier to understand. But in large projects, coding techniques that drive compile times up in the 300–500 ms range (or higher) per function can quite quickly become a problem. My suggestion would be to keep monitoring your compile times, set a reasonable threshold for warnings using the above mentioned compiler flags, and address problems whenever they occur.
+现在,我不是说你要一直让编译时间来决定你写代码的方式。有时可以让它做更多的工作,让你的代码更加清晰并且容易理解。但是在大型的项目中,每个函数要用 300-500 ms 范围(或更多)的时间来编译的编码技术可能很快就会成为一个问题。我的建议是对你的编译时间保持监控,使用上面的编译标记设置一个合理的临界值,并在发现问题的时候解决问题。
-I’m sure the examples above don’t cover all potential areas of compile time improvements, so I’d love to hear from you if you have any other techniques that you’ve found useful to speed up compile times in large Swift projects. Either post a response here on Medium, or contact me on [**Twitter @johnsundell**](https://twitter.com/johnsundell).
+我确信上面的例子肯定没有涵盖所有潜在的编译时间改进,所有我很愿意听到你的意见。如果你有任何有用的改进大型 Swift 项目编译时间的其他的技术,你可以写在 Medium 上,或者在 [**Twitter @johnsundell**](https://twitter.com/johnsundell) 上联系我。
-Thanks for reading! 🚀
+感谢阅读!🚀
From 0a5f85276d3671e3227db1b4cda255eab1f30dec Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Wed, 22 Mar 2017 18:40:14 +0800
Subject: [PATCH 002/945] =?UTF-8?q?=E5=85=A8=E6=96=87=E9=80=9A=E8=AF=91?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...in-javascript-part-1-observable-objects.md | 90 +++++++++++++++----
1 file changed, 73 insertions(+), 17 deletions(-)
diff --git a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
index 238e7673c02..73d504b8c39 100644
--- a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
+++ b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
@@ -7,31 +7,49 @@
![](https://d4a7vd7s8p76l.cloudfront.net/uploads/1484604970-4-7876/observables.png)
# [How to build a reactive engine in JavaScript. Part 1: Observable objects](/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects/) #
+如何用 JavaScript 构建响应式引擎 —— Part 1:看得见的对象
## The reactive way ##
+## 响应的方式 ##
+
With the growing need for robust and interactive web interfaces, many developers have started embracing the reactive programming paradigm.
+随着健壮的可交互的网站界面需求的增多,很多开发者开始拥抱响应式编程规范。
Before we begin implementing our own reactive engine, let’s quickly explain what reactive programming actually is. Wikipedia gives us a classic example of a reactive interface implementation – namely a spreadsheet. Defining a formula such as `=A1+B1` would update the cell whenever either `A1` or `B1` change. Such a formula can be considered a computed value.
+在开始实现我们自己的响应式引擎之前,让我们快速的解释一下到底什么是响应式编程。维基百科给出一个经典的响应式界面实现的例子 —— 叫做 spreadsheet。定义一个准则,对于`=A1+B1`,只要 `A1` 或 `B1` 发生变化, `=A1+B1` 也会随之变化。这个准则被认为是 computed value。
You will learn how to implement computed values in the second part of this reactive series. Before that, we first need a base for our reactivity engine.
+我们将会在这系列教程的 Part 2 部分学习如何实现 computed value。在那之前,我们首先需要对响应式引擎有个基础的了解。
## The engine ##
+## 引擎 ##
Currently there are many different approaches to solving the problem of observing and reacting to the changing application state.
+目前有很多不同解决方案可以观察到应用状态的改变并对其做出反应。
- Angular 1.x has its dirty checking.
- React, because of the way it works – doesn’t actually track changes in the data model. It uses the virtual DOM to diff and patch the DOM.
- Cycle.js and Angular 2 prefer the reactive streams implementations like XStream and Rx.js.
- Libraries like Vue.js, MobX or Ractive.js all use a variation of getters/setters to create observable data models.
+- Angular 1.x 有脏检查。
+- React 由于它工作方式的原因,并不追踪数据模型中的改变。 它用虚拟 DOM 比较并修补 DOM。
+- Cycle.js 和 Angular 2 更喜欢响应流方式实现,类似 XStream 和 Rx.js.
+- 像 Vue.js, MobX 或 Ractive.js 这些库都使用 getters/setters 变量创建可观察的数据模型。
+
In this tutorial, we will go the getters/setters way of observing and reacting to changes.
+在这篇教程中,我们将使用 getters/setters 的方式观察并响应变化。
> Note: To keep the tutorial as simple as possible, the code lacks the support for non-primitive data types or nested properties and many of the required sanity checks, thus by no means should this code be considered production ready. The code below is written using the ES2015 standard and is loosely inspired by the Vue.js reactive engine implementation.
+> 注意:为了让这篇教程尽量保持简单,代码缺少对非初级数据类型或嵌套属性的支持,并且很多内容需要完整性检查,因此这些决不能认为这些代码已经可以用于生产环境。下面的代码是受 Vue.js 启发的响应式引擎的实现,使用 ES2015 标准编写。
+
## The observable object ##
+## 可见对象 ##
Let’s start with a `data` object, whose properties we want to observe.
+让我们从一个 `data` 对象开始,我们想要观察到的是它的属性。
```
let data = {
@@ -42,6 +60,7 @@ let data = {
```
Let’s start by creating two functions that will transform our object’s properties into observable properties using the getter/setter functionality.
+让我们从这里开始,首先创建两个函数,使用 getter/setter 的功能,将对象的属性转换成可见属性。
```
function makeReactive (obj, key) {
@@ -49,16 +68,16 @@ function makeReactive (obj, key) {
Object.defineProperty(obj, key, {
get () {
- return val // Simply return the cached value
+ return val // 简单的返回缓存的 value
},
set (newVal) {
- val = newVal // Save the newVal
- notify(key) // Ignore for now
+ val = newVal // 保存 newVal
+ notify(key) // 暂时忽略这里
}
})
}
-// Iterate through our object keys
+// 循环迭代对象的 keys
function observeData (obj) {
for (let key in obj) {
if (obj.hasOwnProperty(key)) {
@@ -71,38 +90,47 @@ observeData(data)
```
By running `observeData(data)` we transform our object into an object capable of being observed; now we have a way to create notifications whenever the value changes.
+通过运行 `observeData(data)`,将原始的对象转换成具有可见性的对象;现在当对象的 value 发生改变时,我们有获得通知的办法。
## Reacting to changes ##
+## 响应变化 ##
Before we begin *notifying*, we need something that we can actually notify. This is a perfect example where we can use the observer pattern. In this case we will make use of the signals implementation.
+在我们开始接收 *notifying* 前,我们需要一些通知的内容。这是一个使用观察者模式极好的例子。在这个案例中我们将使用 signals implementation。
Let’s start with the `observe` function.
+我们从 `observe` 函数开始。
+
```
-let signals = {} // Signals start as an empty object
+let signals = {} // Signals 从一个空对象开始
function observe (property, signalHandler) {
- if(!signals[property]) signals[property] = [] // If there is NO signal for the given property, we create it and set it to a new array to store the signalHandlers
+ if(!signals[property]) signals[property] = [] // 如果给定属性没在 signal 中,则创建该属性,并将其设置为空数组来存储 signalHandlers
- signals[property].push(signalHandler) // We push the signalHandler into the signal array, which effectively gives us an array of callback functions
+ signals[property].push(signalHandler) // 将 signalHandler 存入 signal 数组,高效的获得一组存在数组中的回调函数
}
```
We can now use the `observe` function like this: `observe('propertyName', callback)`, where `callback` is a function that should be called each time the property’s value has changed. When we **observe** a property multiple times, each callback will be stored inside the corresponding property’s signal array. This way we can store all callbacks and have easy access to them.
+我们现在可以这样用 `observe` 函数: `observe('propertyName', callback)`,每次属性值发生改变的时候 `callback` 函数应该被调用。
Now for the `notify` function that you saw before.
+现在来看一下上文中提到的 `notify` 函数。
```
function notify (signal, newVal) {
- if(!signals[signal] || signals[signal].length < 1) return // Early return if there are no signal handlers
+ if(!signals[signal] || signals[signal].length < 1) return // 如果没有 signal 的处理器则提前 return
- signals[signal].forEach((signalHandler) => signalHandler()) // We call each signalHandler that’s observing the given property
+ signals[signal].forEach((signalHandler) => signalHandler()) // 调用给定属性的每个 signalHandler
}
```
As you can see, now every time one of the properties changes, the assigned signalHandlers will be called.
+如你所见,现在每次一个属性发生变化,就会调用对其分配的 signalHandlers。
So let’s wrap it all up into a factory function that we pass the data object that has to be reactive. I will name mine `Seer`. We end up with something like this:
+所以我们把它全部封装起来做成一个工厂函数,传入想要响应的数据对象,我把它命名为 `Seer`。我们最终得到的像下面这样
```
function Seer (dataObj) {
@@ -110,8 +138,8 @@ function Seer (dataObj) {
observeData(dataObj)
- // Besides the reactive data object, we also want to return and thus expose the observe and notify functions.
- return {
+ // 除了响应的数据对象,我们也需要返回并且暴露出 observe 和 notify 函数。
+ return {
data: dataObj,
observe,
notify
@@ -154,6 +182,7 @@ function Seer (dataObj) {
```
All we need to do now is to create a new reactive object. Thanks to the exposed `notify` and `observe` functions, we can observe and react to the changes made to the object.
+现在我们需要做的就是创建一个新的可响应对象。多亏了暴露出来的 `notify` 和 `observe` 函数,我们可以观察到并响应对象的改变。
```
const App = new Seer({
@@ -163,11 +192,11 @@ const App = new Seer({
age: 25
})
-// To subscribe and react to changes made to the reactive App object:
+// 为了订阅并响应可响应 APP 对象的改变:
App.observe('firstName', () => console.log(App.data.firstName))
App.observe('lastName', () => console.log(App.data.lastName))
-// To trigger the above callbacks simply change the values like this:
+// 为了触发上面的回调函数,像下面这样简单的改变 values:
App.data.firstName = 'Sansa'
App.data.lastName = 'Stark'
@@ -175,58 +204,79 @@ App.data.lastName = 'Stark'
Simple, isn’t it? Now that we have the basic reactivity engine covered, let’s make some use of it.
I mentioned that with the more reactive approach to front-end programming, we should not be concerned with things like manually updating the DOM after each change.
+很简单,是不是?现在我们讲完了基本的响应式引擎,让我们来用用它。
+我提到过随着前端编程可响应式方法的增多,我们不应该总想着在发生改变后手动的更新 DOM。
There are many approaches to this. I guess the most trending one right now is the so called virtual DOM. If you are interested in learning how to create your own virtual DOM implementation, there are already great tutorials for this. However, here we will go with a much simpler approach.
+有很多方法来完成这项任务。我猜现在流行的趋势是用虚拟 DOM 的办法。如果你对学习如何创建你自己的虚拟 DOM 实现感兴趣,已经有很多这方面的教程。然而,这里我们将用到更简单的方法。
Let’s say our HTML looks like this:`html
Title comes here
`
+HTML 看起来像这样: `html
Title comes here
`
The function responsible for updating the DOM would look like this:
+响应式更新 DOM 的函数看起来像这样:
```
// First we need to get the node that we want to keep updating.
+// 首先需要获得想要保持更新的节点。
const h1Node = document.querySelector('h1')
function syncNode (node, obj, property) {
// Initialize the h1’s textContent value with the observed object’s property value
- node.textContent = obj[property]
+ // 用可见对象的属性值初始化 h1 的 textContent 值
+ node.textContent = obj[property]
// Start observing the property using our Seer instance App.observe method.
- App.observe(property, value => node.textContent = obj[property] || '')
+ // 开始用我们的 Seer 的实例 App.observe 观察属性。
+ App.observe(property, value => node.textContent = obj[property] || '')
}
syncNode(h1Node, App.data, 'title')
```
This will work but actually requires a lot of work from us to actually bind all the DOM elements to the desired data models.
+这样做事可行的,但是实际使用它绑定数据模型到 DOM 元素需要大量的工作。
That’s why we can go a step further and automate all of this.
If you are familiar with AngularJS or Vue.js you surely remember using custom HTML attributes like `ng-bind` or `v-text`. We will create something similar here!
Our custom attribute will be called `s-text`. We will look for it to create bindings between the DOM and the data model.
+这就是我们为什么要再向前迈一步,然后将所有这些自动化完成。
+如果你熟悉 AngularJS 或者 Vue.js,你肯定记得使用自定义属性 `ng-bind` 或 `v-text`。我们在这里创建类似的东西。
+我们的自定义属性叫做 `s-text`。我们将寻找在 DOM 和 数据模型之间建立绑定的方式。
+
Let’s update our HTML:
+让我们更新一下 HTML:
```
+
Title comes here
function parseDOM (node, observable) {
// We get all nodes that have the s-text custom attribute
+ // 我们获得所有具有自定义属性 s-text 的节点
const nodes = document.querySelectorAll('[s-text]')
// For each existing node, we call the syncNode function
- nodes.forEach((node) => {
+ // 对于每个存在的节点,我们调用 syncNode 函数
+ nodes.forEach((node) => {
syncNode(node, observable, node.attributes['s-text'].value)
})
}
// Now all we need to do is call it with document.body as the root node. All `s-text` nodes will automatically create bindings to the corresponding reactive property.
+// 现在我们需要做的就是在根节点 document.body 上调用它。所有的 `s-text` 节点将会自动的创建与之对应的响应式属性的绑定。
parseDOM(document.body, App.data)
```
## Summary ##
+## 总结 ##
Now that we have a way to parse the DOM and bind the nodes to the data model, let’s add those two functions into the Seer factory function, where we will parse the DOM on initialization.
+现在我们可以解析 DOM 并且将数据模型绑定到节点上,把这两个函数添加到 Seer 工厂函数中,这样就可以在初始化的时候解析 DOM。
The result should look like this:
+结果应该像下面这样:
```
function Seer (dataObj) {
@@ -293,6 +343,7 @@ function Seer (dataObj) {
```
Example on JsFiddle:
+JsFiddle 上的例子:
HTML
@@ -408,14 +459,19 @@ Result
![Markdown](http://i2.buimg.com/1949/cf89248985467d6f.png)
The above code can be found here: [github.com/shentao/seer](https://github.com/shentao/seer/tree/master)
+上文的代码可以在这里找到: [github.com/shentao/seer](https://github.com/shentao/seer/tree/master)
## To be continued... ##
+## 未完待续…… ##
This is the first part in a series about crafting your own reactivity engine.
+这篇是制作你自己的响应式引擎系列文章中的第一篇。
**The next part will be about creating computed properties, where each has its own trackable dependencies.**
+**下一篇将是关于创建 computed properties,每个属性都有它自己的可追踪依赖。**
Your feedback and ideas on what to cover next are both very welcome in the comments!
+非常欢迎在评论区提出你对于下一篇文章讲述内容的反馈和想法!
Thanks for reading.
-
+感谢阅读。
From e91f4baaa79693df1e607d904cca7d82ce335e44 Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Wed, 22 Mar 2017 19:43:31 +0800
Subject: [PATCH 003/945] =?UTF-8?q?=E5=85=A8=E6=96=87=E6=A0=A1=E5=AF=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...in-javascript-part-1-observable-objects.md | 50 +++++++++----------
1 file changed, 25 insertions(+), 25 deletions(-)
diff --git a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
index 73d504b8c39..b90d3137c03 100644
--- a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
+++ b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
@@ -1,23 +1,22 @@
> * 原文地址:[How to build a reactive engine in JavaScript. Part 1: Observable objects](https://monterail.com/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects)
> * 原文作者:[Damian Dulisz](https://disqus.com/by/damiandulisz/)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[IridescentMia](https://github.com/IridescentMia)
> * 校对者:
![](https://d4a7vd7s8p76l.cloudfront.net/uploads/1484604970-4-7876/observables.png)
# [How to build a reactive engine in JavaScript. Part 1: Observable objects](/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects/) #
-如何用 JavaScript 构建响应式引擎 —— Part 1:看得见的对象
+# 如何用 JavaScript 构建响应式引擎 —— Part 1:可见的对象 #
## The reactive way ##
-
## 响应的方式 ##
With the growing need for robust and interactive web interfaces, many developers have started embracing the reactive programming paradigm.
-随着健壮的可交互的网站界面需求的增多,很多开发者开始拥抱响应式编程规范。
+随着对强健、可交互的网站界面的需求不断增多,很多开发者开始拥抱响应式编程规范。
Before we begin implementing our own reactive engine, let’s quickly explain what reactive programming actually is. Wikipedia gives us a classic example of a reactive interface implementation – namely a spreadsheet. Defining a formula such as `=A1+B1` would update the cell whenever either `A1` or `B1` change. Such a formula can be considered a computed value.
-在开始实现我们自己的响应式引擎之前,让我们快速的解释一下到底什么是响应式编程。维基百科给出一个经典的响应式界面实现的例子 —— 叫做 spreadsheet。定义一个准则,对于`=A1+B1`,只要 `A1` 或 `B1` 发生变化, `=A1+B1` 也会随之变化。这个准则被认为是 computed value。
+在开始实现我们自己的响应式引擎之前,快速的解释一下到底什么是响应式编程。维基百科给出一个经典的响应式界面实现的例子 —— 叫做 spreadsheet。定义一个准则,对于 `=A1+B1`,只要 `A1` 或 `B1` 发生变化,`=A1+B1` 也会随之变化。这个准则被认为是 computed value。
You will learn how to implement computed values in the second part of this reactive series. Before that, we first need a base for our reactivity engine.
我们将会在这系列教程的 Part 2 部分学习如何实现 computed value。在那之前,我们首先需要对响应式引擎有个基础的了解。
@@ -26,7 +25,7 @@ You will learn how to implement computed values in the second part of this react
## 引擎 ##
Currently there are many different approaches to solving the problem of observing and reacting to the changing application state.
-目前有很多不同解决方案可以观察到应用状态的改变并对其做出反应。
+目前有很多不同解决方案可以观察到应用状态的改变,并对其做出反应。
- Angular 1.x has its dirty checking.
- React, because of the way it works – doesn’t actually track changes in the data model. It uses the virtual DOM to diff and patch the DOM.
@@ -34,22 +33,22 @@ Currently there are many different approaches to solving the problem of observin
- Libraries like Vue.js, MobX or Ractive.js all use a variation of getters/setters to create observable data models.
- Angular 1.x 有脏检查。
-- React 由于它工作方式的原因,并不追踪数据模型中的改变。 它用虚拟 DOM 比较并修补 DOM。
-- Cycle.js 和 Angular 2 更喜欢响应流方式实现,类似 XStream 和 Rx.js.
-- 像 Vue.js, MobX 或 Ractive.js 这些库都使用 getters/setters 变量创建可观察的数据模型。
+- React 由于它工作方式的原因,并不追踪数据模型中的改变。它用虚拟 DOM 比较并修补 DOM。
+- Cycle.js 和 Angular 2 更喜欢响应流方式实现,像 XStream 和 Rx.js.
+- 像 Vue.js, MobX 或 Ractive.js 这些库都使用 getters/setters 变量创建可见的数据模型。
In this tutorial, we will go the getters/setters way of observing and reacting to changes.
在这篇教程中,我们将使用 getters/setters 的方式观察并响应变化。
> Note: To keep the tutorial as simple as possible, the code lacks the support for non-primitive data types or nested properties and many of the required sanity checks, thus by no means should this code be considered production ready. The code below is written using the ES2015 standard and is loosely inspired by the Vue.js reactive engine implementation.
-> 注意:为了让这篇教程尽量保持简单,代码缺少对非初级数据类型或嵌套属性的支持,并且很多内容需要完整性检查,因此这些决不能认为这些代码已经可以用于生产环境。下面的代码是受 Vue.js 启发的响应式引擎的实现,使用 ES2015 标准编写。
+> 注意:为了让这篇教程尽量保持简单,代码缺少对非初级数据类型或嵌套属性的支持,并且很多内容需要完整性检查,因此决不能认为这些代码已经可以用于生产环境。下面的代码是受 Vue.js 启发的响应式引擎的实现,使用 ES2015 标准编写。
## The observable object ##
-## 可见对象 ##
+## 可见的对象 ##
Let’s start with a `data` object, whose properties we want to observe.
-让我们从一个 `data` 对象开始,我们想要观察到的是它的属性。
+让我们从一个 `data` 对象开始,我们想要将它的属性可见。
```
let data = {
@@ -60,7 +59,7 @@ let data = {
```
Let’s start by creating two functions that will transform our object’s properties into observable properties using the getter/setter functionality.
-让我们从这里开始,首先创建两个函数,使用 getter/setter 的功能,将对象的属性转换成可见属性。
+首先从创建两个函数开始,使用 getter/setter 的功能,将对象的普通属性转换成可见的属性。
```
function makeReactive (obj, key) {
@@ -90,30 +89,29 @@ observeData(data)
```
By running `observeData(data)` we transform our object into an object capable of being observed; now we have a way to create notifications whenever the value changes.
-通过运行 `observeData(data)`,将原始的对象转换成具有可见性的对象;现在当对象的 value 发生改变时,我们有获得通知的办法。
+通过运行 `observeData(data)`,将原始的对象转换成具有可见性的对象;现在当对象的 value 发生变化时,我们有创建通知的办法。
## Reacting to changes ##
## 响应变化 ##
Before we begin *notifying*, we need something that we can actually notify. This is a perfect example where we can use the observer pattern. In this case we will make use of the signals implementation.
-在我们开始接收 *notifying* 前,我们需要一些通知的内容。这是一个使用观察者模式极好的例子。在这个案例中我们将使用 signals implementation。
+在我们开始接收 *notifying* 前,我们需要一些通知的内容。这里是使用观察者模式的一个极好例子。在这个案例中我们将使用 signals 实现。
Let’s start with the `observe` function.
-
我们从 `observe` 函数开始。
```
let signals = {} // Signals 从一个空对象开始
function observe (property, signalHandler) {
- if(!signals[property]) signals[property] = [] // 如果给定属性没在 signal 中,则创建该属性,并将其设置为空数组来存储 signalHandlers
+ if(!signals[property]) signals[property] = [] // 如果给定属性没在 signal 中,则创建这个属性的 signal,并将其设置为空数组来存储 signalHandlers
- signals[property].push(signalHandler) // 将 signalHandler 存入 signal 数组,高效的获得一组存在数组中的回调函数
+ signals[property].push(signalHandler) // 将 signalHandler 存入 signal 数组,高效的获得一组保存在数组中的回调函数
}
```
We can now use the `observe` function like this: `observe('propertyName', callback)`, where `callback` is a function that should be called each time the property’s value has changed. When we **observe** a property multiple times, each callback will be stored inside the corresponding property’s signal array. This way we can store all callbacks and have easy access to them.
-我们现在可以这样用 `observe` 函数: `observe('propertyName', callback)`,每次属性值发生改变的时候 `callback` 函数应该被调用。
+我们现在可以这样用 `observe` 函数:`observe('propertyName', callback)`,每次属性值发生改变的时候 `callback` 函数应该被调用。当多次在一个属性上调用 **observe** 时,每个回调函数将被存在对应属性的 signal 数组中。这样就可以存储所有的回调函数并且可以很容易的获得到它们。
Now for the `notify` function that you saw before.
现在来看一下上文中提到的 `notify` 函数。
@@ -130,7 +128,7 @@ As you can see, now every time one of the properties changes, the assigned signa
如你所见,现在每次一个属性发生变化,就会调用对其分配的 signalHandlers。
So let’s wrap it all up into a factory function that we pass the data object that has to be reactive. I will name mine `Seer`. We end up with something like this:
-所以我们把它全部封装起来做成一个工厂函数,传入想要响应的数据对象,我把它命名为 `Seer`。我们最终得到的像下面这样
+所以我们把它全部封装起来做成一个工厂函数,传入想要响应的数据对象。我把它命名为 `Seer`。我们最终得到如下:
```
function Seer (dataObj) {
@@ -192,10 +190,12 @@ const App = new Seer({
age: 25
})
+// To subscribe and react to changes made to the reactive App object:
// 为了订阅并响应可响应 APP 对象的改变:
App.observe('firstName', () => console.log(App.data.firstName))
App.observe('lastName', () => console.log(App.data.lastName))
+// To trigger the above callbacks simply change the values like this:
// 为了触发上面的回调函数,像下面这样简单的改变 values:
App.data.firstName = 'Sansa'
App.data.lastName = 'Stark'
@@ -205,10 +205,10 @@ App.data.lastName = 'Stark'
Simple, isn’t it? Now that we have the basic reactivity engine covered, let’s make some use of it.
I mentioned that with the more reactive approach to front-end programming, we should not be concerned with things like manually updating the DOM after each change.
很简单,是不是?现在我们讲完了基本的响应式引擎,让我们来用用它。
-我提到过随着前端编程可响应式方法的增多,我们不应该总想着在发生改变后手动的更新 DOM。
+我提到过随着前端编程可响应式方法的增多,我们不能总想着在发生改变后手动的更新 DOM。
There are many approaches to this. I guess the most trending one right now is the so called virtual DOM. If you are interested in learning how to create your own virtual DOM implementation, there are already great tutorials for this. However, here we will go with a much simpler approach.
-有很多方法来完成这项任务。我猜现在流行的趋势是用虚拟 DOM 的办法。如果你对学习如何创建你自己的虚拟 DOM 实现感兴趣,已经有很多这方面的教程。然而,这里我们将用到更简单的方法。
+有很多方法来完成这项任务。我猜现在最流行的趋势是用虚拟 DOM 的办法。如果你对学习如何创建你自己的虚拟 DOM 实现感兴趣,已经有很多这方面的教程。然而,这里我们将用到更简单的方法。
Let’s say our HTML looks like this:`html
Title comes here
`
HTML 看起来像这样: `html
Title comes here
`
@@ -235,7 +235,7 @@ syncNode(h1Node, App.data, 'title')
```
This will work but actually requires a lot of work from us to actually bind all the DOM elements to the desired data models.
-这样做事可行的,但是实际使用它绑定数据模型到 DOM 元素需要大量的工作。
+这样做是可行的,但是使用它把所有数据模型绑定到 DOM 元素需要大量的工作。
That’s why we can go a step further and automate all of this.
If you are familiar with AngularJS or Vue.js you surely remember using custom HTML attributes like `ng-bind` or `v-text`. We will create something similar here!
@@ -243,7 +243,7 @@ Our custom attribute will be called `s-text`. We will look for it to create bind
这就是我们为什么要再向前迈一步,然后将所有这些自动化完成。
如果你熟悉 AngularJS 或者 Vue.js,你肯定记得使用自定义属性 `ng-bind` 或 `v-text`。我们在这里创建类似的东西。
-我们的自定义属性叫做 `s-text`。我们将寻找在 DOM 和 数据模型之间建立绑定的方式。
+我们的自定义属性叫做 `s-text`。我们将寻找在 DOM 和数据模型之间建立绑定的方式。
Let’s update our HTML:
让我们更新一下 HTML:
@@ -254,7 +254,7 @@ Let’s update our HTML:
Title comes here
function parseDOM (node, observable) {
// We get all nodes that have the s-text custom attribute
- // 我们获得所有具有自定义属性 s-text 的节点
+ // 获得所有具有自定义属性 s-text 的节点
const nodes = document.querySelectorAll('[s-text]')
// For each existing node, we call the syncNode function
From 506f5c0dbab6d41c4916dadf9cc3fb3fadddafda Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Wed, 22 Mar 2017 20:09:10 +0800
Subject: [PATCH 004/945] =?UTF-8?q?=E7=BB=A7=E7=BB=AD=E6=A0=A1=E5=AF=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...-engine-in-javascript-part-1-observable-objects.md | 11 ++++++++---
1 file changed, 8 insertions(+), 3 deletions(-)
diff --git a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
index b90d3137c03..f015154d56f 100644
--- a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
+++ b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
@@ -7,7 +7,7 @@
![](https://d4a7vd7s8p76l.cloudfront.net/uploads/1484604970-4-7876/observables.png)
# [How to build a reactive engine in JavaScript. Part 1: Observable objects](/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects/) #
-# 如何用 JavaScript 构建响应式引擎 —— Part 1:可见的对象 #
+# 如何使用 JavaScript 构建响应式引擎 —— Part 1:可见的对象 #
## The reactive way ##
## 响应的方式 ##
@@ -323,13 +323,15 @@ function Seer (dataObj) {
}
}
// We can safely parse the DOM looking for bindings after we converted the dataObject.
- parseDOM(document.body, obj)
+ //转换数据对象后,可以安全的解析 DOM 绑定。
+ parseDOM(document.body, obj)
}
function syncNode (node, observable, property) {
node.textContent = observable[property]
// We remove the `Seer.` as it is now available for us in our scope.
- observe(property, () => node.textContent = observable[property])
+ // 移除了 `Seer.` 是因为 observe 函数在可获得的作用域范围之内。
+ observe(property, () => node.textContent = observable[property])
}
function parseDOM (node, observable) {
@@ -367,6 +369,7 @@ JS
```
// This code uses ES2015.
// Please use a compatible browser like: Chrome, Opera, Firefox
+// 代码用了 ES2015,使用兼容的浏览器才可以哦,比如 Chrome,Opera,Firefox
function Seer (dataObj) {
let signals = {}
@@ -412,12 +415,14 @@ function Seer (dataObj) {
}
}
// We can safely parse the DOM looking for bindings after we converted the dataObject.
+ //转换数据对象后,可以安全的解析 DOM 绑定。
parseDOM(document.body, obj)
}
function syncNode (node, observable, property) {
node.textContent = observable[property]
// We remove the `Seer.` as it is now available for us in our scope.
+ // 移除了 `Seer.` 是因为 observe 函数在可获得的作用域范围之内。
observe(property, () => node.textContent = observable[property])
}
From a744d2727157fd1838cdc4fde8637dbc2ee0abb1 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E7=8E=8B=E6=96=87=E6=98=8E?=
Date: Thu, 23 Mar 2017 21:23:25 +0800
Subject: [PATCH 005/945] trans
---
...-to-create-a-bubble-selection-animation-on-android.md | 9 +++++++++
1 file changed, 9 insertions(+)
diff --git a/TODO/how-to-create-a-bubble-selection-animation-on-android.md b/TODO/how-to-create-a-bubble-selection-animation-on-android.md
index 51891e6b9d5..ee2661264b1 100644
--- a/TODO/how-to-create-a-bubble-selection-animation-on-android.md
+++ b/TODO/how-to-create-a-bubble-selection-animation-on-android.md
@@ -7,25 +7,34 @@
# How to Create a Bubble Selection Animation on Android #
+# Android 如何实现气泡选择动画 #
*Authors:* [*Irina Galata*](https://github.com/igalata) *, Android Developer;* [*Yulia Serbenenko*](https://dribbble.com/yuyonder) *, UI/UX designer.*
+**作者:[Irina Galata](https://github.com/igalata) Android 开发者;[Yulia Serbenenko](https://dribbble.com/yuyonder) UI/UX 设计师**
There is a growing trend for unifying user experience across platforms: in early days iOS and Android had their own unique feel, but recently they have been growing closer together in the way applications are designed and interactions happen.
+跨平台用户体验统一正处于增长趋势:早些时候 iOS 和安卓有着不同的体验,但是最近在应用设计以及交互方面变得越来越接近。
From [bottom navigation](https://material.io/guidelines/components/bottom-navigation.html#) to split screen feature available in Nougat Android, there is a lot of common between two platforms these days. For designers it means that often we can adjust popular features that were once associated with one platform to apps designed for another one. As for developers, it is a great chance to improve and refine their technical skills.
+从安卓 Nougat 的[底部导航](https://material.io/guidelines/components/bottom-navigation.html#)到分屏特性,两个平台间有了许多相同之处。对设计师而言,我们可以将主流功能设计成两个平台一致(过去需要单独设计)。对开发者而言,这是一个提高、改进开发经验的好机会。
So we decided to create the component with the bubble based interface for Android, drawing our inspiration from selection bubbles in [Apple music](http://www.apple.com/lae/apple-music/) .
+所以我们决定开发一个安卓气泡选择的组件库 —— 灵感来自于[苹果音乐](http://www.apple.com/lae/apple-music/)的气泡选择。
### ***Put design first*** ###
+### **先说设计** ###
Our Bubble Picker is an example of the animation that is equally appealing to different groups of users. Bubbles summarize information into convenient UI elements that are easy to understand and also visually consistent. It makes the interface simple enough for novice users and still feels interesting for experienced ones.
+我们的气泡选择动画是一个好的范例,它对不同的用户群体有着同样的吸引力。气泡以方便的 UI 元素汇总信息,通俗易懂并且视觉一致。它会让交互对新手足够简单并让经验用户依然感兴趣。
This type of animation is very helpful for apps rich in content, where users have to make a choice from a list of options. For example, in our component we used bubbles to hold names of potential destinations for a travel app. Bubbles float freely, and when a user taps on one of them, the chosen bubble grows in size. It gives users a meaningful feedback on their actions and enhances the sense of direct manipulation.
+这种动画类型对丰富应用的内容由很大帮助,主要体现在用户要从一系列选项中进行选择的地方。例如,我们使用气泡来选择旅游应用中潜在目的地名字。气泡自由的浮动,当用户点击一个气泡时,选中的气泡会变大。这给用户很深刻的反馈并增强操作的直观感受。
The component is pretty with a white theme, lots of bright colors and photographs throughout. Moreover, I decided to experiment with gradients in order to add more depth and volume. Gradients might be the major visual in the display and might attract the attention of new visitors.
+组件使用白色主题,布满明亮的颜色和图片十分漂亮。此外,我决定试验渐变来增加深度和体积。渐变可能是主要的显示视觉,会吸引新用户的注意。
From 661578c58437e457eb9a63070683b51d7cb38974 Mon Sep 17 00:00:00 2001
From: DeepMissea <398752853@qq.com>
Date: Fri, 24 Mar 2017 10:59:17 +0800
Subject: [PATCH 006/945] =?UTF-8?q?=E5=88=9D=E6=AD=A5=E7=BF=BB=E8=AF=91?=
=?UTF-8?q?=E5=AE=8C=E6=88=90=E3=80=82?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/a-quick-look-at-semaphores.md | 194 ++++++++++++++---------------
1 file changed, 97 insertions(+), 97 deletions(-)
diff --git a/TODO/a-quick-look-at-semaphores.md b/TODO/a-quick-look-at-semaphores.md
index ac28e36f949..c7346831fb8 100644
--- a/TODO/a-quick-look-at-semaphores.md
+++ b/TODO/a-quick-look-at-semaphores.md
@@ -1,33 +1,32 @@
> * 原文地址:[A Quick Look at Semaphores in Swift 🚦](https://medium.com/swiftly-swift/a-quick-look-at-semaphores-6b7b85233ddb#.61uw6lq2d)
> * 原文作者:[Federico Zanetello](https://medium.com/@zntfdr)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[Deepmissea](http://deepmissea.blue)
> * 校对者:
---
# A Quick Look at Semaphores in Swift 🚦
-First of all, if you’re not familiar with the Grand Central Dispatch (GCD) and Dispatch Queues, please head over [this awesome article](http://www.appcoda.com/grand-central-dispatch/) from [AppCoda](https://medium.com/@appcodamobile).
+首先,如果你对 GCD 和 Dispatch Queue 不熟悉,请看看 [AppCoda](https://medium.com/@appcodamobile) 的[这篇文章](http://www.appcoda.com/grand-central-dispatch/)。
-All right! Time to talk about Semaphores!
+好了!是时候来聊聊信号量了!
![](https://cdn-images-1.medium.com/max/1600/1*8ZCGzvA6DjfR9JoamqauoQ.jpeg)
-### Introduction
+### 引言
-Let’s imagine a group of *writers* that must share a single *pen*.
-Obviously only one *writer *can use the *pen* at any given time.
+让我们想象一下,一群**作家**只能共同使用一只**笔**。显然,在任何指定的时间里,只有一名**作家**可以使用**笔**。
-Now, imagine that those *writers* are our threads and that the *pen* is our *shared resource* (it can be anything: a file, a variable, the right to do something, etc).
+现在,把**作家**想象成我们的线程,把**笔**想象成我们的**共享资源**(可以是任何东西:一个文件、一个变量、做某事的权利等等)。
-How do we make sure that our *resource* is really [mutually exclusive](https://en.wikipedia.org/wiki/Mutual_exclusion)?
+怎么才能确保我们的**资源**是真正[互斥](https://en.wikipedia.org/wiki/Mutual_exclusion)?
![](https://cdn-images-1.medium.com/max/1600/1*nfAYVSYFMB874-z4sfJ_YQ.jpeg)
-### Implementing our own Resource Control Access
+### 实现我们自己的资源控制访问
-Someone may think: well I can just use a *resourceIsAvailable* *Bool* and set it to *true*/*false.*
+有人可能会想:我只要用一个 **Bool** 类型的 **resourceIsAvailable** 变量,然后设置它为 **true** 或者 **false** 就可以互斥了。
```
if (resourceIsAvailable) {
@@ -39,108 +38,109 @@ if (resourceIsAvailable) {
}
```
-The problem is that, on concurrency, **there’s no guarantee of knowing which thread, among all, is going to execute the next step, regardless of their priority**.
+问题是出现在并发上,**在无视他们的优先级时,不知道他们之中,哪个线程将执行下一步。**
-#### Example
+#### 例子
-Imagine that we’ve implemented the code above and that we have two threads, *threadA* and *threadB*, that would like to use a mutual exclusive resource:
+假设我们实现了上面的代码,我们有两个线程,**threadA** 和 **threadB**,他们会使用一个互斥的资源:
-- *threadA* reads the if-condition and see that the resource is available, great!
-- But, before the execution of the next line (*resourceIsAvalilable = false*), the processor turns to *threadB* and it also reads the if-condition.
-- Now we have two threads that believe that the resource is available and both are going to execute the *use-the-resource* block.
+- **threadA** 读取到 if 条件语句,发现资源可用,很棒!
+- 但是,在执行下一行代码(**resourceIsAvalilable = false**)之前,处理器切换到 **threadB**,然后它也读取了 if 条件语句。
+- 现在我们的两个线程都确信资源是可用的,然后他们都会执行**使用资源**部分的代码块。
-Writing thread-safe code without GCD is not an easy task.
+
+不用 GCD 编写线程安全的代码可不是一个容易的任务。
![](https://cdn-images-1.medium.com/max/1600/1*p54pBislRafckGffcDqRdA.png)
-### How Semaphores Work
+### 信号量是如何工作的
-Three steps:
+三步:
-1. Whenever we would like to use one shared resource, we send a **request** to its semaphore;
-2. Once the semaphore gives us the green light (see what I did here?) we can assume that the resource is ours and we can use it;
-3. Once the resource is no longer necessary, we let the semaphore know by sending him a **signal**,allowing him to assign the resource to another thread.
+1. 在我们需要使用一个共享资源的时候,我们发送一个 **request** 给它的信号量;
+2. 一旦信号量给出我们绿灯(看我在这做了什么?),我们就可以假定资源是我们的并使用它;
+3. 一旦不需要资源了,我们通过发送给信号量一个 **signal** 让它知道,然后它可以把资源分配给另一个的线程。
-When this resource is only one and can be used only by onethread at any given time, you can think of these **request/signal** as the resource **lock/unlock**.
+当这个资源只有一个,并且在任何给定的时间里,只有一个线程可以使用,你就可以把这些 **request/signal** 作为资源的 **lock/unlock**。
![](https://cdn-images-1.medium.com/max/1600/1*-_owdkyNPRUQS5a5yjdEkA.jpeg)
-### What’s Happening Behind the Scenes
+### 在幕后发生了什么
-#### The Structure
+#### 结构
-The Semaphore is composed by:
+信号量由下面的两部分组成:
-- a *counter* that let the Semaphore know how many threads can use its resource(s);
-- a *FIFO queue* for tracking the threads waiting for the resource;
+- 一个**计数器**,让信号量知道有多少个线程能使用它的资源;
+- 一个 **FIFO 队列**,用来追踪这些等待资源的线程;
-#### Resource Request: wait()
+#### 请求资源: wait()
-When the semaphore receives a request, it checks if its *counter* is above zero:
+当信号量收到一个请求时,它会检查它的**计数器**是否大于零:
-- if yes, then the semaphore decrements it and gives the thread the green light;
-- otherwise it pushes the thread at the end of its queue;
+- 如果是,那信号量会减一,然后给线程放绿灯;
+- 如果不是,它会把线程添加到它队列的末尾;
-#### Resource Release: signal()
+#### 释放资源: signal()
-Once the semaphore receives a signal, it checks if its FIFO queue has threads in it:
+一旦信号量收到一个信号,它会检查它的 FIFO 队列是否有线程存在:
-- if yes, then the semaphore pulls the first thread and give him the green light;
-- otherwise it increments its counter;
+- 如果有,那么信号量会把第一个线程拉出来,然后给他一个绿灯;
+- 如果没有,那么它会增加它的计数器;
-#### Warning: Busy Waiting
+#### 警告: 忙碌等待
-When a thread sends a *wait() *resource request to the semaphore, the thread **freezes **until the semaphore gives the thread the green light.
+当一个线程发送一个 **wait()** 资源请求给信号量时,线程会**冻结**直到信号量给线程绿灯。
-⚠️️If you do this in the main thread, the whole app will freeze ⚠️️
+⚠️️如果你在在主线程这么做,那整个应用都会冻结⚠️️
![](https://cdn-images-1.medium.com/max/1600/1*3GANzX3n1uEiuhXE49fcrg.jpeg)
-### Using Semaphores in Swift (with GCD)
+### 在 Swift 里使用信号量 (通过 GCD)
-Let’s write some code!
+让我们写一些代码!
-#### Declaration
+#### 声明
-Declaring a Semaphore is simple:
+声明一个信号量很简单:
```
let semaphore = DispatchSemaphore(value: 1)
```
-The *value* parameter is the number of threads that can access to the resource as for the semaphore creation.
+**value** 参数代表信号量创建的可以访问这个资源的线程数量。
-#### Resource Request
+#### 资源请求
-To **request** the *semaphore*’s resource(s), we just call:
+如果要**请求信号量**的资源,我们只需:
```
semaphore.wait()
```
-Note that the semaphore is not physically giving us anything, the resource has to be in the thread’s scope already, we just use the resource only between our request and release calls.
+要知道信号量并不能实质上地给我们任何东西,资源都是在线程的范围内,而我们只是在请求和释放调用之间使用资源。
-Once the semaphore gives us its blessing, the thread resumes its normal execution and can consider the resource his to use.
+一旦信号量给我们开恩了,那线程就会恢复正常执行,并可以考虑使用它的资源。
-#### Resource Release
+#### 资源释放
-To **release** the resource we write:
+要**释放**资源,我们这么写:
```
semaphore.signal()
```
-After sending this signal we aren’t allowed to touch the resource anymore, until we request for it again.
+在发送这个信号后,我们就不能接触到任何资源了,知道我们再次的请求它。
-### Semaphore Playgrounds
+### Playgrounds 中的信号量
-Following [AppCoda](https://medium.com/@appcodamobile)[article](http://www.appcoda.com/grand-central-dispatch/) examples, let’s see this Semaphore in action!
+跟随 [AppCoda](https://medium.com/@appcodamobile) 上[这篇文章](http://www.appcoda.com/grand-central-dispatch/)的例子,让我们看看工作中的信号量!
-> Warning: these are Xcode Playgrounds, as Swift Playgrounds don’t support Logging just yet. Fingers crossed for WWDC17!
+> 注意:这些是 Xcode 中的 Playground,Swift Playground 还不支持日志记录。希望 WWDC17 能一切顺利!
-In these playgrounds we have two threads, one with slightly higher priority than the other, that print 10 times an emoji and incremental numbers.
+在这些 playground 里,我们有两个线程,一个线程的优先级比其他的略微高一些,打印 10 次表情和增加的数字。
-#### Semaphore-less Playground
+#### 没有信号量的 Playground
```
import Foundation
@@ -163,15 +163,15 @@ asyncPrint(queue: lowerPriority, symbol: "🔵")
PlaygroundPage.current.needsIndefiniteExecution = true
```
-As you can Imagine, the higher priority thread finishes first most of the times:
+和你想的一样,多数情况下,高优先级的线程先完成任务:
![](https://cdn-images-1.medium.com/max/1600/1*OjtJO8-44tStXpRS8y1N-A.png)
-#### Semaphore Playground
+#### 有信号量的 Playground
-In this case we will use the same code as before, but we will give the right to print the *emoji+number* sequence only to one thread at a time.
+这次我们会使用和前面一样的代码,但是在同一时间,我们只给一个线程赋予打印**表情+数字**的权利。
-In order to do so we will define one semaphore and update our *asyncPrint* function:
+为了达到这个目的,我们定义了一个信号量并且更新了我们的 **asyncPrint** 函数:
```
import Foundation
@@ -185,14 +185,14 @@ let semaphore = DispatchSemaphore(value: 1)
func asyncPrint(queue: DispatchQueue, symbol: String) {
queue.async {
print("\(symbol) waiting")
- semaphore.wait() // requesting the resource
+ semaphore.wait() // 请求资源
for i in 0...10 {
print(symbol, i)
}
print("\(symbol) signal")
- semaphore.signal() // releasing the resource
+ semaphore.signal() // 释放资源
}
}
@@ -202,50 +202,50 @@ asyncPrint(queue: lowerPriority, symbol: "🔵")
PlaygroundPage.current.needsIndefiniteExecution = true
```
-I’ve also added a couple more *print* commands to see the actual state of each thread during our execution.
+我还添加了一个 **print** 指令,以便我们看到每个线程执行中的实际状态。
![](https://cdn-images-1.medium.com/max/1600/1*g7SMrR7svWNetOqjSGIEYA.png)
-As you can see, when one thread starts printing the sequence, the other thread must wait until the first one ends, then the semaphore will receive the *signal* from the first thread and then, *only then*, the second thread can start printing its own sequence.
+就像你看到的,当一个线程开始打印队列,另一个线程必须等待,直到第一个结束,然后信号量会从第一个线程收到 **signal**,接着,**只有这样**,第二个线程才能开始打印它的队列。
-It doesn’t matter at which point of the sequence the second thread will send the *wait()* request, it will always have to wait until the other thread is done.
+第二个线程在队列的哪个点发送 **wait()** 无关紧要,它会一直出于等待状态直到另一个线程结束。
-**Priority Inversion**
+**优先级翻转**
-Now that we understand how everything works, please take a look at the following log:
+现在我们已经明白每个步骤是如何工作的,请看一下这个日志:
![](https://cdn-images-1.medium.com/max/1600/1*eCFBl9XpF6JYX1b8xwD26w.png)
-In this case, with the exact code above, the processor has decided to execute the low priority thread first.
+在这种情况下,通过上面的代码,处理器决定先执行低优先级的线程。
-When this happens, the high priority thread must wait the low priority thread to finish! This is ok, it can happen.
-The problem is that the low priority thread has low priority even when one high priority thread is waiting for him: this is called [***Priority Inversion***](https://en.wikipedia.org/wiki/Priority_inversion).
+这时,高优先级的线程必须等待低优先级的线程完成!这是真的,它的确会发生。
+问题是即使一个高优先级线程正等待它,低优先级的线程也是低优先级的:这被称为[***优先级反转***](https://en.wikipedia.org/wiki/Priority_inversion)。
-In other programming concepts different than the Semaphore, when this happens the low priority thread will temporarily *inherit* the priority of the highest priority thread that is waiting on him: this is called [***Priority Inheritance***](https://en.wikipedia.org/wiki/Priority_inheritance).
+在不同于信号量的其他编程概念里,当发生这种情况时,低优先级的线程会**继承**等待它的最高优先级线程的优先级,这被称为:[***优先级继承***](https://en.wikipedia.org/wiki/Priority_inheritance)。
-With Semaphores this is not the case because, actually, anybody can call the *signal()* function (not only the thread that is currently using the resource).
+在使用信号量的时候不是这样的,实际上,谁都可以调用 **signal()** 函数(不仅是当前正使用资源的线程)。
-**Thread Starvation**
+**线程饥饿**
-To make things even worse, let’s imagine that between our high & low priority threads there are 1000 more middle-priority threads.
+为了让事情变得更糟,让我们假设在我们的高优先级和低优先级线程之间还有 1000 多个中优先级的线程。
-If we have a case of *Priority Inversion* like above, the high priority thread must wait for the low priority thread, but, most of the time, the processor will execute the middle priority threads, as they have higher priority than our low priority one.
+如果我们有一种像上面那样**优先级反转**的情况,高优先级的线程必须等待低优先级的线程,但是,大多数情况下,处理器会执行中优先级的线程,因为他们的优先级高于我们的低优先级线程。
-In this scenario our high priority thread is being starved of CPU time (hence the concept of [Starvation](https://en.wikipedia.org/wiki/Starvation_%28computer_science%29)).
+这种情况下,我们的高优先级线程正被 CPU 饿的要死(于是有了[饥饿](https://en.wikipedia.org/wiki/Starvation_%28computer_science%29)的概念)。
-#### Solutions
+#### 解决方案
-In my opinion, it’s better to use Semaphores only among threads of the same priority. If this is not your case, I suggest you to look at other solutions such as [Regions](https://en.wikipedia.org/wiki/Critical_section) and [Monitors](https://en.wikipedia.org/wiki/Monitor_%28synchronization%29).
+我的观点是,在使用信号量的时候,线程之间最好都使用相同的优先级。如果这不符合你的情况,我建议你看看其他的解决方案,比如[临界区块](https://en.wikipedia.org/wiki/Critical_section)和[管程](https://en.wikipedia.org/wiki/Monitor_%28synchronization%29).
-### Deadlock Playground
+### Playground 上的死锁
-This time we have two threads that use two mutual exclusive resources “*A*” and “*B*”.
+现在我们有两个线程,使用两个互斥的资源,“**A**” 和 “**B**”。
-If the two resources can be used separately, it makes sense to define one semaphore for each resource. If not, one semaphore can manage both.
+如果两个资源可以分离使用,为每个资源定义一个信号量是有意义的,如果不可以,那一个信号量足以管理两者。
-I want to make an example with the former case (2 resources, 2 semaphores) with a twist: the high priority thread will use first resource “A” and then “B”, while our low priority one will use first resource “B” and then “A”.
+我想用一个用前一种情况(2 个资源, 2 个信号量)做一个例子:高优先级线程会先使用资源 “A”,然后 “B”,而低优先级的线程会先使用 “B”,然后再使用 "A"。
-Here’s the code:
+代码在这:
```
import Foundation
@@ -284,37 +284,37 @@ asyncPrint(queue: lowerPriority, symbol: "🔵", firstResource: "B", firstSemaph
PlaygroundPage.current.needsIndefiniteExecution = true
```
-If we’re lucky, this is what happens:
+如果我们幸运的话,会这样:
![](https://cdn-images-1.medium.com/max/1600/1*_ASgiqbV_o9caE7M7hNBpQ.png)
-Simply, the high priority thread will be served with the first resource, then the second and only later the the processor will move to the low priority thread.
+简单来说就是,第一个资源会先提供给高优先级线程,然后对于第二个资源,处理器只有稍后把它移动到低优先级线程。
-However, if we’re unlucky, this can also happen:
+然而,如果我们不是很幸运的话,那这种情况也会发生:
![](https://cdn-images-1.medium.com/max/1600/1*cVvGM-1NRH7kouSRu2mSRQ.png)
-Both threads didn’t finish their execution! Let’s review the current state:
+两个线程都没有完成他们的执行!让我们检查一下当前的状态:
-- The high priority thread is waiting for the resource “B”, which is held by the low priority thread;
-- The low priority thread is waiting for the resource “A”, which is held by the high priority thread;
+- 高优先级的线程正在等待资源 “B”,可是被低优先级的线程持有;
+- 低优先级的线程正在等待资源 “A”,可是被高优先级的线程持有;
-Both threads are waiting on each other with no possibility to move forward: welcome to a [*Thread Deadlock*](https://en.wikipedia.org/wiki/Deadlock)!
+两个线程都在等待相互的资源,谁也不能向前一步:欢迎来到[**线程死锁**](https://en.wikipedia.org/wiki/Deadlock)!
-#### Solutions
+#### 解决方案
-Avoiding [deadlocks](https://en.wikipedia.org/wiki/Deadlock) is not simple. The best solution would be preventing them by writing code that [can’t possibly reach this state](https://en.wikipedia.org/wiki/Deadlock_prevention_algorithms).
+避免[死锁](https://en.wikipedia.org/wiki/Deadlock)很难。最好的解决方案是编写[不能达到这种状态](https://en.wikipedia.org/wiki/Deadlock_prevention_algorithms)的代码来阻止他们。
-In other OSs, for example, one of the deadlock threads could be killed (in order to release all its resources) with the hope that other threads can continue their execution.
+例如,在其他的操作系统里,为了其他线程的继续执行,其中一个死锁线程可能被杀死(为了释放它的所有资源)。
-…Or you can just use the [Ostrich_Algorithm](https://en.wikipedia.org/wiki/Ostrich_algorithm) 😆.
+...或者你可以使用[鸵鸟算法(Ostrich_Algorithm)](https://en.wikipedia.org/wiki/Ostrich_algorithm) 😆。
![](https://cdn-images-1.medium.com/max/1600/1*Nmcb2GTIk-PO0TNPNPD8Mw.jpeg)
-### Conclusions
+### 结论
-Semaphores are a little nice concept that can be very handy in many applications. Just, be careful: look both ways before crossing.
+信号量是一个很棒的概念,它可以在很多应用里方便的使用,只是要小心:过马路要看两边。
---
-[*Federico*](https://twitter.com/zntfdr) *is a Bangkok-based Software Engineer with a strong passion for Swift, Minimalism, Design, and iOS Development.*
\ No newline at end of file
+[*Federico*](https://twitter.com/zntfdr) *是一名在曼谷的软件工程师,对 Swift、Minimalism、Design 和 iOS 开发有浓厚的热情。*
From 2c81fe3a1d90f8ed1023e5358bf7f71938e80a38 Mon Sep 17 00:00:00 2001
From: DeepMissea <398752853@qq.com>
Date: Fri, 24 Mar 2017 11:00:16 +0800
Subject: [PATCH 007/945] Update a-quick-look-at-semaphores.md
---
TODO/a-quick-look-at-semaphores.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/a-quick-look-at-semaphores.md b/TODO/a-quick-look-at-semaphores.md
index c7346831fb8..ab11f9f6dc9 100644
--- a/TODO/a-quick-look-at-semaphores.md
+++ b/TODO/a-quick-look-at-semaphores.md
@@ -6,7 +6,7 @@
---
-# A Quick Look at Semaphores in Swift 🚦
+# 快速浏览 Swift 中的信号量 🚦
首先,如果你对 GCD 和 Dispatch Queue 不熟悉,请看看 [AppCoda](https://medium.com/@appcodamobile) 的[这篇文章](http://www.appcoda.com/grand-central-dispatch/)。
From 442757b05e448c1df18fc2fefa0bb4b63e2d86fc Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Fri, 24 Mar 2017 18:38:39 +0800
Subject: [PATCH 008/945] =?UTF-8?q?=E4=BF=AE=E6=94=B9=E9=A2=98=E7=9B=AE?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...a-reactive-engine-in-javascript-part-1-observable-objects.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
index aa0993fce23..4b29caa4e82 100644
--- a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
+++ b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
@@ -7,7 +7,7 @@
![](https://d4a7vd7s8p76l.cloudfront.net/uploads/1484604970-4-7876/observables.png)
# [How to build a reactive engine in JavaScript. Part 1: Observable objects](/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects/) #
-# 如何使用 JavaScript 构建响应式引擎 —— Part 1:可见的对象 #
+# 如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象 #
## The reactive way ##
## 响应的方式 ##
From 19ebd9616df134dc75772f98efc5b0f05f5f6a45 Mon Sep 17 00:00:00 2001
From: zhouzihanntu
Date: Fri, 24 Mar 2017 21:32:28 +0800
Subject: [PATCH 009/945] update
---
...odules-in-node-js-everything-you-need-to-know.md | 13 ++++++++++---
1 file changed, 10 insertions(+), 3 deletions(-)
diff --git a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
index fce6fbc0359..5618ebc6fd9 100644
--- a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
+++ b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
@@ -1,23 +1,30 @@
> * 原文地址:[Requiring modules in Node.js: Everything you need to know](https://medium.freecodecamp.com/requiring-modules-in-node-js-everything-you-need-to-know-e7fbd119be8#.wcrwm9c81)
> * 原文作者:[Samer Buna](https://medium.freecodecamp.com/@samerbuna?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[zhouzihanntu](https://github.com/zhouzihanntu)
> * 校对者:
# Requiring modules in Node.js: Everything you need to know #
+# 在 Node.js 中加载模块 #
## How modularity works in Node.js ##
+## 模块化在 Node.js 中的应用##
Node uses two core modules for managing module dependencies:
+Node 提供了两个核心模块来管理模块依赖:
- The `require` module, which appears to be available on the global scope — no need to `require('require')`.
+- `require` 模块在全局范围内可用,不需要 `require('require')`.
- The `module` module, which also appears to be available on the global scope — no need to `require('module')`.
+- `module` 模块同样在全局范围内可用,不需要 `require('module')`.
You can think of the `require` module as the command and the `module` module as the organizer of all required modules.
+你可以把 `require` 模块当做命令 `module` 模块 所有需加载模块的
Requiring a module in Node isn’t that complicated of a concept.
+在 Node 中加载一个模块其实没有概念那么复杂。
```
const config = require('/path/to/file');
@@ -108,7 +115,7 @@ Error: Cannot find module 'find-me'
If you now create a local `node_modules` directory and put a `find-me.js` in there, the `require('find-me')` line will find it.
```
-~/learn-node $ mkdir node_modules
+~/learn-node $ mkdir node_modules
~/learn-node $ echo "console.log('I am not lost');" > node_modules/find-me.js
@@ -302,7 +309,7 @@ In util Module {
I’ve removed some attributes in the above output to keep it brief, but note how the `exports` object now has the attributes we defined in each module. You can put as many attributes as you want on that exports object, and you can actually change the whole object to be something else. For example, to change the exports object to be a function instead of an object, we do the following:
```
-// Add the following line in index.js before the console.log
+// Add the following line in index.js before the console.log
module.exports = function() {};
```
From 3d29b535a0aabb009ce58800ee94e950be9b62d2 Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Sat, 25 Mar 2017 15:46:25 +0800
Subject: [PATCH 010/945] =?UTF-8?q?=E5=85=A8=E6=96=87=E9=80=9A=E8=AF=91?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...operties-javascript-dependency-tracking.md | 165 ++++++++++++++----
1 file changed, 134 insertions(+), 31 deletions(-)
diff --git a/TODO/computed-properties-javascript-dependency-tracking.md b/TODO/computed-properties-javascript-dependency-tracking.md
index ecf8790d675..41ac3611901 100644
--- a/TODO/computed-properties-javascript-dependency-tracking.md
+++ b/TODO/computed-properties-javascript-dependency-tracking.md
@@ -7,16 +7,22 @@
![](https://d4a7vd7s8p76l.cloudfront.net/uploads/56873733-c918-4cd6-bac1-dea44dcc3a9f/Reactive%20engine.png)
# [How to build a reactive engine in JavaScript. Part 2: Computed properties and dependency tracking](/blog/2017/computed-properties-javascript-dependency-tracking/) #
+# 如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪 #
Hey! If you have ever worked with Vue.js, Ember or MobX I’m pretty sure you stumbled upon so-called **computed** properties. They allow you to create functions that can be accessed just like normal values, but once computed they are cached until one of its dependencies has changed. In general this is a concept very similar to getters and in fact, the following implementation will be using getters. In a smart way. ;)
+Hey!如果你用过 Vue.js、Ember 或 MobX,我敢肯定你被 **计算** 属性难倒过。计算属性允许你创建像正常的值一样使用的函数,但是一旦完成计算,他们就被缓存下来直到它的一个依赖发生改变。总的来说,这一概念与 getters 非常相似,并且事实上,下面的实现将会使用 getters。用一种机智的方式。;)
> This is the 2nd part of the How to build a reactive engine in JavaScript series. Before reading any further it is highly recommended to read [Part 1: Observable objects](https://monterail.com/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects), because the following implementation is built on top of the previous article's code.
+> 这是如何使用 JavaScript 构建响应式引擎系列文章的第二部分。在深入阅读前强烈建议读一下[Part 1: 可观察的对象](https://monterail.com/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects),因为接下来的实现是构建于前一篇文章的代码之上的。
## Computed properties ##
+## 计算属性 ##
Let’s say we have a computed property called `fullName` which is a combination of `firstName` and `lastName` with space in between.
+比如有一个计算属性叫 `fullName`,是 `firstName` 和 `lastName` 之间加上空格的组合。
In Vue.js such a computed value could be created like this:
+在 Vue.js 中这样的计算值可以像下面这样创建:
```
data: {
@@ -31,26 +37,36 @@ computed: {
```
Now if we use the `fullName` somewhere in our template, we expect it will be updated whenever `firstName` or `lastName` change. If you come from an AngularJS background you might also remember using expressions inside the template or function calls. Of course, this works the same when using render functions (with JSX or not); it doesn’t really matter.
+现在如果在模板中使用 `fullName`,我们希望它能随着 `firstName` 或 `lastName` 的改变而更新。如果你有使用 AngularJS 的背景,你可能还记得在模板或者函数调用内使用表达式。当然了,使用渲染函数(使用 JSX 或者不使用)的时候和这里是一样的;其实这无关紧要。
Let’s consider the following example:
+来看一下下面的例子:
```
+
{{ firstName + ' ' + lastName }}
+
{{ getFullName() }}
+
{{ fullName }}
```
The result of the above code will be mostly the same. Each time `firstName` or `lastName` changes, the view will update with the headers and show the full name.
+上面代码的执行结果几乎是一样的。每次 `firstName` 或 `lastName` 发生变化,视图将会更新这些 `` 并且显示 full name。
However, what if we use the expression, method call and computed property multiple times? The expression and method call will have to be calculated each time they are accessed, whereas the computed property will be cached after the first computation until one of its dependencies change. It will also persist through the re-render cycles! That’s actually quite a nice optimization if you consider that in event-based modern user interfaces, it’s hard to predict which action the user will take first.
+然而,如果多次使用表达式、函数调用和计算属性呢?使用表达式和函数调用每次都会计算一遍,而计算属性在第一次计算后将会缓存下来,直到它的依赖发生改变。它也会一直坚持 re-render 循环!如果考虑在基于事件模型的用户界面中,很难预测用户会首先执行哪项操作,那么这确实是一个最佳选择方案。
## Basic computed property ##
+## 基础的计算属性 ##
In the previous article, we learned how to track and react to changes inside observable object properties by utilizing an event emitter. We know that when we change the `firstName` it will call all the handlers that subscribed to the `’firstName’` event. Thus it is quite easy to build a computed property by manually subscribing to its dependencies.
This is actually how Ember does it:
+在前面文章中,我们学习了如何通过使用事件发射器追踪和响应可观察对象内的改变。我们知道当改变 `firstName` 时,会调用所有的订阅了 `’firstName’` 事件的处理器。因此通过手动订阅它的依赖来构建计算属性是相当容易的。
+这也是 Ember 实现它的方式:
```
fullName: Ember.computed('firstName', 'lastName', function() {
@@ -59,6 +75,7 @@ fullName: Ember.computed('firstName', 'lastName', function() {
```
The drawback here is that you have to declare the dependencies yourself. Doesn’t seem like a problem until you have computed properties that are a result of a chain of more expensive, complex functions. For example:
+这样做的缺点就是你不得不自己声明依赖。当你的计算属性是一串高开销的、功能复杂的函数的时候,你就知道这的确是个问题了。例如:
```
selectedTransformedList: Ember.computed('story', 'listA', 'listB', 'listC', function() {
@@ -74,32 +91,44 @@ selectedTransformedList: Ember.computed('story', 'listA', 'listB', 'listC', func
```
In the above case, even if `this.story` always equals `’A’`, the computed property will have to be re-evaluated each time one of the lists changes. Even if they are not used for the end result.
+在上面的案例中,即便 `this.story` 总是等于 `’A’`,一旦 lists 发生改变,计算属性将不得不每次都反复求值。
## Dependency tracking ##
+## 依赖追踪 ##
Vue.js and MobX take a different approach to this problem. The difference is, you don’t have to declare the dependencies at all because they are detected automatically each time it is evaluated. Let say `this.story = ‘A’`. The detected dependencies will be:
+Vue.js 和 MobX 在解决这个问题上使用了与上文不同的方法。不同在于,你根本不必声明依赖,因为在计算的时候他们会自动地检测。假定 `this.story = ‘A’`,检测到的依赖会是:
+
* `this.story`
* `this.listA`
+
When `this.story` changes to `’B’` it will collect a fresh set of dependencies and remove the unnecessary ones (`this.listA`) that were used before but was not called anymore.
This way even if the other lists change it won’t trigger a recalculation of `selectedTransformedList`.
That’s smart!
+当 `this.story` 变成 `’B’`,它将会收集一组新的依赖,并移除那些之前用而现在不再使用的多余的依赖(`this.listA`)。这样,尽管其他 lists 发生变化,也不会触发 `selectedTransformedList` 的重计算。
Now it’s a good time to look again at the [code from the previous article - JSFiddle](https://jsfiddle.net/shentao/4k0gk3bx/10/), as the following changes will be based upon that code.
+现在是时候返回来看一看 [上一篇文章中的代码 - JSFiddle](https://jsfiddle.net/shentao/4k0gk3bx/10/),下面的改动将基于这些代码。
> The code in this article is written to be as simple as possible, ignoring many sanity checks and optimizations. By no means is it production ready, as it’s the only purpose is strictly educational.
+> 这篇文章中的代码尽量写的简单,忽略很多完整性检查和优化。绝不是已经可以用于生产环境的,仅仅用于教育目的。
+
Let’s create a new data model:
+我们来创建一个新的数据模型:
```
const App = Seer({
data: {
// observable values
- goodCharacter: 'Cloud Strife',
+ // 可观察的值
+ goodCharacter: 'Cloud Strife',
evilCharacter: 'Sephiroth',
placeholder: 'Choose your side!',
side: null,
// computed property
- selectedCharacter () {
+ // 计算属性
+ selectedCharacter () {
switch (this.side) {
case 'Good':
return `Your character is ${this.goodCharacter}!`
@@ -110,7 +139,8 @@ const App = Seer({
}
},
// computed property depending on other computed
- selectedCharacterSentenceLength () {
+ // 依赖其他计算属性的计算属性
+ selectedCharacterSentenceLength () {
return this.selectedCharacter.length
}
}
@@ -118,11 +148,14 @@ const App = Seer({
```
### Detecting the dependencies ###
+### 检测依赖 ###
To find out the dependencies of the currently evaluated computed property, we need a way to collect the dependencies. As you know, every observable property is already transformed into a getter and setter.
When evaluating the computed property (function) it will access other properties, which will trigger their getters.
+为了找到当前求值计算属性的依赖,需要一种收集依赖的办法。如你所知,每个可观察属性是已经转换成 getter 和 setter 的形式。当对计算属性(函数)求值,需要用到其他的属性,也就是出发他们的 getters。
For example this function:
+例如这个函数:
```
{
@@ -133,33 +166,41 @@ For example this function:
```
will call both the `firstName` and `lastName` getters.
+将会调用 `firstName` 和 `lastName` 的 getters。
Let’s make use of it!
+让我们开始使用它!
We need a way to collect the information that a getter was called when evaluating a computed property. For this to work, first we need a place to store which computed property is currently being evaluated. We can use a simple object for this:
+当对计算属性求值的时候,我们需要收集 getter 被调用的信息。为了完成这项工作,首先需要空间存储当前求值的计算属性。可以用这样的简单对象:
```
let Dep = {
// Name of the currently evaluated computed value
- target: null
+ // 当前求值的计算属性的名字
+ target: null
}
```
We used the `makeReactive` function to transform primitives into observable properties. Let’s create a transform function for computed properties and name it `makeComputed`.
+使用 `makeReactive` 函数将原始属性转换成可观察属性。为计算属性创建一个转换函数并将它命名为 `makeComputed`。
```
function makeComputed (obj, key, computeFunc) {
Object.defineProperty(obj, key, {
get () {
// If there is no target set
- if (!Dep.target) {
+ // 如果没有 target 集合
+ if (!Dep.target) {
// Set the currently evaluated property as the target
- Dep.target = key
+ // 设置 target 为当前求值的属性
+ Dep.target = key
}
const value = computeFunc.call(obj)
// Clear the target context
- Dep.target = null
+ // 清空 target 上下文
+ Dep.target = null
return value
},
set () {
@@ -169,26 +210,32 @@ function makeComputed (obj, key, computeFunc) {
}
// It will be later called in this manner
+// 后面将会用这种方式调用
makeComputed(data, 'fullName', data['fullName'])
```
Okay. Now that the context is available, we can modify our `makeReactive` function that we created in the previous article to make use of that context.
+Okay!既然上下文可以获取了,修改上一篇文章中创建的 `makeReactive` 函数以便使用获取的上下文。
The new `makeReactive` function should look like this:
+新的 `makeReactive` 函数像下面这样:
```
function makeReactive (obj, key) {
let val = obj[key]
// create an empty array for storing dependencies
- let deps = []
+ // 创建空数组用来存依赖
+ let deps = []
Object.defineProperty(obj, key, {
get () {
// Run only when called within a computed property context
- if (Dep.target) {
+ // 只有在计算属性上下文中调用的时候才会执行
+ if (Dep.target) {
// Add the computed property as depending on this value
// if not yet added
- if (!deps.includes(Dep.target)) {
+ // 如果还没添加,把这个计算属性作为依赖这个值添加
+ if (!deps.includes(Dep.target)) {
deps.push(Dep.target)
}
}
@@ -198,9 +245,11 @@ function makeReactive (obj, key) {
val = newVal
// If there are computed properties
// that depend on this value
- if (deps.length) {
+ // 如果有依赖于这个值的计算属性
+ if (deps.length) {
// Notify each computed property observers
- deps.forEach(notify)
+ // 通知每个计算属性的观察者
+ deps.forEach(notify)
}
notify(key)
}
@@ -210,6 +259,8 @@ function makeReactive (obj, key) {
One last thing we need is to slightly modify our `observeData` function so that it runs `makeComputed` instead of `makeReactive` for properties that are functions.
+我们要做的最后一件事就是稍稍改进 `observeData` 函数,以便对于函数形式的属性,它运行 `makeComputed` 而不是 `makeReactive`。
+
```
function observeData (obj) {
for (let key in obj) {
@@ -226,21 +277,27 @@ function observeData (obj) {
```
And that’s basically it! We just created our own implementation of computed properties with dependency tracking.
+基本上就是这样!我们刚刚通过依赖追踪创建了我们自己的计算数学实现。
-Sadly – the above, very naive implementation still lacks some crucial features that can be found in Vue.js and MobX. I guess the most important of those will be caching and removing the dead dependencies. So let’s add them.
+Sadly – the above, very naive implementation still lacks some crucial that can be found in Vue.js and MobX. I guess the most important of those will be caching and removing the dead dependencies. So let’s add them.
+不幸的是 —— 上面的实现是非常基础的,仍然缺少 Vue.js 和 MobX 中可以找到的重要的特性。我猜最重要的就是缓存和移除废弃的依赖。所以我们把它们添上。
## Caching ##
+## 缓存 ##
First, we need to add a place to store the cache. Let’s add our cache management to the `makeComputed` function:
+首先,我们需要空间存储缓存。我们在 `makeComputed` 函数中添加缓存管理器。
```
function makeComputed (obj, key, computeFunc) {
let cache = null
// Observe self to clear cache when deps change
- observe(key, () => {
+ // 自观察,当 deps 改变的时候清除缓存
+ observe(key, () => {
// Clear cache
- cache = null
+ // 清空缓存
+ cache = null
})
Object.defineProperty(obj, key, {
@@ -250,9 +307,11 @@ function makeComputed (obj, key, computeFunc) {
}
// If not cached yet
- if (!cache) {
+ // 当没有缓存
+ if (!cache) {
// calculate new value and save to cache
- cache = computeFunc.call(obj)
+ // 计算新的值并存入缓存
+ cache = computeFunc.call(obj)
}
Dep.target = null
@@ -266,56 +325,74 @@ function makeComputed (obj, key, computeFunc) {
```
That’s it! Now each time we access our computed property after the initial computation, it will return the cached value until it has to be recalculated. Pretty straightforward, right?
+就是这样!现在在初始化计算后,每次读取计算属性,它都会返回缓存的值,直到不得不重新计算。相当简单,是不是?
Thanks to the `observe` method we used inside `makeComputed` during the data transformation process, we ensure to always clean the cache before other signal handlers are executed. This means whenever one of the computed property’s dependencies change, the cache will be cleaned, just before the interface gets updated.
+多亏了 `observe` 函数,在数据转换过程中我们在 `makeComputed` 内部使用,确保在其他 signal 处理器执行前清空缓存。这意味着,计算属性的一个依赖发生变化,缓存将被清空,刚刚好在界面更新前。
## Removing the unnecessary dependencies ##
+## 移除不必要的依赖 ##
So now what’s left is to get rid of dependencies that are no longer valid. This is often a case when our computed properties conditionally depend on different values. What we want to achieve is our computed property only depending on the last used dependencies. The above implementation is flawed in that once a dependency registers that a computed property depends on it, it stays this way forever.
+现在剩下的工作就是清理无效的依赖。当计算属性依赖于不同的值的时候通常是一个案例。我们想达到的效果是计算属性仅依赖最后使用的依赖项。上面的实现在这方面是由缺陷的,一旦计算属性登记了依赖于它,它就一直在那了。
There are probably better ways to handle this, but because we want to keep things really simple let’s just create a secondary dependency list. One that will store a computed property’s dependencies.
To sum things up, our dependency lists:
+可能有更好的方式处理这种情况,但是因为我们想保持简单,我们来创建第二个依赖列表,来存储计算属性的依赖项。
+总结来说,我们的依赖列表:
- List of computed property names that depend on this value (observables or other computed) stored locally. Think: **Those are the values that depend on me.**
- A secondary dependency list that is used to remove dead dependencies and stores the most recent dependencies of a computed property. Think: **Those are the values I’m depending on.**
+- 依赖于这个值(可观察的或者其他的计算的)的计算属性名列表存储在本地。Think:**这些是依赖于我的值。**
+- 第二个依赖列表,用来移除废弃的依赖项并存储计算属性的最新的依赖。Think:**这些值是我依赖的**
+
With those two lists, we can run a filter function to remove the no longer valid dependencies. So let’s start with creating an object to store a secondary dependency list and some utility functions.
+用这两列表,我们可以运行一个过滤函数来移除无效的依赖项。让我们首先创建一个存储第二个依赖列表的对象和一些实用的函数。
```
let Dep = {
target: null,
// Stores the dependencies of computed properties
- subs: {},
+ // 存储计算属性的依赖项
+ subs: {},
// Create a two-way dependency relation between computed properties
// and other computed or observable values
- depend (deps, dep) {
+ // 在计算属性和其他计算或者可观察的值之间创建双向的依赖关联
+ depend (deps, dep) {
// Add the current context (Dep.target) to local deps
// as depending on the current property
// if not yet added
- if (!deps.includes(this.target)) {
+ // 如果还没添加,则添加当前上下文(Dep.target)到本地的 deps,作为依赖于当前属性
+ if (!deps.includes(this.target)) {
deps.push(this.target)
}
// Add the current property as a dependency of the computed value
// if not yet added
- if (!Dep.subs[this.target].includes(dep)) {
+ // 如果还没有添加,将当前属性作为计算值的依赖加入
+ if (!Dep.subs[this.target].includes(dep)) {
Dep.subs[this.target].push(dep)
}
},
getValidDeps (deps, key) {
// Filter only valid dependencies by removing dead dependencies
// that were not used during last computation
- return deps.filter(dep => this.subs[dep].includes(key))
+ // 通过移除在上一次计算中没有使用的废弃依赖,仅仅过滤出有效的依赖
+ return deps.filter(dep => this.subs[dep].includes(key))
},
notifyDeps (deps) {
// notify all existing deps
- deps.forEach(notify)
+ // 通知所有已存在的 deps
+ deps.forEach(notify)
}
}
```
If the `Dep.depend` method doesn’t make much sense right now, wait until we use it. It should become more clear what is actually happening there.
+如果 `Dep.depend` 函数现在还没什么意义,等一下我们就会用到它。这里发生的应该更加清楚了。
First, let’s tune the `makeReactive` transform function.
+首先,来调整 `makeReactive` 转换函数。
```
function makeReactive (obj, key, computeFunc) {
@@ -338,9 +415,11 @@ function makeReactive (obj, key, computeFunc) {
val = newVal
// Clean up dead dependencies
- deps = Dep.getValidDeps(deps, key)
+ // 清除废弃依赖
+ deps = Dep.getValidDeps(deps, key)
// and notify valid deps
- Dep.notifyDeps(deps, key)
+ // 并通知有效的 deps
+ Dep.notifyDeps(deps, key)
notify(key)
}
@@ -350,6 +429,8 @@ function makeReactive (obj, key, computeFunc) {
Almost the same has to be changed inside the `makeComputed` transform function. The difference is that we won’t be using the setter but the signal handler callback we passed to the `observe` function. Why? Because this callback will be called whenever the actual computed value has to update, as its dependencies have changed.
+`makeComputed` 转换函数内部也需要做相似的改动。不同在于不使用 setter 而是用传给 `observe` 函数的 signal 回调处理器。为什么?因为这个回调无论何时计算的值更新了,也就是依赖改变了,都会被调用。
+
```
function makeComputed (obj, key, computeFunc) {
let cache = null
@@ -380,12 +461,14 @@ function makeComputed (obj, key, computeFunc) {
if (!cache) {
// Clear dependencies list to ensure getting a fresh one
- Dep.subs[key] = []
+ // 清空依赖列表以便获得一个新的
+ Dep.subs[key] = []
cache = computeFunc.call(obj)
}
// Clear the target context
- Dep.target = null
+ // 清空目标上下文
+ Dep.target = null
return cache
},
set () {
@@ -396,24 +479,32 @@ function makeComputed (obj, key, computeFunc) {
```
Done! You might have already noticed that it also enables computed properties to be dependent on other computed properties, without having to know about the observables that lie underneath. Pretty sweet, isn’t it?
+完成了!你可能已经注意到,它允许计算属性依赖于其他计算属性,不需要知道背后的可观察的对象。相当不错,是不?
## Asynchronous pitfalls ##
+## 异步陷阱 ##
Now that you know how dependency tracking works, it should be quite obvious why it’s not possible to track asynchronous data inside computed properties both in MobX and Vue.js. It all breaks because even a `setTimeout(callback, 0)` will be called out of the current context where `Dep.target` no longer exists. This means that whatever happens inside the callback won’t be tracked.
+既然你知道了依赖追踪如何工作,也就很明显为什么在 MobX 和 Vue.js 中追踪计算属性中的异步数据是不可能的。这一切会被打破因为即使 `setTimeout(callback, 0)` 将会被当前上下文外被调用,在那里 `Dep.target` 不在存在。这也就意味着在回调函数中无论发生什么都不会被追踪到。
## Bonus: Watchers ##
+## 红利:Watchers ##
The above problem can be, however, partially tackled with watchers. You might know them from Vue.js. Building watchers on top of what we already have is actually really simple. After all, a watcher is just a signal handler called after a given value has changed.
+然而,上面的问题可以通过 watchers 部分的解决。你可能已经在 Vue.js 中了解过它们。在我们已有的基础上创建 watchers 真的很容易。毕竟,watcher是一个给定值发生变化时调用的 signal 处理器。
We just have to add a watchers registration method and trigger it within our Seer function.
+我们只是不得不添加一个 watchers 注册方法并在 Seer 函数内触发它。
+
```
function subscribeWatchers(watchers, context) {
for (let key in watchers) {
if (watchers.hasOwnProperty(key)) {
// We use Function.prototype.bind to bind our data model
// as the new `this` context for our signal handler
- observe(key, watchers[key].bind(context))
+ // 使用 Function.prototype.bind 来绑定数据模型,作为我们 signal 处理器新的 `this` 上下文
+ observe(key, watchers[key].bind(context))
}
}
}
@@ -422,6 +513,7 @@ subscribeWatchers(config.watch, config.data)
```
That’s all! We can use it like this:
+这就是全部了,可以像这样用它:
```
const App = Seer({
@@ -429,11 +521,14 @@ const App = Seer({
goodCharacter: 'Cloud Strife'
},
// here we can declare watchers
- watch: {
+ // 这里可以忽略 watchers
+ watch: {
// watch for 'goodCharacter' changes
+ // 'goodCharacter' 改变时的 watch
goodCharacter () {
// log the value into the console
- console.log(this.goodCharacter)
+ // 在控制台输出值
+ console.log(this.goodCharacter)
}
}
}
@@ -441,20 +536,28 @@ const App = Seer({
```
The complete code is available here:
+完整的代码可以在下面获得:
[https://github.com/shentao/seer/tree/cached-computed](https://github.com/shentao/seer/tree/cached-computed)
You can play with it online here (Opera/Chrome only):
+你可以在线的试试它(仅支持 Opera/Chrome):
[https://jsfiddle.net/oyw72Lyy/](https://jsfiddle.net/oyw72Lyy/)
## Summary ##
+## 总结 ##
I hope you enjoyed the tutorial and that the provided explanation turned out to be good enough to bring some light into what might be happening inside Vue or MobX when using computed properties. Keep in mind that the provided implementation was meant to be quite naive and not on par with the mentioned libraries. It is also not production ready in any way.
-
+我希望你们喜欢这个教程,当使用计算属性的时候,希望我的解释很好的阐明了 Vue 或 MobX 内部的原理。记住本文提供的实现是相当基础的,和提到的库中的实现不是同等水平的。无论如何都不是可以直接用于生产环境的。
## What comes next? ##
+## 接下来讲什么? ##
The 3rd part should include support for nested properties and observing arrays. I might also finally add a way to unsubscribe from the event bus! :D
As for the 4th part – maybe streams? Would you be interested?
+第三部分包含对嵌套属性和可观察数组的支持,我也可能在最后添加从事件中取消订阅的办法! :D
+至于第四部分,也许是数据流?你们感兴趣吗?
Feel free to leave your feedback in the comments!
+在评论区随意反馈意见!
Thanks for reading!
+感谢阅读!
From 370f4f05689ea1c2a5c2520d62b5918a43a3508a Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Sun, 26 Mar 2017 10:48:51 +0800
Subject: [PATCH 011/945] =?UTF-8?q?=E5=86=85=E5=AE=B9=E6=A0=A1=E5=AF=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...operties-javascript-dependency-tracking.md | 74 ++++++++++---------
1 file changed, 40 insertions(+), 34 deletions(-)
diff --git a/TODO/computed-properties-javascript-dependency-tracking.md b/TODO/computed-properties-javascript-dependency-tracking.md
index 41ac3611901..3070a1809fb 100644
--- a/TODO/computed-properties-javascript-dependency-tracking.md
+++ b/TODO/computed-properties-javascript-dependency-tracking.md
@@ -1,7 +1,7 @@
> * 原文地址:[How to build a reactive engine in JavaScript. Part 2: Computed properties and dependency tracking](https://monterail.com/blog/2017/computed-properties-javascript-dependency-tracking)
> * 原文作者:[Damian Dulisz](https://disqus.com/by/damiandulisz/)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[IridescentMia](https://github.com/IridescentMia)
> * 校对者:
![](https://d4a7vd7s8p76l.cloudfront.net/uploads/56873733-c918-4cd6-bac1-dea44dcc3a9f/Reactive%20engine.png)
@@ -19,7 +19,7 @@ Hey!如果你用过 Vue.js、Ember 或 MobX,我敢肯定你被 **计算**
## 计算属性 ##
Let’s say we have a computed property called `fullName` which is a combination of `firstName` and `lastName` with space in between.
-比如有一个计算属性叫 `fullName`,是 `firstName` 和 `lastName` 之间加上空格的组合。
+假设有一个计算属性叫 `fullName`,是 `firstName` 和 `lastName` 之间加上空格的组合。
In Vue.js such a computed value could be created like this:
在 Vue.js 中这样的计算值可以像下面这样创建:
@@ -55,10 +55,10 @@ Let’s consider the following example:
```
The result of the above code will be mostly the same. Each time `firstName` or `lastName` changes, the view will update with the headers and show the full name.
-上面代码的执行结果几乎是一样的。每次 `firstName` 或 `lastName` 发生变化,视图将会更新这些 `` 并且显示 full name。
+上面代码的执行结果几乎是一样的。每次 `firstName` 或 `lastName` 发生变化,视图将会更新这些 `` 并且显示出全名。
However, what if we use the expression, method call and computed property multiple times? The expression and method call will have to be calculated each time they are accessed, whereas the computed property will be cached after the first computation until one of its dependencies change. It will also persist through the re-render cycles! That’s actually quite a nice optimization if you consider that in event-based modern user interfaces, it’s hard to predict which action the user will take first.
-然而,如果多次使用表达式、函数调用和计算属性呢?使用表达式和函数调用每次都会计算一遍,而计算属性在第一次计算后将会缓存下来,直到它的依赖发生改变。它也会一直坚持 re-render 循环!如果考虑在基于事件模型的用户界面中,很难预测用户会首先执行哪项操作,那么这确实是一个最佳选择方案。
+然而,如果多次使用表达式、函数调用和计算属性呢?使用表达式和函数调用每次都会计算一遍,而计算属性在第一次计算后将会缓存下来,直到它的依赖发生改变。它也会在重新渲染的循环中一直保持!如果考虑在基于事件模型的用户界面中,很难预测用户会首先执行哪项操作,那么这确实是一个最优化方案。
## Basic computed property ##
## 基础的计算属性 ##
@@ -75,7 +75,7 @@ fullName: Ember.computed('firstName', 'lastName', function() {
```
The drawback here is that you have to declare the dependencies yourself. Doesn’t seem like a problem until you have computed properties that are a result of a chain of more expensive, complex functions. For example:
-这样做的缺点就是你不得不自己声明依赖。当你的计算属性是一串高开销的、功能复杂的函数的时候,你就知道这的确是个问题了。例如:
+这样做的缺点就是你不得不自己声明依赖。当你的计算属性是一串高开销的、复杂的函数的时候,你就知道这的确是个问题了。例如:
```
selectedTransformedList: Ember.computed('story', 'listA', 'listB', 'listC', function() {
@@ -91,7 +91,7 @@ selectedTransformedList: Ember.computed('story', 'listA', 'listB', 'listC', func
```
In the above case, even if `this.story` always equals `’A’`, the computed property will have to be re-evaluated each time one of the lists changes. Even if they are not used for the end result.
-在上面的案例中,即便 `this.story` 总是等于 `’A’`,一旦 lists 发生改变,计算属性将不得不每次都反复求值。
+在上面的案例中,即便 `this.story` 总是等于 `’A’`,一旦 lists 发生改变,计算属性也将不得不每次都反复计算。
## Dependency tracking ##
## 依赖追踪 ##
@@ -152,7 +152,7 @@ const App = Seer({
To find out the dependencies of the currently evaluated computed property, we need a way to collect the dependencies. As you know, every observable property is already transformed into a getter and setter.
When evaluating the computed property (function) it will access other properties, which will trigger their getters.
-为了找到当前求值计算属性的依赖,需要一种收集依赖的办法。如你所知,每个可观察属性是已经转换成 getter 和 setter 的形式。当对计算属性(函数)求值,需要用到其他的属性,也就是出发他们的 getters。
+为了找到当前求值计算属性的依赖,需要一种收集依赖的办法。如你所知,每个可观察属性是已经转换成 getter 和 setter 的形式。当对计算属性(函数)求值的时候,需要用到其他的属性,也就是触发他们的 getters。
For example this function:
例如这个函数:
@@ -183,7 +183,7 @@ let Dep = {
```
We used the `makeReactive` function to transform primitives into observable properties. Let’s create a transform function for computed properties and name it `makeComputed`.
-使用 `makeReactive` 函数将原始属性转换成可观察属性。为计算属性创建一个转换函数并将它命名为 `makeComputed`。
+我们过去曾用 `makeReactive` 函数将原始属性转换成可观察属性。现在让我们为计算属性创建一个转换函数并将它命名为 `makeComputed`。
```
function makeComputed (obj, key, computeFunc) {
@@ -215,7 +215,7 @@ makeComputed(data, 'fullName', data['fullName'])
```
Okay. Now that the context is available, we can modify our `makeReactive` function that we created in the previous article to make use of that context.
-Okay!既然上下文可以获取了,修改上一篇文章中创建的 `makeReactive` 函数以便使用获取的上下文。
+Okay!既然上下文可以获取了,修改上一篇文章中创建的 `makeReactive` 函数以便使用获取到的上下文。
The new `makeReactive` function should look like this:
新的 `makeReactive` 函数像下面这样:
@@ -234,7 +234,7 @@ function makeReactive (obj, key) {
if (Dep.target) {
// Add the computed property as depending on this value
// if not yet added
- // 如果还没添加,把这个计算属性作为依赖这个值添加
+ // 如果还没添加,则作为依赖这个值的计算属性添加
if (!deps.includes(Dep.target)) {
deps.push(Dep.target)
}
@@ -258,7 +258,6 @@ function makeReactive (obj, key) {
```
One last thing we need is to slightly modify our `observeData` function so that it runs `makeComputed` instead of `makeReactive` for properties that are functions.
-
我们要做的最后一件事就是稍稍改进 `observeData` 函数,以便对于函数形式的属性,它运行 `makeComputed` 而不是 `makeReactive`。
```
@@ -277,7 +276,7 @@ function observeData (obj) {
```
And that’s basically it! We just created our own implementation of computed properties with dependency tracking.
-基本上就是这样!我们刚刚通过依赖追踪创建了我们自己的计算数学实现。
+基本上就是这样!我们刚刚通过依赖追踪创建了我们自己的计算属性实现。
Sadly – the above, very naive implementation still lacks some crucial that can be found in Vue.js and MobX. I guess the most important of those will be caching and removing the dead dependencies. So let’s add them.
不幸的是 —— 上面的实现是非常基础的,仍然缺少 Vue.js 和 MobX 中可以找到的重要的特性。我猜最重要的就是缓存和移除废弃的依赖。所以我们把它们添上。
@@ -328,13 +327,13 @@ That’s it! Now each time we access our computed property after the initial com
就是这样!现在在初始化计算后,每次读取计算属性,它都会返回缓存的值,直到不得不重新计算。相当简单,是不是?
Thanks to the `observe` method we used inside `makeComputed` during the data transformation process, we ensure to always clean the cache before other signal handlers are executed. This means whenever one of the computed property’s dependencies change, the cache will be cleaned, just before the interface gets updated.
-多亏了 `observe` 函数,在数据转换过程中我们在 `makeComputed` 内部使用,确保在其他 signal 处理器执行前清空缓存。这意味着,计算属性的一个依赖发生变化,缓存将被清空,刚刚好在界面更新前。
+多亏了 `observe` 函数,在数据转换过程中我们在 `makeComputed` 内部使用,确保在其他信号处理器执行前清空缓存。这意味着,计算属性的一个依赖发生变化,缓存将被清空,刚刚好在界面更新前完成。
## Removing the unnecessary dependencies ##
## 移除不必要的依赖 ##
So now what’s left is to get rid of dependencies that are no longer valid. This is often a case when our computed properties conditionally depend on different values. What we want to achieve is our computed property only depending on the last used dependencies. The above implementation is flawed in that once a dependency registers that a computed property depends on it, it stays this way forever.
-现在剩下的工作就是清理无效的依赖。当计算属性依赖于不同的值的时候通常是一个案例。我们想达到的效果是计算属性仅依赖最后使用的依赖项。上面的实现在这方面是由缺陷的,一旦计算属性登记了依赖于它,它就一直在那了。
+现在剩下的工作就是清理无效的依赖。当计算属性依赖于不同的值的时候通常是一个案例。我们想达到的效果是计算属性仅依赖最后使用到的依赖。上面的实现在这方面是有缺陷的,一旦计算属性登记了依赖于它,它就一直在那了。
There are probably better ways to handle this, but because we want to keep things really simple let’s just create a secondary dependency list. One that will store a computed property’s dependencies.
To sum things up, our dependency lists:
@@ -344,21 +343,21 @@ To sum things up, our dependency lists:
- List of computed property names that depend on this value (observables or other computed) stored locally. Think: **Those are the values that depend on me.**
- A secondary dependency list that is used to remove dead dependencies and stores the most recent dependencies of a computed property. Think: **Those are the values I’m depending on.**
-- 依赖于这个值(可观察的或者其他的计算的)的计算属性名列表存储在本地。Think:**这些是依赖于我的值。**
-- 第二个依赖列表,用来移除废弃的依赖项并存储计算属性的最新的依赖。Think:**这些值是我依赖的**
+- 依赖于这个值(可观察的或者其他的计算后的)的计算属性名列表存储在本地。可以这样想:**这些是依赖于我的值。**
+- 第二个依赖列表,用来移除废弃的依赖并存储计算属性的最新的依赖。可以这样想:**这些值是我依赖的**
With those two lists, we can run a filter function to remove the no longer valid dependencies. So let’s start with creating an object to store a secondary dependency list and some utility functions.
-用这两列表,我们可以运行一个过滤函数来移除无效的依赖项。让我们首先创建一个存储第二个依赖列表的对象和一些实用的函数。
+用这两列表,我们可以运行一个过滤函数来移除无效的依赖。让我们首先创建一个存储第二个依赖列表的对象和一些实用的函数。
```
let Dep = {
target: null,
// Stores the dependencies of computed properties
- // 存储计算属性的依赖项
+ // 存储计算属性的依赖
subs: {},
// Create a two-way dependency relation between computed properties
// and other computed or observable values
- // 在计算属性和其他计算或者可观察的值之间创建双向的依赖关联
+ // 在计算属性和其他计算后的或者可观察的值之间创建双向的依赖关系
depend (deps, dep) {
// Add the current context (Dep.target) to local deps
// as depending on the current property
@@ -389,7 +388,7 @@ let Dep = {
```
If the `Dep.depend` method doesn’t make much sense right now, wait until we use it. It should become more clear what is actually happening there.
-如果 `Dep.depend` 函数现在还没什么意义,等一下我们就会用到它。这里发生的应该更加清楚了。
+如果 `Dep.depend` 函数现在还没什么意义,等一下我们就会用到它。这里做的什么应该就更清楚了。
First, let’s tune the `makeReactive` transform function.
首先,来调整 `makeReactive` 转换函数。
@@ -402,11 +401,13 @@ function makeReactive (obj, key, computeFunc) {
Object.defineProperty(obj, key, {
get () {
// Run only when getting within a computed value context
- if (Dep.target) {
+ // 只有当在计算值的上下文内时才执行
+ if (Dep.target) {
// Add Dep.target as depending on this value
// this will mutate the deps Array
// as we’re passing a reference to it
- Dep.depend(deps, key)
+ // 将 Dep.target 作为依赖于这个值添加,浙江使 deps 数组发生变化,因为我们给它传了一个引用
+ Dep.depend(deps, key)
}
return val
@@ -429,19 +430,21 @@ function makeReactive (obj, key, computeFunc) {
Almost the same has to be changed inside the `makeComputed` transform function. The difference is that we won’t be using the setter but the signal handler callback we passed to the `observe` function. Why? Because this callback will be called whenever the actual computed value has to update, as its dependencies have changed.
-`makeComputed` 转换函数内部也需要做相似的改动。不同在于不使用 setter 而是用传给 `observe` 函数的 signal 回调处理器。为什么?因为这个回调无论何时计算的值更新了,也就是依赖改变了,都会被调用。
+`makeComputed` 转换函数内部也需要做相似的改动。不同在于不使用 setter 而是用传给 `observe` 函数的信号回调处理器。为什么?因为这个回调无论何时计算的值更新了,也就是依赖改变了,都会被调用。
```
function makeComputed (obj, key, computeFunc) {
let cache = null
// Create a local deps list similar to makeReactive deps
- let deps = []
+ // 创建一个本地的 deps 列表,相似于 makeReactive 的 deps
+ let deps = []
observe(key, () => {
cache = null
// Clean up and notify valid deps
- deps = Dep.getValidDeps(deps, key)
+ // 清空并通知有效的 deps
+ deps = Dep.getValidDeps(deps, key)
Dep.notifyDeps(deps, key)
})
@@ -449,19 +452,22 @@ function makeComputed (obj, key, computeFunc) {
get () {
// If evaluated during the evaluation of
// another computed property
- if (Dep.target) {
+ // 如果如果在其他计算属性正在计算的时候计算
+ if (Dep.target) {
// Create a dependency relationship
// between those two computed properties
- Dep.depend(deps, key)
+ // 在这两个计算属性之间创建一个依赖关系
+ Dep.depend(deps, key)
}
// Normalize Dep.target back to self
// This makes it possible to build a dependency tree
// instead of a flat structure
- Dep.target = key
+ // 将 Dep.target 标准化成它原本的样子,这使得构建一个依赖树成为可能,而不是一个扁平化的结构
+ Dep.target = key
if (!cache) {
// Clear dependencies list to ensure getting a fresh one
- // 清空依赖列表以便获得一个新的
+ // 清空依赖列表以获得一个新的
Dep.subs[key] = []
cache = computeFunc.call(obj)
}
@@ -485,16 +491,15 @@ Done! You might have already noticed that it also enables computed properties to
## 异步陷阱 ##
Now that you know how dependency tracking works, it should be quite obvious why it’s not possible to track asynchronous data inside computed properties both in MobX and Vue.js. It all breaks because even a `setTimeout(callback, 0)` will be called out of the current context where `Dep.target` no longer exists. This means that whatever happens inside the callback won’t be tracked.
-既然你知道了依赖追踪如何工作,也就很明显为什么在 MobX 和 Vue.js 中追踪计算属性中的异步数据是不可能的。这一切会被打破因为即使 `setTimeout(callback, 0)` 将会被当前上下文外被调用,在那里 `Dep.target` 不在存在。这也就意味着在回调函数中无论发生什么都不会被追踪到。
+既然你知道了依赖追踪如何工作,也就很明显为什么在 MobX 和 Vue.js 中追踪计算属性中的异步数据是不可能的。这一切会被打破,因为即使 `setTimeout(callback, 0)` 将会在当前上下文外被调用,在那里 `Dep.target` 不在存在。这也就意味着在回调函数中无论发生什么都不会被追踪到。
## Bonus: Watchers ##
## 红利:Watchers ##
The above problem can be, however, partially tackled with watchers. You might know them from Vue.js. Building watchers on top of what we already have is actually really simple. After all, a watcher is just a signal handler called after a given value has changed.
-然而,上面的问题可以通过 watchers 部分的解决。你可能已经在 Vue.js 中了解过它们。在我们已有的基础上创建 watchers 真的很容易。毕竟,watcher是一个给定值发生变化时调用的 signal 处理器。
+然而,上面的问题可以通过 watchers 部分的解决。你可能已经在 Vue.js 中了解过它们。在我们已有的基础上创建 watchers 真的很容易。毕竟,watcher 是一个给定值发生变化时调用的信号处理器。
We just have to add a watchers registration method and trigger it within our Seer function.
-
我们只是不得不添加一个 watchers 注册方法并在 Seer 函数内触发它。
```
@@ -503,7 +508,7 @@ function subscribeWatchers(watchers, context) {
if (watchers.hasOwnProperty(key)) {
// We use Function.prototype.bind to bind our data model
// as the new `this` context for our signal handler
- // 使用 Function.prototype.bind 来绑定数据模型,作为我们 signal 处理器新的 `this` 上下文
+ // 使用 Function.prototype.bind 来绑定数据模型,作为我们信号处理器新的 `this` 上下文
observe(key, watchers[key].bind(context))
}
}
@@ -548,12 +553,13 @@ You can play with it online here (Opera/Chrome only):
I hope you enjoyed the tutorial and that the provided explanation turned out to be good enough to bring some light into what might be happening inside Vue or MobX when using computed properties. Keep in mind that the provided implementation was meant to be quite naive and not on par with the mentioned libraries. It is also not production ready in any way.
我希望你们喜欢这个教程,当使用计算属性的时候,希望我的解释很好的阐明了 Vue 或 MobX 内部的原理。记住本文提供的实现是相当基础的,和提到的库中的实现不是同等水平的。无论如何都不是可以直接用于生产环境的。
+
## What comes next? ##
## 接下来讲什么? ##
The 3rd part should include support for nested properties and observing arrays. I might also finally add a way to unsubscribe from the event bus! :D
As for the 4th part – maybe streams? Would you be interested?
-第三部分包含对嵌套属性和可观察数组的支持,我也可能在最后添加从事件中取消订阅的办法! :D
+第三部分涵盖了对嵌套属性和可观察数组的支持,我也可能在最后添加从事件中取消订阅的办法! :D
至于第四部分,也许是数据流?你们感兴趣吗?
Feel free to leave your feedback in the comments!
From 5cffbd29e19baf0ac674d67b430df9ea6895f933 Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Sun, 26 Mar 2017 10:52:54 +0800
Subject: [PATCH 012/945] =?UTF-8?q?=E6=A0=BC=E5=BC=8F=E6=A0=A1=E5=AF=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/computed-properties-javascript-dependency-tracking.md | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/TODO/computed-properties-javascript-dependency-tracking.md b/TODO/computed-properties-javascript-dependency-tracking.md
index 3070a1809fb..27be4bc4077 100644
--- a/TODO/computed-properties-javascript-dependency-tracking.md
+++ b/TODO/computed-properties-javascript-dependency-tracking.md
@@ -10,10 +10,10 @@
# 如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪 #
Hey! If you have ever worked with Vue.js, Ember or MobX I’m pretty sure you stumbled upon so-called **computed** properties. They allow you to create functions that can be accessed just like normal values, but once computed they are cached until one of its dependencies has changed. In general this is a concept very similar to getters and in fact, the following implementation will be using getters. In a smart way. ;)
-Hey!如果你用过 Vue.js、Ember 或 MobX,我敢肯定你被 **计算** 属性难倒过。计算属性允许你创建像正常的值一样使用的函数,但是一旦完成计算,他们就被缓存下来直到它的一个依赖发生改变。总的来说,这一概念与 getters 非常相似,并且事实上,下面的实现将会使用 getters。用一种机智的方式。;)
+Hey!如果你用过 Vue.js、Ember 或 MobX,我敢肯定你被 **计算** 属性难倒过。计算属性允许你创建像正常的值一样使用的函数,但是一旦完成计算,他们就被缓存下来直到它的一个依赖发生改变。总的来说,这一概念与 getters 非常相似,并且事实上,下面的实现将会使用 getters。用一种机智的方式。 ;)
> This is the 2nd part of the How to build a reactive engine in JavaScript series. Before reading any further it is highly recommended to read [Part 1: Observable objects](https://monterail.com/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects), because the following implementation is built on top of the previous article's code.
-> 这是如何使用 JavaScript 构建响应式引擎系列文章的第二部分。在深入阅读前强烈建议读一下[Part 1: 可观察的对象](https://monterail.com/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects),因为接下来的实现是构建于前一篇文章的代码之上的。
+> 这是如何使用 JavaScript 构建响应式引擎系列文章的第二部分。在深入阅读前强烈建议读一下 [Part 1: 可观察的对象](https://monterail.com/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects),因为接下来的实现是构建于前一篇文章的代码基础之上的。
## Computed properties ##
## 计算属性 ##
@@ -344,7 +344,7 @@ To sum things up, our dependency lists:
- A secondary dependency list that is used to remove dead dependencies and stores the most recent dependencies of a computed property. Think: **Those are the values I’m depending on.**
- 依赖于这个值(可观察的或者其他的计算后的)的计算属性名列表存储在本地。可以这样想:**这些是依赖于我的值。**
-- 第二个依赖列表,用来移除废弃的依赖并存储计算属性的最新的依赖。可以这样想:**这些值是我依赖的**
+- 第二个依赖列表,用来移除废弃的依赖并存储计算属性的最新的依赖。可以这样想:**这些值是我依赖的。**
With those two lists, we can run a filter function to remove the no longer valid dependencies. So let’s start with creating an object to store a secondary dependency list and some utility functions.
用这两列表,我们可以运行一个过滤函数来移除无效的依赖。让我们首先创建一个存储第二个依赖列表的对象和一些实用的函数。
From 61928edffc77c47caf3cc30887a6a4acce6be61a Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Sun, 26 Mar 2017 11:02:56 +0800
Subject: [PATCH 013/945] Update
how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
---
...in-javascript-part-1-observable-objects.md | 28 +++++++++----------
1 file changed, 14 insertions(+), 14 deletions(-)
diff --git a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
index 4b29caa4e82..363c4beaf5f 100644
--- a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
+++ b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
@@ -16,7 +16,7 @@ With the growing need for robust and interactive web interfaces, many developers
随着对强健、可交互的网站界面的需求不断增多,很多开发者开始拥抱响应式编程规范。
Before we begin implementing our own reactive engine, let’s quickly explain what reactive programming actually is. Wikipedia gives us a classic example of a reactive interface implementation – namely a spreadsheet. Defining a formula such as `=A1+B1` would update the cell whenever either `A1` or `B1` change. Such a formula can be considered a computed value.
-在开始实现我们自己的响应式引擎之前,快速的解释一下到底什么是响应式编程。维基百科给出一个经典的响应式界面实现的例子 —— 叫做 spreadsheet。定义一个准则,对于 `=A1+B1`,只要 `A1` 或 `B1` 发生变化,`=A1+B1` 也会随之变化。这个准则被认为是 computed value。
+在开始实现我们自己的响应式引擎之前,快速地解释一下到底什么是响应式编程。维基百科给出一个经典的响应式界面实现的例子 —— 叫做 spreadsheet。定义一个准则,对于 `=A1+B1`,只要 `A1` 或 `B1` 发生变化,`=A1+B1` 也会随之变化。这个准则被认为是 computed value。
You will learn how to implement computed values in the second part of this reactive series. Before that, we first need a base for our reactivity engine.
我们将会在这系列教程的 Part 2 部分学习如何实现 computed value。在那之前,我们首先需要对响应式引擎有个基础的了解。
@@ -33,9 +33,9 @@ Currently there are many different approaches to solving the problem of observin
- Libraries like Vue.js, MobX or Ractive.js all use a variation of getters/setters to create observable data models.
- Angular 1.x 有脏检查。
-- React 由于它工作方式的原因,并不追踪数据模型中的改变。它用虚拟 DOM 比较并修补 DOM。
-- Cycle.js 和 Angular 2 更喜欢响应流方式实现,像 XStream 和 Rx.js.
-- 像 Vue.js, MobX 或 Ractive.js 这些库都使用 getters/setters 变量创建可见的数据模型。
+- React 由于它工作方式,并不追踪数据模型中的改变。它用虚拟 DOM 比较并修补 DOM。
+- Cycle.js 和 Angular 2 更喜欢响应流方式实现,像 XStream 和 Rx.js。
+- 像 Vue.js, MobX 或 Ractive.js 这些库都使用 getters/setters 变量创建可观察的数据模型。
In this tutorial, we will go the getters/setters way of observing and reacting to changes.
在这篇教程中,我们将使用 getters/setters 的方式观察并响应变化。
@@ -45,10 +45,10 @@ In this tutorial, we will go the getters/setters way of observing and reacting t
> 注意:为了让这篇教程尽量保持简单,代码缺少对非初级数据类型或嵌套属性的支持,并且很多内容需要完整性检查,因此决不能认为这些代码已经可以用于生产环境。下面的代码是受 Vue.js 启发的响应式引擎的实现,使用 ES2015 标准编写。
## The observable object ##
-## 可见的对象 ##
+## 可观察的对象 ##
Let’s start with a `data` object, whose properties we want to observe.
-让我们从一个 `data` 对象开始,我们想要将它的属性可见。
+让我们从一个 `data` 对象开始,我们想要观察它的属性。
```
let data = {
@@ -59,7 +59,7 @@ let data = {
```
Let’s start by creating two functions that will transform our object’s properties into observable properties using the getter/setter functionality.
-首先从创建两个函数开始,使用 getter/setter 的功能,将对象的普通属性转换成可见的属性。
+首先从创建两个函数开始,使用 getter/setter 的功能,将对象的普通属性转换成可观察的属性。
```
function makeReactive (obj, key) {
@@ -67,7 +67,7 @@ function makeReactive (obj, key) {
Object.defineProperty(obj, key, {
get () {
- return val // 简单的返回缓存的 value
+ return val // 简单地返回缓存的 value
},
set (newVal) {
val = newVal // 保存 newVal
@@ -106,12 +106,12 @@ let signals = {} // Signals 从一个空对象开始
function observe (property, signalHandler) {
if(!signals[property]) signals[property] = [] // 如果给定属性没在 signal 中,则创建这个属性的 signal,并将其设置为空数组来存储 signalHandlers
- signals[property].push(signalHandler) // 将 signalHandler 存入 signal 数组,高效的获得一组保存在数组中的回调函数
+ signals[property].push(signalHandler) // 将 signalHandler 存入 signal 数组,高效地获得一组保存在数组中的回调函数
}
```
We can now use the `observe` function like this: `observe('propertyName', callback)`, where `callback` is a function that should be called each time the property’s value has changed. When we **observe** a property multiple times, each callback will be stored inside the corresponding property’s signal array. This way we can store all callbacks and have easy access to them.
-我们现在可以这样用 `observe` 函数:`observe('propertyName', callback)`,每次属性值发生改变的时候 `callback` 函数应该被调用。当多次在一个属性上调用 **observe** 时,每个回调函数将被存在对应属性的 signal 数组中。这样就可以存储所有的回调函数并且可以很容易的获得到它们。
+我们现在可以这样用 `observe` 函数:`observe('propertyName', callback)`,每次属性值发生改变的时候 `callback` 函数应该被调用。当多次在一个属性上调用 **observe** 时,每个回调函数将被存在对应属性的 signal 数组中。这样就可以存储所有的回调函数并且可以很容易地获得到它们。
Now for the `notify` function that you saw before.
现在来看一下上文中提到的 `notify` 函数。
@@ -196,7 +196,7 @@ App.observe('firstName', () => console.log(App.data.firstName))
App.observe('lastName', () => console.log(App.data.lastName))
// To trigger the above callbacks simply change the values like this:
-// 为了触发上面的回调函数,像下面这样简单的改变 values:
+// 为了触发上面的回调函数,像下面这样简单地改变 values:
App.data.firstName = 'Sansa'
App.data.lastName = 'Stark'
@@ -205,7 +205,7 @@ App.data.lastName = 'Stark'
Simple, isn’t it? Now that we have the basic reactivity engine covered, let’s make some use of it.
I mentioned that with the more reactive approach to front-end programming, we should not be concerned with things like manually updating the DOM after each change.
很简单,是不是?现在我们讲完了基本的响应式引擎,让我们来用用它。
-我提到过随着前端编程可响应式方法的增多,我们不能总想着在发生改变后手动的更新 DOM。
+我提到过随着前端编程可响应式方法的增多,我们不能总想着在发生改变后手动地更新 DOM。
There are many approaches to this. I guess the most trending one right now is the so called virtual DOM. If you are interested in learning how to create your own virtual DOM implementation, there are already great tutorials for this. However, here we will go with a much simpler approach.
有很多方法来完成这项任务。我猜现在最流行的趋势是用虚拟 DOM 的办法。如果你对学习如何创建你自己的虚拟 DOM 实现感兴趣,已经有很多这方面的教程。然而,这里我们将用到更简单的方法。
@@ -323,7 +323,7 @@ function Seer (dataObj) {
}
}
// We can safely parse the DOM looking for bindings after we converted the dataObject.
- //转换数据对象后,可以安全的解析 DOM 绑定。
+ //转换数据对象后,可以安全地解析 DOM 绑定。
parseDOM(document.body, obj)
}
@@ -415,7 +415,7 @@ function Seer (dataObj) {
}
}
// We can safely parse the DOM looking for bindings after we converted the dataObject.
- //转换数据对象后,可以安全的解析 DOM 绑定。
+ //转换数据对象后,可以安全地解析 DOM 绑定。
parseDOM(document.body, obj)
}
From 1f2f169eb8fb118c91dac3d798e0054bc231f9da Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Mon, 27 Mar 2017 14:39:45 +0800
Subject: [PATCH 014/945] =?UTF-8?q?=E4=BA=8C=E6=A0=A1?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...e-engine-in-javascript-part-1-observable-objects.md | 10 +++++-----
1 file changed, 5 insertions(+), 5 deletions(-)
diff --git a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
index 363c4beaf5f..6e7bfb04857 100644
--- a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
+++ b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
@@ -10,13 +10,13 @@
# 如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象 #
## The reactive way ##
-## 响应的方式 ##
+## 响应式的方式 ##
With the growing need for robust and interactive web interfaces, many developers have started embracing the reactive programming paradigm.
随着对强健、可交互的网站界面的需求不断增多,很多开发者开始拥抱响应式编程规范。
Before we begin implementing our own reactive engine, let’s quickly explain what reactive programming actually is. Wikipedia gives us a classic example of a reactive interface implementation – namely a spreadsheet. Defining a formula such as `=A1+B1` would update the cell whenever either `A1` or `B1` change. Such a formula can be considered a computed value.
-在开始实现我们自己的响应式引擎之前,快速地解释一下到底什么是响应式编程。维基百科给出一个经典的响应式界面实现的例子 —— 叫做 spreadsheet。定义一个准则,对于 `=A1+B1`,只要 `A1` 或 `B1` 发生变化,`=A1+B1` 也会随之变化。这个准则被认为是 computed value。
+在开始实现我们自己的响应式引擎之前,快速地解释一下到底什么是响应式编程。维基百科给出一个经典的响应式界面实现的例子 —— 叫做 spreadsheet。定义一个准则,对于 `=A1+B1`,只要 `A1` 或 `B1` 发生变化,`=A1+B1` 也会随之变化。这样的准则也可以被理解为是一种 computed value。
You will learn how to implement computed values in the second part of this reactive series. Before that, we first need a base for our reactivity engine.
我们将会在这系列教程的 Part 2 部分学习如何实现 computed value。在那之前,我们首先需要对响应式引擎有个基础的了解。
@@ -34,7 +34,7 @@ Currently there are many different approaches to solving the problem of observin
- Angular 1.x 有脏检查。
- React 由于它工作方式,并不追踪数据模型中的改变。它用虚拟 DOM 比较并修补 DOM。
-- Cycle.js 和 Angular 2 更喜欢响应流方式实现,像 XStream 和 Rx.js。
+- Cycle.js 和 Angular 2 更倾向于响应流方式实现,像 XStream 和 Rx.js。
- 像 Vue.js, MobX 或 Ractive.js 这些库都使用 getters/setters 变量创建可观察的数据模型。
In this tutorial, we will go the getters/setters way of observing and reacting to changes.
@@ -89,7 +89,7 @@ observeData(data)
```
By running `observeData(data)` we transform our object into an object capable of being observed; now we have a way to create notifications whenever the value changes.
-通过运行 `observeData(data)`,将原始的对象转换成具有可见性的对象;现在当对象的 value 发生变化时,我们有创建通知的办法。
+通过运行 `observeData(data)`,将原始的对象转换成可被观察的对象;现在当对象的 value 发生变化时,我们有创建通知的办法。
## Reacting to changes ##
## 响应变化 ##
@@ -136,7 +136,7 @@ function Seer (dataObj) {
observeData(dataObj)
- // 除了响应的数据对象,我们也需要返回并且暴露出 observe 和 notify 函数。
+ // 除了响应式的数据对象,我们也需要返回并且暴露出 observe 和 notify 函数。
return {
data: dataObj,
observe,
From dd1c4298275d57b9762a9d8bbd9f7e418a8a8134 Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Mon, 27 Mar 2017 14:42:47 +0800
Subject: [PATCH 015/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E6=A0=A1=E5=AF=B9?=
=?UTF-8?q?=E8=80=85?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...a-reactive-engine-in-javascript-part-1-observable-objects.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
index 6e7bfb04857..411cb7218e6 100644
--- a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
+++ b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
@@ -2,7 +2,7 @@
> * 原文作者:本文已获原作者 [Damian Dulisz](https://disqus.com/by/damiandulisz/) 授权
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[IridescentMia](https://github.com/IridescentMia)
-> * 校对者:
+> * 校对者:[reid3290](https://github.com/reid3290),[malcolmyu](https://github.com/malcolmyu)
![](https://d4a7vd7s8p76l.cloudfront.net/uploads/1484604970-4-7876/observables.png)
From 9a5900a2acd35a9b70da41c172954b9309a0d201 Mon Sep 17 00:00:00 2001
From: reid
Date: Mon, 27 Mar 2017 16:12:47 +0800
Subject: [PATCH 016/945] =?UTF-8?q?=E5=88=9D=E7=A8=BF=E7=BF=BB=E8=AF=91?=
=?UTF-8?q?=E5=AE=8C=E6=88=90?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/setstate-gate-abc.md | 243 ++++++++++++++++++++------------------
1 file changed, 125 insertions(+), 118 deletions(-)
diff --git a/TODO/setstate-gate-abc.md b/TODO/setstate-gate-abc.md
index d3f579800fc..8c8f024b4df 100644
--- a/TODO/setstate-gate-abc.md
+++ b/TODO/setstate-gate-abc.md
@@ -1,106 +1,108 @@
> * 原文地址:[setState() Gate](https://medium.com/javascript-scene/setstate-gate-abc10a9b2d82#.z148awo8n)
> * 原文作者:[Eric Elliott](https://medium.com/@_ericelliott?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[reid3290](https://github.com/reid3290)
> * 校对者:
-# setState() Gate #
+# setState() 门 #
-## Navigating React setState() Behavior Confusion ##
+## React setState() 解惑 ##
+
+> 译注:本文纯属撕逼,实际营养价值不高,希望从技术层面深入了解 `setState()` 的同学可以参考[[译] React 未来之函数式 setState](https://juejin.im/post/58cfcf6e44d9040068478fc6)。对 `setState()` 不了解的同学可能会感到本文不知所云,特此说明。
-It all started last week. 3 different React learners encountered 3 different obstacles trying to use `setState()` in their projects. I mentor new React users a lot, and consult with teams making transitions from other architectures to React.
+一切都开始于上周。3 位 React 初学者尝试在项目中使用 `setState()` 时遇到了 3 种不同的问题。我指导过很多 React 新手,也为从其他技术架构转向 React 的团队提供咨询。
-One of those learners was working on a production project that is a good fit for Redux, so instead of working out how to fix the timing with `setState()`, I recommended that we just replace `setState()` with Redux, which has the effect of removing the timing of state updates from the component drawing the DOM. Then the module simply has to decide what to render based on the props from the store, and the timing complexity is magically side-stepped.
+其中一位初学者正在开发一个十分适合使用 Redux 的生产项目,我建议他直接用 Redux 替换掉 `setState()` 而不是去正面解决 `setState()` 的同步问题(the timing with `setState()`),因为使用 Redux 能避免组件状态(state)在组件渲染的过程中发生改变。Redux 简单地利用来自 store 属性(props)来决定如何渲染界面,巧妙地避过了复杂的同步问题。
-That inspired this tweet:
+因此也就有了下面这条推特:
-[“React has a setState() problem: Asking newbies to use setState() is a recipe for headaches. Advanced users have learned to avoid it. ;)](https://twitter.com/_ericelliott)
+[“React 有个 setState() 问题:叫新手使用 setState() 毫无好处(a recipe for headaches)。高手们已经学会了如何避免使用它"](https://twitter.com/_ericelliott)
-After that, some advanced users chimed in to correct me:
+之后,有些高手就来纠正我了:
-[“React team member checking in. Please learn to use setState before other approaches.”](https://twitter.com/dan_abramov/status/842490428440150017?ref_src=twsrc%5Etfw)
+[“我是 React 团队的一员。在尝试其他方法之前,请学会使用 setState。”](https://twitter.com/dan_abramov/status/842490428440150017?ref_src=twsrc%5Etfw)
-[“Those ‘adanced’ users will get left behind when we turn on async scheduling by default in React 17”](https://twitter.com/acdlite/status/842499250822950912?ref_src=twsrc%5Etfw)
+[“那些所谓‘高手’们怕是要落伍了,因为 React 17 将会默认采用异步调度。”](https://twitter.com/acdlite/status/842499250822950912?ref_src=twsrc%5Etfw)
-On that second point:
+对于第二点:
-[“Fiber has a strategy for pausing, splitting, rebasing, aborting updates that doesn’t work if you deviate from component state”](https://twitter.com/acdlite/status/842506455232143360?ref_src=twsrc%5Etfw)
+[“Fiber 有一种用于暂停、切分、重建和取消更新的策略,但如果你背离了组件状态,那此策略便无法正常工作了。”](https://twitter.com/acdlite/status/842506455232143360?ref_src=twsrc%5Etfw)
-Both fair points. Other people made memes:
+貌似都没错,可是码农们就要骂娘了:
-It’s great to make fun of a frustrating situation, but let’s not pretend there’s no problem.
+面对困境“呵呵”两下并无妨,不过千万别呵呵过后就对问题视而不见了。
-In my very next meeting with a different mentee, *he was also confused* about how `setState()` works, and had just given up and stuffed his state in a closure, which of course wouldn’t automatically trigger a render if the closure state changed.
+在和另一个初学者交流的时候,我发现他也对 `setState()` 的工作机制感到困惑。后来他放弃了,他把状态塞在一个闭包里;显而易见,闭包中状态的改变是不会触发 render 函数自动执行的。
-Given the *constant influx* of confused React newbies, I stand by the first part of my tweet, but if I had it to do again, I’d change the second part *a little,* because some advanced users (notably, lots of Facebook and Netflix) use `setState()` extensively:
+考虑到深感困惑的初学者之多,我还是坚持我上述推文中前半句的观点;但如果可以重来的话,我会对后半句稍作修改,因为确有很多高手(主要是 Facebook 和 Netfix 的工程师)广泛地使用 `setState()`:
-> “React has a setState() problem: Asking newbies to use setState() is a recipe for headaches. Advanced users have secret cures.”
+> “React 有个 setState() 问题:叫新手使用 setState() 毫无好处,但高手们自有神技。“
-Of course, Twitter would probably still lose its collective mind. After all, React is *perfect*, and we must all agree that `setState()` is beautiful just as it is, or face ridicule and scorn.
+当然,推特还是有可能会丧失其集体智慧(lose its collective mind)(译注:个人认为这句应该是指当网络上大多数人持某一观点时,那即使该观点是错的,那你也不能指出其错误,否则就会招致集体攻讦;或者说,真理有时候只掌握在少数人手里)。 毕竟,React 是**完美的**, 我们都必须承认 `setState` 的美妙优雅,否则只会遭到冷嘲热讽。
-If `setState()` confuses you, it’s *your fault.* You must be crazy or stupid. (Have I mentioned that [the JavaScript community has a bullying problem?](https://medium.com/javascript-scene/the-js-community-has-a-bullying-problem-96c10f11c85d#.wagjqz54o) )
+如果 `setState()` 令你感到困惑,那都是**你的错** —— 你要么是疯子,要么是傻瓜。(我好像忘了说 [Javascript 的社区霸凌问题了](https://medium.com/javascript-scene/the-js-community-has-a-bullying-problem-96c10f11c85d#.wagjqz54o) )
-Let’s check our egos for a moment and stop patting ourselves on the back for our `setState()` mastery while we mock everybody who hasn’t learned the same lessons.
+好了,当你嘲笑所有初学者的时候,先反省反省自己吧,别以为掌握了 `setState()` 就可以得意忘形了。
-That behavior is absurd, elitist, and very uninviting to newcomers. If people frequently get confused about an API, it could be an opportunity to improve that API, or at least improve the documentation.
+那种行为是荒谬可笑的,是精英主义论的,会让新手们感到十分讨厌。如果人们经常对某个 API 感到困惑的话,那就该改进 API 本身的设计了,或者至少应该改进下文档。
-Making the community and our tools more friendly and inviting is good for everybody.
+让我们的社区和工具变得更加友好和充满魅力对所有人来说都是件好事。
-### What’s Wrong with setState()? ###
+### setState() 究竟有何问题? ###
-This question has two answers:
+这个问题可以有两个答案:
-1. Not much. It (mostly) behaves like it needs to to solve the problem it’s designed to solve.
-2. Learning curve. Users new to React and `setState()` frequently encounter obstacles while trying to do things that *just work* with vanilla JS and direct DOM manipulation.
+1. 没啥问题。(大部分情况下)它表现地和设计期望一样,足以解决目标问题。
+2. 学习曲线问题。对新手而言,一些用纯 JS 和直接的 DOM 操作可以实现的效果,用 React 和 `setState` 实现起来就会苦难重重。
-React is designed to make it easier to build apps, but:
+React 的设计目标就是要简化网站和应用的开发,但是:
-- You can’t just grab bits of DOM and update them any way you like.
-- You can’t just set the state to anything at any time, depending on any data source you like.
-- You can’t just observe the rendered DOM or element visibility on screen at any part of the component lifecycle, which limits when and how you can use `setState()` for render dependent state (the state you’re working on may not have been rendered to the screen, yet).
+- 你不能直接操作 DOM。
+- 你不能随心所欲地(于任何时间任何地点用任何数据)更新状态。
+- 在组件的声明周期中,你并不总是能在屏幕上直接观察到渲染后的 DOM 元素,这限制了 `setState()` 的使用时机和方式(因为你有些状态可能还没有渲染到屏幕上)。
-In all of these cases, the confusion is caused by the *(intentional, good)* limitations of the React component lifecycle.
+在这几种情况下,困惑都来源于 React 组件生命周期的限制性(这些限制是刻意设计的,是好的)。
-#### Dependent State ####
+#### 从属状态(Dependent State) ####
-When we’re updating state, sometimes the value of the update depends on things that React tries to help us with:
+在更新状态的时候,更新结果可能依赖于:
-- The current state
-- Previous attempts to update state in the same cycle
-- The rendered DOM (e.g., component coordinates, visibility, calculated CSS values, etc…)
+- 当前状态
+- 同一循环中之前的更新操作
+- 当前已渲染的 DOM (例如:组件的坐标位置、可见性、CSS 计算值等等)
-If you try to simply update the state in a straightforward way when you have these kinds of dependent state, React’s behavior might surprise you in an obnoxiously hard-to-debug way. Frequently, whatever you just tried to do simply doesn’t work. You’ll end up with incorrect state, or you’ll see an error in the console.
+当存在这几种从属状态的时候,如果你还想简单直接地更新状态,那 React 的表现行为会让你大吃一惊,并且是以一种令人憎恶又难以调试的方式。大多数情况下,你的代码根本无法工作:要么状态不对,要么控制台一堆错误。
-My gripe with `setState()` is that its restrictive behavior is not made obvious to newcomers in the API documentation, and common patterns for dealing with its restrictive behavior are not well explained. This forces users to resort to trial and error, Google, and help from other community members, when there could be better guide-posts built into `setState()` and it’s API documentation.
+我之所以吐槽 `setState()`,是因为它这种具有限制性的表现行为在 API 文档中并没有详细说明,关于处理这种限制性行为的各种通用模式也未能阐述清楚。这迫使初学者只能不断试错、Google 或者从其他社区成员那里寻求帮助,但实际上在文档中本该就有更好的新手指南。
-The current API documentation for `setState()` leads with this:
+当前关于 `setState()` 的文档开头如下:
```
setState(nextState, callback)
```
-> Performs a shallow merge of nextState into current state. This is the primary method you use to trigger UI updates from event handlers and server request callbacks.
+> 将 nextState 浅合并到当前状态。这是在事件处理函数和服务器请求回调函数中触发 UI 更新的主要方法。
-It does make very brief mention at the end that it has async behavior:
+在末尾确实也提到了其异步行为:
-> There is no guarantee of synchronous operation of calls to `setState` and calls may be batched for performance gains.
+> 调用 `setState` 并不保证同步操作,考虑到性能问题,可能会对多次调用作批处理化。
-The consequence of both of those things together is the root of many userland bugs:
+这就是很多用户层 bug 的根本原因:
```
-// assuming state.count === 0
+// 假设 state.count === 0
this.setState({count: state.count + 1});
this.setState({count: state.count + 1});
this.setState({count: state.count + 1});
-// state.count === 1, not 3
+// state.count === 1, 而不是 3
```
-It’s essentially equivalent to:
+本质上等同于:
```
Object.assign(state,
@@ -110,38 +112,38 @@ Object.assign(state,
); // {count: 1}
```
-This is not mentioned explicitly in the API docs (it is covered elsewhere in a special guide).
+这在文档中并未显式说明(在另外一份特殊指南中提到了)。
-The API docs also makes mention of a function alternative to the `setState()` signature:
+文档还提到了另外一种函数式的 `setState()` 语法:
-> It’s also possible to pass a function with the signature `function(state, props) => newState`. This enqueues an atomic update that consults the previous value of state and props before setting any values.
+> 也可以传递一个签名为 `function(state, props) => newState` 的函数作为参数。这会将一个原子性的更新操作加入更新队列,在设置任何值之前,此操作会查询前一刻的属性和状态。
> `...`
-> `setState()` does not immediately mutate `this.state` but creates a pending state transition. Accessing `this.state` after calling this method can potentially return the existing value.
+> `setState()` 并不会立即改变 `this.state` ,而是会创建一个待执行的变动。调用此方法后访问 `this.state` 有可能会得到当前已存在的状态。
-The API docs are dropping some breadcrumbs, but they don’t really explain the behavior that newbies frequently encounter in a way that clearly guides the reader on the right path, and though React is famous for generating useful errors in dev mode, no such warnings get logged when `setState()` timing bugs crop up.
+API 文档省略了一些细节,但未能以一种清晰明了的方式阐明初学者经常遇到的怪异表现。尽管 React 在提供开发模式下的有效错误信息方面做得很好,但当 `setState()` 的同步问题出现 bug 的时候控制台却没有任何警告。
-[![](https://ww2.sinaimg.cn/large/006tNc79gy1fdwma23qp0j30jk077mxt.jpg)](https://twitter.com/JikkuJose/status/842915627899670528?ref_src=twsrc%5Etfw)
+[Jikku Jose ](https://twitter.com/JikkuJose/status/842915627899670528?ref_src=twsrc%5Etfw)
-[![](https://ww3.sinaimg.cn/large/006tNc79gy1fdwmac1hwlj30ji06rq3h.jpg)](https://twitter.com/PierB/status/842590294776451072?ref_src=twsrc%5Etfw)
+[Pier Bover ](https://twitter.com/PierB/status/842590294776451072?ref_src=twsrc%5Etfw)
-Lifecycle timing issues account for a lot of the questions asked about `setState()` on StackOverflow. Of course, React is very popular, so those questions have been [asked](http://stackoverflow.com/questions/25996891/react-js-understanding-setstate)[many](http://stackoverflow.com/questions/35248748/calling-setstate-in-a-loop-only-updates-state-1-time) [times](http://stackoverflow.com/questions/30338577/reactjs-concurrent-setstate-race-condition/30341560#30341560), with answers of various quality and correctness.
+StackOverflow 上有关 `setState()` 的问题大都要归结于组件的生命周期问题。毫无疑问,React 非常流行,因此那些问题都被[问](http://stackoverflow.com/questions/25996891/react-js-understanding-setstate)[烂](http://stackoverflow.com/questions/35248748/calling-setstate-in-a-loop-only-updates-state-1-time)[了](http://stackoverflow.com/questions/30338577/reactjs-concurrent-setstate-race-condition/30341560#30341560),也有着各种良莠不齐的回答。
-So how can newbies learn the right way to manage `setState()` timing issues?
+那么,初学者究竟该如何掌握 `setState()` 呢?
-There is more in-depth information in a separate guide in the React docs called [“State and Lifecycle”](https://facebook.github.io/react/docs/state-and-lifecycle.html) :
+在 React 的文档中还有一份名为 [“状态和生命周期”](https://facebook.github.io/react/docs/state-and-lifecycle.html)的指南,该指南提供了更多深入内容:
-> “…To fix it, use a second form of `setState()` that accepts a function rather than an object. That function will receive the previous state as the first argument, and the props at the time the update is applied as the second argument:”
+> “…要解决此问题,请使用 `setState()` 的第二种形式 —— 以一个函数而不是对象作为参数,此函数的第一个参数是前一刻的状态,第二个参数是状态更新时的属性:”
```
-// Correct
+// 正确用法
this.setState((prevState, props) => ({
count: prevState.count + props.increment
}));
```
-This function-parameter form (sometimes called “functional `setState()`”) works more like this:
+这个函数参数形式(有时被称为“函数式 `setState()`”)工作机制更像:
```
[
@@ -153,9 +155,9 @@ This function-parameter form (sometimes called “functional `setState()`”) wo
}), {count: 0}); // {count: 3}
```
-Not sure how reduce works? See [“Reduce”](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d#.8d8kw0l40) from [“Composing Software”](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.7k9w6v9ok) .
+不明白 reduce 的工作机制? 参见 [“Composing Software”](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.7k9w6v9ok) 的 [“Reduce”](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d#.8d8kw0l40) 教程。
-The key is the **updater function**:
+关键点在于**更新函数**:
```
(prevState, props) => ({
@@ -163,56 +165,60 @@ The key is the **updater function**:
})
```
-This is basically a reducer, where `prevState` acts like an accumulator, and `props` acts as the source for the new update data. Like reducers from Redux, you can reduce with this function using any standard reduce utility (including `Array.prototype.reduce()`). Also like Redux, the reducer should be a [pure function](https://medium.com/javascript-scene/master-the-javascript-interview-what-is-a-pure-function-d1c076bec976) .
+这基本上是个 reducer,其中 `prevState` 类似于一个累加器,而 `props` 则像是新的数据源。类似于 Redux 中的 reducers,你可以使用任何标准的 reduce 工具库对该函数进行 reduce(包括 `Array.prototype.reduce()`)。同样类似于 Redux,reducer 应该是 [纯函数](https://medium.com/javascript-scene/master-the-javascript-interview-what-is-a-pure-function-d1c076bec976) 。
-> Note: Trying to directly mutate `prevState` is a common source of confusion among new users.
+> 注意:企图直接修改 `prevState` 通常会给初学者带来很多困惑。
-These properties and expectations of the updater function are not mentioned in the API documentation, so the rare, lucky newbie who chances across the fact that the functional `setState()` form does something useful that isn’t supported by the object literal form is probably still going to be confused.
+API 文档中并未提及此更新函数的这些特性和表现,罕见的是,一些“幸运”的初学者碰巧了解到函数式 `setState()` 可以实现一些对象字面量形式无法实现的功能,可是结果还是可能会困惑不解。
-### Just A Newbie Problem? ###
+### 仅仅是新手才有的问题吗? ###
-I still bump into rough edges now and then when I’m dealing with forms or DOM element coordinates because, when you use `setState()`, you have to deal with the component lifecycle directly. When you use a container component or store and pass your state through props, React handles the timing issues for you.
+直到现在,在处理表单或是 DOM 元素坐标位置的时候,我还是会时不时得掉到坑里去。当你使用 `setState()` 的时候,你必须直接处理组件生命周期的相关问题;但当你使用容器组件或是通过属性来存储和传递状态的话,React 则会替你处理同步问题。
-Shared mutable state and state locks can be painful to navigate [*regardless of your experience level*](https://medium.com/@mweststrate/3-reasons-why-i-stopped-using-react-setstate-ab73fc67a42e#.saj7jn6wh) *.* Experienced users are just better at identifying the problem quickly and jumping to a handy workaround.
+ [**无论你有经验与否**](https://medium.com/@mweststrate/3-reasons-why-i-stopped-using-react-setstate-ab73fc67a42e#.saj7jn6wh) ,处理共享的可变状态和状态锁(state locks)都是很棘手的。经验丰富之人只不过是能更加快速地定位问题,然后找出一个巧妙的解决方案罢了。
-Since newbies haven’t seen the problem before, and aren’t aware of workarounds, it just happens to hit them hardest.
+因为初学者从未遇到过这种问题,也不知道该如何解决,所以是掉坑里摔得最惨的。
[](https://twitter.com/_ericelliott/status/842546271944564737?ref_src=twsrc%5Etfw)
[](https://twitter.com/dan_abramov/status/842548605525331969?ref_src=twsrc%5Etfw)
-You can fight with React over when things happen, or you can let React do its thing and go with the flow. That’s what I mean when I say that Redux is *sometimes* easier than `setState()`, *even for beginners.*
+当问题发生时,你当然可以选择和 React 斗个你死我活;不过,你也可以选择让 React 顺其自然的工作。这就是我说**即使是对初学者而言**,Redux **有时** 都比 `setState` 更简单的原因。
-In concurrent systems, updates to state are typically handled in 1 of 2 ways:
+在并发系统中,处理状态的更新通常采用以下两种方法中的一种:
-- Locking or restricting access to state updates while other things are using the state, (e.g., `setState()`)or…
-- Employing immutability to eliminate shared mutable state, which allows unrestricted access to state, and new state creation at any point in time. (e.g., Redux)
+- 当其他程序(或代码)正在访问状态时,禁止状态的更新(例如 `setState()`)(译注:即常见的锁机制)
+- 引入不可变性来消除共享的可变状态,从而实现对状态的无限制访问,并且可以在任何时间创建新状态(例如 Redux)
-In my opinion (after teaching both techniques to lots of students), the first way is much more error prone and confusing than the second way. When state updates are simply blocked (or in the case of `setState()`, batched or deferred), the correct solution to the problem is not immediately clear.
+在我看来(在向很多学生教授过这两种方法之后),相比于第二种方法,第一种方法更加容易导致错误,也更加容易令人困惑。当状态更新被简单地阻塞时(在 `setState` 的例子中,也可以叫批处理化或延迟执行),解决问题的正确方法并不十分清晰明了。
-My default reaction when I encounter a `setState()` timing issue is simple: Move my state management up the tree, either to Redux (or MobX), or to a container component. I usually use and recommend Redux for [lots of reasons](https://medium.com/javascript-scene/10-tips-for-better-redux-architecture-69250425af44) , but obviously, that’s *not the right advice for everybody.*
+当遇到 `setState()` 的同步问题时,我的直觉反应其实是很简单的:将状态的管理上移到 Redux(或 MobX) 或容器组件中。基于[很多原因](https://medium.com/javascript-scene/10-tips-for-better-redux-architecture-69250425af44) ,我自己使用同时也推荐他人使用 Redux,但很显然,这**并不是一条放之四海而皆准的建议**。
-Redux has its own *gigantic learning curve,* but sidesteps shared mutable state and state update timing complexity, so I find that once I teach students to avoid mutations, it’s pretty smooth sailing, *without too many gotchas or roadblocks.*
+Redux 自有其**陡峭**的学习曲线,但它规避了共享的可变状态以及状态更新同步等复杂问题。因此我发现,一旦我教会了学生如何避免可变性,接下来基本就**康庄大道直奔小康**了。
-A newbie trying to learn Redux without any functional programming experience is probably going to have more trouble with Redux than they would with `setState()` — but at least there is a great set of [free](https://egghead.io/courses/getting-started-with-redux) [courses](https://egghead.io/courses/building-react-applications-with-idiomatic-redux) on the topic, by the author of Redux.
+对于没有任何函数式编程经验的新手而言,学习 Redux 遇到的问题可能会比学习 `setState()` 遇到的更多 —— 但是,Redux 至少有很多其作者亲自讲授的[免费](https://egghead.io/courses/getting-started-with-redux) [教程](https://egghead.io/courses/building-react-applications-with-idiomatic-redux)
-React should take a page out of the Redux book: A great video tutorial on common React patterns and `setState()` gotchas would make an amazing addition to the React home-page.
+React 应当向 Redux 学习:有关 React 编程模式和 `setState()` 踩坑的视频教程定能让 React 主页锦上添花。
-#### Decide the State Before You Render ####
+#### 在渲染之前决定状态 ####
-Moving state management to a container component (or Redux) forces you to think differently about your component state by making it clear that *before you can render* a component, *its state must already be decided* (because you have to pass it in as props).
+将状态管理移到容器组件(或 Redux)中能促使你从另一个角度思考组件状态问题,因为这种情况下,在组件**渲染之前**,其**状态必须是既定的**(因为你必须将其作为属性传下去)。
-That’s worth repeating:
+重要的事情说三篇:
-> Before you render, decide the state!
+> 渲染之前,决定状态!
+>
+> 渲染之前,决定状态!
+>
+> 渲染之前,决定状态!
-An obvious corollary is that trying to use `setState()` inside your `render()` method is an anti-pattern.
+说完三篇之后就可以得到一个显然的推论:在 `render()` 函数中调用 `setState()` 是反模式的。
-Calculating dependent state inside your render method is fine (e.g., if you have `firstName` and `lastName` and you want to calculate `fullName`, it’s OK to do that in `render()`), but I prefer to calculate dependent state in a container and pass it in as props to presentation components.
+在 `render` 函数中计算从属状态是 OK 的(比如说,状态中有 `firstName` 和 `lastName`,据此你计算出 `fullName`,在 `render` 函数中这样做完全是 OK 的),但我还是倾向于在容器组件中计算出从属状态,然后通过属性将其传递给表现性组件(presentation components)。
-### How Can setState() be Fixed? ###
+### setState() 该怎么治? ###
-My preference would be to deprecate the object literal form of `setState()`. I know it’s (superficially) easier to understand and more convenient, but it’s also how a lot of new users get stuck, and I think it’s self-evident that somebody doing this:
+我倾向于废弃掉对象字面量形式的 `setState()`,我知道这(表面上看)更加易于理解也更加方便(译者:“这”指对象字面量形式的 `setState()`),但它也是坑之所在啊。用脚指头都能猜到,肯定有人这样写:
```
state.count; // 0
@@ -221,73 +227,74 @@ this.setState({count: state.count + 1});
this.setState({count: state.count + 1});
```
-Expects to see `{count: 3}` afterwards. I have not yet seen a case where a batched object merge on the same property was expected behavior. I would argue that if such cases exist, they’re too tightly coupled to implementation details of React to be advisable valid use-cases.
+然后天真就地以为 `{count: 3}`。批量化处理后对象的同名属性被合并掉的情况几乎不可能是用户所期望的行为,反正我是没见过这种例子。要是真存在这种情况,那我必须说这跟 React 的实现细节耦合地太紧密了,根本不能作为有效参考用例。
-I would also like to see the API section of the `setState()` docs link to the in-depth [“State and Lifecycle”](https://facebook.github.io/react/docs/state-and-lifecycle.html) guide, to provide much more detail on this topic to users who are trying to learn the ins and outs of `setState()`. Because it does not operate synchronously or return anything meaningful, simply describing its function signature without more thoroughly discussing its effects and behavior is not successfully onboarding new users.
+我也希望 API 文档中有关 `setState()` 的章节能够加上[“状态和声明周期”](https://facebook.github.io/react/docs/state-and-lifecycle.html)这一深度指南的链接,这能给那些想要全面学习 `setState()` 的用户更多的细节内容。`setState()` 并非同步操作,也无任何有意义的返回结果,仅仅是简单地描述其函数签名而没有深入地探讨其各种影响和表现,这对初学者是极不友好的。
-They have to resort to hours of troubleshooting, Google searches, StackOverflow, and GitHub issues.
+初学者必须花上大量时间去找出问题:Google 上搜、StackOverflow 上搜、GitHub issues 里搜。
-### Why is setState() so Strict? ###
+### setState() 为何如此严苛? ###
-The quirky behavior of setState() is not a bug. It’s a feature. In fact, you might say *it’s the whole reason that React exists in the first place.*
+setState() 的怪异表现并非 bug,而是特性。实际上,甚至可以说**这是 React 之所以存在的根本原因**。
-One of the driving motivations for React was to ensure deterministic renders: Given some application state, render some specific output. Ideally, given the same state, always render the same output.
+React 的一大创作动机就是保证确定性渲染:给定应用状态,渲染出特定结果。理想情况下,给定状态相同,渲染结果也应相同。
-In order to make that happen, React has to *manage mutation* by limiting when it can happen. We don’t just grab hold of the DOM and mutate it in place. Instead, React renders the DOM, and when some state changes, React decides how to render again. *We don’t render the DOM. React does.*
+为了达到此目的,当发生变化时,React 通过采取一些限制性手段来**管理**变化。我们不能随意取得某些 DOM 节点然后就地修改之。相反,React 负责 DOM 渲染;当状态发生改变时,也由React 决定如何重绘。**我们不渲染 DOM,而是由 React 来负责**。
-But to do this in a way that doesn’t retrigger renders during the update cycle, React introduces a rule:
+为了避免在状态更新的过程中触发重绘,React 引入了一条规则:
-The state that React uses to render can’t mutate during the DOM render process. *We don’t decide when component state gets updated. React does.*
+React 用于渲染的状态不会在 DOM 渲染的过程中发生改变。**我们不能决定组件状态何时得到更新,而是由 React 来决定**。
-Hence the confusion. When you call `setState()`, you think you’re setting the state. You’re not.
+困惑就此而来。当你调用 `setState()` 时,你以为你设置了状态,其实并没有。
-“You keep using that word. I don’t think it means what you think it means.”
+“接着装逼,你以为你所以为的就是你以为所以为的吗?”
-### When Should We Use setState()? ###
+### setState() 该在何时使用? ###
-I use `setState()` almost exclusively for self-contained units of functionality that don’t need to persist state. In other words, things like reusable form validation components, custom date or time block selection widgets, data visualization widgets that let you customize their view state, etc…
+我一般只在不需要持久化状态的自包含功能单元中使用 `setState()`,例如可复用的表单校验组件、自定义的日期或时间选择部件(widget)、可自定义界面的数据可视化部件等。
-I call components like that “widgets”, and they’re really made up of two or more components: a container for internal state management, and one or more child components which handle the actual DOM and presentation aspects.
+我称这种组件为“小部件(widget)”,它们一般由两个或两个以上组件构成:一个负责内部状态管理的容器组件,一个或一个以上负责界面显示的子组件
-Here are some simple litmus tests:
+几条立见分晓的检验方法(litmus tests):
-- Do other components rely on the state?
-- Do you need to persist the state? (Save it to local storage or send it to a server?)
+- 其他组件是否依赖于状态?
+- 是否需要持久化状态?(存储于 local storage 或服务器)
-If the answers to both of those questions is “no”, maybe it’s OK to use `setState()`. Otherwise, you might want to consider something else.
+如果这两个问题的答案都是“否”的话,那使用 `setState()` 基本是没问题的;否则,就要另作考虑了。
-At Facebook, as far as I understand, they use `setState()` managed by a [Relay container](https://facebook.github.io/relay/) to encapsulate different parts of the Facebook UI like mini applications inside the larger Facebook application. For them, it’s a great way to colocate their many complex data dependencies with the components that actually use them.
+据我所知,Facebook 使用受管于 [Relay container](https://facebook.github.io/relay/) 的 `setState()` 来包装 Facebook UI 的各个不同部分,例如大型 Facebook 应用内部的迷你型应用。于 Facebook 而言,将复杂的数据依赖和需要实际使用这些数据的组件放在一起是很好的。
-I recommend similar strategies for very large (enterprise scale) applications. If your app has a whole lot of code (hundreds of thousands of LOC+), this may be a good strategy for you, too — but there’s no reason the approach can’t also scale down as well.
+对于大型(企业级)应用我推荐这种策略。如果你的应用代码量非常大(十万行以上),那此策略可能是很好的 —— 但这并意味着这种方式就不能应用于小型应用中。
-There’s also no reason you can’t use a similar approach by instead breaking those different pieces into actually separate mini-applications which get composed into the larger application. I have done that with Redux for enterprise software. For example, I often separate analytics dashboards, messaging, admin, team/user role management, and billing management into totally separate apps with their own Redux stores. Such apps can share a domain along with common login/session management using API tokens and OAuth so that the apps feel like one connected app.
+在将一个大型应用拆分成多个独立的迷你型应用的场景下,这种方式也是适用的。我自己就结合 Redux为企业级应用这样做过。例如,我经常将分析面板、消息管理、系统管理、团队/成员角色管理以及账单管理等模块拆分成多个独立的应用,每个应用都有其自己的 Redux store。通过 API tokens 和 OAuth,这些应用共享同一个域下的登录/session 管理,感觉就像是一个互有连接的应用。
-For most apps, I recommend **defaulting to Redux**. It’s worth noting that Dan Abramov (the creator of Redux) disagrees with me on that point. He rightly favors keeping apps as simple as they can be until they can’t be that simple anymore. The conventional community wisdom says “don’t use Redux until you feel the pain.”
+对于大多数应用,我建议**默认使用 Redux**。需要指出的是,Dan Abramov(Redux 的作者)在这一点上和我持相反的观点。他喜欢应用尽可能地保持简单,这当然没错。传统社区有句格言如是说:“除非真得感到痛苦,否则别使用 Redux”。
-My response is this:
+而我的反应是这样的:
-“Those who are unaware they are walking in darkness will never seek the light”.
-As I mentioned already, *in some cases,* Redux is a simpler path than `setState()`. Redux simplifies state management by eliminating entire classes of bugs related to shared mutable state and timing dependencies.
+“不知道自己正走在黑暗中的人是永远不会去搜寻光明的“。
+
+正如我说过的,**在某些情况下**,Redux 比 `setState()` 更简单。通过消除一切和共享的可变状态以及同步依赖有关的 bug,Redux 简化了状态管理问题。
-Do learn `setState()`, but even if you decide you don’t want to use Redux in your app, **you should still learn Redux.** It will teach you new ways to think about application state, and probably help you simplify your application state no matter what other solution you choose for your app.
+`setState()` 肯定要学,但即使你不想使用 Redux,你也应该学学 Redux。无论你采用何种解决方案,它都能让你从新的角度思考去应用的状态管理问题,也可能能帮你简化应用状态。
-For apps with a whole lot of derived state, [MobX](https://github.com/mobxjs/mobx) is probably a better solution than `setState()` or Redux, because it’s very good at efficiently managing and organizing calculated state.
+对于有大量派生(derived )状态的应用而言, [MobX](https://github.com/mobxjs/mobx) 可能会比 `setState()` 和 Redux 都要好,因为它非常擅于高效地管理和组织需要通过计算得到的(calculated )状态。
-Because of its granular observable subscription model, it’s also very good at rendering LOTS of dynamic DOM elements efficiently (tens of thousands). So if you’re building a graphical game, or a console that monitors all the instances of your enterprise microservices, it might be a great choice for visually displaying all that complex information in realtime.
+得利于其细粒度的、可观察的订阅模型,MobX也很擅于高效渲染大量(数以万计)动态 DOM 节点。因此,如果你正在开发的是一款图形游戏,或者是一个监控所有企业级微服务实例的控制台,那 MobX 可能是个很好的选择,它非常有利于实时地可视化展示这种复杂的信息。
-### Next Steps ###
+### 接下来 ###
-Want to learn a whole lot more about building software with React and Redux?
+想要全面学习如何用 React 和 Redux 开发软件?
-[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/) . If you’re not a member, you’re missing out!
+[跟着 Eric Elliott 学 Javacript](http://ericelliottjs.com/product/lifetime-access-pass/),机不可失时不再来!
[
](https://ericelliottjs.com/product/lifetime-access-pass/)
-***Eric Elliott*** *is the author of* [*“Programming JavaScript Applications”*](http://pjabook.com) *(O’Reilly), and* [*“Learn JavaScript with Eric Elliott”*](http://ericelliottjs.com/product/lifetime-access-pass/) . He has contributed to software experiences for **Adobe Systems** , **Zumba Fitness** , **The Wall Street Journal** , **ESPN** , **BBC** , and top recording artists including **Usher** , **Frank Ocean** , **Metallica**, and many more.
+**Eric Elliott** 是 [**“编写 JavaScript 应用”**](http://pjabook.com) (O’Reilly) 以及 [**“跟着 Eric Elliott 学 Javascript”**](http://ericelliottjs.com/product/lifetime-access-pass/) 两书的**作者**。他为许多公司和组织作过贡献,例如 **Adobe Systems**、**Zumba Fitness**、**The Wall Street Journal**、**ESPN**和**BBC**等 , 也是很多机构的顶级艺术家,包括但不限于 **Usher** , **Frank Ocean** , **Metallica**。
-*He spends most of his time in the San Francisco Bay Area with the most beautiful woman in the world.*
+大多数时间,他都在 San Francisco Bay Area,同这世上最美的女子在一起(译注:这是怕老婆呢还是怕老婆呢还是怕老婆呢?)。
\ No newline at end of file
From 312d6c3e62c36f666027f9008fe1081f8a5e4cea Mon Sep 17 00:00:00 2001
From: xiaoyusilen
Date: Mon, 27 Mar 2017 17:37:50 +0800
Subject: [PATCH 017/945] complete translation
---
TODO/go-function-calls-redux.md | 56 ++++++++++++++++-----------------
1 file changed, 28 insertions(+), 28 deletions(-)
diff --git a/TODO/go-function-calls-redux.md b/TODO/go-function-calls-redux.md
index 3ee8548a7b9..2e8b6efa515 100644
--- a/TODO/go-function-calls-redux.md
+++ b/TODO/go-function-calls-redux.md
@@ -1,18 +1,18 @@
> * 原文地址:[Go Function Calls Redux](https://hackernoon.com/go-function-calls-redux-609fdd1c90fd#.jsh5r78wp)
> * 原文作者:[Phil Pearl](https://hackernoon.com/@philpearl?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[xiaoyusilen](http://xiaoyu.world)
> * 校对者:
-# Go Function Calls Redux #
+# Go 函数调用 Redux #
-Some time ago in a [previous post](https://syslog.ravelin.com/anatomy-of-a-function-call-in-go-f6fc81b80ecc#.gpqsgzmjc) I promised to take a further look at how function calls and call stacks in Go work. I’ve think found a neat way to make good on that promise, so here goes.
+前段时间在一篇[文章](https://syslog.ravelin.com/anatomy-of-a-function-call-in-go-f6fc81b80ecc#.gpqsgzmjc)中我答应写一篇进一步分析 Go 中如何进行函数调用和调用堆栈在 Go 中如何工作的文章。现在我找到了一种简洁的方式来向大家展示上述内容,所以有了现在这篇文章。
-So what is a call stack? Well, it’s an area of memory used to hold local variables and call parameters, and to track where functions should return to. Each goroutine has it’s own stack. You could almost say a goroutine is its stack.
+什么是调用堆栈?它是一个用于保存局部变量和调用参数的内存区域,并且跟踪每个函数应该返回到哪里去。每个 goroutine 都有它自己的堆栈。你甚至可以说每个 goroutine 就是它自己的堆栈。
-Here’s the code I’m going to use to show the stack in action. It’s just a sequence of simple function calls. main() calls [f1(0xdeadbeef)](https://en.wikipedia.org/wiki/Hexspeak) , which then calls `f2(0xabad1dea)`, which calls `f3(0xbaddcafe)`. `f3()` then adds one to it’s parameter, and stores it in a local variable called `local`. It then takes the address of `local` and prints out memory starting at that address. Because `local` is on the stack, this prints the stack.
+下面是我用于演示堆栈的代码。就是一系列简单的函数调用,main() 函数调用 [f1(0xdeadbeef)](https://en.wikipedia.org/wiki/Hexspeak),然后调用 `f2(0xabad1dea)`,再调用 `f3(0xbaddcafe)`。然后 f3() 将其中一个作为它的参数,并且将它存储在名为 `local` 的本地变量中。然后获取 `local` 的内存地址并且从那里开始输出。因为 `loacl` 在栈内,所以输出的就是栈。
-```
+```go
package main
import (
@@ -59,9 +59,9 @@ func showFunc(at uintptr) {
}
```
-Here’s the output of the program. It is a dump of memory starting at the address of `local`, shown as a list of 8-byte integers in hex. The address of each integer is on the left, and the int at the address is on the right.
+下面是上述代码的输出结果。它是从 `local` 的地址开始的内存转储,是以十六进制形式展示的8字节列表。左边是每个整数的存储地址,右边是地址内存储的整数。
-We know `local` should equal 0xBADDCAFE + 1, or 0xBADDCAFF, and this is indeed what we see at the start of the dump.
+我们知道 `local` 应该等于 0xBADDCAFE + 1,或者 0xBADDCAFF,这确实是我们转储开始时看到的。
```
C42003FF28: BADDCAFF
@@ -94,13 +94,13 @@ C42003FFC0: C4200001A0
102752A is runtime.main /usr/local/Cellar/go/1.8/libexec/src/runtime/proc.go 194
```
-- The next number is 0xC42003FF48, which is the address of the 5th line of the dump.
-- After that we have 0x1088BEB. It turns out this is an address of executable code, and if we feed it into `runtime.FuncForPC` we see it is the address of line 19 of main.go, which is the last line of `f2()`. This is the address we return to when `f3()` returns.
-- Next we have 0xBADDCAFE, the parameter to our call to `f3()`
+- 下一个数字是 0xC42003FF48,它是转储的第五行的地址。
+- 然后我们可以得到 0x1088BEB。事实上这是一个可执行代码的地址,如果我们将它作为 `runtime.FuncForPC` 的参数,我们知道它是 main.go 的第19行代码的地址,也是 f2() 的最后一行代码。这是 f3() 返回时我们得到的地址。
+- 接下来我们得到 0xBADDCAFE,这是我们调用 `f3()` 时的参数。
-If we carry on we continue to see this pattern. Below I’ve marked up the memory dump showing how the stack pointers track back through the dump and where the function parameters and return addresses sit.
+如果继续我们将看到类似上面的输出结果。下面我已经标记了内存转储,显示堆栈指针如何跟踪转储,参数和返回地址在哪里。
-```
+```go
C42003FF28: BADDCAFF Local variable in f3()
+-C42003FF30: C42003FF48
| C42003FF38: 1088BEB return to f2() main.go line 19
@@ -113,18 +113,18 @@ If we carry on we continue to see this pattern. Below I’ve marked up the memor
| C42003FF70: DEADBEEF f1() parameter
+-C42003FF78: C42003FFD0
C42003FF80: 102752A return to runtime.main()
-
```
-From this we can see many things.
+通过这些我们可以看出:
-- First, the stack starts at a high address, and the stack address reduces as function calls are made.
-- When a function call is made, the caller pushes the parameters onto the stack, then the return address (the address of the next instruction in the calling function), then a pointer to a higher point in the stack.
-- This pointer is used to find the previous function call on the stack when unwinding the stack when the call returns.
-- Local variables are stored after the stack pointer.
+- 首先,堆栈从高地址开始,堆栈地址随着函数调用变小。
+- 当进行函数调用时,调用者将参数放入栈内,然后返回地址(调用函数中的下一条指令的地址),接着指向堆栈中较高的指针。
+- 当调用返回时,这个指针用于在堆栈中查找先前调用的函数。
+- 局部变量存储在堆栈指针之后。
-We can use the same technique to look at some slightly more complicated function calls. I’ve added more parameters, and some return values to `f2()` in this version.
-```
+我们可以使用相同的技巧来分析一些稍微复杂的函数调用。这次,我添加了更多的参数,`f2()` 函数也返回了更多的值。
+
+```go
package main
import (
@@ -153,9 +153,9 @@ func f3(val int) {
}
```
-This time I’ve jumped straight to the marked-up output.
+这次我们直接看被我标记好的输出结果。
-```
+```go
C42003FF10: BADDCAFF local variable in f3()
+-C42003FF18: C42003FF30
| C42003FF20: 1088BFB return to f2()
@@ -173,11 +173,11 @@ This time I’ve jumped straight to the marked-up output.
C42003FF80: 102752A return to runtime.main()
```
-From this we can see that
+从结果中我们可以看出:
-- the calling function makes space for the return values of the called function before the function parameters. (Note the values are uninitialised because the function hasn’t returned yet!)
-- parameters are pushed onto the stack in reverse order.
+- 调用函数在函数参数之前为被调用函数的返回值提供空间。(注意这些值是没有初始化的,因为这个函数还没有返回!)
+- 参数在栈内的顺序与入栈顺序相反。
-Hopefully all that made sense! If you got this far and enjoyed it or learned something, please hit that heart-button. I can’t earn internet points without you.
+希望我都讲清楚了。既然你已经看到这儿了,如果喜欢我的这篇文章或者可以从中学到一点什么的话,那么请给我点个赞。不然我就没办法获得积分。
-*By day, Phil fights crime at* [*ravelin.com*](https://ravelin.com) *. You can join him:* [*https://angel.co/ravelin/jobs*](https://angel.co/ravelin/jobs) .
+**Phil 白天在 [ravelin.com](https://ravelin.com) 的工作主要是防止网上欺诈,你可以加入他 https://angel.co/ravelin/jobs。**
\ No newline at end of file
From e92fbf56ab77d5c1b77d98af7d51b16dd4d3c209 Mon Sep 17 00:00:00 2001
From: luoyaqifei
Date: Mon, 27 Mar 2017 20:58:14 +0800
Subject: [PATCH 018/945] =?UTF-8?q?=E5=9C=A8=20Apache=20&=20Nginx=20?=
=?UTF-8?q?=E6=97=A5=E5=BF=97=E9=87=8C=E6=A3=80=E6=B5=8B=E7=88=AC=E8=99=AB?=
=?UTF-8?q?=E6=9C=BA=E5=99=A8=E4=BA=BA?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/detect-bots-apache-nginx-logs.md | 89 +++++++++++++--------------
1 file changed, 44 insertions(+), 45 deletions(-)
diff --git a/TODO/detect-bots-apache-nginx-logs.md b/TODO/detect-bots-apache-nginx-logs.md
index f0377d039c1..b5497b85278 100644
--- a/TODO/detect-bots-apache-nginx-logs.md
+++ b/TODO/detect-bots-apache-nginx-logs.md
@@ -1,66 +1,66 @@
> * 原文地址:[Detecting Bots in Apache & Nginx Logs](http://tech.marksblogg.com/detect-bots-apache-nginx-logs.html)
> * 原文作者:[Mark Litwintschik](http://tech.marksblogg.com/)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[luoyaqifei](http://www.zengmingxia.com)
> * 校对者:
-# Detecting Bots in Apache & Nginx Logs
+# 在 Apache & Nginx 日志里检测爬虫机器人
-As browser plugins that block JavaScript-based tracking beacons now enjoy a 9-figure user base, web traffic logs can be a good place to get a better feel for how many people are visiting your website. But anyone that has monitored a web traffic log for more than a few minutes is aware there is an army of bots crawling websites. But being able to separate bot and human-generated traffic in web server logs can be challenging.
+现在阻止基于 JavaScript 追踪的浏览器插件享有九位数的用户量,从这一事实可以看出,web 流量日志可以成为一个很好的、能够感知有多少人在访问你的网站的地方。但是任何监测过 web 流量日志一段时间的人都知道,有成群结队的爬虫机器人在爬网站。然而,在 web 服务器日志里分辨出机器人和人为产生的流量是一个难题。
-In this blog I'll walk through the steps I went through to build an IPv4 ownership and browser string-based bot detection script.
+在本文中,我将带你们重现那些我在创建一个基于 IPv4 所属和浏览器字串(browser string)的机器人检测脚本时用过的步骤。
-The code used in this blog can be found in this [gist](https://gist.github.com/marklit/80b875ccab8b215bfa0ecdfaa5000e7b).
+本文中用到的代码在这个 [代码片段](https://gist.github.com/marklit/80b875ccab8b215bfa0ecdfaa5000e7b) 里。
-## IP Address Ownership Databases
+## IP 地址所属数据库
-I'll first install Python and some dependencies. The following was run on a fresh Ubuntu 14.04.3 LTS installation.
+我首先会安装 Python 和一些依赖包。接下来的指令会在一个新的 Ubuntu 14.04.3 LTS 安装过程中执行。
$ sudo apt-get update
$ sudo apt-get install \
python-dev \
python-pip \
python-virtualenv
-
-I'll then create a Python virtual environment and activate it. This should ease any issues with permissions when installing libraries via pip.
+
+接下来我要创建一个 Python 虚拟环境,并且激活它。通过 pip 安装库时,会很容易导致权限问题。
$ virtualenv findbots
$ source findbots/bin/activate
-
-MaxMind offer a free database of country and city registration information for IPv4 addresses. Along with this dataset they've released a Python-based library called "geoip2" that can map their datasets to memory-mapped files and use a C-based Python extension to perform very fast lookups.
-The following will install their library and download and unpack their city-level dataset.
+MaxMind 提供了一个免费的数据库,数据库里有 IPv4 地址对应的国家和城市注册信息。和这些数据集一起,他们还发布了一个基于 Python 的库,叫 “geoip2”,这个库可以将他们的数据集映射到内存映射的文件里,并且用基于 C 的 Python 扩展来执行非常快的查询。
+
+下面的命令会安装它们的包,下载、解压它们在城市那一层的数据集。
$ pip install geoip2
$ curl -O http://geolite.maxmind.com/download/geoip/database/GeoLite2-City.mmdb.gz
$ gunzip GeoLite2-City.mmdb.gz
-
-I had a look at some web traffic logs and grep'ed out hits where "robots.txt" was being requested. From that list I spot-checked some of the more frequently-appearing IP addresses and found a number of hosting and cloud providers being listed as the owners of these IPs. I wanted to see if it was possible to put together a list, however incomplete, of IPv4 addresses under these providers' ownership.
-Google have a DNS-based mechanism for collecting a list of their IP addresses they use for their cloud offering. This first call will give you a list of hosts to query.
+我看过一些 web 流量日志,并且抓取出来一些恰好请求了「robots.txt」的流量。从那个列表里,我重点检查了经常出现的 IP 地址中的一些,发现里面有不少作为 IP 所有者的主机和云提供商。我想知道是不是有可能攒出来一个列表,无论完不完整,包括了这些提供商所有的 IPv4 地址。
+
+Google 有一个基于 DNS 的机制,用于收集它们用于提供云的 IP 地址列表。这个最初的调用将给你一系列可以查询的主机。
$ dig -t txt _cloud-netblocks.googleusercontent.com | grep spf
-
+
```
_cloud-netblocks.googleusercontent.com. 5 IN TXT "v=spf1 include:_cloud-netblocks1.googleusercontent.com include:_cloud-netblocks2.googleusercontent.com include:_cloud-netblocks3.googleusercontent.com include:_cloud-netblocks4.googleusercontent.com include:_cloud-netblocks5.googleusercontent.com ?all"
```
-
-The above states that _cloud-netblocks[1-5].googleusercontent.com will contain SPF records that contain IPv4 and IPv6 CIDR addresses they use. Querying all five addresses like the following should give you an up-to-date listing.
+
+以上阐明了 _cloud-netblocks[1-5].googleusercontent.com 将包含 SPF 记录,这些记录里包括他们实用的 IPv4 和 IPv6 CIDR 地址。像如下这样查询所有的五个地址,应当会给你一个最新的列表。
$ dig -t txt _cloud-netblocks1.googleusercontent.com | grep spf
-
+
```
_cloud-netblocks1.googleusercontent.com. 5 IN TXT "v=spf1 ip4:8.34.208.0/20 ip4:8.35.192.0/21 ip4:8.35.200.0/23 ip4:108.59.80.0/20 ip4:108.170.192.0/20 ip4:108.170.208.0/21 ip4:108.170.216.0/22 ip4:108.170.220.0/23 ip4:108.170.222.0/24 ?all"
```
-I published a [blog post](bulk-ip-address-whois-python-hadoop.html#ipv4-whois-mapreduce-job) last March where I attempted to scrape WHOIS details for the entirety of the IPv4 address space using a Hadoop-based MapReduce job. The job itself ran for about two hours before terminating prematurely. I was left with an incomplete but still sizeable dataset of 235,532 WHOIS records. The dataset is a year old now but should still prove valuable, if not somewhat dated.
+去年三月我尝试着抓取了使用一个基于 Hadoop 的 MapReduce 任务的 IPv4 地址空间的 WHOIS 细节,并且发布了一篇 [博客文章](bulk-ip-address-whois-python-hadoop.html#ipv4-whois-mapreduce-job)。这个任务在过早结束之前,跑了接近两个小时,留给了我一份虽然不完整,但是大小可观的数据集,里面有 235,532 个 WHOIS 记录。这个数据集已经存在一年之久了,如果不需要那么新的数据的话,应该还是有价值的。
$ ls -l
-
+
```
-rw-rw-r-- 1 mark mark 5946203 Mar 31 2016 part-00001
-rw-rw-r-- 1 mark mark 5887326 Mar 31 2016 part-00002
@@ -69,7 +69,7 @@ I published a [blog post](bulk-ip-address-whois-python-hadoop.html#ipv4-whois-ma
-rw-rw-r-- 1 mark mark 5961162 Mar 31 2016 part-00155
```
-When I spot-checked the IP ownership of bots hitting "robots.txt", in addition to Google, six firms came up a lot: Amazon, Baidu, Digital Ocean, Hetzner, Linode and New Dream Network. I ran the following commands to try and pick out their IPv4 WHOIS records.
+当我重点检查那些爬到「robots.txt」机器人的 IP 所属的时候,除了 Google,六家公司也出现了很多次:Amazon、百度、Digital Ocean、Hetzner、Linode 和 New Dream Network。我跑了以下的命令,尝试去取出它们的 IPv4 WHOIS 记录。
$ grep -i 'amazon' part-00* > amzn
$ grep -i 'baidu' part-00* > baidu
@@ -77,9 +77,9 @@ When I spot-checked the IP ownership of bots hitting "robots.txt", in addition t
$ grep -i 'hetzner' part-00* > hetzner
$ grep -i 'linode' part-00* > linode
$ grep -i 'new dream network' part-00* > dream
-
-I had to parse out double-encoded JSON strings that were embedded with file name and frequency count information from the above six files. I use the following code in iPython to get the distinctive CIDR blocks.
+
+我需要从以上六个文件中,解析二次编码的 JSON 字符串,这些字符串包含了文件名和频率次数信息。我使用了 iPython 代码来获得不同的 CIDR 块,代码如下:
```
import json
@@ -107,7 +107,7 @@ for _name in ['amzn', 'baidu', 'digital_ocean',
print _name, parse_cidrs(_name)
```
-Here is an example WHOIS record once it's been cleaned up. I've truncated out the contact information.
+下面是一份清理完毕的 WHOIS 记录实例,我已经去掉了联系信息。
```
{
@@ -156,23 +156,23 @@ Here is an example WHOIS record once it's been cleaned up. I've truncated out th
}
```
-The list of seven firms isn't an exhaustive list of where bot traffic originates from. I found a lot of bot traffic from residential IPs in the Ukraine, Chinese IPs where the originating organisation is difficult to distinguish in addition to a distributed army of bots connecting from all around the world. If I wanted an exhaustive list of IPs used by bots I could look into [HTTP header order](http://geocar.sdf1.org/browser-verification.html), examine TCP/IP behaviour, hunt down [forged IP registrations](http://go.whiteops.com/rs/179-SQE-823/images/WO_Methbot_Operation_WP.pdf) (see page 28), the list goes on and it's a bit of a cat-and-mouse game to be honest.
+这份七个公司的列表不是一个关于爬虫机器人来源的全面的列表。我发现很多爬虫流量来源于一些在乌克兰、中国的住宅 IP,源头很难分辨,除了分布式爬虫战队,从世界各地连接。说实话,如果我想要一个全面的爬虫机器人实用的 IP 列表,我只需要看看 [HTTP 头的顺序](http://geocar.sdf1.org/browser-verification.html),检查下 TCP/IP 的行为,搜寻[伪造 IP 注册](http://go.whiteops.com/rs/179-SQE-823/images/WO_Methbot_Operation_WP.pdf)(请看 28 页),列表就出来了,并且这就像猫和老鼠的游戏一样。
-## Installing Libraries
+## 安装库
-For this project I'll be using a number of well-written libraries. [Apache Log Parser](https://github.com/rory/apache-log-parser) can parse lines in Apache and Nginx-generated traffic logs. The library supports parsing over 30 different types of information from log files and I've found it remarkably flexible and reliable. [Python User Agents](https://github.com/selwin/python-user-agents) can parse user agent strings as well as perform some basic classification of the agent being used. [Colorama](https://github.com/tartley/colorama) assists in creating colourful ANSI output. [Netaddr](https://github.com/drkjam/netaddr/) is a mature and well-maintained network address manipulation library.
+对于这个项目而言,我会实用一些写得很好的库。[Apache Log Parser](https://github.com/rory/apache-log-parser) 可以解析 Apache 和 Nginx 生成的流量日志。这个库支持从日志文件中解析超过 30 种不同类型的信息,并且我发现,它相当弹性、可靠。[Python User Agents](https://github.com/selwin/python-user-agents) 可以解析用户代理的字符串,并执行一些代理使用的基本分类操作。[Colorama](https://github.com/tartley/colorama) 协助创建有高亮的 ANSI 输出。[Netaddr](https://github.com/drkjam/netaddr/) 是一种成熟的、维护得很好的网络地址操作库。
$ pip install -e git+https://github.com/rory/apache-log-parser.git#egg=apache-log-parser \
-e git+https://github.com/selwin/python-user-agents.git#egg=python-user-agents \
colorama \
netaddr
-
-## The Bot Monitoring Script
-The following walks through the contents of monitor.py. This script accepts web traffic logs piped in from stdin. This means you can tail a log on a remote server via ssh and run this script locally.
+## 爬虫机器人监控脚本
-I'll first import two libraries from the Python Standard Library and the five external libraries installed via pip.
+接下来的部分是跑 monitor.py 的内容。这段脚本从 stdin(标准输入) 管道中接收 web 流量日志。这说明你可以通过 ssh 在远程服务器上看日志,在本地跑这段脚本。
+
+我先从 Python 标准库里 import(导入) 两个库,并通过 pip 安装了五个外部库。
```
import sys
@@ -185,11 +185,11 @@ from netaddr import IPNetwork, IPAddress
from user_agents import parse
```
-In the following I've setup MaxMind's geoip2 library to use the "GeoLite2-City.mmdb" city-level library.
+接下来我设置好 MaxMind 的 geoip2 库,以使用「GeoLite2-City.mmdb」城市级别的库。
-I've also setup apache_log_parser to work with the format my web logs are being stored in. Your log format may vary so please take the time to compare your web server's traffic logging configuration against the library's [format documentation](https://github.com/rory/apache-log-parser#supported-values).
+我还设置了 apache_log_parser,来处理存储的 web 日志格式。你的日志格式可能不一样,所以可能需要花点时间比较下你的 web 服务器的流量日志配置与这个库的 [格式文档](https://github.com/rory/apache-log-parser#supported-values)。
-Finally, I have a dictionary of the CIDR blocks I found being owned by the seven firms. Included in this list is Baidu, which isn't a hosting or cloud provider per-se but nonetheless run bots that haven't always identified themselves by their user agent.
+最后,我有一个我发现的属于那七家公司的 CIDR 块的字典。在这个列表里,从本质上来说,百度不是一家主机或者云提供商,但是跑着很多无法通过它们的用户代理所识别的机器人。
```
reader = geoip2.database.Reader('GeoLite2-City.mmdb')
@@ -278,7 +278,7 @@ CIDRS = {
}
```
-I've created a utility function where I can pass in an IPv4 address and a list of CIDR blocks and it'll tell me if the IP address belongs to any of the given CIDR blocks.
+我创建了一个工具函数,可以传入一个 IPv4 地址和一个 CIDR 块列表,它会告诉我这个 IP 地址是不是属于给定的这些 CIDR 块中的任何一个。
```
def in_block(ip, block):
@@ -288,7 +288,7 @@ def in_block(ip, block):
if _ip in IPNetwork(cidr)])
```
-The following function takes objects of a request and the browser agent used and tries to determine if the source of traffic and/or the browser agent are that of a bot. The browser agent object is put together by the Python User Agents library and it already has some tests for determining if a user agent string is that of a known bot. I've further expanded these tests with a number of tokens I saw slip through the library's classification system. I also iterate through the CIDR blocks to see if the the remote host's IPv4 address is within any of them.
+接下来的函数传入一个请求对象,浏览器代理使用并尝试判断这个流量或这个浏览器代理的来源是不是一个爬虫机器人。浏览器代理对象被 Python 用户代理库放在一起,并有一些测试来确定用户代理字串是否一个已知的爬虫机器人。我已经用一些我从库的分类系统中看到的 token 来扩展这些测试。同时我在 CIDR 块迭代,来判断远程主机的 IPv4 地址是否在里面。
```
def bot_test(req, agent):
@@ -318,11 +318,11 @@ def bot_test(req, agent):
return is_bot
```
-The following is the main section of the script. Here web traffic logs are read in from stdin line by line. Each line of content is parsed for a tokenised version of the request, user agent and URI being requested. These objects make it easier to work with the data without the complexity of having to parse them on the fly.
+下面是脚本的主要部分。web 流量日志从标准输入里一行行地读入。内容的每一行都被解析成请求的一个带 token 的版本,用户代理和 URI 被请求。这些对象让与这些数据打交道变得更容易,不需要去麻烦地在空中解析它们。
-I attempt to look up the city and country associated with the IPv4 address using MaxMind's library. If there is any sort of failure these are simply set to None.
+我尝试着用 MaxMind 的库查询与这些 IPv4 相关的城市和国家。
-After the bot test I prepare the output. If the request is seen to be that of a bot it'll highlight the output with a red background.
+在机器人测试后,我准备输出。如果请求看起来是从机器人处发送的,它会被标成红色背景,高亮在输出上。
```
if __name__ == '__main__':
@@ -369,18 +369,18 @@ if __name__ == '__main__':
Style.RESET_ALL
```
-## Bot Detection in Action
+## 机器人检测实战
-Below is an example where I concatenate out the last 100 lines of a web traffic log while continuing to follow it and pipe the contents into the monitoring script.
+接下来是一个例子,关于我是怎么连接输出 web 流量日志的最后一百行的,当我接着做并且将内容放进监测脚本的管道中时。
```
$ ssh server \
'tail -n100 -f access.log' \
| python monitor.py
```
-
-Requests suspected to have originated from bots will be highlighted with a red background and a "b" prefix. Non-bot traffic will be prefixed with "h" for human. The following is an example output from the script sans the ANSI background colours.
+
+有可能来源于机器人的请求将使用红色背景和「b」前缀高亮。不存在机器人的流量将被打上「h」的前缀,代表 human(人)。下面是从脚本出来的样例输出,不过没有 ANSI 背景色。
...
b US Indianapolis /robots.txt Python Requests 2.2 Linux 3.2.0
@@ -396,4 +396,3 @@ Requests suspected to have originated from bots will be highlighted with a red b
h IE Dublin /theme/css/syntax.css Chrome 56.0.2924 Mac OS X 10.12.0
h IE Dublin /theme/images/mark.jpg Chrome 56.0.2924 Mac OS X 10.12.0
b SG Singapore /./theme/images/mark.jpg Slack-ImgProxy Spider Spider Desktop Amazon IP
-
\ No newline at end of file
From 24cc145535f0afc795f9fefdcba4a053f1fa1eb5 Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Tue, 28 Mar 2017 09:11:39 +0800
Subject: [PATCH 019/945] =?UTF-8?q?=E5=88=A0=E9=99=A4=E8=8B=B1=E6=96=87?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...in-javascript-part-1-observable-objects.md | 65 -------------------
1 file changed, 65 deletions(-)
diff --git a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
index 411cb7218e6..2fcb5272a33 100644
--- a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
+++ b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
@@ -6,48 +6,31 @@
![](https://d4a7vd7s8p76l.cloudfront.net/uploads/1484604970-4-7876/observables.png)
-# [How to build a reactive engine in JavaScript. Part 1: Observable objects](/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects/) #
# 如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象 #
-## The reactive way ##
## 响应式的方式 ##
-With the growing need for robust and interactive web interfaces, many developers have started embracing the reactive programming paradigm.
随着对强健、可交互的网站界面的需求不断增多,很多开发者开始拥抱响应式编程规范。
-Before we begin implementing our own reactive engine, let’s quickly explain what reactive programming actually is. Wikipedia gives us a classic example of a reactive interface implementation – namely a spreadsheet. Defining a formula such as `=A1+B1` would update the cell whenever either `A1` or `B1` change. Such a formula can be considered a computed value.
在开始实现我们自己的响应式引擎之前,快速地解释一下到底什么是响应式编程。维基百科给出一个经典的响应式界面实现的例子 —— 叫做 spreadsheet。定义一个准则,对于 `=A1+B1`,只要 `A1` 或 `B1` 发生变化,`=A1+B1` 也会随之变化。这样的准则也可以被理解为是一种 computed value。
-You will learn how to implement computed values in the second part of this reactive series. Before that, we first need a base for our reactivity engine.
我们将会在这系列教程的 Part 2 部分学习如何实现 computed value。在那之前,我们首先需要对响应式引擎有个基础的了解。
-## The engine ##
## 引擎 ##
-Currently there are many different approaches to solving the problem of observing and reacting to the changing application state.
目前有很多不同解决方案可以观察到应用状态的改变,并对其做出反应。
-- Angular 1.x has its dirty checking.
-- React, because of the way it works – doesn’t actually track changes in the data model. It uses the virtual DOM to diff and patch the DOM.
-- Cycle.js and Angular 2 prefer the reactive streams implementations like XStream and Rx.js.
-- Libraries like Vue.js, MobX or Ractive.js all use a variation of getters/setters to create observable data models.
-
- Angular 1.x 有脏检查。
- React 由于它工作方式,并不追踪数据模型中的改变。它用虚拟 DOM 比较并修补 DOM。
- Cycle.js 和 Angular 2 更倾向于响应流方式实现,像 XStream 和 Rx.js。
- 像 Vue.js, MobX 或 Ractive.js 这些库都使用 getters/setters 变量创建可观察的数据模型。
-In this tutorial, we will go the getters/setters way of observing and reacting to changes.
在这篇教程中,我们将使用 getters/setters 的方式观察并响应变化。
-> Note: To keep the tutorial as simple as possible, the code lacks the support for non-primitive data types or nested properties and many of the required sanity checks, thus by no means should this code be considered production ready. The code below is written using the ES2015 standard and is loosely inspired by the Vue.js reactive engine implementation.
-
> 注意:为了让这篇教程尽量保持简单,代码缺少对非初级数据类型或嵌套属性的支持,并且很多内容需要完整性检查,因此决不能认为这些代码已经可以用于生产环境。下面的代码是受 Vue.js 启发的响应式引擎的实现,使用 ES2015 标准编写。
-## The observable object ##
## 可观察的对象 ##
-Let’s start with a `data` object, whose properties we want to observe.
让我们从一个 `data` 对象开始,我们想要观察它的属性。
```
@@ -58,7 +41,6 @@ let data = {
}
```
-Let’s start by creating two functions that will transform our object’s properties into observable properties using the getter/setter functionality.
首先从创建两个函数开始,使用 getter/setter 的功能,将对象的普通属性转换成可观察的属性。
```
@@ -88,16 +70,12 @@ function observeData (obj) {
observeData(data)
```
-By running `observeData(data)` we transform our object into an object capable of being observed; now we have a way to create notifications whenever the value changes.
通过运行 `observeData(data)`,将原始的对象转换成可被观察的对象;现在当对象的 value 发生变化时,我们有创建通知的办法。
-## Reacting to changes ##
## 响应变化 ##
-Before we begin *notifying*, we need something that we can actually notify. This is a perfect example where we can use the observer pattern. In this case we will make use of the signals implementation.
在我们开始接收 *notifying* 前,我们需要一些通知的内容。这里是使用观察者模式的一个极好例子。在这个案例中我们将使用 signals 实现。
-Let’s start with the `observe` function.
我们从 `observe` 函数开始。
```
@@ -110,10 +88,8 @@ function observe (property, signalHandler) {
}
```
-We can now use the `observe` function like this: `observe('propertyName', callback)`, where `callback` is a function that should be called each time the property’s value has changed. When we **observe** a property multiple times, each callback will be stored inside the corresponding property’s signal array. This way we can store all callbacks and have easy access to them.
我们现在可以这样用 `observe` 函数:`observe('propertyName', callback)`,每次属性值发生改变的时候 `callback` 函数应该被调用。当多次在一个属性上调用 **observe** 时,每个回调函数将被存在对应属性的 signal 数组中。这样就可以存储所有的回调函数并且可以很容易地获得到它们。
-Now for the `notify` function that you saw before.
现在来看一下上文中提到的 `notify` 函数。
```
@@ -124,10 +100,8 @@ function notify (signal, newVal) {
}
```
-As you can see, now every time one of the properties changes, the assigned signalHandlers will be called.
如你所见,现在每次一个属性发生变化,就会调用对其分配的 signalHandlers。
-So let’s wrap it all up into a factory function that we pass the data object that has to be reactive. I will name mine `Seer`. We end up with something like this:
所以我们把它全部封装起来做成一个工厂函数,传入想要响应的数据对象。我把它命名为 `Seer`。我们最终得到如下:
```
@@ -179,7 +153,6 @@ function Seer (dataObj) {
}
```
-All we need to do now is to create a new reactive object. Thanks to the exposed `notify` and `observe` functions, we can observe and react to the changes made to the object.
现在我们需要做的就是创建一个新的可响应对象。多亏了暴露出来的 `notify` 和 `observe` 函数,我们可以观察到并响应对象的改变。
```
@@ -190,43 +163,33 @@ const App = new Seer({
age: 25
})
-// To subscribe and react to changes made to the reactive App object:
// 为了订阅并响应可响应 APP 对象的改变:
App.observe('firstName', () => console.log(App.data.firstName))
App.observe('lastName', () => console.log(App.data.lastName))
-// To trigger the above callbacks simply change the values like this:
// 为了触发上面的回调函数,像下面这样简单地改变 values:
App.data.firstName = 'Sansa'
App.data.lastName = 'Stark'
```
-Simple, isn’t it? Now that we have the basic reactivity engine covered, let’s make some use of it.
-I mentioned that with the more reactive approach to front-end programming, we should not be concerned with things like manually updating the DOM after each change.
很简单,是不是?现在我们讲完了基本的响应式引擎,让我们来用用它。
我提到过随着前端编程可响应式方法的增多,我们不能总想着在发生改变后手动地更新 DOM。
-There are many approaches to this. I guess the most trending one right now is the so called virtual DOM. If you are interested in learning how to create your own virtual DOM implementation, there are already great tutorials for this. However, here we will go with a much simpler approach.
有很多方法来完成这项任务。我猜现在最流行的趋势是用虚拟 DOM 的办法。如果你对学习如何创建你自己的虚拟 DOM 实现感兴趣,已经有很多这方面的教程。然而,这里我们将用到更简单的方法。
-Let’s say our HTML looks like this:`html
Title comes here
`
HTML 看起来像这样: `html
Title comes here
`
-The function responsible for updating the DOM would look like this:
响应式更新 DOM 的函数看起来像这样:
```
-// First we need to get the node that we want to keep updating.
// 首先需要获得想要保持更新的节点。
const h1Node = document.querySelector('h1')
function syncNode (node, obj, property) {
- // Initialize the h1’s textContent value with the observed object’s property value
// 用可见对象的属性值初始化 h1 的 textContent 值
node.textContent = obj[property]
- // Start observing the property using our Seer instance App.observe method.
// 开始用我们的 Seer 的实例 App.observe 观察属性。
App.observe(property, value => node.textContent = obj[property] || '')
}
@@ -234,48 +197,35 @@ function syncNode (node, obj, property) {
syncNode(h1Node, App.data, 'title')
```
-This will work but actually requires a lot of work from us to actually bind all the DOM elements to the desired data models.
这样做是可行的,但是使用它把所有数据模型绑定到 DOM 元素需要大量的工作。
-That’s why we can go a step further and automate all of this.
-If you are familiar with AngularJS or Vue.js you surely remember using custom HTML attributes like `ng-bind` or `v-text`. We will create something similar here!
-Our custom attribute will be called `s-text`. We will look for it to create bindings between the DOM and the data model.
-
这就是我们为什么要再向前迈一步,然后将所有这些自动化完成。
如果你熟悉 AngularJS 或者 Vue.js,你肯定记得使用自定义属性 `ng-bind` 或 `v-text`。我们在这里创建类似的东西。
我们的自定义属性叫做 `s-text`。我们将寻找在 DOM 和数据模型之间建立绑定的方式。
-Let’s update our HTML:
让我们更新一下 HTML:
```
-
Title comes here
function parseDOM (node, observable) {
- // We get all nodes that have the s-text custom attribute
// 获得所有具有自定义属性 s-text 的节点
const nodes = document.querySelectorAll('[s-text]')
- // For each existing node, we call the syncNode function
// 对于每个存在的节点,我们调用 syncNode 函数
nodes.forEach((node) => {
syncNode(node, observable, node.attributes['s-text'].value)
})
}
-// Now all we need to do is call it with document.body as the root node. All `s-text` nodes will automatically create bindings to the corresponding reactive property.
// 现在我们需要做的就是在根节点 document.body 上调用它。所有的 `s-text` 节点将会自动的创建与之对应的响应式属性的绑定。
parseDOM(document.body, App.data)
```
-## Summary ##
## 总结 ##
-Now that we have a way to parse the DOM and bind the nodes to the data model, let’s add those two functions into the Seer factory function, where we will parse the DOM on initialization.
现在我们可以解析 DOM 并且将数据模型绑定到节点上,把这两个函数添加到 Seer 工厂函数中,这样就可以在初始化的时候解析 DOM。
-The result should look like this:
结果应该像下面这样:
```
@@ -322,14 +272,12 @@ function Seer (dataObj) {
makeReactive(obj, key)
}
}
- // We can safely parse the DOM looking for bindings after we converted the dataObject.
//转换数据对象后,可以安全地解析 DOM 绑定。
parseDOM(document.body, obj)
}
function syncNode (node, observable, property) {
node.textContent = observable[property]
- // We remove the `Seer.` as it is now available for us in our scope.
// 移除了 `Seer.` 是因为 observe 函数在可获得的作用域范围之内。
observe(property, () => node.textContent = observable[property])
}
@@ -344,7 +292,6 @@ function Seer (dataObj) {
}
```
-Example on JsFiddle:
JsFiddle 上的例子:
HTML
@@ -367,10 +314,7 @@ HTML
JS
```
-// This code uses ES2015.
-// Please use a compatible browser like: Chrome, Opera, Firefox
// 代码用了 ES2015,使用兼容的浏览器才可以哦,比如 Chrome,Opera,Firefox
-
function Seer (dataObj) {
let signals = {}
@@ -414,14 +358,12 @@ function Seer (dataObj) {
makeReactive(obj, key)
}
}
- // We can safely parse the DOM looking for bindings after we converted the dataObject.
//转换数据对象后,可以安全地解析 DOM 绑定。
parseDOM(document.body, obj)
}
function syncNode (node, observable, property) {
node.textContent = observable[property]
- // We remove the `Seer.` as it is now available for us in our scope.
// 移除了 `Seer.` 是因为 observe 函数在可获得的作用域范围之内。
observe(property, () => node.textContent = observable[property])
}
@@ -463,23 +405,16 @@ Result
![Markdown](http://i2.buimg.com/1949/cf89248985467d6f.png)
-The above code can be found here: [github.com/shentao/seer](https://github.com/shentao/seer/tree/master)
上文的代码可以在这里找到: [github.com/shentao/seer](https://github.com/shentao/seer/tree/master)
-## To be continued... ##
## 未完待续…… ##
-This is the first part in a series about crafting your own reactivity engine.
这篇是制作你自己的响应式引擎系列文章中的第一篇。
-**The next part will be about creating computed properties, where each has its own trackable dependencies.**
**下一篇将是关于创建 computed properties,每个属性都有它自己的可追踪依赖。**
-Your feedback and ideas on what to cover next are both very welcome in the comments!
非常欢迎在评论区提出你对于下一篇文章讲述内容的反馈和想法!
-Thanks for reading.
-
感谢阅读。
---
From 4c3b930b904ff06b37997f0fa0ea873bf11b5fd1 Mon Sep 17 00:00:00 2001
From: zhouzihanntu
Date: Tue, 28 Mar 2017 19:05:01 +0800
Subject: [PATCH 020/945] update file
---
...-in-node-js-everything-you-need-to-know.md | 21 +++++++++++++++----
1 file changed, 17 insertions(+), 4 deletions(-)
diff --git a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
index 5618ebc6fd9..fef57b8f8e0 100644
--- a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
+++ b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
@@ -16,41 +16,52 @@ Node uses two core modules for managing module dependencies:
Node 提供了两个核心模块来管理模块依赖:
- The `require` module, which appears to be available on the global scope — no need to `require('require')`.
-- `require` 模块在全局范围内可用,不需要 `require('require')`.
+- `require` 模块在全局范围内可用,不需要 `require('require')`.
- The `module` module, which also appears to be available on the global scope — no need to `require('module')`.
- `module` 模块同样在全局范围内可用,不需要 `require('module')`.
You can think of the `require` module as the command and the `module` module as the organizer of all required modules.
-你可以把 `require` 模块当做命令 `module` 模块 所有需加载模块的
+你可以将 `require` 模块理解为命令,将 `module` 模块理解为组织所有需加载模块的工具。
Requiring a module in Node isn’t that complicated of a concept.
-在 Node 中加载一个模块其实没有概念那么复杂。
+在 Node 中加载一个模块其实不像概念那么复杂。
```
const config = require('/path/to/file');
```
The main object exported by the `require` module is a function (as used in the above example). When Node invokes that `require()` function with a local file path as the function’s only argument, Node goes through the following sequence of steps:
+`require` 模块导出的主对象是一个函数(如上例)。当 Node 将本地文件路径作为唯一参数调用 `require()` 时,Node 将执行以下步骤:
- **Resolving**: To find the absolute path of the file.
+- **解析**:找到该文件的绝对路径。
- **Loading**: To determine the type of the file content.
+- **加载**:确定文件内容的类型。
- **Wrapping**: To give the file its private scope. This is what makes both the `require` and `module` objects local to every file we require.
+- **打包**:为文件划分私有作用域,这样 `require` 和 `module` 两个对象对于我们要加载的每个模块来说就都是本地的。
- **Evaluating**: This is what the VM eventually does with the loaded code.
+- **评估**:最后由虚拟机对加载得到的代码做评估。
- **Caching**: So that when we require this file again, we don’t go over all the steps another time.
+- **缓存**:当再次加载该文件时,无需再重复以上所有步骤。
In this article, I’ll attempt to explain with examples these different stages and how the affect the way we write modules in Node.
+在本文中,我将试着用示例说明这些不同阶段,以及它们如何影响我们在 Node 中编写模块的方式。
Let me first create a directory to host all the examples using my terminal:
+我先使用终端创建一个目录来托管本文中的所有示例:
```
mkdir ~/learn-node && cd ~/learn-node
```
All the commands in the rest of this article will be run from within `~/learn-node`.
+本文余下部分的所有命令都将在 `~/learn-node` 目录下运行。
#### Resolving a local path ####
+#### 解析本地路径 ####
Let me introduce you to the `module` object. You can check it out in a simple REPL session:
+首先,让我来介绍一下 `module` 对象。你可以通过一个简单的 REPL 会话查看它:
```
~/learn-node $ node
@@ -62,10 +73,12 @@ Module {
filename: null,
loaded: false,
children: [],
- paths: [ ... ] }
+ paths: [ ... ]
+}
```
Every module object gets an `id` property to identify it. This `id` is usually the full path to the file, but in a REPL session it’s simply `.`
+每个模块对象都有一个用于识别该对象的 `id` 属性。这个 `id` 通常是该文件的完整路径,但在 REPL 会话中 `.`
Node modules have a one-to-one relation with files on the file-system. We require a module by loading the content of a file into memory.
From 25f4ecdc994ed93cbce6cb494ae279769da16e0f Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Tue, 28 Mar 2017 23:57:30 +0800
Subject: [PATCH 021/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20=E6=88=91=E6=98=AF?=
=?UTF-8?q?=E5=A6=82=E4=BD=95=E5=81=9A=E5=88=B0=E5=9C=A8=205=20=E5=88=86?=
=?UTF-8?q?=E9=92=9F=E4=B9=8B=E5=86=85=E5=B0=86=E5=BA=94=E7=94=A8=E5=A4=A7?=
=?UTF-8?q?=E5=B0=8F=E5=87=8F=E5=B0=91=2060%=20=E7=9A=84=20=E7=9A=84?=
=?UTF-8?q?=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 4 ++--
android.md | 1 +
2 files changed, 3 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index 0916246a4b9..6715a5e0da6 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [437](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [438](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -20,10 +20,10 @@
## Android
+* [我是如何做到在 5 分钟之内将应用大小减少 60% 的](https://juejin.im/post/58d9b6a1a22b9d0064719f9e/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([jifaxu](https://github.com/jifaxu) 翻译)
* [拉模式和推模式,命令式和响应式 – 响应式编程 [Android RxJava2](这到底是什么):第二部分](https://juejin.im/post/58d78547a22b9d006465ca57/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([XHShirley](https://github.com/XHShirley) 翻译)
* [离线支持:不再『稍后重试』](https://juejin.im/post/58d491a8128fe1006cb6e750/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([skyar2009](https://github.com/skyar2009) 翻译)
* [当发布安卓开源库时我希望知道的东西](https://juejin.im/post/58d247b00ce4630057e92e9c/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([jifaxu](https://github.com/jifaxu) 翻译)
-* [单元测试试图告诉我们关于 Activity 的什么事情:第二部分](https://gold.xitu.io/entry/58cf988b2f301e007e54c434/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([tanglie1993](https://github.com/tanglie1993/) 翻译)
* [所有 Android 译文>>](https://github.com/xitu/gold-miner/blob/master/android.md)
diff --git a/android.md b/android.md
index 42c60c048d2..c845f087d8d 100644
--- a/android.md
+++ b/android.md
@@ -1,3 +1,4 @@
+* [我是如何做到在 5 分钟之内将应用大小减少 60% 的](https://juejin.im/post/58d9b6a1a22b9d0064719f9e/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([jifaxu](https://github.com/jifaxu) 翻译)
* [拉模式和推模式,命令式和响应式 – 响应式编程 [Android RxJava2](这到底是什么):第二部分](https://juejin.im/post/58d78547a22b9d006465ca57/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([XHShirley](https://github.com/XHShirley) 翻译)
* [离线支持:不再『稍后重试』](https://juejin.im/post/58d491a8128fe1006cb6e750/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([skyar2009](https://github.com/skyar2009) 翻译)
* [当发布安卓开源库时我希望知道的东西](https://juejin.im/post/58d247b00ce4630057e92e9c/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([jifaxu](https://github.com/jifaxu) 翻译)
From f94325f3ded55e13add63a11fa4b68e5a9376c6c Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Wed, 29 Mar 2017 00:05:59 +0800
Subject: [PATCH 022/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20=E5=A6=82=E4=BD=95?=
=?UTF-8?q?=E5=9C=A8=20ChromeOS=20=E4=B8=8B=E7=94=A8=20Go=20=E6=90=AD?=
=?UTF-8?q?=E5=BB=BA=20Web=20=E6=9C=8D=E5=8A=A1=20=E7=9A=84=E7=A7=AF?=
=?UTF-8?q?=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 4 ++--
backend.md | 1 +
2 files changed, 3 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index 6715a5e0da6..e8015a4450f 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [438](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [439](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -55,10 +55,10 @@
## 后端
+* [如何在 ChromeOS 下用 Go 搭建 Web 服务](https://juejin.im/post/58d9e1711b69e6006bc38b1a/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([xiaoyusilen](https://github.com/xiaoyusilen) 翻译)
* [在你沉迷于包的海洋之前,还是了解一下运行时 Node.js 的本身](https://juejin.im/post/58cf4a3144d90400690b7be7/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([fghpdf](https://github.com/fghpdf) 翻译)
* [Pull request review 的十大错误](https://juejin.im/post/58ce3b3e61ff4b006c988f63/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([luoyaqifei](https://github.com/luoyaqifei) 翻译)
* [震惊!RxJava 5 个不为人知的小秘密](https://juejin.im/post/58cb833b8ac247218c2632e5/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([skyar2009](https://github.com/skyar2009) 翻译)
-* [跨站请求伪造已死!](https://juejin.im/post/58c669b6a22b9d0058b3c630?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([XatMassacrE](https://github.com/XatMassacrE) 翻译)
* [所有后端译文>>](https://github.com/xitu/gold-miner/blob/master/backend.md)
## 教程
diff --git a/backend.md b/backend.md
index 6b61f3952af..2e7926a7cba 100644
--- a/backend.md
+++ b/backend.md
@@ -1,3 +1,4 @@
+* [如何在 ChromeOS 下用 Go 搭建 Web 服务](https://juejin.im/post/58d9e1711b69e6006bc38b1a/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([xiaoyusilen](https://github.com/xiaoyusilen) 翻译)
* [在你沉迷于包的海洋之前,还是了解一下运行时 Node.js 的本身](https://juejin.im/post/58cf4a3144d90400690b7be7/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([fghpdf](https://github.com/fghpdf) 翻译)
* [Pull request review 的十大错误](https://juejin.im/post/58ce3b3e61ff4b006c988f63/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([luoyaqifei](https://github.com/luoyaqifei) 翻译)
* [震惊!RxJava 5 个不为人知的小秘密](https://juejin.im/post/58cb833b8ac247218c2632e5/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([skyar2009](https://github.com/skyar2009) 翻译)
From 54ae0b20c8db6c81ecac08698a38ffa91e75c8b6 Mon Sep 17 00:00:00 2001
From: sqrtthree
Date: Wed, 29 Mar 2017 10:38:36 +0800
Subject: [PATCH 023/945] =?UTF-8?q?=E8=B0=83=E6=95=B4=E5=9B=BE=E7=89=87?=
=?UTF-8?q?=E9=93=BE=E6=8E=A5?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/modern-javascript-for-ancient-web-developers.md | 5 +----
1 file changed, 1 insertion(+), 4 deletions(-)
diff --git a/TODO/modern-javascript-for-ancient-web-developers.md b/TODO/modern-javascript-for-ancient-web-developers.md
index 7a306e5172b..e39f2691bd5 100644
--- a/TODO/modern-javascript-for-ancient-web-developers.md
+++ b/TODO/modern-javascript-for-ancient-web-developers.md
@@ -46,10 +46,7 @@ The other difficult thing about learning JavaScript in 2017: getting set up will
The JS world has a *lot* of work to do in this regard. Again, this area of modern JavaScript is a constantly moving target, but my Local Friendly Modern JS Engineers tell me that [this tutorial from Jonathan Verrecchia](https://github.com/verekia/js-stack-from-scratch) is currently the definitive guide to building a modern JavaScript stack. For now.
-![Markdown](http://i1.piimg.com/1949/95cedaf271a8c352.png)
-[**verekia/js-stack-from-scratch**](https://github.com/verekia/js-stack-from-scratch)
-
-[*js-stack-from-scratch - 🎉 V2 release! 🎉 - Step-by-step tutorial to build a modern JavaScript stack.*github.com](https://github.com/verekia/js-stack-from-scratch)
+[![Markdown](http://i1.piimg.com/1949/95cedaf271a8c352.png)](https://github.com/verekia/js-stack-from-scratch)
### Tutorial / Project / Throw It Away / Repeat ###
From 85c0cf5ac1f92ba9560e65e0e408f21d0c367101 Mon Sep 17 00:00:00 2001
From: reid
Date: Wed, 29 Mar 2017 10:56:08 +0800
Subject: [PATCH 024/945] =?UTF-8?q?=E4=B8=80=E6=A0=A1?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/setstate-gate-abc.md | 16 ++++++++--------
1 file changed, 8 insertions(+), 8 deletions(-)
diff --git a/TODO/setstate-gate-abc.md b/TODO/setstate-gate-abc.md
index 8c8f024b4df..40ca0843b22 100644
--- a/TODO/setstate-gate-abc.md
+++ b/TODO/setstate-gate-abc.md
@@ -45,13 +45,13 @@
当然,推特还是有可能会丧失其集体智慧(lose its collective mind)(译注:个人认为这句应该是指当网络上大多数人持某一观点时,那即使该观点是错的,那你也不能指出其错误,否则就会招致集体攻讦;或者说,真理有时候只掌握在少数人手里)。 毕竟,React 是**完美的**, 我们都必须承认 `setState` 的美妙优雅,否则只会遭到冷嘲热讽。
-如果 `setState()` 令你感到困惑,那都是**你的错** —— 你要么是疯子,要么是傻瓜。(我好像忘了说 [Javascript 的社区霸凌问题了](https://medium.com/javascript-scene/the-js-community-has-a-bullying-problem-96c10f11c85d#.wagjqz54o) )
+如果 `setState()` 令你感到困惑,那都是**你的错** —— 你要么是疯子,要么是傻瓜。(我好像忘了说 [Javascript 社区的霸凌问题了](https://medium.com/javascript-scene/the-js-community-has-a-bullying-problem-96c10f11c85d#.wagjqz54o) )
好了,当你嘲笑所有初学者的时候,先反省反省自己吧,别以为掌握了 `setState()` 就可以得意忘形了。
那种行为是荒谬可笑的,是精英主义论的,会让新手们感到十分讨厌。如果人们经常对某个 API 感到困惑的话,那就该改进 API 本身的设计了,或者至少应该改进下文档。
-让我们的社区和工具变得更加友好和充满魅力对所有人来说都是件好事。
+让我们的社区和工具变得更加友好对所有人来说都是件好事。
### setState() 究竟有何问题? ###
@@ -63,7 +63,7 @@
React 的设计目标就是要简化网站和应用的开发,但是:
- 你不能直接操作 DOM。
-- 你不能随心所欲地(于任何时间任何地点用任何数据)更新状态。
+- 你不能随心所欲地(于任何时间依赖任意数据源)更新状态。
- 在组件的声明周期中,你并不总是能在屏幕上直接观察到渲染后的 DOM 元素,这限制了 `setState()` 的使用时机和方式(因为你有些状态可能还没有渲染到屏幕上)。
在这几种情况下,困惑都来源于 React 组件生命周期的限制性(这些限制是刻意设计的,是好的)。
@@ -73,10 +73,10 @@ React 的设计目标就是要简化网站和应用的开发,但是:
在更新状态的时候,更新结果可能依赖于:
- 当前状态
-- 同一循环中之前的更新操作
+- 同一周期中之前的更新操作
- 当前已渲染的 DOM (例如:组件的坐标位置、可见性、CSS 计算值等等)
-当存在这几种从属状态的时候,如果你还想简单直接地更新状态,那 React 的表现行为会让你大吃一惊,并且是以一种令人憎恶又难以调试的方式。大多数情况下,你的代码根本无法工作:要么状态不对,要么控制台一堆错误。
+当存在这几种从属状态的时候,如果你还想简单直接地更新状态,那 React 的表现行为会让你大吃一惊,并且是以一种令人憎恶又难以调试的方式。大多数情况下,你的代码根本无法工作:要么状态不对,要么控制台有错误。
我之所以吐槽 `setState()`,是因为它这种具有限制性的表现行为在 API 文档中并没有详细说明,关于处理这种限制性行为的各种通用模式也未能阐述清楚。这迫使初学者只能不断试错、Google 或者从其他社区成员那里寻求帮助,但实际上在文档中本该就有更好的新手指南。
@@ -173,7 +173,7 @@ API 文档中并未提及此更新函数的这些特性和表现,罕见的是
### 仅仅是新手才有的问题吗? ###
-直到现在,在处理表单或是 DOM 元素坐标位置的时候,我还是会时不时得掉到坑里去。当你使用 `setState()` 的时候,你必须直接处理组件生命周期的相关问题;但当你使用容器组件或是通过属性来存储和传递状态的话,React 则会替你处理同步问题。
+直到现在,在处理表单或是 DOM 元素坐标位置的时候,我还是会时不时得掉到坑里去。当你使用 `setState()` 的时候,你必须直接处理组件生命周期的相关问题;但当你使用容器组件或是通过属性来存储和传递状态的时候,React 则会替你处理同步问题。
[**无论你有经验与否**](https://medium.com/@mweststrate/3-reasons-why-i-stopped-using-react-setstate-ab73fc67a42e#.saj7jn6wh) ,处理共享的可变状态和状态锁(state locks)都是很棘手的。经验丰富之人只不过是能更加快速地定位问题,然后找出一个巧妙的解决方案罢了。
@@ -264,7 +264,7 @@ React 用于渲染的状态不会在 DOM 渲染的过程中发生改变。**我
如果这两个问题的答案都是“否”的话,那使用 `setState()` 基本是没问题的;否则,就要另作考虑了。
-据我所知,Facebook 使用受管于 [Relay container](https://facebook.github.io/relay/) 的 `setState()` 来包装 Facebook UI 的各个不同部分,例如大型 Facebook 应用内部的迷你型应用。于 Facebook 而言,将复杂的数据依赖和需要实际使用这些数据的组件放在一起是很好的。
+据我所知,Facebook 使用受管于 [Relay container](https://facebook.github.io/relay/) 的 `setState()` 来包装 Facebook UI 的各个不同部分,例如大型 Facebook 应用内部的迷你型应用。于 Facebook 而言,以这种方式将复杂的数据依赖和需要实际使用这些数据的组件放在一起是很好的。
对于大型(企业级)应用我推荐这种策略。如果你的应用代码量非常大(十万行以上),那此策略可能是很好的 —— 但这并意味着这种方式就不能应用于小型应用中。
@@ -297,4 +297,4 @@ React 用于渲染的状态不会在 DOM 渲染的过程中发生改变。**我
**Eric Elliott** 是 [**“编写 JavaScript 应用”**](http://pjabook.com) (O’Reilly) 以及 [**“跟着 Eric Elliott 学 Javascript”**](http://ericelliottjs.com/product/lifetime-access-pass/) 两书的**作者**。他为许多公司和组织作过贡献,例如 **Adobe Systems**、**Zumba Fitness**、**The Wall Street Journal**、**ESPN**和**BBC**等 , 也是很多机构的顶级艺术家,包括但不限于 **Usher** , **Frank Ocean** , **Metallica**。
-大多数时间,他都在 San Francisco Bay Area,同这世上最美的女子在一起(译注:这是怕老婆呢还是怕老婆呢还是怕老婆呢?)。
\ No newline at end of file
+大多数时间,他都在 San Francisco Bay Area,同这世上最美的女子在一起(译注:这是怕老婆呢还是怕老婆呢还是怕老婆呢?)。
From fa775fb725f0617c96a42d856d69c9e754846c71 Mon Sep 17 00:00:00 2001
From: zhouzihanntu
Date: Wed, 29 Mar 2017 19:14:06 +0800
Subject: [PATCH 025/945] update file
---
...-in-node-js-everything-you-need-to-know.md | 133 ++++++++----------
1 file changed, 60 insertions(+), 73 deletions(-)
diff --git a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
index fef57b8f8e0..0f35c99a5d3 100644
--- a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
+++ b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
@@ -5,63 +5,46 @@
> * 校对者:
# Requiring modules in Node.js: Everything you need to know #
-# 在 Node.js 中加载模块 #
+# 在 Node.js 中引用模块 #
-## How modularity works in Node.js ##
-## 模块化在 Node.js 中的应用##
+## Node.js 中模块化的工作原理 ##
-Node uses two core modules for managing module dependencies:
Node 提供了两个核心模块来管理模块依赖:
-- The `require` module, which appears to be available on the global scope — no need to `require('require')`.
-- `require` 模块在全局范围内可用,不需要 `require('require')`.
-- The `module` module, which also appears to be available on the global scope — no need to `require('module')`.
-- `module` 模块同样在全局范围内可用,不需要 `require('module')`.
+- `require` 模块在全局范围内可用,不需要写 `require('require')`.
+- `module` 模块同样在全局范围内可用,不需要写 `require('module')`.
-You can think of the `require` module as the command and the `module` module as the organizer of all required modules.
-你可以将 `require` 模块理解为命令,将 `module` 模块理解为组织所有需加载模块的工具。
+你可以将 `require` 模块理解为命令,将 `module` 模块理解为组织所有被引用模块的工具。
-Requiring a module in Node isn’t that complicated of a concept.
-在 Node 中加载一个模块其实不像概念那么复杂。
+在 Node 中引入一个模块其实不像概念那么复杂。
```
const config = require('/path/to/file');
```
-The main object exported by the `require` module is a function (as used in the above example). When Node invokes that `require()` function with a local file path as the function’s only argument, Node goes through the following sequence of steps:
`require` 模块导出的主对象是一个函数(如上例)。当 Node 将本地文件路径作为唯一参数调用 `require()` 时,Node 将执行以下步骤:
-- **Resolving**: To find the absolute path of the file.
- **解析**:找到该文件的绝对路径。
-- **Loading**: To determine the type of the file content.
- **加载**:确定文件内容的类型。
-- **Wrapping**: To give the file its private scope. This is what makes both the `require` and `module` objects local to every file we require.
-- **打包**:为文件划分私有作用域,这样 `require` 和 `module` 两个对象对于我们要加载的每个模块来说就都是本地的。
-- **Evaluating**: This is what the VM eventually does with the loaded code.
+- **打包**:为文件划分私有作用域,这样 `require` 和 `module` 两个对象对于我们要引入的每个模块来说就都是本地的。
- **评估**:最后由虚拟机对加载得到的代码做评估。
-- **Caching**: So that when we require this file again, we don’t go over all the steps another time.
-- **缓存**:当再次加载该文件时,无需再重复以上所有步骤。
+- **缓存**:当再次引用该文件时,无需再重复以上所有步骤。
-In this article, I’ll attempt to explain with examples these different stages and how the affect the way we write modules in Node.
-在本文中,我将试着用示例说明这些不同阶段,以及它们如何影响我们在 Node 中编写模块的方式。
+在本文中,我将试着用示例说明这些不同阶段的运行原理,以及它们如何影响我们在 Node 中编写模块的方式。
-Let me first create a directory to host all the examples using my terminal:
我先使用终端创建一个目录来托管本文中的所有示例:
```
mkdir ~/learn-node && cd ~/learn-node
```
-All the commands in the rest of this article will be run from within `~/learn-node`.
本文余下部分的所有命令都将在 `~/learn-node` 目录下运行。
-#### Resolving a local path ####
#### 解析本地路径 ####
-Let me introduce you to the `module` object. You can check it out in a simple REPL session:
-首先,让我来介绍一下 `module` 对象。你可以通过一个简单的 REPL 会话查看它:
+首先,让我来介绍一下 `module` 对象。你可以在一个简单的 REPL 会话中查看该对象:
```
~/learn-node $ node
@@ -77,20 +60,19 @@ Module {
}
```
-Every module object gets an `id` property to identify it. This `id` is usually the full path to the file, but in a REPL session it’s simply `.`
-每个模块对象都有一个用于识别该对象的 `id` 属性。这个 `id` 通常是该文件的完整路径,但在 REPL 会话中 `.`
+每个模块对象都有一个用于识别该对象的 `id` 属性。这个 `id` 通常是该文件的完整路径,但在 REPL 会话中只会显示为 ``。
-Node modules have a one-to-one relation with files on the file-system. We require a module by loading the content of a file into memory.
+Node 模块与文件系统中的文件有着一一对应的关系。我们通过加载模块对应的文件内容到内存中来实现模块引用。
-However, since Node allows many ways to require a file (for example, with a relative path or a pre-configured path), before we can load the content of a file into the memory we need to find the absolute location of that file.
+然而,由于 Node 允许使用许多方式引入文件(例如,使用相对路径或预先配置的路径),我们需要在将文件的内容加载到内存前找到该文件的绝对位置。
-When we require a `'find-me'` module, without specifying a path:
+例如,我们不声明路径,直接引入一个 `'find-me'` 模块时:
```
require('find-me');
```
-Node will look for `find-me.js` in all the paths specified by `module.paths` — in order.
+Node 会在 `module.paths` 声明的所有路径中依次查找 `find-me.js` 。
```
~/learn-node $ node
@@ -105,9 +87,9 @@ Node will look for `find-me.js` in all the paths specified by `module.paths`
'/usr/local/Cellar/node/7.7.1/lib/node' ]
```
-The paths list is basically a list of node_modules directories under every directory from the current directory to the root directory. It also includes a few legacy directories whose use is not recommended.
+Node 从当前目录开始一级级向上寻找 node_modules 目录,这个数组大致就是当前目录到所有 node_modules 目录的相对路径。其中还包括一些为了兼容性保留的目录,不推荐使用。
-If Node can’t find `find-me.js` in any of these paths, it will throw a “cannot find module error.”
+如果 Node 在以上路径中都无法找到 `find-me.js` ,将抛出一个 “找不到该模块” 错误。
```
~/learn-node $ node
@@ -125,7 +107,7 @@ Error: Cannot find module 'find-me'
at REPLServer.onLine (repl.js:533:10)
```
-If you now create a local `node_modules` directory and put a `find-me.js` in there, the `require('find-me')` line will find it.
+如果你现在创建一个本地的 `node_modules` 目录,并向目录中添加一个 `find-me.js` 文件,就能通过 `require('find-me')` 找到它了。
```
~/learn-node $ mkdir node_modules
@@ -139,14 +121,14 @@ I am not lost
>
```
-If another `find-me.js` file existed in any of the other paths, for example, if we have a `node_modules` directory under the home directory and we have a different `find-me.js` file in there:
+如果在其他路径下也有 `find-me.js` 文件呢?例如,我们在主目录下的 `node_modules` 目录中放置一个不同的 `find-me.js` 文件:
```
$ mkdir ~/node_modules
$ echo "console.log('I am the root of all problems');" > ~/node_modules/find-me.js
```
-When we `require('find-me')` from within the `learn-node` directory — which has its own `node_modules/find-me.js`, the `find-me.js` file under the home directory will not be loaded at all:
+当我们在 `learn-node` 目录下执行 `require('find-me')` 时,`learn-node` 目录会加载自己的 `node_modules/find-me.js`,主目录下的 `find-me.js` 文件并不会被加载:
```
~/learn-node $ node
@@ -156,7 +138,7 @@ I am not lost
>
```
-If we remove the local `node_modules` directory under `~/learn-node` and try to require `find-me` one more time, the file under the home’s `node_modules` directory would be used:
+此时,如果我们将 `~/learn-node` 下的 `node_modules` 移除,再一次引入 `find-me` 模块,那么主目录下的 `node_modules` 将会被加载:
```
~/learn-node $ rm -r node_modules/
@@ -168,9 +150,9 @@ I am the root of all problems
>
```
-#### Requiring a folder ####
+#### 引入文件夹 ####
-Modules don’t have to be files. We can also create a `find-me` folder under `node_modules` and place an `index.js` file in there. The same `require('find-me')` line will use that folder’s `index.js` file:
+模块不一定是单个文件。我们也可以在 `node_modules` 目录下创建一个 `find-me` 文件夹,然后向其中添加一个 `index.js` 文件。`require('find-me')` 会引用该文件夹下的 `index.js` 文件:
```
~/learn-node $ mkdir -p node_modules/find-me
@@ -184,9 +166,9 @@ Found again.
>
```
-Note how it ignored the home directory’s `node_modules` path again since we have a local one now.
+>注意,由于我们现在有一个本地目录,它再次忽略了主目录的 `node_modules` 路径。
-An `index.js` file will be used by default when we require a folder, but we can control what file name to start with under the folder using the `main` property in `package.json`. For example, to make the `require('find-me')` line resolve to a different file under the `find-me` folder, all we need to do is add a `package.json` file in there and specify which file should be used to resolve this folder:
+当我们引入一个文件夹时,将默认执行 `index.js` 文件,但是我们可以通过 `package.json` 中的 `main` 属性指定主入口文件。例如,要令 `require('find-me')` 解析到 `find-me` 文件夹下的另一个文件,我们只需要在该文件夹下添加一个 `package.json` 文件来声明解析该文件夹时引用的文件:
```
~/learn-node $ echo "console.log('I rule');" > node_modules/find-me/start.js
@@ -200,9 +182,9 @@ I rule
>
```
-#### require.resolve ####
+#### require.resolve 方法 ####
-If you want to only resolve the module and not execute it, you can use the `require.resolve` function. This behaves exactly the same as the main `require` function, but does not load the file. It will still throw an error if the file does not exist and it will return the full path to the file when found.
+如果你只想解析模块而不运行,此时可以使用 `require.resolve` 函数。这个方法与 `require` 的主要功能完全相同,但是不加载文件。如果文件不存在,它仍会抛出错误,并在找到文件时返回文件的完整路径。
```
> require.resolve('find-me');
@@ -222,34 +204,34 @@ Error: Cannot find module 'not-there'
>
```
-This can be used, for example, to check whether an optional package is installed or not and only use it when it’s available.
+这个方法可以用于检查一个可选安装包是否安装,并仅在该包可用时使用。
-#### Relative and absolute paths ####
+#### 相对路径和绝对路径 ####
-Besides resolving modules from within the `node_modules` directories, we can also place the module anywhere we want and require it with either relative paths (`./` and `../`) or with absolute paths starting with `/`.
+除了从 `node_modules` 目录中解析模块以外,我们还可以将模块放置在任意位置,使用相对路径( `./` 和 `../` )或以 `/` 开头的绝对路径引入。
-If, for example, the `find-me.js` file was under a `lib` folder instead of the `node_modules` folder, we can require it with:
+举个例子,如果 `find-me.js` 文件并不在 `node_modules` 中,而在 `lib` 文件夹中。我们可以使用以下代码引入它:
```
require('./lib/find-me');
```
-#### Parent-child relation between files ####
+#### 文件间的父子关系 ####
-Create a `lib/util.js` file and add a `console.log` line there to identify it. Also, `console.log` the `module` object itself:
+现在我们来创建一个 `lib/util.js` 文件,向文件添加一行 `console.log` 代码作为标识。打印出 `module` 对象本身:
```
~/learn-node $ mkdir lib
~/learn-node $ echo "console.log('In util', module);" > lib/util.js
```
-Do the same for an `index.js` file, which is what we’ll be executing with the node command. Make this `index.js` file require `lib/util.js`:
+同样的,向 `index.js` 文件中也添加一行打印 `module` 对象的代码,并在文件中引入 `lib/util.js`,我们将使用 node 命令运行该文件:
```
~/learn-node $ echo "console.log('In index', module); require('./lib/util');" > index.js
```
-Now execute the `index.js` file with node:
+用 node 运行 `index.js` 文件:
```
~/learn-node $ node index.js
@@ -279,25 +261,27 @@ In util Module {
paths: [...] }
```
-Note how the main `index` module `(id: '.')` is now listed as the parent for the `lib/util` module. However, the `lib/util` module was not listed as a child of the `index` module. Instead, we have the `[Circular]` value there because this is a circular reference. If Node prints the `lib/util` module object, it will go into an infinite loop. That’s why it simply replaces the `lib/util` reference with `[Circular]`.
+>注意:`index` 主模块 `(id: '.')` 现在被列为 `lib/util` 模块的父类。但 `lib/util` 模块并没有被列为 `index` 模块的子目录。相反,我们在这里得到的值是 `[Circular]`,因为这是一个循环引用。如果 Node 打印 `lib/util` 模块对象,将进入一个无限循环。这就是为什么 Node 使用 `[Circular]` 代替了 `lib/util` 引用。
-More importantly now, what happens if the `lib/util` module required the main `index` module? This is where we get into what’s known as the circular modular dependency, which is allowed in Node.
+
+重点来了,如果我们在 `lib/util` 模块中引入 `index` 主模块会发生什么?这就是 Node 中所支持的循环模块依赖关系。
To understand it better, let’s first understand a few other concepts on the module object.
+为了更好理解循环模块依赖,我们先来了解一些关于 module 对象的概念。
-#### exports, module.exports, and synchronous loading of modules ####
+#### exports、module.exports 和模块异步加载 ####
-In any module, exports is a special object. If you’ve noticed above, every time we’ve printed a module object, it had an exports property which has been an empty object so far. We can add any attribute to this special exports object. For example, let’s export an id attribute for `index.js` and `lib/util.js`:
+在所有模块中,exports 都是一个特殊对象。你可能注意到了,以上我们每打印一个 module 对象时,它都有一个空的 exports 属性。我们可以向这个特殊的 exports 对象添加任意属性。例如,我们现在为 `index.js` 和 `lib/util.js` 的 exports 对象添加一个 id 属性:
```
-// Add the following line at the top of lib/util.js
+// 在 lib/util.js 顶部添加以下代码
exports.id = 'lib/util';
-// Add the following line at the top of index.js
+// 在 index.js 顶部添加以下代码
exports.id = 'index';
```
-When we now execute `index.js`, we’ll see these attributes as managed on each file’s `module` object:
+然后运行 `index.js`,我们将看到:
```
~/learn-node $ node index.js
@@ -319,15 +303,15 @@ In util Module {
... }
```
-I’ve removed some attributes in the above output to keep it brief, but note how the `exports` object now has the attributes we defined in each module. You can put as many attributes as you want on that exports object, and you can actually change the whole object to be something else. For example, to change the exports object to be a function instead of an object, we do the following:
+为了保持示例简短,我删除了以上输出中的一些属性,但请注意:`exports` 对象现在拥有我们在各模块中定义的属性。你可以向 exports 对象添加任意多的属性,也可以直接将整个对象改为其它对象。例如,我们可以通过以下方式将 exports 对象更改为一个函数:
```
-// Add the following line in index.js before the console.log
+// 将以下代码添加在 index.js 中的 console.log 语句前
module.exports = function() {};
```
-When you run `index.js` now, you’ll see how the `exports` object is a function:
+再次运行 `index.js`,你将看到 `exports` 对象是一个函数:
```
~/learn-node $ node index.js
@@ -338,9 +322,9 @@ In index Module {
... }
```
-Note how we did not do `exports = function() {}` to make the `exports` object into a function. We can’t actually do that because the `exports` variable inside each module is just a reference to `module.exports` which manages the exported properties. When we reassign the `exports` variable, that reference is lost and we would be introducing a new variable instead of changing the `module.exports` object.
+>注意:我们并没有使用 `exports = function() {}` 来将 `exports` 对象更改为函数。实际上,由于各模块中的 `exports` 变量仅仅是对管理输出属性的 `module.exports` 的引用,当我们对 `exports` 变量重新赋值时,引用就会丢失,此时我们只是引入了一个新的变量,而没有对 `module.exports` 做修改。
-The `module.exports` object in every module is what the `require` function returns when we require that module. For example, change the `require('./lib/util')` line in `index.js` into:
+各模块中的 `module.exports` 对象就是我们在引入该模块时 `require` 函数的返回值。例如,我们将 `index.js` 中的 `require('./lib/util')` 改为:
```
const UTIL = require('./lib/util');
@@ -348,24 +332,24 @@ const UTIL = require('./lib/util');
console.log('UTIL:', UTIL);
```
-The above will capture the properties exported in `lib/util` into the `UTIL` constant. When we run `index.js` now, the very last line will output:
+以上代码会将 `lib/util` 输出的属性赋值给 `UTIL` 常量。我们现在运行 `index.js`,最后一行将输出以下结果:
```
UTIL: { id: 'lib/util' }
```
-Let’s also talk about the `loaded` attribute on every module. So far, every time we printed a module object, we saw a `loaded` attribute on that object with a value of `false`.
+我们再来谈谈各模块中的 `loaded` 属性。到目前为止我们打印的所有 module 对象中都有一个值为 `false` 的 `loaded` 属性。
-The `module` module uses the `loaded` attribute to track which modules have been loaded (true value) and which modules are still being loaded (false value). We can, for example, see the `index.js` module fully loaded if we print its `module` object on the next cycle of the event loop using a `setImmediate` call:
+`module` 模块使用 `loaded` 属性对模块的加载状态进行跟踪,判断哪些模块已经加载完成(值为 true)以及哪些模块仍在加载(值为 false)。例如,要判断 `index.js` 模块是否已完全加载,我们可以在下一个事件循环中使用一个 `setImmediate` 回调打印出他的 `module` 对象。
```
-// In index.js
+// index.js 中
setImmediate(() => {
console.log('The index.js module object is now loaded!', module)
});
```
-The output of that would be:
+以上输出将得到:
```
The index.js module object is now loaded! Module {
@@ -390,23 +374,26 @@ The index.js module object is now loaded! Module {
'/node_modules' ] }
```
-Note how in this delayed `console.log` output both `lib/util.js` and `index.js` are fully loaded.
+>注意:这个延迟的 `console.log` 的输出显示了 `lib/util.js` 和 `index.js` 都已完全加载。
The `exports` object becomes complete when Node finishes loading the module (and labels it so). The whole process of requiring/loading a module is *synchronous.* That’s why we were able to see the modules fully loaded after one cycle of the event loop.
+`exports` 对象在 Node 完成引入/加载所有模块并标记时(becomes complete……构建完成??)。引入一个模块的整个过程是 **同步的**,因此我们才能在一个事件循环结束后看见模块被完全加载。
-This also means that we cannot change the `exports` object asynchronously. We can’t, for example, do the following in any module:
+这也意味着我们无法异步地更改 `exports` 对象。例如,我们在任何模块中都无法执行以下操作:
```
fs.readFile('/etc/passwd', (err, data) => {
if (err) throw err;
- exports.data = data; // Will not work.
+ exports.data = data; // 无效
});
```
#### Circular module dependency ####
+#### 循环模块依赖 ####
Let’s now try to answer the important question about circular dependency in Node: What happens when module 1 requires module 2, and module 2 requires module 1?
+我们现在来回答关于 Node 中循环依赖的重要问题:当模块1
To find out, let’s create the following two files under `lib/`, `module1.js` and `module.js` and have them require each other:
From b90d1564b47e41aaafbf62e1b433bf23e7b6f233 Mon Sep 17 00:00:00 2001
From: sqrtthree
Date: Wed, 29 Mar 2017 20:50:30 +0800
Subject: [PATCH 026/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E6=96=87=E7=AB=A0?=
=?UTF-8?q?=E9=93=BE=E6=8E=A5?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...a-reactive-engine-in-javascript-part-1-observable-objects.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
index 2fcb5272a33..767d52a0a11 100644
--- a/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
+++ b/TODO/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects.md
@@ -411,7 +411,7 @@ Result
这篇是制作你自己的响应式引擎系列文章中的第一篇。
-**下一篇将是关于创建 computed properties,每个属性都有它自己的可追踪依赖。**
+**[下一篇](https://github.com/xitu/gold-miner/blob/master/TODO/computed-properties-javascript-dependency-tracking.md) 将是关于创建 computed properties,每个属性都有它自己的可追踪依赖。**
非常欢迎在评论区提出你对于下一篇文章讲述内容的反馈和想法!
From 7778344210b68df78f507275904b1214e7e09c7c Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Wed, 29 Mar 2017 21:51:54 +0800
Subject: [PATCH 027/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20=E6=90=9C=E7=B4=A2?=
=?UTF-8?q?=E7=BB=93=E6=9E=9C=E9=A1=B5=E7=9A=84=E6=9C=80=E4=BD=B3=E5=AE=9E?=
=?UTF-8?q?=E8=B7=B5=20=E7=9A=84=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 4 ++--
design.md | 1 +
2 files changed, 3 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index e8015a4450f..d2e4e284886 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [439](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [440](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -72,10 +72,10 @@
## 设计
+* [搜索结果页的最佳实践](https://juejin.im/post/58da37c761ff4b00607287a6/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([sunui](https://github.com/sunui) 翻译)
* [某些2017年的 UX 趋势啊,扎心了](https://juejin.im/post/58cf5dc22f301e007e52fb2b/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([Ruixi](https://github.com/Ruixi) 翻译)
* [像开发人员一样做设计](https://gold.xitu.io/entry/58b7ba6f8fd9c56d16be6bb0/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([Airmacho](https://github.com/Airmacho) 翻译)
* [设计预期的用户体验](https://gold.xitu.io/entry/58b2d8e9570c3500696f53a5/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([jifaxu](https://github.com/jifaxu) 翻译)
-* [优秀产品背后的设计原则](https://gold.xitu.io/entry/58b04c49570c35006960d764/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([jifaxu](https://github.com/jifaxu) 翻译)
* [所有设计译文>>](https://github.com/xitu/gold-miner/blob/master/design.md)
diff --git a/design.md b/design.md
index a885ed92043..d2d30a65ad9 100644
--- a/design.md
+++ b/design.md
@@ -1,3 +1,4 @@
+* [搜索结果页的最佳实践](https://juejin.im/post/58da37c761ff4b00607287a6/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([sunui](https://github.com/sunui) 翻译)
* [某些2017年的 UX 趋势啊,扎心了](https://juejin.im/post/58cf5dc22f301e007e52fb2b/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([Ruixi](https://github.com/Ruixi) 翻译)
* [像开发人员一样做设计](https://gold.xitu.io/entry/58b7ba6f8fd9c56d16be6bb0/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([Airmacho](https://github.com/Airmacho) 翻译)
* [设计预期的用户体验](https://gold.xitu.io/entry/58b2d8e9570c3500696f53a5/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([jifaxu](https://github.com/jifaxu) 翻译)
From 5fc917662c2b7704987d79be59514ff233f23ac6 Mon Sep 17 00:00:00 2001
From: sqrtthree
Date: Thu, 30 Mar 2017 11:16:53 +0800
Subject: [PATCH 028/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E6=8E=88=E6=9D=83?=
=?UTF-8?q?=E4=BF=A1=E6=81=AF?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/beyond-browser-web-desktop-apps.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/beyond-browser-web-desktop-apps.md b/TODO/beyond-browser-web-desktop-apps.md
index 57d1a9da13e..5cea035fd0d 100644
--- a/TODO/beyond-browser-web-desktop-apps.md
+++ b/TODO/beyond-browser-web-desktop-apps.md
@@ -1,5 +1,5 @@
> * 原文地址:[Beyond The Browser: From Web Apps To Desktop Apps](https://www.smashingmagazine.com/2017/03/beyond-browser-web-desktop-apps/)
-> * 原文作者:[Adam Lynch](https://www.smashingmagazine.com/author/adamlynch/)
+> * 原文作者:本文已获原作者 [Adam Lynch](https://www.smashingmagazine.com/author/adamlynch/) 授权
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:
> * 校对者:
From 67eed56dabde3a166958af57954f9973aa30af67 Mon Sep 17 00:00:00 2001
From: DeepMissea <398752853@qq.com>
Date: Thu, 30 Mar 2017 12:46:16 +0800
Subject: [PATCH 029/945] =?UTF-8?q?=E6=A0=B9=E6=8D=AE=E4=B8=A4=E4=BD=8D?=
=?UTF-8?q?=E6=A0=A1=E5=AF=B9=E8=80=85=E7=9A=84=E6=84=8F=E8=A7=81=E6=9B=B4?=
=?UTF-8?q?=E6=94=B9=E3=80=82?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/improving-swift-compile-times.md | 24 ++++++++++++------------
1 file changed, 12 insertions(+), 12 deletions(-)
diff --git a/TODO/improving-swift-compile-times.md b/TODO/improving-swift-compile-times.md
index 889f74d6cba..5ead3f5915f 100644
--- a/TODO/improving-swift-compile-times.md
+++ b/TODO/improving-swift-compile-times.md
@@ -2,19 +2,19 @@
* 原文作者:[John Sundell](https://medium.com/@johnsundell?source=post_header_lockup)
* 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
* 译者:[Deepmissea](http://deepmissea.blue)
-* 校对者:
+* 校对者:[atuooo](http://atuo.xyz),[1992chenlu](https://github.com/1992chenlu)
-# 提高 Swift 的编译时间
+# 优化 Swift 的编译时间
-在 Swift 所有的魅力中,有一件是可能会相当麻烦,那就是在用 Swift 编写更大规模的项目时,它**一般**会编译多久。尽管 Swift 编译器在保证运行时安全方面做的更多,但是它的编译时间要比 Objective-C 编译时间要长很多。我想研究一下,是否我们可以帮助编译器让他工作的更快。
+在 Swift 所有的特性中,有一件事有时会相当恼人,那就是在用 Swift 编写更大规模的项目时,它**一般**会编译多久。尽管 Swift 编译器在保证运行时安全方面做的更多,但是它的编译时间要比 Objective-C 编译时间长很多。(所以)我想研究一下,是否我们可以帮助编译器让他工作的更快。
-所以,上周我在 [Hyper](http://www.hyper.no) 投身于我们 Swift 的大项目之一。它大哥有 350 个源文件以及 30,000 行的代码。最后我设法将这个项目的平均构建时间减少了 [20%](https://twitter.com/johnsundell/status/837318595973611521)。所以我想在我这周的博客上详细的介绍我是怎么做的。
+所以,上周我投身于 [Hyper](http://www.hyper.no) 上的一个较大的 Swift 项目。它大概有 350 个源文件以及 30,000 行的代码。最后我设法将这个项目的平均构建时间减少了 [20%](https://twitter.com/johnsundell/status/837318595973611521)。所以我想在我这周的博客上详细的介绍我是怎么做的。
-现在,在我们开始之前,我只想说**我不想这篇文章以任何形式的方式来批判 Swift 或它的团队工作**。我知道 Swift 编译器的开发者,包含 Apple 公司和开源社区,都在持续地对编译器速度、功能和稳定性做出重大改进。希望这篇博文能随着时间的流逝而感到多余,但在那之前,我只是想提供一些我发现可以提升编译速度的实用技巧。
+现在,在我们开始之前,我只想说**我不想这篇文章以任何形式的方式来批判 Swift 或它的团队工作**。我知道 Swift 编译器的开发者,包含 Apple 公司和开源社区,都在持续地对编译器速度、功能和稳定性做出重大改进。希望这篇博文能随着时间的流逝而显得多余,但在那之前,我只是想提供一些我发现可以提升编译速度的实用技巧。
#### Step 1: 采集数据
-在开始优化工作之前,建立一个能衡量你改进的基准总是好的。我是通过在 Xcode 里添加两个简单的脚本到应用的 target 作为**运行脚本阶段**来实现的。
+在开始优化工作之前,建立一个能衡量你改进的基准总是好的。我是通过在 Xcode 里,给应用的 target 添加两个简单的脚本作为**运行脚本阶段**来实现的。
在**编译源文件**之前,添加下面的脚本:
@@ -35,23 +35,23 @@ echo "[Start] $startime$newline[End] $endtime$newline[Delta] $deltatime" > "buil
现在,这个脚本只会测算编译器编译**应用自己的源文件**的时间(为了测量出整个引用的编译时间,你可以使用 Xcode 的特性来挂载(hook)到 **Build Starts** 和 **Build Succeeds** 上)。由于编译时间非常依赖于编译它的设备,所以我也 **git ignored 了 buildtimes.log 文件**。
-接下来,我想突出哪些个别代码块耗费了额外的长时间来编译,以便识别瓶颈,这样我就可以修复它。要做到这个,就需要通过向 Xcode 中 Build Setting 里的 **Other Swift Flags** 传递下面的参数给 Swift 编译器来设置一个临界值:
+接下来,我想突出哪些个别代码块耗费了额外的长时间来编译,以便识别瓶颈,这样我就可以修复它。要做到这个,只需要通过向 Xcode 中 Build Setting 里的 **Other Swift Flags** 传递下面的参数给 Swift 编译器来设置一个临界值:
```
-Xfrontend -warn-long-function-bodies=500
```
-使用上面的参数后,在你的项目中,如果有任何函数耗费了超过 500 毫秒的编译时间,你就会得到一个警告。这是我开始设置的临界值(并且随着我对更多瓶颈的修复,这个标准在不断的降低)。
+使用上面的参数后,在你的项目中,如果有任何函数耗费了超过 500 毫秒的编译时间,你就会得到一个警告。这是我开始设置的临界值(并且随着我对更多瓶颈的修复,这个值在不断的降低)。
#### Step 2: 消除所有的警告
-在设置了函数编译时间过长的警告之后,你可能会在项目中开始发现一些。最开始,你会觉得编译时间过长的函数是随机的,但是很快模式(patterns)就开始出现了。这里我注意到了两个使用 Swift 3.0 编译器编译函数时间过长的常见模式:
+在设置了函数编译时间过长的警告之后,你可能会在项目中开始发现一些。最开始,你会觉得编译时间过长的函数是随机的,但是很快模式(patterns)就开始出现了。这里我注意到了两个使 Swift 3.0 编译器编译函数时间过长的常见模式:
**自定义运算符(特别是带有通用参数的重载)**
-当 Swift 出现时,对于大多数 iOS 和 macOS 开发者来说,运算符重载是全新的概念之一,但就像许多新鲜事物一样,我们很兴奋的使用它们。现在,我不打算在这讨论自定义或重载运算符是好是坏,但它们的确对编译时间有很大影响,尤其是如果使用更加负载的表达式。
+当 Swift 出现时,对于大多数 iOS 和 macOS 开发者来说,运算符重载是全新的概念之一,但就像许多新鲜事物一样,我们很兴奋的使用它们。现在,我不打算在这讨论自定义或重载运算符是好是坏,但它们的确对编译时间有很大影响,尤其是如果使用更加复杂的表达式。
-思考下面的运算符,添加了两个 **IntegerConvertible** 类型的数字来形成自定义的数字类型:
+思考下面的运算符,它将两个 **IntegerConvertible** 类型的数字加起来,构成了自定义的数字类型:
```
func +
Date: Thu, 30 Mar 2017 16:42:02 +0800
Subject: [PATCH 030/945] =?UTF-8?q?=E4=B8=80=E6=A0=A1?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...mputed-properties-javascript-dependency-tracking.md | 10 +++++-----
1 file changed, 5 insertions(+), 5 deletions(-)
diff --git a/TODO/computed-properties-javascript-dependency-tracking.md b/TODO/computed-properties-javascript-dependency-tracking.md
index 27be4bc4077..d8256d2fd38 100644
--- a/TODO/computed-properties-javascript-dependency-tracking.md
+++ b/TODO/computed-properties-javascript-dependency-tracking.md
@@ -10,7 +10,7 @@
# 如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪 #
Hey! If you have ever worked with Vue.js, Ember or MobX I’m pretty sure you stumbled upon so-called **computed** properties. They allow you to create functions that can be accessed just like normal values, but once computed they are cached until one of its dependencies has changed. In general this is a concept very similar to getters and in fact, the following implementation will be using getters. In a smart way. ;)
-Hey!如果你用过 Vue.js、Ember 或 MobX,我敢肯定你被 **计算** 属性难倒过。计算属性允许你创建像正常的值一样使用的函数,但是一旦完成计算,他们就被缓存下来直到它的一个依赖发生改变。总的来说,这一概念与 getters 非常相似,并且事实上,下面的实现将会使用 getters。用一种机智的方式。 ;)
+Hey!如果你用过 Vue.js、Ember 或 MobX,我敢肯定你被 **计算** 属性难倒过。计算属性允许你创建像正常的值一样使用的函数,但是一旦完成计算,他们就被缓存下来直到它的一个依赖发生改变。总的来说,这一概念与 getters 非常相似,实际上下面的实现也将会用到 getters。只不过实现的方式更加聪明一点。 ;)
> This is the 2nd part of the How to build a reactive engine in JavaScript series. Before reading any further it is highly recommended to read [Part 1: Observable objects](https://monterail.com/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects), because the following implementation is built on top of the previous article's code.
> 这是如何使用 JavaScript 构建响应式引擎系列文章的第二部分。在深入阅读前强烈建议读一下 [Part 1: 可观察的对象](https://monterail.com/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects),因为接下来的实现是构建于前一篇文章的代码基础之上的。
@@ -37,7 +37,7 @@ computed: {
```
Now if we use the `fullName` somewhere in our template, we expect it will be updated whenever `firstName` or `lastName` change. If you come from an AngularJS background you might also remember using expressions inside the template or function calls. Of course, this works the same when using render functions (with JSX or not); it doesn’t really matter.
-现在如果在模板中使用 `fullName`,我们希望它能随着 `firstName` 或 `lastName` 的改变而更新。如果你有使用 AngularJS 的背景,你可能还记得在模板或者函数调用内使用表达式。当然了,使用渲染函数(使用 JSX 或者不使用)的时候和这里是一样的;其实这无关紧要。
+现在如果在模板中使用 `fullName`,我们希望它能随着 `firstName` 或 `lastName` 的改变而更新。如果你有使用 AngularJS 的背景,你可能还记得在模板或者函数调用内使用表达式。当然了,使用渲染函数(不管用不用 JSX)的时候和这里是一样的;其实这无关紧要。
Let’s consider the following example:
来看一下下面的例子:
@@ -58,7 +58,7 @@ The result of the above code will be mostly the same. Each time `firstName` or `
上面代码的执行结果几乎是一样的。每次 `firstName` 或 `lastName` 发生变化,视图将会更新这些 `` 并且显示出全名。
However, what if we use the expression, method call and computed property multiple times? The expression and method call will have to be calculated each time they are accessed, whereas the computed property will be cached after the first computation until one of its dependencies change. It will also persist through the re-render cycles! That’s actually quite a nice optimization if you consider that in event-based modern user interfaces, it’s hard to predict which action the user will take first.
-然而,如果多次使用表达式、函数调用和计算属性呢?使用表达式和函数调用每次都会计算一遍,而计算属性在第一次计算后将会缓存下来,直到它的依赖发生改变。它也会在重新渲染的循环中一直保持!如果考虑在基于事件模型的用户界面中,很难预测用户会首先执行哪项操作,那么这确实是一个最优化方案。
+然而,如果多次使用表达式、函数调用和计算属性呢?使用表达式和函数调用每次都会计算一遍,而计算属性在第一次计算后将会缓存下来,直到它的依赖发生改变。它也会在重新渲染的周期中一直保持!如果考虑在基于事件模型的现代用户界面中,很难预测用户会首先执行哪项操作,那么这确实是一个最优化方案。
## Basic computed property ##
## 基础的计算属性 ##
@@ -169,7 +169,7 @@ will call both the `firstName` and `lastName` getters.
将会调用 `firstName` 和 `lastName` 的 getters。
Let’s make use of it!
-让我们开始使用它!
+让我们利用这一点!
We need a way to collect the information that a getter was called when evaluating a computed property. For this to work, first we need a place to store which computed property is currently being evaluated. We can use a simple object for this:
当对计算属性求值的时候,我们需要收集 getter 被调用的信息。为了完成这项工作,首先需要空间存储当前求值的计算属性。可以用这样的简单对象:
@@ -491,7 +491,7 @@ Done! You might have already noticed that it also enables computed properties to
## 异步陷阱 ##
Now that you know how dependency tracking works, it should be quite obvious why it’s not possible to track asynchronous data inside computed properties both in MobX and Vue.js. It all breaks because even a `setTimeout(callback, 0)` will be called out of the current context where `Dep.target` no longer exists. This means that whatever happens inside the callback won’t be tracked.
-既然你知道了依赖追踪如何工作,也就很明显为什么在 MobX 和 Vue.js 中追踪计算属性中的异步数据是不可能的。这一切会被打破,因为即使 `setTimeout(callback, 0)` 将会在当前上下文外被调用,在那里 `Dep.target` 不在存在。这也就意味着在回调函数中无论发生什么都不会被追踪到。
+既然你知道了依赖追踪如何工作,在 MobX 和 Vue.js 中不能追踪计算属性种的异步数据的原因就很明显了。这一切会被打破,因为即使 `setTimeout(callback, 0)` 将会在当前上下文外被调用,在那里 `Dep.target` 不在存在。这也就意味着在回调函数中无论发生什么都不会被追踪到。
## Bonus: Watchers ##
## 红利:Watchers ##
From ca72be4d378ef6ef111a8eadad23b2174c9da392 Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Thu, 30 Mar 2017 16:51:01 +0800
Subject: [PATCH 031/945] =?UTF-8?q?=E4=BA=8C=E6=A0=A1?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...operties-javascript-dependency-tracking.md | 19 +++++++++----------
1 file changed, 9 insertions(+), 10 deletions(-)
diff --git a/TODO/computed-properties-javascript-dependency-tracking.md b/TODO/computed-properties-javascript-dependency-tracking.md
index d8256d2fd38..2a177f8e0c7 100644
--- a/TODO/computed-properties-javascript-dependency-tracking.md
+++ b/TODO/computed-properties-javascript-dependency-tracking.md
@@ -65,8 +65,8 @@ However, what if we use the expression, method call and computed property multip
In the previous article, we learned how to track and react to changes inside observable object properties by utilizing an event emitter. We know that when we change the `firstName` it will call all the handlers that subscribed to the `’firstName’` event. Thus it is quite easy to build a computed property by manually subscribing to its dependencies.
This is actually how Ember does it:
-在前面文章中,我们学习了如何通过使用事件发射器追踪和响应可观察对象内的改变。我们知道当改变 `firstName` 时,会调用所有的订阅了 `’firstName’` 事件的处理器。因此通过手动订阅它的依赖来构建计算属性是相当容易的。
-这也是 Ember 实现它的方式:
+在前面文章中,我们学习了如何通过使用事件发射器追踪和响应可观察对象属性内的改变。我们知道当改变 `firstName` 时,会调用所有的订阅了 `’firstName’` 事件的处理器。因此通过手动订阅它的依赖来构建计算属性是相当容易的。
+这也是 Ember 实现计算属性的方式:
```
fullName: Ember.computed('firstName', 'lastName', function() {
@@ -75,7 +75,7 @@ fullName: Ember.computed('firstName', 'lastName', function() {
```
The drawback here is that you have to declare the dependencies yourself. Doesn’t seem like a problem until you have computed properties that are a result of a chain of more expensive, complex functions. For example:
-这样做的缺点就是你不得不自己声明依赖。当你的计算属性是一串高开销的、复杂的函数的时候,你就知道这的确是个问题了。例如:
+这样做的缺点就是你不得不自己声明依赖。当你的计算属性是一串高开销的、复杂的函数的运行结果时候,你就知道这的确是个问题了。例如:
```
selectedTransformedList: Ember.computed('story', 'listA', 'listB', 'listC', function() {
@@ -105,7 +105,7 @@ Vue.js 和 MobX 在解决这个问题上使用了与上文不同的方法。不
When `this.story` changes to `’B’` it will collect a fresh set of dependencies and remove the unnecessary ones (`this.listA`) that were used before but was not called anymore.
This way even if the other lists change it won’t trigger a recalculation of `selectedTransformedList`.
That’s smart!
-当 `this.story` 变成 `’B’`,它将会收集一组新的依赖,并移除那些之前用而现在不再使用的多余的依赖(`this.listA`)。这样,尽管其他 lists 发生变化,也不会触发 `selectedTransformedList` 的重计算。
+当 `this.story` 变成 `’B’`,它将会收集一组新的依赖,并移除那些之前用而现在不再使用的多余的依赖(`this.listA`)。这样,尽管其他 lists 发生变化,也不会触发 `selectedTransformedList` 的重计算。真聪明!
Now it’s a good time to look again at the [code from the previous article - JSFiddle](https://jsfiddle.net/shentao/4k0gk3bx/10/), as the following changes will be based upon that code.
现在是时候返回来看一看 [上一篇文章中的代码 - JSFiddle](https://jsfiddle.net/shentao/4k0gk3bx/10/),下面的改动将基于这些代码。
@@ -387,8 +387,7 @@ let Dep = {
}
```
-If the `Dep.depend` method doesn’t make much sense right now, wait until we use it. It should become more clear what is actually happening there.
-如果 `Dep.depend` 函数现在还没什么意义,等一下我们就会用到它。这里做的什么应该就更清楚了。
+If the `Dep.depend` method doesn’t make much sense right now, wait until we use it. It should become more clear what is actually `Dep.depend` 函数现在还看不出用处,但我们待会就会用到它。那时在这里它的用处会更清楚。
First, let’s tune the `makeReactive` transform function.
首先,来调整 `makeReactive` 转换函数。
@@ -467,7 +466,7 @@ function makeComputed (obj, key, computeFunc) {
if (!cache) {
// Clear dependencies list to ensure getting a fresh one
- // 清空依赖列表以获得一个新的
+ // 清空依赖列表以获得一个新的列表
Dep.subs[key] = []
cache = computeFunc.call(obj)
}
@@ -497,7 +496,7 @@ Now that you know how dependency tracking works, it should be quite obvious why
## 红利:Watchers ##
The above problem can be, however, partially tackled with watchers. You might know them from Vue.js. Building watchers on top of what we already have is actually really simple. After all, a watcher is just a signal handler called after a given value has changed.
-然而,上面的问题可以通过 watchers 部分的解决。你可能已经在 Vue.js 中了解过它们。在我们已有的基础上创建 watchers 真的很容易。毕竟,watcher 是一个给定值发生变化时调用的信号处理器。
+然而,上面的问题可以通过 watchers 部分解决。你可能已经在 Vue.js 中了解过它们。在我们已有的基础上创建 watchers 真的很容易。毕竟,watcher 是一个给定值发生变化时调用的信号处理器。
We just have to add a watchers registration method and trigger it within our Seer function.
我们只是不得不添加一个 watchers 注册方法并在 Seer 函数内触发它。
@@ -545,7 +544,7 @@ The complete code is available here:
[https://github.com/shentao/seer/tree/cached-computed](https://github.com/shentao/seer/tree/cached-computed)
You can play with it online here (Opera/Chrome only):
-你可以在线的试试它(仅支持 Opera/Chrome):
+你可以在线的试玩(仅支持 Opera/Chrome):
[https://jsfiddle.net/oyw72Lyy/](https://jsfiddle.net/oyw72Lyy/)
## Summary ##
@@ -563,7 +562,7 @@ As for the 4th part – maybe streams? Would you be interested?
至于第四部分,也许是数据流?你们感兴趣吗?
Feel free to leave your feedback in the comments!
-在评论区随意反馈意见!
+欢迎在评论区随意反馈意见!
Thanks for reading!
感谢阅读!
From 1974ab65a1ed02b5590b64668bd0c3ad58565496 Mon Sep 17 00:00:00 2001
From: IridescentMia
Date: Thu, 30 Mar 2017 17:04:18 +0800
Subject: [PATCH 032/945] =?UTF-8?q?=E5=88=A0=E9=99=A4=E8=8B=B1=E6=96=87?=
=?UTF-8?q?=E5=8E=9F=E6=96=87=E6=B7=BB=E5=8A=A0=E6=A0=A1=E5=AF=B9=E8=80=85?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...operties-javascript-dependency-tracking.md | 141 ++----------------
1 file changed, 9 insertions(+), 132 deletions(-)
diff --git a/TODO/computed-properties-javascript-dependency-tracking.md b/TODO/computed-properties-javascript-dependency-tracking.md
index 2a177f8e0c7..c5aff2a2d4c 100644
--- a/TODO/computed-properties-javascript-dependency-tracking.md
+++ b/TODO/computed-properties-javascript-dependency-tracking.md
@@ -1,27 +1,21 @@
> * 原文地址:[How to build a reactive engine in JavaScript. Part 2: Computed properties and dependency tracking](https://monterail.com/blog/2017/computed-properties-javascript-dependency-tracking)
-> * 原文作者:[Damian Dulisz](https://disqus.com/by/damiandulisz/)
+> * 原文作者:本文已获原作者 [Damian Dulisz](https://disqus.com/by/damiandulisz/) 授权
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[IridescentMia](https://github.com/IridescentMia)
-> * 校对者:
+> * 校对者:[malcolmyu](https://github.com/malcolmyu),[AceLeeWinnie](https://github.com/AceLeeWinnie)
![](https://d4a7vd7s8p76l.cloudfront.net/uploads/56873733-c918-4cd6-bac1-dea44dcc3a9f/Reactive%20engine.png)
-# [How to build a reactive engine in JavaScript. Part 2: Computed properties and dependency tracking](/blog/2017/computed-properties-javascript-dependency-tracking/) #
# 如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪 #
-Hey! If you have ever worked with Vue.js, Ember or MobX I’m pretty sure you stumbled upon so-called **computed** properties. They allow you to create functions that can be accessed just like normal values, but once computed they are cached until one of its dependencies has changed. In general this is a concept very similar to getters and in fact, the following implementation will be using getters. In a smart way. ;)
Hey!如果你用过 Vue.js、Ember 或 MobX,我敢肯定你被 **计算** 属性难倒过。计算属性允许你创建像正常的值一样使用的函数,但是一旦完成计算,他们就被缓存下来直到它的一个依赖发生改变。总的来说,这一概念与 getters 非常相似,实际上下面的实现也将会用到 getters。只不过实现的方式更加聪明一点。 ;)
-> This is the 2nd part of the How to build a reactive engine in JavaScript series. Before reading any further it is highly recommended to read [Part 1: Observable objects](https://monterail.com/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects), because the following implementation is built on top of the previous article's code.
-> 这是如何使用 JavaScript 构建响应式引擎系列文章的第二部分。在深入阅读前强烈建议读一下 [Part 1: 可观察的对象](https://monterail.com/blog/2016/how-to-build-a-reactive-engine-in-javascript-part-1-observable-objects),因为接下来的实现是构建于前一篇文章的代码基础之上的。
+> 这是如何使用 JavaScript 构建响应式引擎系列文章的第二部分。在深入阅读前强烈建议读一下 [Part 1: 可观察的对象](https://juejin.im/post/58dc9da661ff4b0061547ca0),因为接下来的实现是构建于前一篇文章的代码基础之上的。
-## Computed properties ##
## 计算属性 ##
-Let’s say we have a computed property called `fullName` which is a combination of `firstName` and `lastName` with space in between.
假设有一个计算属性叫 `fullName`,是 `firstName` 和 `lastName` 之间加上空格的组合。
-In Vue.js such a computed value could be created like this:
在 Vue.js 中这样的计算值可以像下面这样创建:
```
@@ -36,35 +30,25 @@ computed: {
}
```
-Now if we use the `fullName` somewhere in our template, we expect it will be updated whenever `firstName` or `lastName` change. If you come from an AngularJS background you might also remember using expressions inside the template or function calls. Of course, this works the same when using render functions (with JSX or not); it doesn’t really matter.
现在如果在模板中使用 `fullName`,我们希望它能随着 `firstName` 或 `lastName` 的改变而更新。如果你有使用 AngularJS 的背景,你可能还记得在模板或者函数调用内使用表达式。当然了,使用渲染函数(不管用不用 JSX)的时候和这里是一样的;其实这无关紧要。
-Let’s consider the following example:
来看一下下面的例子:
```
-
{{ firstName + ' ' + lastName }}
-
{{ getFullName() }}
-
{{ fullName }}
```
-The result of the above code will be mostly the same. Each time `firstName` or `lastName` changes, the view will update with the headers and show the full name.
上面代码的执行结果几乎是一样的。每次 `firstName` 或 `lastName` 发生变化,视图将会更新这些 `` 并且显示出全名。
-However, what if we use the expression, method call and computed property multiple times? The expression and method call will have to be calculated each time they are accessed, whereas the computed property will be cached after the first computation until one of its dependencies change. It will also persist through the re-render cycles! That’s actually quite a nice optimization if you consider that in event-based modern user interfaces, it’s hard to predict which action the user will take first.
然而,如果多次使用表达式、函数调用和计算属性呢?使用表达式和函数调用每次都会计算一遍,而计算属性在第一次计算后将会缓存下来,直到它的依赖发生改变。它也会在重新渲染的周期中一直保持!如果考虑在基于事件模型的现代用户界面中,很难预测用户会首先执行哪项操作,那么这确实是一个最优化方案。
-## Basic computed property ##
## 基础的计算属性 ##
-In the previous article, we learned how to track and react to changes inside observable object properties by utilizing an event emitter. We know that when we change the `firstName` it will call all the handlers that subscribed to the `’firstName’` event. Thus it is quite easy to build a computed property by manually subscribing to its dependencies.
-This is actually how Ember does it:
在前面文章中,我们学习了如何通过使用事件发射器追踪和响应可观察对象属性内的改变。我们知道当改变 `firstName` 时,会调用所有的订阅了 `’firstName’` 事件的处理器。因此通过手动订阅它的依赖来构建计算属性是相当容易的。
这也是 Ember 实现计算属性的方式:
@@ -74,7 +58,6 @@ fullName: Ember.computed('firstName', 'lastName', function() {
})
```
-The drawback here is that you have to declare the dependencies yourself. Doesn’t seem like a problem until you have computed properties that are a result of a chain of more expensive, complex functions. For example:
这样做的缺点就是你不得不自己声明依赖。当你的计算属性是一串高开销的、复杂的函数的运行结果时候,你就知道这的确是个问题了。例如:
```
@@ -90,43 +73,31 @@ selectedTransformedList: Ember.computed('story', 'listA', 'listB', 'listC', func
})
```
-In the above case, even if `this.story` always equals `’A’`, the computed property will have to be re-evaluated each time one of the lists changes. Even if they are not used for the end result.
在上面的案例中,即便 `this.story` 总是等于 `’A’`,一旦 lists 发生改变,计算属性也将不得不每次都反复计算。
-## Dependency tracking ##
## 依赖追踪 ##
-Vue.js and MobX take a different approach to this problem. The difference is, you don’t have to declare the dependencies at all because they are detected automatically each time it is evaluated. Let say `this.story = ‘A’`. The detected dependencies will be:
Vue.js 和 MobX 在解决这个问题上使用了与上文不同的方法。不同在于,你根本不必声明依赖,因为在计算的时候他们会自动地检测。假定 `this.story = ‘A’`,检测到的依赖会是:
* `this.story`
* `this.listA`
-When `this.story` changes to `’B’` it will collect a fresh set of dependencies and remove the unnecessary ones (`this.listA`) that were used before but was not called anymore.
-This way even if the other lists change it won’t trigger a recalculation of `selectedTransformedList`.
-That’s smart!
当 `this.story` 变成 `’B’`,它将会收集一组新的依赖,并移除那些之前用而现在不再使用的多余的依赖(`this.listA`)。这样,尽管其他 lists 发生变化,也不会触发 `selectedTransformedList` 的重计算。真聪明!
-Now it’s a good time to look again at the [code from the previous article - JSFiddle](https://jsfiddle.net/shentao/4k0gk3bx/10/), as the following changes will be based upon that code.
现在是时候返回来看一看 [上一篇文章中的代码 - JSFiddle](https://jsfiddle.net/shentao/4k0gk3bx/10/),下面的改动将基于这些代码。
-> The code in this article is written to be as simple as possible, ignoring many sanity checks and optimizations. By no means is it production ready, as it’s the only purpose is strictly educational.
-
> 这篇文章中的代码尽量写的简单,忽略很多完整性检查和优化。绝不是已经可以用于生产环境的,仅仅用于教育目的。
-Let’s create a new data model:
我们来创建一个新的数据模型:
```
const App = Seer({
data: {
- // observable values
// 可观察的值
goodCharacter: 'Cloud Strife',
evilCharacter: 'Sephiroth',
placeholder: 'Choose your side!',
side: null,
- // computed property
// 计算属性
selectedCharacter () {
switch (this.side) {
@@ -138,7 +109,6 @@ const App = Seer({
return this.placeholder
}
},
- // computed property depending on other computed
// 依赖其他计算属性的计算属性
selectedCharacterSentenceLength () {
return this.selectedCharacter.length
@@ -147,14 +117,10 @@ const App = Seer({
})
```
-### Detecting the dependencies ###
### 检测依赖 ###
-To find out the dependencies of the currently evaluated computed property, we need a way to collect the dependencies. As you know, every observable property is already transformed into a getter and setter.
-When evaluating the computed property (function) it will access other properties, which will trigger their getters.
为了找到当前求值计算属性的依赖,需要一种收集依赖的办法。如你所知,每个可观察属性是已经转换成 getter 和 setter 的形式。当对计算属性(函数)求值的时候,需要用到其他的属性,也就是触发他们的 getters。
-For example this function:
例如这个函数:
```
@@ -165,40 +131,31 @@ For example this function:
}
```
-will call both the `firstName` and `lastName` getters.
将会调用 `firstName` 和 `lastName` 的 getters。
-Let’s make use of it!
让我们利用这一点!
-We need a way to collect the information that a getter was called when evaluating a computed property. For this to work, first we need a place to store which computed property is currently being evaluated. We can use a simple object for this:
当对计算属性求值的时候,我们需要收集 getter 被调用的信息。为了完成这项工作,首先需要空间存储当前求值的计算属性。可以用这样的简单对象:
```
let Dep = {
- // Name of the currently evaluated computed value
// 当前求值的计算属性的名字
target: null
}
```
-We used the `makeReactive` function to transform primitives into observable properties. Let’s create a transform function for computed properties and name it `makeComputed`.
我们过去曾用 `makeReactive` 函数将原始属性转换成可观察属性。现在让我们为计算属性创建一个转换函数并将它命名为 `makeComputed`。
```
function makeComputed (obj, key, computeFunc) {
Object.defineProperty(obj, key, {
get () {
- // If there is no target set
// 如果没有 target 集合
if (!Dep.target) {
- // Set the currently evaluated property as the target
// 设置 target 为当前求值的属性
Dep.target = key
}
const value = computeFunc.call(obj)
-
- // Clear the target context
// 清空 target 上下文
Dep.target = null
return value
@@ -209,31 +166,24 @@ function makeComputed (obj, key, computeFunc) {
})
}
-// It will be later called in this manner
// 后面将会用这种方式调用
makeComputed(data, 'fullName', data['fullName'])
```
-Okay. Now that the context is available, we can modify our `makeReactive` function that we created in the previous article to make use of that context.
Okay!既然上下文可以获取了,修改上一篇文章中创建的 `makeReactive` 函数以便使用获取到的上下文。
-The new `makeReactive` function should look like this:
新的 `makeReactive` 函数像下面这样:
```
function makeReactive (obj, key) {
let val = obj[key]
- // create an empty array for storing dependencies
// 创建空数组用来存依赖
let deps = []
Object.defineProperty(obj, key, {
get () {
- // Run only when called within a computed property context
// 只有在计算属性上下文中调用的时候才会执行
if (Dep.target) {
- // Add the computed property as depending on this value
- // if not yet added
// 如果还没添加,则作为依赖这个值的计算属性添加
if (!deps.includes(Dep.target)) {
deps.push(Dep.target)
@@ -243,11 +193,8 @@ function makeReactive (obj, key) {
},
set (newVal) {
val = newVal
- // If there are computed properties
- // that depend on this value
// 如果有依赖于这个值的计算属性
if (deps.length) {
- // Notify each computed property observers
// 通知每个计算属性的观察者
deps.forEach(notify)
}
@@ -257,7 +204,6 @@ function makeReactive (obj, key) {
}
```
-One last thing we need is to slightly modify our `observeData` function so that it runs `makeComputed` instead of `makeReactive` for properties that are functions.
我们要做的最后一件事就是稍稍改进 `observeData` 函数,以便对于函数形式的属性,它运行 `makeComputed` 而不是 `makeReactive`。
```
@@ -275,26 +221,20 @@ function observeData (obj) {
}
```
-And that’s basically it! We just created our own implementation of computed properties with dependency tracking.
基本上就是这样!我们刚刚通过依赖追踪创建了我们自己的计算属性实现。
-Sadly – the above, very naive implementation still lacks some crucial that can be found in Vue.js and MobX. I guess the most important of those will be caching and removing the dead dependencies. So let’s add them.
不幸的是 —— 上面的实现是非常基础的,仍然缺少 Vue.js 和 MobX 中可以找到的重要的特性。我猜最重要的就是缓存和移除废弃的依赖。所以我们把它们添上。
-## Caching ##
## 缓存 ##
-First, we need to add a place to store the cache. Let’s add our cache management to the `makeComputed` function:
首先,我们需要空间存储缓存。我们在 `makeComputed` 函数中添加缓存管理器。
```
function makeComputed (obj, key, computeFunc) {
let cache = null
- // Observe self to clear cache when deps change
// 自观察,当 deps 改变的时候清除缓存
observe(key, () => {
- // Clear cache
// 清空缓存
cache = null
})
@@ -304,11 +244,8 @@ function makeComputed (obj, key, computeFunc) {
if (!Dep.target) {
Dep.target = key
}
-
- // If not cached yet
// 当没有缓存
if (!cache) {
- // calculate new value and save to cache
// 计算新的值并存入缓存
cache = computeFunc.call(obj)
}
@@ -323,73 +260,50 @@ function makeComputed (obj, key, computeFunc) {
}
```
-That’s it! Now each time we access our computed property after the initial computation, it will return the cached value until it has to be recalculated. Pretty straightforward, right?
就是这样!现在在初始化计算后,每次读取计算属性,它都会返回缓存的值,直到不得不重新计算。相当简单,是不是?
-Thanks to the `observe` method we used inside `makeComputed` during the data transformation process, we ensure to always clean the cache before other signal handlers are executed. This means whenever one of the computed property’s dependencies change, the cache will be cleaned, just before the interface gets updated.
多亏了 `observe` 函数,在数据转换过程中我们在 `makeComputed` 内部使用,确保在其他信号处理器执行前清空缓存。这意味着,计算属性的一个依赖发生变化,缓存将被清空,刚刚好在界面更新前完成。
-## Removing the unnecessary dependencies ##
## 移除不必要的依赖 ##
-So now what’s left is to get rid of dependencies that are no longer valid. This is often a case when our computed properties conditionally depend on different values. What we want to achieve is our computed property only depending on the last used dependencies. The above implementation is flawed in that once a dependency registers that a computed property depends on it, it stays this way forever.
现在剩下的工作就是清理无效的依赖。当计算属性依赖于不同的值的时候通常是一个案例。我们想达到的效果是计算属性仅依赖最后使用到的依赖。上面的实现在这方面是有缺陷的,一旦计算属性登记了依赖于它,它就一直在那了。
-There are probably better ways to handle this, but because we want to keep things really simple let’s just create a secondary dependency list. One that will store a computed property’s dependencies.
-To sum things up, our dependency lists:
可能有更好的方式处理这种情况,但是因为我们想保持简单,我们来创建第二个依赖列表,来存储计算属性的依赖项。
总结来说,我们的依赖列表:
-- List of computed property names that depend on this value (observables or other computed) stored locally. Think: **Those are the values that depend on me.**
-- A secondary dependency list that is used to remove dead dependencies and stores the most recent dependencies of a computed property. Think: **Those are the values I’m depending on.**
-
- 依赖于这个值(可观察的或者其他的计算后的)的计算属性名列表存储在本地。可以这样想:**这些是依赖于我的值。**
- 第二个依赖列表,用来移除废弃的依赖并存储计算属性的最新的依赖。可以这样想:**这些值是我依赖的。**
-With those two lists, we can run a filter function to remove the no longer valid dependencies. So let’s start with creating an object to store a secondary dependency list and some utility functions.
用这两列表,我们可以运行一个过滤函数来移除无效的依赖。让我们首先创建一个存储第二个依赖列表的对象和一些实用的函数。
```
let Dep = {
target: null,
- // Stores the dependencies of computed properties
// 存储计算属性的依赖
subs: {},
- // Create a two-way dependency relation between computed properties
- // and other computed or observable values
// 在计算属性和其他计算后的或者可观察的值之间创建双向的依赖关系
depend (deps, dep) {
- // Add the current context (Dep.target) to local deps
- // as depending on the current property
- // if not yet added
// 如果还没添加,则添加当前上下文(Dep.target)到本地的 deps,作为依赖于当前属性
if (!deps.includes(this.target)) {
deps.push(this.target)
}
- // Add the current property as a dependency of the computed value
- // if not yet added
// 如果还没有添加,将当前属性作为计算值的依赖加入
if (!Dep.subs[this.target].includes(dep)) {
Dep.subs[this.target].push(dep)
}
},
getValidDeps (deps, key) {
- // Filter only valid dependencies by removing dead dependencies
- // that were not used during last computation
// 通过移除在上一次计算中没有使用的废弃依赖,仅仅过滤出有效的依赖
return deps.filter(dep => this.subs[dep].includes(key))
},
notifyDeps (deps) {
- // notify all existing deps
// 通知所有已存在的 deps
deps.forEach(notify)
}
}
```
+ `Dep.depend` 函数现在还看不出用处,但我们待会就会用到它。那时在这里它的用处会更清楚。
-If the `Dep.depend` method doesn’t make much sense right now, wait until we use it. It should become more clear what is actually `Dep.depend` 函数现在还看不出用处,但我们待会就会用到它。那时在这里它的用处会更清楚。
-
-First, let’s tune the `makeReactive` transform function.
首先,来调整 `makeReactive` 转换函数。
```
@@ -399,12 +313,8 @@ function makeReactive (obj, key, computeFunc) {
Object.defineProperty(obj, key, {
get () {
- // Run only when getting within a computed value context
// 只有当在计算值的上下文内时才执行
if (Dep.target) {
- // Add Dep.target as depending on this value
- // this will mutate the deps Array
- // as we’re passing a reference to it
// 将 Dep.target 作为依赖于这个值添加,浙江使 deps 数组发生变化,因为我们给它传了一个引用
Dep.depend(deps, key)
}
@@ -413,11 +323,8 @@ function makeReactive (obj, key, computeFunc) {
},
set (newVal) {
val = newVal
-
- // Clean up dead dependencies
// 清除废弃依赖
deps = Dep.getValidDeps(deps, key)
- // and notify valid deps
// 并通知有效的 deps
Dep.notifyDeps(deps, key)
@@ -427,21 +334,16 @@ function makeReactive (obj, key, computeFunc) {
}
```
-Almost the same has to be changed inside the `makeComputed` transform function. The difference is that we won’t be using the setter but the signal handler callback we passed to the `observe` function. Why? Because this callback will be called whenever the actual computed value has to update, as its dependencies have changed.
-
`makeComputed` 转换函数内部也需要做相似的改动。不同在于不使用 setter 而是用传给 `observe` 函数的信号回调处理器。为什么?因为这个回调无论何时计算的值更新了,也就是依赖改变了,都会被调用。
```
function makeComputed (obj, key, computeFunc) {
let cache = null
- // Create a local deps list similar to makeReactive deps
// 创建一个本地的 deps 列表,相似于 makeReactive 的 deps
let deps = []
observe(key, () => {
cache = null
-
- // Clean up and notify valid deps
// 清空并通知有效的 deps
deps = Dep.getValidDeps(deps, key)
Dep.notifyDeps(deps, key)
@@ -449,29 +351,20 @@ function makeComputed (obj, key, computeFunc) {
Object.defineProperty(obj, key, {
get () {
- // If evaluated during the evaluation of
- // another computed property
// 如果如果在其他计算属性正在计算的时候计算
if (Dep.target) {
- // Create a dependency relationship
- // between those two computed properties
// 在这两个计算属性之间创建一个依赖关系
Dep.depend(deps, key)
}
- // Normalize Dep.target back to self
- // This makes it possible to build a dependency tree
- // instead of a flat structure
// 将 Dep.target 标准化成它原本的样子,这使得构建一个依赖树成为可能,而不是一个扁平化的结构
Dep.target = key
if (!cache) {
- // Clear dependencies list to ensure getting a fresh one
// 清空依赖列表以获得一个新的列表
Dep.subs[key] = []
cache = computeFunc.call(obj)
}
- // Clear the target context
// 清空目标上下文
Dep.target = null
return cache
@@ -483,30 +376,22 @@ function makeComputed (obj, key, computeFunc) {
}
```
-Done! You might have already noticed that it also enables computed properties to be dependent on other computed properties, without having to know about the observables that lie underneath. Pretty sweet, isn’t it?
完成了!你可能已经注意到,它允许计算属性依赖于其他计算属性,不需要知道背后的可观察的对象。相当不错,是不?
-## Asynchronous pitfalls ##
## 异步陷阱 ##
-Now that you know how dependency tracking works, it should be quite obvious why it’s not possible to track asynchronous data inside computed properties both in MobX and Vue.js. It all breaks because even a `setTimeout(callback, 0)` will be called out of the current context where `Dep.target` no longer exists. This means that whatever happens inside the callback won’t be tracked.
既然你知道了依赖追踪如何工作,在 MobX 和 Vue.js 中不能追踪计算属性种的异步数据的原因就很明显了。这一切会被打破,因为即使 `setTimeout(callback, 0)` 将会在当前上下文外被调用,在那里 `Dep.target` 不在存在。这也就意味着在回调函数中无论发生什么都不会被追踪到。
-## Bonus: Watchers ##
## 红利:Watchers ##
-The above problem can be, however, partially tackled with watchers. You might know them from Vue.js. Building watchers on top of what we already have is actually really simple. After all, a watcher is just a signal handler called after a given value has changed.
然而,上面的问题可以通过 watchers 部分解决。你可能已经在 Vue.js 中了解过它们。在我们已有的基础上创建 watchers 真的很容易。毕竟,watcher 是一个给定值发生变化时调用的信号处理器。
-We just have to add a watchers registration method and trigger it within our Seer function.
我们只是不得不添加一个 watchers 注册方法并在 Seer 函数内触发它。
```
function subscribeWatchers(watchers, context) {
for (let key in watchers) {
if (watchers.hasOwnProperty(key)) {
- // We use Function.prototype.bind to bind our data model
- // as the new `this` context for our signal handler
// 使用 Function.prototype.bind 来绑定数据模型,作为我们信号处理器新的 `this` 上下文
observe(key, watchers[key].bind(context))
}
@@ -516,7 +401,6 @@ function subscribeWatchers(watchers, context) {
subscribeWatchers(config.watch, config.data)
```
-That’s all! We can use it like this:
这就是全部了,可以像这样用它:
```
@@ -524,13 +408,10 @@ const App = Seer({
data: {
goodCharacter: 'Cloud Strife'
},
- // here we can declare watchers
// 这里可以忽略 watchers
watch: {
- // watch for 'goodCharacter' changes
// 'goodCharacter' 改变时的 watch
goodCharacter () {
- // log the value into the console
// 在控制台输出值
console.log(this.goodCharacter)
}
@@ -539,30 +420,26 @@ const App = Seer({
```
-The complete code is available here:
完整的代码可以在下面获得:
[https://github.com/shentao/seer/tree/cached-computed](https://github.com/shentao/seer/tree/cached-computed)
-You can play with it online here (Opera/Chrome only):
你可以在线的试玩(仅支持 Opera/Chrome):
[https://jsfiddle.net/oyw72Lyy/](https://jsfiddle.net/oyw72Lyy/)
-## Summary ##
## 总结 ##
-I hope you enjoyed the tutorial and that the provided explanation turned out to be good enough to bring some light into what might be happening inside Vue or MobX when using computed properties. Keep in mind that the provided implementation was meant to be quite naive and not on par with the mentioned libraries. It is also not production ready in any way.
我希望你们喜欢这个教程,当使用计算属性的时候,希望我的解释很好的阐明了 Vue 或 MobX 内部的原理。记住本文提供的实现是相当基础的,和提到的库中的实现不是同等水平的。无论如何都不是可以直接用于生产环境的。
-## What comes next? ##
## 接下来讲什么? ##
-The 3rd part should include support for nested properties and observing arrays. I might also finally add a way to unsubscribe from the event bus! :D
-As for the 4th part – maybe streams? Would you be interested?
第三部分涵盖了对嵌套属性和可观察数组的支持,我也可能在最后添加从事件中取消订阅的办法! :D
至于第四部分,也许是数据流?你们感兴趣吗?
-Feel free to leave your feedback in the comments!
欢迎在评论区随意反馈意见!
-Thanks for reading!
感谢阅读!
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
+
From 9825a2c1d000050cab41e811a16c145efaaadcb0 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E7=8E=8B=E6=96=87=E6=98=8E?=
Date: Thu, 30 Mar 2017 20:48:52 +0800
Subject: [PATCH 033/945] trans
---
...a-bubble-selection-animation-on-android.md | 130 +++++++++---------
1 file changed, 62 insertions(+), 68 deletions(-)
diff --git a/TODO/how-to-create-a-bubble-selection-animation-on-android.md b/TODO/how-to-create-a-bubble-selection-animation-on-android.md
index ee2661264b1..c5c56b98d7e 100644
--- a/TODO/how-to-create-a-bubble-selection-animation-on-android.md
+++ b/TODO/how-to-create-a-bubble-selection-animation-on-android.md
@@ -1,66 +1,58 @@
> * 原文地址:[How to Create a Bubble Selection Animation on Android](https://medium.com/@igalata13/how-to-create-a-bubble-selection-animation-on-android-627044da4854#.7iwkfupy7)
> * 原文作者:[Irina Galata](https://medium.com/@igalata13?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[skyar2009](https://github.com/skyar2009)
> * 校对者:
-# How to Create a Bubble Selection Animation on Android #
# Android 如何实现气泡选择动画 #
-*Authors:* [*Irina Galata*](https://github.com/igalata) *, Android Developer;* [*Yulia Serbenenko*](https://dribbble.com/yuyonder) *, UI/UX designer.*
**作者:[Irina Galata](https://github.com/igalata) Android 开发者;[Yulia Serbenenko](https://dribbble.com/yuyonder) UI/UX 设计师**
-There is a growing trend for unifying user experience across platforms: in early days iOS and Android had their own unique feel, but recently they have been growing closer together in the way applications are designed and interactions happen.
跨平台用户体验统一正处于增长趋势:早些时候 iOS 和安卓有着不同的体验,但是最近在应用设计以及交互方面变得越来越接近。
-From [bottom navigation](https://material.io/guidelines/components/bottom-navigation.html#) to split screen feature available in Nougat Android, there is a lot of common between two platforms these days. For designers it means that often we can adjust popular features that were once associated with one platform to apps designed for another one. As for developers, it is a great chance to improve and refine their technical skills.
从安卓 Nougat 的[底部导航](https://material.io/guidelines/components/bottom-navigation.html#)到分屏特性,两个平台间有了许多相同之处。对设计师而言,我们可以将主流功能设计成两个平台一致(过去需要单独设计)。对开发者而言,这是一个提高、改进开发经验的好机会。
-So we decided to create the component with the bubble based interface for Android, drawing our inspiration from selection bubbles in [Apple music](http://www.apple.com/lae/apple-music/) .
所以我们决定开发一个安卓气泡选择的组件库 —— 灵感来自于[苹果音乐](http://www.apple.com/lae/apple-music/)的气泡选择。
-### ***Put design first*** ###
### **先说设计** ###
-Our Bubble Picker is an example of the animation that is equally appealing to different groups of users. Bubbles summarize information into convenient UI elements that are easy to understand and also visually consistent. It makes the interface simple enough for novice users and still feels interesting for experienced ones.
-我们的气泡选择动画是一个好的范例,它对不同的用户群体有着同样的吸引力。气泡以方便的 UI 元素汇总信息,通俗易懂并且视觉一致。它会让交互对新手足够简单并让经验用户依然感兴趣。
+我们的气泡选择动画是一个好的范例,它对不同的用户群体有着同样的吸引力。气泡以方便的 UI 元素汇总信息,通俗易懂并且视觉一致。它让交互对新手足够简单的同时还能吸引老司机兴趣。
-This type of animation is very helpful for apps rich in content, where users have to make a choice from a list of options. For example, in our component we used bubbles to hold names of potential destinations for a travel app. Bubbles float freely, and when a user taps on one of them, the chosen bubble grows in size. It gives users a meaningful feedback on their actions and enhances the sense of direct manipulation.
这种动画类型对丰富应用的内容由很大帮助,主要体现在用户要从一系列选项中进行选择的地方。例如,我们使用气泡来选择旅游应用中潜在目的地名字。气泡自由的浮动,当用户点击一个气泡时,选中的气泡会变大。这给用户很深刻的反馈并增强操作的直观感受。
-The component is pretty with a white theme, lots of bright colors and photographs throughout. Moreover, I decided to experiment with gradients in order to add more depth and volume. Gradients might be the major visual in the display and might attract the attention of new visitors.
组件使用白色主题,布满明亮的颜色和图片十分漂亮。此外,我决定试验渐变来增加深度和体积。渐变可能是主要的显示视觉,会吸引新用户的注意。
-Bubble Picker’s gradients
+气泡选择的渐变
-We provide developers with an opportunity to customize all UI elements, therefore our component will suit any app.
+我们允许开发者自定义所有的 UI 元素,所以我们的组件适合任意的应用。
-### **Review developer’s challenges** ###
+### **再看开发者的挑战** ###
-The first issue I’ve faced when I decided to implement this animation was to choose tools for the development. It was clear for me that rendering of such a fast animation on Canvas wouldn’t be efficient enough, so decided to use OpenGL (Open Graphics Library). It’s a cross-platform application programming interface for 2D and 3D graphics rendering. Fortunately, Android supports some versions of the OpenGL.
+当我决定实现这个动画时,我面临的第一个问题就是使用什么工具开发。我清楚知道绘制如此快速的动画在 Canvas 上绘制的效率是不够的,所以决定使用 OpenGL (Open Graphics Library)。OpenGL 是一个跨平台的 2D 和 3D 图形绘制应用开发接口。幸运地是,Android 支持部分版本的 OpenGL。
-I needed to make circles move in a natural way, just like gas bubbles do in a glass of fizzy drink. There are plenty of physics engines available for Android, but I had some requirements for them and this fact made it significantly more difficult to make a choice. My requirements were the following: the engine should be lightweight and easily embeddable in the Android library. Most engines are developed for games and they required adapting the project structure to them. After some research I found JBox2D (Java port of the Box2D engine written in C++), and since our animation isn’t supposed to be used with a great number of physical bodies (e.g 200 or more), it was enough to use a Java port but not the original engine.
+我需要圆自然的运动,就像碳酸饮料中的气泡那样。对 Android 来说有许多可用的物理引擎,同时我又有一些特定需要,使得选择变得更加困难。我的需求是:引擎要轻量级并且方便嵌入 Android 库。多数的引擎是为游戏开发的,并且它们需要调整工程结构来适应它们。功夫不负有心人,我最终找到了 JBox2D(C++ 引擎 Box2D 的 Java 版),因为我们的动画不需要支持大量的物理实体(例如 200+),使用非原版的 Java 版引擎已经足够了。
-Also further in this article I’ll explain my choice of the programming language (Kotlin) and what advantages it has in my opinion. To find out more about the difference between Java and Kotlin review my previous [article](https://yalantis.com/blog/kotlin-vs-java-syntax/)
+此外,本文后面我会解释我为什么选择 Kotlin 语言开发,以及这样做的好处。需要了解 Java 和 Kotlin 更多不同之处可以阅读我之前的[文章](https://yalantis.com/blog/kotlin-vs-java-syntax/)。
-**How to create shaders?**
+**如何创建着色器?**
-Firstly, it’s important to understand that the building block in OpenGL is a triangle, since it’s the simplest shape that can approximate other shapes. So any shape that you create will consist of 1 or more triangles. To implement our animation I used two combined triangles for every body, so it looks like a square, where I can draw the circle.
+首先,我们需要理解 OpenGL 中的基础构件三角形,因为它是和其它形状类似且最简单的形状。所以你绘制的任意图形都是由一个或多个三角形组成。在动画实现中,我使用两个关联的三角形代表一个实体,所以我画圆的地方像一个正方形。
-To render created shape you need to write at least two shaders — vertex shader and fragment shader. Their difference is described by their names. Vertex shader will be executed for each vertex of each triangle, and fragment shader will be executed for every pixel-sized part of the triangles.
+绘制一个形状至少需要两个着色器 —— 顶点着色器和片段着色器。通过名字就可以区分他们的用途。顶点着色器负责绘制每个三角形的顶点,片段着色器负责绘制三角形中每个像素。
-Fragments and vertices of the triangle
-Vertex shaders are used to control transformations of the shape (e.g scaling, position, rotation), while fragment shaders are responsible for the color of the sample.
+三角形的片段和顶点
+
+顶点着色器负责控制图形的变化(例如:大小、位置、旋转),片段着色器负责形状的颜色。
```
// language=GLSL
@@ -78,7 +70,7 @@ val vertexShader = """
```
-Vertex shader
+顶点着色器
```
// language=GLSL
@@ -96,49 +88,51 @@ val fragmentShader = """
```
-Fragment shader
-Shaders are written in GLSL (OpenGL Shading Language) and are required to be compiled at runtime. If you code in Java the most convenient way is to write your shaders in separate file and retrieve them using input stream. As you see Kotlin lets developers create shaders in classes in easier way. You can put any multiline code in triple quotes `"""`.
+片段着色器
+
+着色器使用 GLSL(OpenGL 着色语言) 编写,需要运行时编译。如果项目使用的是 Java,那么最方便的方式是在另一个文件编写你的着色器,然后使用输入流读取。如上述示例代码所示,Kotlin 可以简单地在类中创建着色器。你可以在 `"""` 中间添加任意的 GLSL 代码。
-In GLSL there are several types of variables:
+GLSL 中有许多类型的变量:
-- value of the `uniform` variableisthe same for all vertices and fragments
-- `attribute` variable is different for each vertex
-- the `varying` variable is used to pass data from vertex shader to fragment shader and its value will be linearly interpolated for each fragment
+- 顶点和片段的 `uniform` 变量的值是相同的
+- 每个顶点的 `attribute` 变量是不同的
+- `varying` 变量负责从顶点着色器向片段着色器传递数据,它的值由片段线性地插入。
-`u_Matrix` variable contains the translation matrix with `x` and `y`, which should be added to the initial position of the circle, and obviously its value should be equal for all vertices of the shape and the type of this variable is `uniform`, while the position of the vertices will differ, so `a_Position` variable is `attribute` *.* `a_UV`variable is needed for two purposes:
+`u_Matrix` 变量包含由圆初始化位置的 `x` 和 `y` 构成的变化矩阵,显然它的值对图形的所有顶点拉说都是相同的,类型为 `uniform`,然而顶点的位置是不同的,所以 `a_Position` 变量是 `attribute` 类型。`a_UV` 变量有两个用途:
-1. To find out the distance between current fragment and the center of the square. Depending on this distance I can change the color of the fragment to draw a circle.
-2. To properly place the texture(the photo and the name of the country) in the center of a shape.
+1. 确定当前片段和正方形中心位置的距离。根据这个距离,我可以调整片段的颜色而实现画圆。
+2. 正确地将文理(照片和国家的名字)置于图形的中心位置。
-The center of the circle
+圆的中心
-`a_UV` contains `x` and `y` values which are different for each vertex and lie between 0 and 1. In the vertex shader I just pass the value of the `a_UV` to `v_UV` variable, so the second one could be interpolated for every fragment. And as a result the `v_UV` variable of a fragment in the center of a shape will contain the [0.5, 0.5] value. To find out the distance I used `distance()` method, which receives two points.
+`a_UV` 包含 `x` 和 `y`,它们的值每个顶点都不同,取值范围是 0 ~ 1。我只给顶点着色器 `a_UV` 和 `v_UV` 两个入参,因此每个片段都可以插入 `v_UV`。并且对于片段中心点的 `v_UV` 值为 [0.5, 0.5]。我使用 `distance()` 方法计算两个点的距离。
-**Using** `smoothstep` **to draw antialiased circles**
+**使用** `smoothstep` **绘制平滑的圆**
-Initially my fragment shader looked a bit different:
+起初片段着色器看上去不太一样:
`gl_FragColor = distance < 0.5 ? texture2D(u_Text, v_UV) : u_BgColor;`
-I changed the fragment color depending on the distance from the center without any antialiasing. And the result was not so impressive — the edges of the circles were notched.
+我根据点到中心的距离调整片段的颜色,没有采取抗锯齿手段。当然结果差强人意 —— 圆的边是凹凸不平的。
-Not antialiased circles
-So the `smoothstep` function was the solution. It smoothly interpolates from 0 to 1 based on `distance` compared to the start and end point of the transition between the texture and the background. Thus the alpha of the texture on the distance from 0 to 0.49 is 1, on the 0.5 and above it is 0, and between the 0.49 and 0.5 it is interpolated, so the edges of the circles would be antialiased.
+有锯齿的圆
+
+解决方案是 `smoothstep`。它根据到纹理与背景的变换起始点的距离平滑的从0到1变化。因此距离 0 到 0.49 时纹理的透明度为 1,大于等于 0.5 时为 0,0.49 和 0.5 之间时平滑变化,如此圆的边就平滑了。
-Antialiased circles
+无锯齿圆
-**How to use textures to display the images and text in OpenGL?**
+**OpenGL 中如何使用 texture 显示图像和文本?**
-Every circle in this animation can have two states — normal and selected. In the normal state the texture of a circle contains text and color, in the selected state it also contains an image. So for every circle I needed to create two different textures.
+在动画中圆有两种状态 —— 普通和选中。在普通状态下圆的纹理包含文字和颜色,在选中状态下同时包含图像。因此我需要为每个圆创建两个不同的纹理。
-To create the texture I use a Bitmap instance where I draw all the elements and bind the texture.
+我使用 Bitmap 实例来创建纹理,绘制所有元素。
```
fun bindTextures(textureIds: IntArray, index: Int) {
@@ -180,11 +174,11 @@ fun bindTextures(textureIds: IntArray, index: Int) {
}
```
-And after that I pass the texture unit to the `u_Text` variable. And to get the actual color of a fragment I use `texture2D()` method which receives the texture unit and the position of the fragment respective to its vertices.
+之后我将纹理单元赋值给 `u_Text` 变量。我使用 `texture2()` 方法获取片段的真实颜色,`texture2()` 接收纹理单元和片段顶点的位置两个参数。
-**Using JBox2D to make the bubbles move**
+**使用 JBox2D 让气泡动起来**
-The animation is pretty simple when talking about the physics. The main object is a `World` instance. All the bodies must be created using the world.
+关于动画的物理特性十分的简单。主要的对象是 `World` 实例,所有的实体创建都需要它。
```
class CircleBody(world: World, var position: Vec2, var radius: Float, var increasedRadius: Float) {
@@ -224,11 +218,11 @@ class CircleBody(world: World, var position: Vec2, var radius: Float, var increa
```
-As you see it’s easy to create the body: you need to specify the body type (e.g dynamic, static, kinematic), its position, radius, shape, density and fixture.
+如你所见创建实体很简单:需要指定实体的类型(例如:动态、静态、运动学)、位置、半径、形状、密度以及运动。
-Every time the surface is drawing, it’s necessary to call `step()` method of the `World` instance to move all the bodies. After that you can draw all shapes at their new positions.
+每次画面绘制,都需要调用 `World` 的 `step()` 方法移动所有的实体。之后你可以在图形的新位置进行绘制。
-The issue I’ve faced is that world can have a gravity only as a direction, but not a point. JBox2D doesn’t support orbital gravity. As a result I couldn’t move the circles to the center of the screen. So I had to implement gravitation by myself.
+我遇到的问题是 `World` 的重力只能是一个方向,而不能是一个点。JBox2D 不支持轨道重力。因此将圆移动到屏幕中心是无法实现的,所以我只能自己来实现引力。
```
private val currentGravity: Float
@@ -248,18 +242,18 @@ private fun move(body: CircleBody) {
-Engine
+引擎
-Gravitation challenge
+引力挑战
-So every time the world moves I calculate the appropriate force and apply it to each body and it looks like the circles are affected by gravitation.
+每次发生移动时,我计算出力的大小并作用于每个实体,看上去就像圆受引力作用在移动。
-**Detecting user’s touches in GlSurfaceView**
+**GlSurfaceView 中检测用户触摸事件**
-`GLSurfaceView` like any other Android view can react to user’s touch.
+`GLSurfaceView` 和其它的 Android view 一样可以响应用户的点击事件。
```
override fun onTouchEvent(event: MotionEvent): Boolean {
@@ -299,7 +293,7 @@ private fun isSwipe(event: MotionEvent) = Math.abs(event.x - previousX) > 20 &&
-So the `GLSurfaceView` intercepts all the touches, and its renderer handles all of them.
+`GLSurfaceView` 拦截所有的点击,并用渲染器进行处理。
```
fun swipe(x: Float, y: Float) = Engine.swipe(x.convert(glView.width, scaleX),
@@ -311,7 +305,7 @@ fun Float.convert(size: Int, scale: Float) = (2f * (this / size.toFloat()) - 1f)
```
-Renderer
+渲染器
```
fun swipe(x: Float, y: Float) {
@@ -325,13 +319,13 @@ fun release() {
}
```
-Engine
+引擎
-When a user swipes the screen, I change the gravity center to the position of the user’s touch, so for the users it looks like they control the movements of the bubbles. And when users stop swiping I return the bubbles to their initial state.
+用户点击屏幕时,我将重力中心设为用户点击点,这样看起来就像用户在控制气泡的移动。用户停止移动后我会将气泡恢复到初始位置。
-**Finding the bubble by the coordinates of the user’s touches**
+**根据用户点击坐标查找气泡**
-When user clicks on the circle, I receive the touch position on the screen in `onTouchEvent()` method. But I also need to find the clicked circle in the coordinate system of the OpenGL. By default the center of the `GLSurfaceView` has [0, 0] position, and the `x` and `y` values lie between -1 and 1. So I also have to consider the ratio of the screen sides.
+当用户点击圆时,我从 `onTouchEvent()` 方法获取屏幕点击点。但是我也需要找到 OpenGL 坐标系中点击的圆。`GLSurfaceView` 的默认中心位置坐标为 [0, 0],`x` `y` 取值范围为 -1 到 1。所以我需要考虑屏幕的比例。
```
private fun getItem(position: Vec2) = position.let {
@@ -341,19 +335,19 @@ private fun getItem(position: Vec2) = position.let {
}
```
-Renderer
+渲染器
-And when I find the selected circle, I change its radius and texture.
+当找到选择的圆后,我会修改它的半径和纹理。
-### Feel free to use it in your projects! ###
+### 你可以随机的使用本组件! ###
-Our vibrant component is a great way to make an app more content-focused, original and fun.
+我们的组件可以让应用更聚焦内容、原始以及充满乐趣。
-**Check Bubble Picker on** [**GitHub**](https://github.com/igalata/Bubble-Picker) , [**Google Play**](https://play.google.com/store/apps/details?id=com.igalata.bubblepickerdemo) **and** [**Dribbble**](https://dribbble.com/shots/3349372-Bubble-Picker-Open-Source-Component) **.**
+**以下途径可以获取 Bubble Picker :** [**GitHub**](https://github.com/igalata/Bubble-Picker) , [**Google Play**](https://play.google.com/store/apps/details?id=com.igalata.bubblepickerdemo) **以及** [**Dribbble**](https://dribbble.com/shots/3349372-Bubble-Picker-Open-Source-Component) **。**
-It is just the first version of the component, but we surely plan to develop it further. We would like to give developers a possibility to customize physical behavior of the bubbles and specify url to add the image to the animation. In addition, we plan to add some new features (e.g removing of bubbles).
+这只是组件的第一个版本,但我们肯定会有后续的迭代。我们将支持自定义气泡的物理特性和通过 url 添加动画的图像。此外,我们还计划添加一些新特性(例如:移除气泡)。
-Don’t hesitate to send us your experiments, we are curious to see how you use our Bubble Picker. And do let us know if you have any questions or suggestion regarding the animation.
+不要犹豫把您的实验发给我们,我们非常想知道您是怎样使用 Bublle Picker 的。如果您有任何问题或者建议,欢迎随时联系我们。
-We are going to publish more awesome things soon. Stay tuned!
+我们将会继续发布一些炫酷的东西。敬请期待!
\ No newline at end of file
From 3a2479ad2bf970e00221fae45c488ad9ccd7e967 Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Thu, 30 Mar 2017 22:01:05 +0800
Subject: [PATCH 034/945] creat article modules-vs-microservices.md
---
TODO/modules-vs-microservices.md | 84 ++++++++++++++++++++++++++++++++
1 file changed, 84 insertions(+)
create mode 100644 TODO/modules-vs-microservices.md
diff --git a/TODO/modules-vs-microservices.md b/TODO/modules-vs-microservices.md
new file mode 100644
index 00000000000..b513f021efa
--- /dev/null
+++ b/TODO/modules-vs-microservices.md
@@ -0,0 +1,84 @@
+> * 原文地址:[Modules vs. microservices](https://www.oreilly.com/ideas/modules-vs-microservices)
+> * 原文作者:[Sander Mak](https://www.oreilly.com/people/sander_mak)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+# Modules vs. microservices
+
+Apply modular system design principles while avoiding the operational complexity of microservices.
+
+![](https://d3tdunqjn7n0wj.cloudfront.net/360x240/container-227877_1920-0db52b796e6b80d98f6df2d01a6ee4fb.jpg)
+
+Much has been said about moving from monoliths to microservices. Besides rolling off the tongue nicely, it also seems like a no-brainer to chop up a monolith into microservices. But is this approach really the best choice for your organization? It’s true that there are many drawbacks to maintaining a messy monolithic application. But there is a compelling alternative which is often overlooked: modular application development. In this article, we'll explore what this alternative entails and show how it relates to building microservices.
+
+## Microservices for modularity
+
+"With microservices we can finally have teams work independently", or "our monolith is too complex, which slows us down." These expressions are just a few of the many reasons that lead development teams down the path of microservices. Another one is the need for scalability and resilience. What developers collectively seem to be yearning for is a modular approach to system design and development. Modularity in software development can be boiled down into three guiding principles:
+
+[![](https://cdn.oreillystatic.com/oreilly/email/programming-newsletter-20160205.jpg)](![](https://cdn.oreillystatic.com/oreilly/email/programming-newsletter-20160205.jpg))
+
+- **Strong encapsulation**: hide implementation details inside components, leading to low coupling between different parts. Teams can work in isolation on decoupled parts of the system.
+- **Well-defined interfaces**: you can't hide everything (or else your system won't do anything meaningful), so well-defined and stable APIs between components are a must. A component can be replaced by any implementation that conforms to the interface specification.
+- **Explicit dependencies**: having a modular system means distinct components must work together. You'd better have a good way of expressing (and verifying) their relationships.
+
+Many of these principles can be realized with microservices. A microservice can be implemented in any way, as long as it exposes a well-defined interface (oftentimes a REST API) for other services. Its implementation details are internal to the service, and can change without system-wide impact or coordination. Dependencies between microservices are typically not quite explicit at development-time, leading to possible service orchestration failures at run-time. Let's just say this last modularity principle could use some love in most microservice architectures.
+
+So, microservices realize important modularity principles, leading to tangible benefits:
+
+-
+Teams can work and scale independently.
+-
+Microservices are small and focused, reducing complexity.
+-
+Services can be internally changed or replaced without global impact.
+
+What's not to like? Well, along the way you've gone from a single (albeit slightly obese) application to a distributed system of microservices. This brings an enormous amount of operational complexity to the table. Suddenly, you need to continuously deploy many different (possibly containerized) services. New concerns arise: service discovery, distributed logging, tracing and so on. You are now even more prone to the [fallacies of distributed computing](https://en.wikipedia.org/wiki/Fallacies_of_distributed_computing). Versioning of interfaces and configuration management become a major concern. The list goes on and on.
+
+It turns out there is as much complexity in the connections between microservices as there is in the combined business logic of all individual microservices. And to get here, you can't just take your monolith and chop it up. Whereas 'spaghetti code' in monolithic codebases is problematic, putting a network boundary in between escalates these entanglement issues to downright painful.
+
+## The modular alternative
+
+Does this mean we are either relegated to the messy monolith, or must drown in the complexity of microservice madness? Modularity can be achieved by other means as well. What's essential is that we can effectively draw and enforce boundaries during development. But we can achieve this by creating a well-structured monolith as well. Of course, that means embracing any help we can get from the programming language and development tooling to enforce the principles of modularity.
+
+In Java, for example, there are several module systems that can help in structuring an application. OSGi is the most well-known one, but with the release of Java 9 a native module system is added to the Java platform itself. Modules are now part of the language and platform as a first-class construct. Java modules can express dependencies on other modules, and publicly export interfaces while strongly encapsulating implementation classes. Even the Java platform itself (an enormous codebase) has been modularized using the new Java module system. You can learn more about modular development with Java 9 in my forthcoming book, [Java 9 Modularity](https://www.safaribooksonline.com/library/view/java-9-modularity/9781491954157/?utm_source=newsite&utm_medium=content&utm_campaign=lgen&utm_content=modules-vs-microservices-inline), now available in early release.
+
+[![](https://d3ansictanv2wj.cloudfront.net/software-architecture-16-cta-1d5409f91b486d57eed2d816223fa119.jpg)](https://conferences.oreilly.com/software-architecture/sa-ny?intcmp=il-prog-confreg-article-sany17_new_site)
+
+Other languages offer similar mechanisms. For instance, JavaScript got a [module system](http://exploringjs.com/es6/ch_modules.html) as of ES2015. Before that, Node.js already offered a non-standard module system for JavaScript back-ends. However, as a dynamic language, JavaScript has weaker support for enforcing interfaces (types) and encapsulation between modules. You can consider using TypeScript on top of JavaScript to get back this advantage again. Microsoft's .Net Framework does have strong typing like Java, but it doesn't have a direct equivalent to Java's upcoming module system in terms of strong encapsulation and explicit dependencies between assemblies. Still, a good modular architecture can be achieved by using Inversion-of-Control patterns which are standardized in [.Net Core](https://msdn.microsoft.com/en-us/magazine/mt707534.aspx) and by creating logically related assemblies. Even C++ is [considering the addition](http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2016/n4610.pdf) of a module system in a future revision. Many languages are gaining appreciation for modularization, which is in itself a striking development.
+
+When you make a conscious effort to use the modularity features of your development platform, you can achieve the same modularity benefits that we ascribed to microservices earlier. Essentially, the better the module system, the more help you get during development. Different teams can work on different parts, where only the well-defined interfaces are touch points between the teams. Still, at deployment time the modules come together in a single deployment unit. This way you can prevent the substantial complexity and costs associated with moving to microservices development and management. True, this means you can't build each module on a different tech-stack. But is your organization really ready for that anyway?
+
+## Designing modules
+
+Creating good modules requires the same design rigor as creating good microservices. A module should model (part of) a single bounded context of the domain. Choosing microservice boundaries is an architecturally significant decision with costly ramifications when done wrong. Module boundaries in a modular application are easier to change. Refactoring across modules is typically supported by the type-system and the compiler. Redrawing microservice boundaries involves a lot of inter-personal communication to not let things blow up at run-time. And be honest, how often do you get your boundaries right the first time, or even the second?
+
+In many ways, modules in statically typed languages offer better constructs for well-defined interfaces. Calling a method through a typed interface exposed by another module is much more robust against changes than calling a REST endpoint on another microservice. REST+JSON is ubiquitous, but it is not the hallmark of well-typed interoperability in the absence of (compiler-checked) schemas. Add in the fact that traversing the network including (de)serialization still isn't free, and the picture becomes even bleaker. What's more, many module systems allow you to express your dependencies on other modules. When these dependencies are violated, the module system will not allow it. Dependencies between microservices only materialize at run-time, leading to hard to debug systems.
+
+Modules are natural units for code-ownership as well. Teams can be responsible for one or more modules in the system. The only thing shared with other teams is the public API of their modules. At run-time, there's less isolation between modules in comparison with microservices. Everything still runs in the same process, after all.
+
+[![](https://d3ansictanv2wj.cloudfront.net/safari-topic-cta-1f60e6f96856da19ba3cb25660472ca5.jpg)](https://www.safaribooksonline.com/home/?utm_source=newsite&utm_medium=content&utm_campaign=lgen&utm_content=software-architecture-post-safari-right-rail-cta)
+
+There's no reason why a module in a monolith can't own its data just like a good microservice does. Sharing within the modular application then happens through well-defined interfaces or messages between modules, not through a shared datastore. The big difference with microservices is that everything happens in-process. Eventual consistency concerns should not be underestimated. With modules, eventual consistency can be a deliberate, strategic choice. Or, you can just 'logically' separate data while storing them in the same datastore and still use cross-domain transactions for the time being. For microservices, there is no choice: eventual consistency is a given and you need to adapt.
+
+## When are microservices right for your organization?
+
+So when should you turn to microservices? Until now, we've mainly focused on tackling complexity through modularity. For that, both microservices and modular applications will do. But there are different challenges besides the ones addressed so far.
+
+When your organization is at the scale of Google or Netflix, it makes complete sense to embrace microservices. You have the capacity to build your own platform and toolkits, and the number of engineers prohibits any reasonable monolithic approach. But most organizations don't operate at this scale. Even if you think your organization will become a billion-dollar unicorn one day, starting out with a modularized monolith won't do much harm.
+
+Another good reason to spin up separate microservices is if different services are inherently better suited to different technology stacks. Then again, you must have the scale to attract talent across these disparate stacks and keep those platforms up and running.
+
+Microservices also enable independent deployment of different parts of the system, something that is harder (or even impossible) in most modular platforms. Isolated deployments add to the resilience and fault-tolerance of the system. Furthermore, the scaling characteristics may be different for each microservice. Different microservices can be deployed to matching hardware. The modularized monolith can be scaled horizontally as well, but you scale out all modules together. That may not always work out for the best, though in practice, you can get quite far with this approach.
+
+## Conclusion
+
+As always, the best option is finding a middle-ground. There's a place for both approaches, and which is best really depends on the environment, organisation and the application itself. Why not start with a modular application? You can always choose to move to microservices later. Then, instead of having to surgically untangle your monolith, you have sensible module boundaries cut out already. It's not even an exclusive choice: you can also use modules to structure microservices internally. The question then becomes, why do microservices have to be 'micro'?
+
+Even if you do depart from a single modularized application, your services don't have to be tiny to be maintainable. Again, applying the principles of modularity within services allows them to scale in complexity beyond what you'd normally ascribe to microservices. There's a place for both modules and microservices in this picture. Real cost-savings can be achieved by reducing the number of services in your architecture. Modules can help structure and scale services just as they can help structure a single monolithic application.
+
+If you're after the benefits of modularity, make sure you don't trick yourself into a microservices-only mindset. Explore the in-process modularity features or frameworks of your favorite technology stack. You’ll get support to enforce modular design, instead of having to just rely on conventions to avoid spaghetti code. Then, make a deliberate choice whether you want to incur the complexity penalty of microservices. Sometimes you just have to, but often, you can find a better way forward.
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 25b93838b268f4ba80d9fd573ff5a3ab465baf4c Mon Sep 17 00:00:00 2001
From: zhouzihanntu
Date: Thu, 30 Mar 2017 22:13:50 +0800
Subject: [PATCH 035/945] update file
---
...-in-node-js-everything-you-need-to-know.md | 57 ++++++++++---------
1 file changed, 29 insertions(+), 28 deletions(-)
diff --git a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
index 0f35c99a5d3..4074986cbae 100644
--- a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
+++ b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
@@ -266,7 +266,6 @@ In util Module {
重点来了,如果我们在 `lib/util` 模块中引入 `index` 主模块会发生什么?这就是 Node 中所支持的循环模块依赖关系。
-To understand it better, let’s first understand a few other concepts on the module object.
为了更好理解循环模块依赖,我们先来了解一些关于 module 对象的概念。
#### exports、module.exports 和模块异步加载 ####
@@ -389,13 +388,11 @@ fs.readFile('/etc/passwd', (err, data) => {
});
```
-#### Circular module dependency ####
#### 循环模块依赖 ####
-Let’s now try to answer the important question about circular dependency in Node: What happens when module 1 requires module 2, and module 2 requires module 1?
-我们现在来回答关于 Node 中循环依赖的重要问题:当模块1
+我们现在来回答关于 Node 中循环依赖的重要问题:当我们在模块1中引用模块2,在模块2中引用模块1时会发生什么?
-To find out, let’s create the following two files under `lib/`, `module1.js` and `module.js` and have them require each other:
+为了找到答案,我们在 `lib/` 下创建 `module1.js` 和 `module.js` 两个文件并让它们互相引用:
```
// lib/module1.js
@@ -413,24 +410,25 @@ const Module1 = require('./module1');
console.log('Module1 is partially loaded here', Module1);
```
-When we run `module1.js` we see the following:
+执行 `module1.js` 后,我们将看到:
```
~/learn-node $ node lib/module1.js
Module1 is partially loaded here { a: 1 }
```
-We required `module2` before `module1` was fully loaded, and since `module2` required `module1` while it wasn’t fully loaded, what we get from the `exports` object at that point are all the properties exported prior to the circular dependency. Only the `a` property was reported because both `b` and `c` were exported after `module2` required and printed `module1`.
+我们在 `module1` 加载完成前引用了 `module2`,而此时 `module1` 尚未加载完,我们从当前的 `exports` 对象中得到的是在循环依赖之前导出的所有属性。这里只列出的只有属性 `a`,因为属性 `b` 和 `c` 都是在 `module2` 引入并打印了 `module1` 后导出的。
-Node keeps this really simple. During the loading of a module, it builds the `exports` object. You can require the module before it’s done loading and you’ll just get a partial exports object with whatever was defined so far.
+Node 使这个过程变得非常简单。它在模块加载时构建 `exports` 对象。你可以在该模块完成加载前引用它,而你将得到此时已定义的部分导出对象。
#### JSON and C/C++ addons ####
+#### 使用 JSON 和 C/C++ 插件 ####
-We can natively require JSON files and C++ addon files with the require function. You don’t even need to specify a file extension to do so.
+我们可以使用自带的 require 函数引用 JSON 文件和 C++ 插件。你甚至不需要为此去指定一个文件扩展。
-If a file extension was not specified, the first thing Node will try to resolve is a `.js` file. If it can’t find a `.js` file, it will try a `.json` file and it will parse the `.json` file if found as a JSON text file. After that, it will try to find a binary `.node` file. However, to remove ambiguity, you should probably specify a file extension when requiring anything other than `.js` files.
+如果一个文件扩展未被声明,Node 会在第一时间解析 `.js` 文件。如果没有找到 `.js` 文件,它将继续寻找 `.json` 文件并在找到一个 JSON 文本文件后将其解析为 `.json` 文件。随后,Node 将会查找二进制的 `.node` 文件。但是为了避免歧义,你最好在引用除 `.js` 文件以外的文件类型时声明文件扩展。
-Requiring JSON files is useful if, for example, everything you need to manage in that file is some static configuration values, or some values that you periodically read from an external source. For example, if we had the following `config.json` file:
+如果你需要在文件中放置的内容都是一些静态的配置信息,或定期从外部来源读取的一些值,那么使用 JSON 文件非常有用。例如,我们有以下 `config.json` 文件:
```
{
@@ -439,28 +437,29 @@ Requiring JSON files is useful if, for example, everything you need to manage in
}
```
-We can require it directly like this:
+我们可以这样直接引用它:
```
const { host, port } = require('./config');
-```
-
-console.log(`Server will run at [http://${host}:${port}`](http://$%7Bhost%7D:$%7Bport%7D`));
+console.log(`Server will run at [http://${host}:${port}](http://$%7Bhost%7D:$%7Bport%7D`));
-Running the above code will have this output:
+```
+执行以上代码将输出以下结果:
+```
Server will run at [http://localhost:8080](http://localhost:8080)
+```
-If Node can’t find a `.js` or a `.json` file, it will look for a `.node` file and it would interpret the file as a compiled addon module.
+如果 Node 找不到 `.js` 或 `.json` 文件,它将查找一个 `.node` 文件并将其解释为一个编译后的插件模块。
-The Node documentation site has a [sample addon file](https://nodejs.org/api/addons.html#addons_hello_world) which is written in C++. It’s a simple module that exposes a `hello()` function and the hello function outputs “world.”
+Node 文档站点有一个用 C++ 编写的[插件示例](https://nodejs.org/api/addons.html#addons_hello_world),该示例模块提供了一个输出 “world” 的 `hello()` 函数。
-You can use the `node-gyp` package to compile and build the `.cc` file into a `.addon` file. You just need to configure a [binding.gyp](https://nodejs.org/api/addons.html#addons_building) file to tell `node-gyp` what to do.
+你可以使用 `node-gyp` 插件将 `.cc` 文件编译成 `.addon` 文件。只需要配置一个 [binding.gyp](https://nodejs.org/api/addons.html#addons_building) 文件来告诉 `node-gyp` 要做什么。
-Once you have the `addon.node` file (or whatever name you specify in `binding.gyp`) then you can natively require it just like any other module:
+有了 `addon.node` 文件(你可以在 `binding.gyp` 中声明任意文件名),你就可以像引用其他模块一样引用它了。
```
const addon = require('./addon');
@@ -468,29 +467,31 @@ const addon = require('./addon');
console.log(addon.hello());
```
-We can actually see the support of the three extensions by looking at `require.extensions`.
+我们可以通过 `require.extensions` 查看 Node 对这三类扩展的支持。
-Looking at the functions for each extension, you can clearly see what Node will do with each. It uses `module._compile` for `.js` files, `JSON.parse` for `.json` files, and `process.dlopen` for `.node` files.
+你可以从各个扩展对应的函数中清楚了解 Node 对它们分别所做的操作:对 `.js` 文件使用 `module._compile`,对 `.json` 文件使用 `JSON.parse`,对 `.node` 文件使用 `process.dlopen`。
#### All code you write in Node will be wrapped in functions ####
+#### 你在 Node 中写的所有代码都将被包装成函数 ####
-Node’s wrapping of modules is often misunderstood. To understand it, let me remind you about the `exports`/`module.exports` relation.
+Node 的模块打包经常被误解。要了解它的原理,请回忆一下 `exports` 与 `module.exports` 的关系。
-We can use the `exports` object to export properties, but we cannot replace the `exports` object directly because it’s just a reference to `module.exports`
+我们可以使用 `exports` 对象导出属性,但是由于 `exports` 对象仅仅是对 `module.exports` 的一个引用,我们无法直接对其执行替换操作。
```
-exports.id = 42; // This is ok.
+exports.id = 42; // 有效
-exports = { id: 42 }; // This will not work.
+exports = { id: 42 }; // 无效
-module.exports = { id: 42 }; // This is ok.
+module.exports = { id: 42 }; // 有效
```
-How exactly does this `exports` object, which appears to be global for every module, get defined as a reference on the `module` object?
+这个 `exports` 对象看起来对所有模块都是全局的,它是如何被定义成 `module` 对象的引用的呢?
Let me ask one more question before explaining Node’s wrapping process.
+在解释 Node 的打包进程前,再
In a browser, when we declare a variable in a script like this:
From 1626c74e23a1b7e26ca815142938b94f1c8805d0 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Thu, 30 Mar 2017 22:58:49 +0800
Subject: [PATCH 036/945] Create
o-h-yeah-what-we-look-forward-to-in-android-o.md
---
...ah-what-we-look-forward-to-in-android-o.md | 90 +++++++++++++++++++
1 file changed, 90 insertions(+)
create mode 100644 TODO/o-h-yeah-what-we-look-forward-to-in-android-o.md
diff --git a/TODO/o-h-yeah-what-we-look-forward-to-in-android-o.md b/TODO/o-h-yeah-what-we-look-forward-to-in-android-o.md
new file mode 100644
index 00000000000..b380df4dfc4
--- /dev/null
+++ b/TODO/o-h-yeah-what-we-look-forward-to-in-android-o.md
@@ -0,0 +1,90 @@
+> * 原文地址:[O-h yeah! What we look forward to in Android O](https://www.novoda.com/blog/o-h-yeah-what-we-look-forward-to-in-android-o/)
+> * 原文作者:[Novoda](https://www.novoda.com/blog/author/novoda/)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+# O-h yeah! What we look forward to in Android O
+
+[/blog/author/novoda/](/blog/author/novoda/)
+[Team Novoda](/blog/author/novoda/)Joint Thinkers
+
+Novoda has a reputation of building the most desirable apps for Android and iOS. We believe living and sharing a hack-and-tell culture is one way to maintain top-shelf quality.
+
+
+This week Google has announced the new Android O Preview programme. Like most Android developers out there, we poured over the documentation to find out what new feature tickles our fancies the most. Here’s what people at Novoda are looking forward to.
+
+![Android O-Some](https://2.bp.blogspot.com/-WSPrWvuvCvc/WM80F43fu4I/AAAAAAAAGtU/N73vMkriLX8rH-lt1t2cns9YSuJlBHr_wCLcB/s1600/android-o-logo.png)
+
+## Sebastiano Poggi
+
+> There are a lot of very interesting new APIs and features in this release; to me, being a UI kind of person, there are a few that have been making me giggle with delight.
+
+### Wide gamut and diverse colour spaces support 🌈
+
+![Adobe RGB vs sRGB colour spaces](https://developer.android.com/reference/android/images/graphics/colorspace_adobe_rgb.png)First and foremost, I'm *extremely* happy that Android will be able to handle **colour spaces** properly. No more being limited to sRGB, now apps will be able to properly display images that are stored in the Adobe RGB, ProPhoto RGB and many others. The new APIs will also help you with [converting between colour spaces](https://developer.android.com/reference/android/graphics/ColorSpace.Adaptation.html). The [documentation](https://developer.android.com/reference/android/graphics/ColorSpace.html) seems excellent as well, even just to learn about colour spaces!
+
+*Fun fact:* all the [named colour space graphs](https://developer.android.com/reference/android/graphics/ColorSpace.Named.html) in the docs have been generated with the new [`ColorSpace.Renderer`](https://developer.android.com/reference/android/graphics/ColorSpace.Renderer.html) on Android.
+
+### First-party and first-class fonts support ❤️
+
+For years now Android developers have been forced to use clever hacks to implement fonts support in apps, such as [Calligraphy](https://github.com/InflationX/Calligraphy) or custom span-based solutions. Unfortunately there is still a lot of limitations with that approach, such as the layout preview in Android Studio not showing the custom fonts, or the inability to apply the fonts in some unusual cases (such as `TabLayout`‘s tabs that get their `textAppearance` separately from their inflation ಠ_ಠ). So I’m very, very happy that all these things will be sorted out in the next version of the OS, with the introduction of **font resources**. It took some time, but we’re getting there. Hopefully it’ll get to the [support library](https://twitter.com/chrisbanes/status/844274842279051264), too…
+
+### Adaptive Icons ⚪ ⬛ 🔴 ⬜
+
+![Three dimensional animation of an Adaptive Icon](https://d2mxuefqeaa7sj.cloudfront.net/s_D495BEC1F83AAA38C0FCFF599E996A34C92045AC1FD3533493D989F431CA82C0_1490194268969_NB_Icon_Layers_3D_03_ext.gif)
+
+I think this might have been introduced to put a leash on OEMs that like to mess a bit too much with things. Embracing the current (mal-)practice of adding a background to icons is not *exactly* new—[round launcher icons](https://developer.android.com/about/versions/nougat/android-7.1.html#circular-icons) in Android N were a first step in that direction. **Adaptive icons** will take it the next level though, allowing OEMs and launcher developers to specify a mask to apply to an application-provided background image. This way, icons will fit into whatever style the context they show up into dictates, without having to ship all possible variations.
+
+In addition, the new assets are supposed to be substantially larger than the previous images, with a lot of leeway for animating the icons:
+
+![Parallax animation on an Adaptive Icon](https://d2mxuefqeaa7sj.cloudfront.net/s_D495BEC1F83AAA38C0FCFF599E996A34C92045AC1FD3533493D989F431CA82C0_1490194498483_Single_Icon_Parallax_Demo_01_2x_ext.gif)![Zoom/pop animation on an Adaptive Icon](https://d2mxuefqeaa7sj.cloudfront.net/s_D495BEC1F83AAA38C0FCFF599E996A34C92045AC1FD3533493D989F431CA82C0_1490194498352_Single_Icon_Pickup_Drop_01_2x_ext.gif)
+
+## Ataul Munim
+
+> A few things caught my eye which will be interesting for those of you interested in inclusive design.
+
+### Accessibility Button
+
+Accessibility services (like Google TalkBack) will be able to request an additional button in the navigation bar for devices with soft navigation keys.
+
+This button will provide a service-specific shortcut. So far, only TalkBack has implemented it—though I think they had the inside track on this one!
+
+It’ll be interesting to see how services use (and mis-use) this feature, and how the system will deal with multiple services that are vying for the same space.
+
+### Fingerprint gestures
+
+Many of you will already be using fingerprint gestures as convenient shortcuts for frequently performed tasks on your phones, from scrolling through content to pulling down the notification shade.
+
+Fingerprint gestures being made available to accessibility services can only be considered a good thing, providing this addition is transparent to app developers and is handled wholly by Android or the service in question.
+
+I wonder if it could be used to bring back something similar to the optical trackball!
+
+### Autosizing TextView 👓
+
+This is the one I’m a little worried about. All too often we see apps that don’t cater for users that make use of the system font size selector (available in Settings > Display), resulting in clipped text and thoroughly confused readers.
+
+I suspect we’ll see this issue exacerbated by apps using the autosizing TextView, though if I put my Hopeful Hat on, maybe it'll prompt designers and developers to consider what their apps will look like with various text sizes.
+
+## [Paul Blundell](http://twitter.com/blundell_apps)
+
+> Everyone is thinking, there are no desserts beginning with the letter O... While I'm excited for the new APIs, I'm more excited that perhaps M will get more device adoption now that people is looking at O!
+
+### AutoFill APIs
+
+Didn't realise I needed it until it was pointed out it was missing. So many times do I give up logging into an app because I know if I go on the mobile website my login will be autofilled. No more.
+
+Two points from the announcement:
+
+> Users can select an autofill app, similar to the way they select a keyboard app. The autofill app stores and secures user data, such as addresses, user names, and even passwords.
+
+Interestingly, this pushes security concerns onto 3rd parties. I wonder who will come out on top. With Google having their own [Smart Lock](https://get.google.com/smartlock/) concept, I imagine Google may be releasing an autofill app alongside O.
+
+> For apps that want to handle autofill, we're adding new APIs to implement an Autofill service.
+
+Oh boy, I want to play with this. I've read about the security researchers who created [invisible input fields](https://github.com/anttiviljami/browser-autofill-phishing) on a webpage to contain *credit card* details, that would get autofilled while the user only saw the *name* field. Users were tricked into allowing autofill and, when they submitted the form, unintentionally disclosed their card details. Looking forward to seeing how Android tackles this and other security questions around Autofill.
+
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 7c87cf2a6114d27898c23dd4dc3aff690714138b Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Thu, 30 Mar 2017 23:00:25 +0800
Subject: [PATCH 037/945] Update
o-h-yeah-what-we-look-forward-to-in-android-o.md
---
TODO/o-h-yeah-what-we-look-forward-to-in-android-o.md | 2 --
1 file changed, 2 deletions(-)
diff --git a/TODO/o-h-yeah-what-we-look-forward-to-in-android-o.md b/TODO/o-h-yeah-what-we-look-forward-to-in-android-o.md
index b380df4dfc4..a6903d3ef88 100644
--- a/TODO/o-h-yeah-what-we-look-forward-to-in-android-o.md
+++ b/TODO/o-h-yeah-what-we-look-forward-to-in-android-o.md
@@ -6,8 +6,6 @@
# O-h yeah! What we look forward to in Android O
-[/blog/author/novoda/](/blog/author/novoda/)
-[Team Novoda](/blog/author/novoda/)Joint Thinkers
Novoda has a reputation of building the most desirable apps for Android and iOS. We believe living and sharing a hack-and-tell culture is one way to maintain top-shelf quality.
From b71a343c7b788c560e92ba5da56a0cc40631dc26 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Thu, 30 Mar 2017 23:16:34 +0800
Subject: [PATCH 038/945] Create preload-prefetch-and-priorities-in-chrome.md
---
...eload-prefetch-and-priorities-in-chrome.md | 252 ++++++++++++++++++
1 file changed, 252 insertions(+)
create mode 100644 TODO/preload-prefetch-and-priorities-in-chrome.md
diff --git a/TODO/preload-prefetch-and-priorities-in-chrome.md b/TODO/preload-prefetch-and-priorities-in-chrome.md
new file mode 100644
index 00000000000..7e8eb74705e
--- /dev/null
+++ b/TODO/preload-prefetch-and-priorities-in-chrome.md
@@ -0,0 +1,252 @@
+> * 原文地址:[Preload, Prefetch And Priorities in Chrome](https://www.teambition.com/project/583d8744180aa4d012496f03/tasks/scrum/583d8744fa1e93bf18a85a7a/task/58dc6b68fd0faca50d444dbc)
+> * 原文作者:[Addy Osmani](https://medium.com/@addyosmani?source=post_header_lockup)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+
+
+
+# **Preload, Prefetch And Priorities in Chrome**
+
+Today we’ll dive into insights from Chrome’s networking stack to provide clarity on how web loading primitives (like [****](https://w3c.github.io/preload/) & [****](https://w3c.github.io/resource-hints/)) work behind the scenes so you can be more effective with them.
+
+As covered well in [other articles](https://www.smashingmagazine.com/2016/02/preload-what-is-it-good-for/), **preload is a declarative fetch, allowing you to force the browser to make a request for a resource without blocking the document’s** [**onload**](https://developer.mozilla.org/en-US/docs/Web/API/GlobalEventHandlers/onload) **event**.
+
+**Prefetch****is a hint to the browser that a resource might be needed**, but delegates deciding whether and when loading it is a good idea or not to the browser.
+
+
+
+Preload can decouple the load event from script parse time. If you haven’t used it before, read ‘[Preload: What is it Good For?](https://www.smashingmagazine.com/2016/02/preload-what-is-it-good-for/)’ by Yoav Weiss
+
+### Preload success stories in production sites
+
+Before we dive into the details, here’s a quick summary of some positive impact to loading metrics that have been observed using preload in the last year:
+
+Housing.com saw a [**~10% improvement in Time to Interactive**](https://twitter.com/HousingEngg/status/844169796891508737) when they switched to preloading key late-discovered scripts for their Progressive Web App:
+
+
+
+Shopify’s switch to [preloading Web Fonts](https://www.bramstein.com/writing/preload-hints-for-web-fonts.html) saw a[**50%**](https://twitter.com/ShopifyEng/status/844245243948163072) **(1.2 second) improvement in time-to-text-paint** on Chrome desktop (cable). This removed their flash-of-invisible text completely.
+
+
+
+Left: with preload, Right: without ([video](https://video.twimg.com/tweet_video/C7dcmxaUwAAUhPX.mp4))
+
+
+
+Web Font loading using
+
+Treebo, one of India’s largest hotel chains **shaved** [**1 second**](https://twitter.com/__lakshya/status/844429211867791361) **off both time to First Paint and Time to Interactive** for their desktop experience over 3G, by preloading their header image and key Webpack bundles:
+
+
+
+Similarly, by switching to preloading their key bundles, Flipkart [**shaved**](https://twitter.com/adityapunjani/status/844250835802619905) **a great deal of main thread idle** before route chunks get evaluated on their PWA (trace from a low-end phone over 3G):
+
+
+
+Top: without preload, Bottom: with preload
+
+And the Chrome Data Saver team saw **time to first contentful paint improvements of** [**12% on average**](https://medium.com/reloading/a-link-rel-preload-analysis-from-the-chrome-data-saver-team-5edf54b08715#.bgj9qkqfr)for pages that could use preload on scripts and CSS stylesheets.
+
+As for prefetch, it’s widely used and at Google we still use it in [Search results pages](https://plus.google.com/+IlyaGrigorik/posts/ahSpGgohSDo) to prefetch critical resources that can speed up rendering destination pages.
+
+Preload is used in production by large sites for a number of use-cases and you can find more of them later on in the article. Before that, let’s dive into how the network stack actually treats preload vs prefetch.
+
+### When should you vs ?
+
+**Tip:** **Preload resources you have high-confidence will be used in the current page. Prefetch resources likely to be used for future navigations across multiple navigation boundaries.**
+
+Preload is an early fetch instruction to the browser to request a resource *needed* for a page (key scripts, Web Fonts, hero images).
+
+Prefetch serves a slightly different use case — a future navigation by the user (e.g between views or pages) where fetched resources and requests need to persist across navigations. If Page A initiates a prefetch request for critical resources needed for Page B, the critical resource and navigation requests can be completed in parallel. If we used preload for this use case, it would be immediately cancelled on Page A’s unload.
+
+Between preload and prefetch, we get solutions for loading critical resources for the current navigation _or_ a future navigation.
+
+### What is the caching behavior for and ?
+
+[Chrome has four caches](https://calendar.perfplanet.com/2016/a-tale-of-four-caches/): the HTTP cache, memory cache, Service Worker cache & Push cache. Both preload and prefetched resources are stored in the **HTTP cache.**
+
+When a resource is **preloaded or prefetched** is travels up from the net stack through to the HTTP cache and into the renderer’s memory cache. If the resource can be cached (e.g there’s a valid [cache-control](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Cache-Control) with valid max-age), it is stored in the HTTP cache and is available for **current and future sessions**. If the resource is **not cacheable**, it does not get stored in the HTTP cache. Instead, it goes up to the memory cache and stays there until it gets used.
+
+### How does Chrome’s network prioritisation handle preload and prefetch?
+
+Here’s a break-down ([courtesy](https://docs.google.com/document/d/1bCDuq9H1ih9iNjgzyAL0gpwNFiEP4TZS-YLRp_RuMlc/edit#) of Pat Meenan) showing how different resources are prioritized in Blink as of Chrome 46 and beyond:
+
+
+
+Preload using “as” or fetch using “type” use the priority of the type they are requesting. (e.g. preload as=style will use Highest priority). With no “as” they will behave like an XHR. “Early” is defined as being requested before any non-preloaded images have been requested (“late” is after). Thanks to Paul Irish for updating this table with the DevTools priorities mapping to the Net and Blink priorities.
+
+Let’s talk about this table for a moment.
+
+**Scripts get different priorities based on where they are in the document and whether they are async, defer or blocking:**
+
+- Blocking scripts requested before the first image (an image early in the document) are Net:Medium
+- Blocking scripts requested after the first image is fetched are Net:Low
+- Async/defer/injected scripts (regardless of where they are in the document) are Net:Lowest
+
+**Images (that are visible and in the viewport) have a higher priority (Net:Medium) than those that are not in the viewport (Net:Lowest)**, so to some extent Chrome will do it’s best to pseudo-lazy-load those images for you. Images start off with a lower priority and after layout is done and they are discovered to be in the viewport, will get a priority boost (but note that images already in flight when layout completes won’t be reprioritized).
+
+Preloaded resources using the “as” attribute will have the **same resource priority** as the **type** of resource they are requesting. For example, preload as=“style” will get the highest priority while as=”script” will get a low or medium priority. These resources are **also subject to the same CSP policies** (e.g script is subject to script-src).
+
+Preloaded resources without an “as” will otherwise be requested with the same priority as async XHR (so High).
+
+If you’re interested in understanding what priority a resource was loaded with, this information is exposed in DevTools via both the Network section of Timeline/Performance:
+
+
+
+and in the Network panel behind the “Priority” column:
+
+
+
+### What happens when a page tries to preload a resource that has already been cached in the Service Worker cache, the HTTP cache or both?
+
+This is going to be a large “it depends” but generally, something good should almost always happen in this case — the resource won’t be refetched from the network unless it has expired from the HTTP cache or the Service Worker intentionally refetches it.
+
+If the resource is in the HTTP cache (between the SW Cache & the network) then preload should get a cache hit from the same resource.
+
+### Are there risks with these primitives of wasting a user’s bandwidth?
+
+**With “preload” or “prefetch”, you’re running some risk of wasting a user’s bandwidth, especially if the resource is not cacheable.**
+
+Unused preloads trigger a console warning in Chrome, ~3 seconds after *onload:*
+
+
+
+The reason for this warning is you’re probably using preload to try warming the cache for other resources you need to improve performance but if these preloaded resources aren’t being used, you’re doing extra work for no reason. On mobile, this sums up to wasting a user’s data plans, so be mindful of what you’re preloading.
+
+### What can cause double fetches?
+
+Preload and prefetch are blunt tools and it isn’t hard to find yourself [double-fetching](https://bugs.chromium.org/p/chromium/issues/list?can=2&q=preload%20double%20owner%3Ayoav%40yoav.ws) if you aren’t careful.
+
+**Don’t use “prefetch” as a fallback for “preload”**. They’re again, used for different purposes and often end up causing [double fetches](https://twitter.com/yoavweiss/status/824957889991303168) while this probably isn’t your intention. Use preload if it’s supported for warming the cache for current sessions otherwise prefetch for future sessions. Don’t use one in place of the other.
+
+
+
+**Don’t rely on fetch() working with “preload”… just yet.** In Chrome if you try to use preload with the fetch() API you will end up triggering a double download. This doesn’t currently occur with XHR and we have an [open bug](https://bugs.chromium.org/p/chromium/issues/detail?id=652228) to try addressing it.
+
+**Supply an “as” when preloading or you’ll negate any benefits!**
+
+If you don’t supply a valid “as” when specifying what to preload, for example, scripts, you will end up [fetching twice](https://twitter.com/DasSurma/status/808791438171537408).
+
+**Preloaded fonts without crossorigin will double fetch! **Ensure you’re adding a [crossorigin attribute](https://developer.mozilla.org/en-US/docs/Web/HTML/CORS_settings_attributes) when fetching fonts using preload otherwise they will be double downloaded. They’re requested using anonymous mode CORS. This advice applies even if fonts are on the same origin as the page. This is applicable to other anonymous fetches too (e.g XHR by default).
+
+**Resources with an integrity attribute can’t reuse preloaded resources (for now) and can also cause double fetches.** The `[integrity](https://bugs.chromium.org/p/chromium/issues/detail?id=677022)` attribute for link elements has not yet been implemented and there’s an open [spec issue](https://github.com/w3c/webappsec-subresource-integrity/issues/26) about it. This means the presence of any integrity metadata will currently discard preloaded resources. In the wild, it can also result in duplicate requests where you have to make a trade-off between security and performance.
+
+Finally, although it won’t cause double fetches, this is generally good advice:
+
+**Don’t try preloading absolutely everything! **Instead, select specific late discovered resources that you want to load earlier and use preload to tell the browser about them.
+
+### Should I just preload all the assets that my page requests in the head? Is there a recommended limit like “only preload ~6 things”?
+
+This is a good example of **Tools, not rules. **How much you preload may well factor in how much network contention you’re going to have with other resources also being loaded on your page, your user’s available bandwidth and other network conditions.
+
+Preload resources that are likely to be discovered late in your page, but are otherwise important to fetch as early as possible. With scripts, preloading your key bundles is good as it separates fetching from execution in a way that just using say,
+```
+
+As you may realize, this URL will make the browser run the injected script and send the user’s cookies, potentially including confidential session cookies, to evil.com!
+
+To help protect users against reflective XSS attacks, some browsers have implemented protection mechanisms. These mechanisms try to identify these attacks by looking for matching code patterns in the HTTP request and response. Internet Explorer was the first browser to introduce such a mechanism with its XSS filter, introduced in Internet Explorer 8 back in 2008, and WebKit later introduced XSS Auditor, available today in Chrome and Safari. (Firefox has no similar mechanism built in, but users can use add-ons to gain this functionality.) These various protection mechanisms are not perfect: They may fail to detect a real XSS attack (a false negative), and in other cases may block legitimate code (a false positive). Due to the latter, browsers allow users to disable the XSS filter via the settings. Unfortunately, this is typically a global setting, which turns off this security feature completely for all web apps loaded by the browser.
+
+Luckily, there is a way for a web app to override this configuration and ensure that the XSS filter is turned on for the web app being loaded by the browser. This is done via the `X-XSS-Protection` header. This header, supported by Internet Explorer (from version 8), Edge, Chrome and Safari, instructs the browser to turn on or off the browser’s built-in protection mechanism and to override the browser’s local configuration.
+
+`X-XSS-Protection` directives include these:
+
+- `1` or `0`
+
+This enables or disables the filter.
+- `mode=block`
+
+This instructs the browser to prevent the entire page from rendering when an XSS attack is detected.
+
+I recommend always turning on the XSS filter, as well as block mode, to maximize user protection. Such a response header looks like this:
+
+```
+X-XSS-Protection: 1; mode=block
+```
+
+Here’s how you would configure this response header in Node.js:
+
+```
+functionrequestHandler(req, res){
+ res.setHeader('X-XSS-Protection','1;mode=block');}
+```
+
+### Controlling Framing ###
+
+An iframe (or HTML inline frame element, if you want to be more formal) is a DOM element that allows a web app to be nested within a parent web app. This powerful element enables some important web use cases, such as embedding third-party content into web apps, but it also has significant drawbacks, such as not being SEO-friendly and not playing nice with browser navigation — the list goes on.
+
+One of the caveats of iframes is that it makes clickjacking easier. Clickjacking is an attack that tricks the user into clicking something different than what they think they’re clicking. To understand a simple implementation of clickjacking, consider the HTML markup below, which tries to trick the user into buying a toaster when they think they are clicking to win a prize!
+
+```
+
+
+
+
+
+
+```
+
+Clickjacking has many malicious applications, such as tricking the user into confirming a Facebook like, purchasing an item online and even submitting confidential information. Malicious web apps can leverage iframes for clickjacking by embedding a legitimate web app inside their malicious web app, rendering the iframe invisible with the `opacity: 0` CSS rule, and placing the iframe’s click target directly on top of an innocent-looking button rendered by the malicious web app. A user who clicks the innocent-looking button will trigger a click on the embedded web app — without at all knowing the effect of their click.
+
+An effective way to block this attack is by restricting your web app from being framed. `X-Frame-Options`, specified in [RFC 7034](https://www.ietf.org/rfc/rfc7034.txt), is designed to do exactly that! This header instructs the browser to apply limitations on whether your web app can be embedded within another web page, thus blocking a malicious web page from tricking users into invoking various transactions on your web app. You can either block framing completely using the `DENY` directive, whitelist specific domains using the `ALLOW-FROM` directive, or whitelist only the web app’s origin using the `SAMEORIGIN` directive.
+
+My recommendation is to use the `SAMEORIGIN` directive, which enables iframes to be leveraged for apps on the same domain — which may be useful at times — and which maintains security. This recommended header looks like this:
+
+```
+X-Frame-Options: SAMEORIGIN
+```
+
+Here’s an example of a configuration of this header to enable framing on the same origin in Node.js:
+
+```
+functionrequestHandler(req, res){
+ res.setHeader('X-Frame-Options','SAMEORIGIN');}
+```
+
+### Explicitly Whitelisting Sources ###
+
+As we’ve noted earlier, you can add in-depth security to your web app by enabling the browser’s XSS filter. However, note that this mechanism is limited, is not supported by all browsers (Firefox, for instance, does not have an XSS filter) and relies on pattern-matching techniques that can be tricked.
+
+Another layer of in-depth protection against XSS and other attacks can be achieved by explicitly whitelisting trusted sources and operations — which is what Content Security Policy (CSP) enables web app developers to do.
+
+CSP is a [W3C specification](https://www.w3.org/TR/2016/WD-CSP3-20160901/) that defines a powerful browser-based security mechanism, enabling granular control over resource-loading and script execution in a web app. With CSP, you can whitelist specific domains for operations such as script-loading, AJAX calls, image-loading and style sheet-loading. You can enable or disable inline scripts or dynamic scripts (the notorious `eval`) and control framing by whitelisting specific domains for framing. Another cool feature of CSP is that it allows you to configure a real-time reporting target, so that you can monitor your app in real time for CSP blocking operations.
+
+This explicit whitelisting of resource loading and execution provides in-depth security that in many cases will fend off attacks. For example, by using CSP to disallow inline scripts, you can fend off many of the reflective XSS attack variants that rely on injecting inline scripts into the DOM.
+
+CSP is a relatively complex header, with a lot of directives, and I won’t go into the details of the various directives. HTML5 Rocks has a [great tutorial](https://www.html5rocks.com/en/tutorials/security/content-security-policy/) that provides an overview of CSP, and I highly recommend reading it and learning how to use CSP in your web app.
+
+Here’s a simple example of a CSP configuration to allow script-loading from the app’s origin only and to block dynamic script execution (`eval`) and inline scripts (as usual, on Node.js):
+
+```
+functionrequestHandler(req, res){
+ res.setHeader('Content-Security-Policy',"script-src 'self'");}
+```
+
+### Preventing Content-Type Sniffing ###
+
+In an effort to make the user experience as seamless as possible, many browsers have implemented a feature called content-type sniffing, or MIME sniffing. This feature enables the browser to detect the type of a resource provided as part of an HTTP response by “sniffing” the actual resource bits, regardless of the resource type declared through the `Content-Type` response header. While this feature is indeed useful in some cases, it introduces a vulnerability and an attack vector known as a MIME confusion attack. A MIME-sniffing vulnerability enables an attacker to inject a malicious resource, such as a malicious executable script, masquerading as an innocent resource, such as an image. With MIME sniffing, the browser will ignore the declared image content type, and instead of rendering an image will execute the malicious script.
+
+Luckily, the `X-Content-Type-Options` response header mitigates this vulnerability! This header, introduced in Internet Explorer 8 back in 2008 and currently supported by most major browsers (Safari is the only major browser not to support it), instructs the browser not to use sniffing when handling fetched resources. Because `X-Content-Type-Options` was only formally specified as part of the [“Fetch” specification](https://fetch.spec.whatwg.org/#x-content-type-options-header), the actual implementation varies across browsers; some (Internet Explorer and Edge) completely avoid MIME sniffing, whereas others (Firefox) still MIME sniff but rather block executable resources (JavaScript and CSS) when an inconsistency between declared and actual types is detected. The latter is in line with the latest Fetch specification.
+
+`X-Content-Type-Options` is a simple response header, with only one directive: `nosniff`. This header looks like this: `X-Content-Type-Options: nosniff`. Here’s an example of a configuration of the header:
+
+```
+functionrequestHandler(req, res){
+ res.setHeader('X-Content-Type-Options','nosniff');}
+```
+
+### Summary ###
+
+In this article, we have seen how to leverage HTTP headers to reinforce the security of your web app, to fend off attacks and to mitigate vulnerabilities.
+
+#### Takeaways ####
+
+- Disable caching for confidential information using the `Cache-Control` header.
+- Enforce HTTPS using the `Strict-Transport-Security` header, and add your domain to Chrome’s preload list.
+- Make your web app more robust against XSS by leveraging the `X-XSS-Protection` header.
+- Block clickjacking using the `X-Frame-Options` header.
+- Leverage `Content-Security-Policy` to whitelist specific sources and endpoints.
+- Prevent MIME-sniffing attacks using the `X-Content-Type-Options` header.
+
+Remember that for the web to be truly awesome and engaging, it has to be secure. Leverage HTTP headers to build a more secure web!
+
+
+
+(**Disclaimer:** The content of this post is my own and doesn’t represent my past or current employers in any way whatsoever.)
+
+*Front page image credits: [Pexels.com](https://www.pexels.com/photo/coffee-writing-computer-blogging-34600/).*
+
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 70b8b484b9c34c825b0ca589e98fd852646ecb93 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Fri, 7 Apr 2017 20:54:36 +0800
Subject: [PATCH 127/945] Create webpack-and-rollup-the-same-but-different.md
---
...bpack-and-rollup-the-same-but-different.md | 58 +++++++++++++++++++
1 file changed, 58 insertions(+)
create mode 100644 TODO/webpack-and-rollup-the-same-but-different.md
diff --git a/TODO/webpack-and-rollup-the-same-but-different.md b/TODO/webpack-and-rollup-the-same-but-different.md
new file mode 100644
index 00000000000..7bc3f470f12
--- /dev/null
+++ b/TODO/webpack-and-rollup-the-same-but-different.md
@@ -0,0 +1,58 @@
+> * 原文地址:[Webpack and Rollup: the same but different](https://medium.com/webpack/webpack-and-rollup-the-same-but-different-a41ad427058c)
+> * 原文作者:[Rich Harris](https://medium.com/@Rich_Harris?source=post_header_lockup)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+# Webpack and Rollup: the same but different #
+
+![](https://cdn-images-1.medium.com/max/1000/1*rtjClMZ8sq3cLFT9Aq8Xyg.png)
+
+This week, Facebook merged a [monster pull request](https://github.com/facebook/react/pull/9327) into React that replaced its existing build process with one based on [Rollup](https://rollupjs.org/) , [prompting](https://twitter.com/stanlemon/status/849366789825994752) [several](https://twitter.com/MrMohtas/status/849362334988595201) [people](https://twitter.com/kyleholzinger/status/849683292760797184) to ask ‘why did you choose Rollup over webpack’?
+
+Which is a completely reasonable question. [Webpack](https://webpack.js.org/) is one of the modern JavaScript community’s greatest success stories, with millions of downloads every month powering tens of thousands of websites and applications. It has a large ecosystem, dozens of contributors, and — unusually for a community open source project — [meaningful financial support](https://opencollective.com/webpack) .
+
+By comparison, Rollup is a minnow. But React isn’t alone — Vue, Ember, Preact, D3, Three.js, Moment, and dozens of other well-known libraries also use Rollup. So what’s going on? Why can’t we have just one JavaScript module bundler that everyone agrees on?
+
+### A tale of two bundlers ###
+
+webpack was started in 2012 by [Tobias Koppers](https://medium.com/@sokra) to solve a hard problem that existing tools didn’t address: building complex single-page applications (SPAs). Two features in particular changed everything:
+
+1. **Code-splitting** makes it possible to break your app apart into manageable chunks that can be loaded on-demand, meaning your users get an interactive site much faster than if they had to wait for the whole application to download and parse. You *can* do this manually, but, well… good luck.
+2. **Static assets** such as images and CSS can be imported into your app and treated as just another node in the dependency graph. No more carefully placing your files in the right folders and hacked-together scripts for adding hashes to file URLs — webpack can take care of it for you.
+
+Rollup was created for a different reason: to build flat distributables of JavaScript libraries as efficiently as possible, taking advantage of the ingenious design of ES2015 modules. Other module bundlers — webpack included — work by wrapping each module in a function, putting them in a bundle with a browser-friendly implementation of `require`, and evaluating them one-by-one. That’s great if you need things like on-demand loading, but otherwise it’s a bit of a waste, and it [gets worse if you have lots of modules](https://nolanlawson.com/2016/08/15/the-cost-of-small-modules/).
+
+ES2015 modules enable a different approach, which Rollup uses. All your code is put in the same place and evaluates in one go, resulting in leaner, simpler code that starts up faster. You can [see it for yourself with the Rollup REPL](https://rollupjs.org/repl).
+
+But there’s a trade-off: code-splitting is a much hairier problem, and at the time of writing Rollup doesn’t support it. Similarly, Rollup doesn’t do hot module replacement (HMR). And perhaps the biggest pain point for people coming to Rollup — while it handles most CommonJS files (via a [plugin](https://github.com/rollup/rollup-plugin-commonjs) ), some things just don’t translate to ES2015, whereas webpack handles everything you throw at it with aplomb.
+
+### So which should I use? ###
+
+By now, hopefully it’s clear why both tools coexist and support each other — they serve different purposes. The tl;dr is this:
+
+> Use webpack for apps, and Rollup for libraries
+
+That’s not a hard and fast rule — lots of sites and apps are built with Rollup, and lots of libraries are built with webpack. But it’s a good rule of thumb.
+
+If you need code-splitting, or you have lots of static assets, or you’re building something with lots of CommonJS dependencies, Webpack is a better choice. If your codebase is ES2015 modules and you’re making something to be used by other people, you probably want Rollup.
+
+### Package authors: use `pkg.module!` ###
+
+For a long time, using JavaScript libraries was a bit of a crapshoot, because you and the library author effectively had to agree on a module system. If you were using Browserify but she preferred AMD, you would have to duct tape things together before you could actually build anything. The [Universal Module Definition](https://github.com/umdjs/umd) (UMD) format *sort of* fixed that, but because it wasn’t enforced anywhere you never knew quite what you were going to get.
+
+ES2015 changes all that, because `import` and `export` are part of the language. In the future, there’ll be no ambiguity, and things will work a lot more seamlessly. Unfortunately, because browsers (mostly) and Node don’t yet support `import` and `export`, we still need to ship UMD files (or CommonJS, if you’re building something Node-only).
+
+By adding a `"module": "dist/my-library.es.js"` entry to your library’s package.json file (aka `pkg.module`), it’s possible to serve UMD and ES2015 at the same time, right now. **That’s important because Webpack and Rollup can both use** `pkg.module` **to generate the most efficient code possible** — in some cases, they can even both [tree-shake](https://webpack.js.org/guides/tree-shaking/) unused parts of your library away.
+
+*Learn more about`pkg.module` on the [*Rollup wiki*](https://github.com/rollup/rollup/wiki/pkg.module) .*
+
+Hopefully this article makes the relationship between the two projects a bit clearer. If you still have questions, find us on Twitter at [rich_harris](https://twitter.com/rich_harris)/[rollupjs](https://twitter.com/rollupjs) and [thelarkinn](https://twitter.com/thelarkinn) . Happy bundling!
+
+Our thanks to Rich Harris for writing this article. We believe that collaboration in open source is incredibly vital to ensure we push technology and the web forward together.
+
+No time to help contribute? Want to give back in other ways? Become a Backer or Sponsor to webpack by [donating to our open collective](https://opencollective.com/webpack). Open Collective not only helps support the Core Team, but also supports contributors who have spent significant time improving our organization on their free time! ❤
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 0f630d62fc21721b39491c8886c93e1f933480ce Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Fri, 7 Apr 2017 21:25:48 +0800
Subject: [PATCH 128/945] Create
the-time-i-had-to-crack-my-own-reddit-password.md
---
...e-i-had-to-crack-my-own-reddit-password.md | 564 ++++++++++++++++++
1 file changed, 564 insertions(+)
create mode 100644 TODO/the-time-i-had-to-crack-my-own-reddit-password.md
diff --git a/TODO/the-time-i-had-to-crack-my-own-reddit-password.md b/TODO/the-time-i-had-to-crack-my-own-reddit-password.md
new file mode 100644
index 00000000000..a7cd6da89cd
--- /dev/null
+++ b/TODO/the-time-i-had-to-crack-my-own-reddit-password.md
@@ -0,0 +1,564 @@
+> * 原文地址:[That time I had to crack my own Reddit password](https://medium.freecodecamp.com/the-time-i-had-to-crack-my-own-reddit-password-a6077c0a13b4)
+> * 原文作者:[Haseeb Qureshi](https://medium.freecodecamp.com/@hosseeb?source=post_header_lockup)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+# That time I had to crack my own Reddit password #
+
+
+
+Hack the planet, everybody.
+
+I have no self-control.
+
+Luckily, I know this about myself. This allows me to consciously engineer my life so that despite having the emotional maturity of a heroin-addicted lab rat, I’m occasionally able to get things done.
+
+![](https://media.giphy.com/media/gOH54eiriYIwM/giphy.gif)
+
+
+Mm, a waste of time!
+
+I waste a lot of time on Reddit. If I want to procrastinate on something, I’ll often open a new tab and dive down a Reddit-hole. But sometimes you need to turn on the blinders and dial down distractions. 2015 was one of these times — I was singularly focused on improving as a programmer, and Redditing was becoming a liability.
+
+I needed an abstinence plan.
+
+So it occurred to me: how about I lock myself out of my account?
+
+**Here’s what I did:**
+
+
+
+I set a random password on my account. Then I asked a friend to e-mail me this password on a certain date. With that, I’d have a foolproof way to lock myself out of Reddit. (Also changed the e-mail for password recovery to cover all the bases.)
+
+This should have worked.
+
+Unfortunately it turns out, friends are very susceptible to social engineering. The technical terminology for this is that they are “nice to you” and will give you back your password if you “beg them.”
+
+![](https://media.giphy.com/media/uB6rsQFg5yPzW/giphy.gif)
+
+Don’t look at me like that.
+
+After a few rounds of this failure mode, I needed a more robust solution. A little Google searching, and I came across this:
+
+
+
+
+
+Looks legit.
+
+Perfect — an automated, friend-less solution! (I’d alienated most of them by now, so that was a big selling point.)
+
+A bit sketchy looking, but hey, any port in a storm.
+
+
+
+For a while I set this up this routine — during the week I’d e-mail myself my password, on the weekends I’d receive the password, load up on internet junk food, and then lock myself out again once the week began. It worked quite well from what I remember.
+
+Eventually I got so busy with programming stuff, I completely forgot about it.
+
+### **Cut to two years later.** ###
+
+I’m now gainfully employed at Airbnb. And Airbnb, it so happens, has a large test suite. This means waiting, and waiting of course means internet rabbit holes.
+
+I decide to scrounge up my old account and find my Reddit password.
+
+
+
+Oh no.That’s not good.
+
+I didn’t remember doing this, but I must have gotten so fed up with myself that** I locked myself out until 2018**. I also set it to “hide,” so I couldn’t view the contents of the e-mail until it’s sent.
+
+What do I do? Do I just have to create a new Reddit account and start from scratch? But that’s *so much work.*
+
+I could write in to LetterMeLater and explain that I didn’t mean to do this. But they would probably take a while to get back to me. We’ve already established I’m wildly impatient. Plus this site doesn’t look like it has a support team. Not to mention it would be an embarrassing e-mail exchange. I started brainstorming elaborate explanations involving dead relatives about why I needed access to the e-mail…
+
+All of my options were messy. I was walking home that night from the office pondering my predicament, when suddenly it hit me.
+
+**The search bar.**
+
+I pulled up the app on my mobile phone and tried it:
+
+
+
+Hmm.
+
+Okay. So it’s indexing the subject for sure. What about the body?
+
+
+
+I try a few letters, and voila. It’s definitely got the body indexed. Remember: the body consisted entirely of my password.
+
+*Essentially, I’ve been given an interface to perform substring queries.* By entering in a string into the search bar, the search results will confirm whether my password contains this substring.
+
+**We’re in business.**
+
+I hurry into my apartment, drop my bag, and pull out my laptop.
+
+Algorithms problem: you are given a function `substring?(str)`, which returns true or false depending on whether a password contains any given substring. *Given this function, write an algorithm that can deduce the hidden password.*
+
+### The Algorithm ###
+
+So let’s think about this. A few things I know about my password: I know it was a long string with some random characters, probably something along the lines of `asgoihej2409g`. I *probably* didn’t include any upper-case characters (and Reddit doesn’t enforce that as a password constraint) so let’s assume for now that I didn’t — in case I did, we can just expand the search space later if the initial algorithm fails.
+
+We also have a subject line as part of the string we’re querying. And we know the subject is “password”.
+
+
+
+Let’s pretend the body is 6 characters long. So we’ve got six slots of characters, some of which may appear in the subject line, some of which certainly don’t. So if we take all of the characters that aren’t in the subject and try searching for each of them, we know for sure we’ll hit a unique letter that’s in the password. Think like a game of Wheel of Fortune.
+
+![](https://cdn-images-1.medium.com/max/800/1*LOzh--_Ujutrh_OKhjfNaw.png)
+
+We keep trying letters one by one until we hit a match for something that’s not in our subject line. Say we hit it.
+
+
+
+Once I’ve found my first letter, I don’t actually know where in this string I am. But I know I can start building out a bigger substring by appending different characters to the end of this until I hit another substring match.
+
+We’ll potentially have to iterate through every character in our alphabet to find it. Any of those characters could be correct, so on average it’ll hit somewhere around the middle, so given an alphabet of size `A`, it should average out to `A/2` guesses per letter (let’s assume the subject is small and there are no repeating patterns of 2+ characters).
+
+
+
+I’ll keep building this substring until it eventually hits the end and no characters can extend it further.
+
+
+
+But that’s not enough — most likely, there will be a prefix to the string that I missed, because I started in a random place. Easy enough: all I have to do is now repeat the process, except going backwards.
+
+
+
+Once the process terminates, I should be able to reconstruct the password. In total, I’ll need to figure out`L` characters(where `L` is the length), and need to expend on average `A/2` guesses per character (where `A` is the alphabet size), so total guesses = `A/2 * L`.
+
+To be precise, I also have to add another `2A` to the number of guesses for ascertaining that the string has terminated on each end. So the total is `A/2 * L + 2A`, which we can factor as `A(L/2 + 2)`.
+
+Let’s assume we have 20 characters in our password, and an alphabet consisting of `a-z` (26) and `0–9` (10), so a total alphabet size of 36. So we’re looking at an average of `36 * (20/2 + 2) = 36 * 12 = 432` iterations.
+
+Damn.
+
+This is actually doable.
+
+![](https://media.giphy.com/media/119cVU19ICcAKc/giphy.gif)
+
+Programming IRL
+
+### The Implementation ###
+
+First things first: I need to write a client that can programmatically query the search box. This will serve as my substring oracle. Obviously this site has no API, so I’ll need to scrape the website directly.
+
+Looks like the URL format for searching is just a simple query string, `www.lettermelater.com/account.php?**qe=#{query_here}**`. That’s easy enough.
+
+Let’s start writing this script. I’m going to use the Faraday gem for making web requests, since it has a simple interface that I know well.
+
+I’ll start by making an API class.
+
+```
+require 'faraday'
+
+class Api
+ BASE_URL = 'http://www.lettermelater.com/account.php'
+
+ def self.get(query)
+ Faraday.get(BASE_URL, qe: query)
+ end
+end
+```
+
+Of course, we don’t expect this to work yet, as our script won’t be authenticated into any account. As we can see, the response returns a 302 redirect with an error message provided in the cookie.
+
+```
+[10] pry(main)> Api.get(“foo”)
+=> #”Tue, 04 Apr 2017 15:35:07 GMT”,
+“server”=>”Apache”,
+“x-powered-by”=>”PHP/5.2.17",
+“set-cookie”=>”msg_error=You+must+be+signed+in+to+see+this+page.”,
+“location”=>”.?pg=account.php”,
+“content-length”=>”0",
+“connection”=>”close”,
+“content-type”=>”text/html; charset=utf-8"},
+status=302>
+```
+
+So how do we sign in? We need to send in our [cookies](http://stackoverflow.com/questions/17769011/how-does-cookie-based-authentication-work) in the header, of course. Using Chrome inspector we can trivially grab them.
+
+
+
+(Not going to show my real cookie here, obviously. Interestingly, looks like it’s storing `user_id` client-side which is always a great sign.)
+
+Through process of elimination, I realize that it needs both `code` and `user_id` to authenticate me… sigh.
+
+So I add these to the script. (This is a fake cookie, just for illustration.)
+
+```
+
+require 'faraday'
+
+class Api
+ BASE_URL = 'http://www.lettermelater.com/account.php'
+ COOKIE = 'code=13fa1fa011169ab29007fcad17b2ae; user_id=279789'
+
+ def self.get(query)
+ Faraday.get(BASE_URL, { qe: query }, Cookie: COOKIE).body
+ end
+end
+```
+
+```
+[29] pry(main)> Api.get(“foo”)
+=> “\n\n\n\n\t\n\t\n\t\n\tLetterMeLater.com — Account Information…
+
+[30] pry(main)> _.include?(“Haseeb”)
+=> true
+```
+
+It’s got my name in there, so we’re definitely logged in!
+
+We’ve got the scraping down, now we just have to parse the result. Luckily, this pretty easy — we know it’s a hit if the e-mail result shows up on the page, so we just need to look for any string that’s unique when the result is present. The string “password” appears nowhere else, so that will do just nicely.
+
+
+
+```
+def self.include?(substring)
+ get(substring).include?(‘password’)
+end
+```
+
+That’s all we need for our API class. We can now do substring queries entirely in Ruby.
+
+```
+[31] pry(main)> Api.include?('password')
+=> true
+[32] pry(main)> Api.include?('f')
+=> false
+[33] pry(main)> Api.include?('g')
+=> true
+```
+
+Now that we know that works, let’s stub out the API while we develop our algorithm. Making HTTP requests is going to be really slow and we might trigger some rate-limiting as we’re experimenting. If we assume our API is correct, once we get the rest of the algorithm working, everything should just work once we swap the real API back in.
+
+So here’s the stubbed API, with a random secret string:
+
+```
+class ApiStub
+ SECRET_PASSWORD = 'g420hpfjpefoj490rjgsd'
+
+ def self.include?(substring)
+ SECRET_PASSWORD.include?(substring)
+ end
+end
+```
+
+We’ll inject the stubbed API into the class while we’re testing. Then for the final run, we’ll use the real API to query for the real password.
+
+So let’s get started with this class. From a high level, recalling my algorithm diagram, it goes in three steps:
+
+1. First, find the first letter that’s not in the subject but exists in the password. This is our starting off point.
+2. Build those letters forward until we fall off the end of the string.
+3. Build that substring backwards until we hit the beginning of the string.
+
+Then we’re done!
+
+Let’s start with initialization. We’ll inject the API, and other than that we just need to initialize the current password chunk to be an empty string.
+
+```
+class PasswordCracker
+ def initialize(api)
+ @api = api
+ @password = ''
+ end
+end
+```
+
+Now let’s write three methods, following the steps we outlined.
+
+```
+ def crack!
+ find_starting_letter!
+ build_forward!
+ build_backward!
+ @password
+ end
+```
+
+
+Perfect. Now the rest of the implementation can take place in private methods.
+
+For finding the first letter, we need to iterate over each character in the alphabet that’s not contained in the subject. To construct this alphabet, we’re going to use `a-z` and `0–9`. Ruby allows us to do this pretty easily with ranges:
+
+```
+ALPHABET = ((‘a’..’z’).to_a + (‘0’..’9').to_a).shuffle
+```
+
+I prefer to shuffle this to remove any bias in the password’s letter distribution. This will make our algorithm query A/2 times on average per character, even if the password is non-randomly distributed.
+
+We also want to set the subject as a constant:
+
+```
+SUBJECT = ‘password’
+```
+
+That’s all the setup we need. Now time to write `find_starting_letter`. This needs to iterate through each candidate letter (in the alphabet but not in the subject) until it finds a match.
+
+```
+ private
+
+ def find_starting_letter!
+ candidate_letters = ALPHABET - SUBJECT.chars
+ @password = candidate_letters.find { |char| @api.include?(char) }
+ end
+```
+
+In testing, looks like this works perfectly:
+
+```
+PasswordCracker.new(ApiStub).send(:find_starting_letter!) # => 'f'
+```
+
+Now for the heavy lifting.
+
+I’m going to do this recursively, because it makes the structure very elegant.
+
+```
+def build_forward!
+ puts "Current password: #{@password}"
+ ALPHABET.each do |char|
+ guess = @password + char
+
+ if @api.include?(guess)
+ @password = guess
+ build_forward!
+ # once I'm done building forward, jump out of all stack frames
+ return
+ end
+ end
+ end
+```
+
+The code is surprisingly straightforward. Let’s see if it works with our stub API.
+
+```
+[63] pry(main)> PasswordCracker.new(ApiStub).crack!
+f
+fj
+fjp
+fjpe
+fjpef
+fjpefo
+fjpefoj
+fjpefoj4
+fjpefoj49
+fjpefoj490
+fjpefoj490r
+fjpefoj490rj
+fjpefoj490rjg
+fjpefoj490rjgs
+fjpefoj490rjgsd
+=> “fjpefoj490rjgsd”
+```
+
+Awesome. We’ve got a suffix, now just to build backward and complete the string. This should look very similar.
+
+```
+def build_backward!
+ puts "Current password: #{@password}"
+ ALPHABET.each do |char|
+ guess = char + @password
+
+ if @api.include?(guess)
+ @password = guess
+ build_backward!
+ return
+ end
+ end
+```
+
+
+In fact, there’s only two lines of difference here: how we construct the `guess`, and the name of the recursive call. There’s an obvious refactoring here, so let’s do it.
+
+```
+def build!(forward:)
+ puts "Current password: #{@password}"
+ ALPHABET.each do |char|
+ guess = forward ? @password + char : char + @password
+
+ if @api.include?(guess)
+ @password = guess
+ build!(forward: forward)
+ return
+ end
+ end
+ end
+```
+
+Now these other calls simply reduce to:
+
+```
+ def build_forward!
+ build!(forward: true)
+ end
+
+ def build_backward!
+ build!(forward: false)
+ end
+```
+
+And let’s see how it works in action:
+
+
+```
+Apps-MacBook:password-recovery haseeb$ ruby letter_me_now.rb
+Current password: 9
+Current password: 90
+Current password: 90r
+Current password: 90rj
+Current password: 90rjg
+Current password: 90rjgs
+Current password: 90rjgsd
+Current password: 90rjgsd
+Current password: 490rjgsd
+Current password: j490rjgsd
+Current password: oj490rjgsd
+Current password: foj490rjgsd
+Current password: efoj490rjgsd
+Current password: pefoj490rjgsd
+Current password: jpefoj490rjgsd
+Current password: fjpefoj490rjgsd
+Current password: pfjpefoj490rjgsd
+Current password: hpfjpefoj490rjgsd
+Current password: 0hpfjpefoj490rjgsd
+Current password: 20hpfjpefoj490rjgsd
+Current password: 420hpfjpefoj490rjgsd
+Current password: g420hpfjpefoj490rjgsd
+g420hpfjpefoj490rjgsd
+```
+
+Beautiful. Now let’s just add some more print statements and a bit of extra logging, and we’ll have our finished `PasswordCracker`.
+
+```
+require 'faraday'
+
+class PasswordCracker
+ ALPHABET = (('a'..'z').to_a + ('0'..'9').to_a).shuffle
+ SUBJECT = 'password'
+
+ def initialize(api)
+ @api = api
+ @password = ''
+ end
+
+ def crack!
+ find_starting_letter!
+ puts "Found first letter: #{@password}"
+ puts "\nBuilding forward!\n"
+ build_forward!
+ puts "\nBuilding backward!\n"
+ build_backward!
+ puts "Done! The result is #{@password}."
+ puts "We found it in #{@api.iterations} iterations"
+ @password
+ end
+
+ private
+
+ def find_starting_letter!
+ candidate_letters = ALPHABET - SUBJECT.chars
+ @password = candidate_letters.find { |char| @api.include?(char) }
+ end
+
+ def build_forward!
+ build!(forward: true)
+ end
+
+ def build_backward!
+ build!(forward: false)
+ end
+
+ def build!(forward:)
+ puts "Current password: #{@password}"
+ ALPHABET.each do |char|
+ guess = forward ? @password + char : char + @password
+
+ if @api.include?(guess)
+ @password = guess
+ build!(forward: forward)
+ return
+ end
+ end
+ end
+end
+
+class Api
+ BASE_URL = 'http://www.lettermelater.com/account.php'
+ COOKIE = 'code=13fa1fa011169ab29007fcad17b2ae; user_id=279789'
+ @iterations = 0
+
+ def self.get(query)
+ @iterations += 1
+ Faraday.get(BASE_URL, { qe: query }, Cookie: COOKIE).body
+ end
+
+ def self.include?(substring)
+ get(substring).include?('password')
+ end
+
+ def self.iterations
+ @iterations
+ end
+end
+
+class ApiStub
+ SECRET_PASSWORD = 'g420hpfjpefoj490rjgsd'
+ @iterations = 0
+
+ def self.include?(substring)
+ @iterations += 1
+ SECRET_PASSWORD.include?(substring)
+ end
+
+ def self.iterations
+ @iterations
+ end
+end
+```
+
+And now… the magic moment. Let’s swap the stub with the real API and see what happens.
+
+### The Moment of Truth ###
+
+Cross your fingers…
+
+`PasswordCracker.new(Api).crack!`
+
+
+
+(Sped up 3x)
+
+Boom. 443 iterations.
+
+Tried it out on Reddit, and login was successful.
+
+Wow.
+
+It… actually worked.
+
+Recall our original formula for the number of iterations: `A(N/2 + 2)`. The true password was 22 characters, so our formula would estimate `36 * (22/2 + 2) = 36 * 13 = 468` iterations. Our real password took 443 iterations, so our estimate was within 5% of the observed runtime.
+
+**Math.**
+
+![](https://media.giphy.com/media/26xBI73gWquCBBCDe/giphy.gif)
+
+tfw wtf ftw
+
+**It works.**
+
+Embarrassing support e-mail averted. Reddit rabbit-holing restored. It’s now confirmed: programming is, indeed, magic.
+
+(The downside is I am now going to have to find a new technique to lock myself out of my accounts.)
+
+And with that, I’m gonna get back to my internet rabbit-holes. Thanks for reading, and give it a like if you enjoyed this!
+
+*—Haseeb*
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 8b1ccb184fbd2898c18da673889fa2466202e965 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Fri, 7 Apr 2017 21:48:45 +0800
Subject: [PATCH 129/945] Create
creating-usability-with-motion-the-ux-in-motion-manifesto.md
---
...-with-motion-the-ux-in-motion-manifesto.md | 452 ++++++++++++++++++
1 file changed, 452 insertions(+)
create mode 100644 TODO/creating-usability-with-motion-the-ux-in-motion-manifesto.md
diff --git a/TODO/creating-usability-with-motion-the-ux-in-motion-manifesto.md b/TODO/creating-usability-with-motion-the-ux-in-motion-manifesto.md
new file mode 100644
index 00000000000..b7982b9da14
--- /dev/null
+++ b/TODO/creating-usability-with-motion-the-ux-in-motion-manifesto.md
@@ -0,0 +1,452 @@
+> * 原文地址:[Creating Usability with Motion: The UX in Motion Manifesto](https://medium.com/@ux_in_motion/creating-usability-with-motion-the-ux-in-motion-manifesto-a87a4584ddc)
+> * 原文作者:[Issara Willenskomer](https://medium.com/@ux_in_motion?source=post_header_lockup)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+
+
+# Creating Usability with Motion: The UX in Motion Manifesto #
+
+The following manifesto represents my answer to the question — “As a UX or UI, designer, how do I know when and where to implement motion to support usability?”
+
+Over the last 5 years, it has been my privilege to coach and mentor UX & UI designers in over 40 countries, and at hundreds of the top brands and design consultancies through my workshops and tutorials on UI Animation.
+
+After over fifteen years studying motion in user interfaces, I have come to the conclusion that there are 12 specific opportunities to support usability in your UX projects using motion.
+
+I call these opportunities ‘The 12 Principles of UX in Motion,’ and they can be stacked and combined synergistically in a myriad of innovative ways.
+
+I’ve broken the manifesto into 5 parts:
+
+1. Addressing the topic of UI Animation — it’s not what you think
+2. Realtime vs non-realtime interactions
+3. Four ways that motion supports usability
+4. Principles, Techniques, Properties and Values
+5. The 12 Principles of UX in Motion
+
+As a quick plug, if you want me to speak at your conference or lead an onsite workshop for your team on the exciting topic of motion and usability, go [here](https://uxinmotion.net/workshops-and-speaking/) . If you want to attend a class in your city go [here](https://uxinmotion.net/workshops-and-speaking/#classes) . Finally, if you want me to consult on your project, you can go [here](https://uxinmotion.net/consulting/) . To get added to my list, go [here](http://uxinmotion.net/joinnow)
+
+### It’s not about UI Animation ###
+
+Because the topic of motion in user interfaces is mostly understood by designers to be ‘UI Animation’—which it is not — I feel like I need to create a bit of context before we jump into the 12 Principles.
+
+‘UI Animation’ is typically thought of by designers as something that makes the user experience more delightful, but overall doesn’t add much value. Thus, UI Animation is often treated like the red-headed stepchild of UX (apologies to red-headed stepchildren everywhere). If at all, it usually comes at the end, as a final lipstick pass.
+
+Additionally, motion in the context of user interfaces has understandably been held to be under the domain of Disney’s 12 Principles of Animation, something I argue against in my article ‘[UI Animation Principles — Disney is Dead](https://medium.com/@ux_in_motion/ui-animation-principles-disney-is-dead-8bf6c66207f9) .’
+
+In my manifesto, I will be making the case that UI Animation is to the ‘12 UX in Motion Principles’ as construction is to architecture.
+
+By this I mean, that while a structure needs to be physically built to exist (requiring construction), the guiding hand which determines *what* gets built comes from the domain of Principles.
+
+Animation is all about tools. Principles are the practical application of ideas that guide the usage of tools and as such, Principles provide high leverage opportunities for designers.
+
+What most designers think of as ‘UI Animation’ is in fact the execution of a higher modality of design: the temporal behavior of interface objects during realtime and non-realtime events.
+
+
+### Realtime vs non-realtime interactions ###
+
+At this juncture, it is important to distinguish between ‘state’ and ‘act.’ The *state* of something in UX is fundamentally static, like a design comp. The *act *of something in UX is fundamentally temporal, and motion based. An object can be in the *state *of being masked or it can be in the *act* of being masked. If it is the latter, we know that motion is involved and in a way that could support usability.
+
+Additionally, all temporal aspects of interaction can be thought of as happening in realtime or non-realtime. Realtime means that the user is directly interacting with the objects in the user interface. Non-realtime means that the object behavior is post-interactive: it occurs *after *a user action, and is transitional.
+
+
+
+
+
+This is an important distinction.
+
+Realtime interactions can also be though of as ‘direct manipulation,’ in that the user is interacting with the interface objects directly and *immediately*. The behavior of the interface is happening *as the user is using it.*
+
+Non-realtime interactions happen only *after *input from the user and have the effect of briefly locking the user out of the user experience until the transition completes.
+
+These distinctions will give us leverage as we continue our journey into the world of the 12 Principles of UX in Motion.
+
+
+### Motion supports usability in four ways ###
+
+These four pillars represent the four ways where the temporal behavior of user experiences supports usability.
+
+#### Expectation ####
+
+Expectation fall into two areas — how users perceive what an object *is*, and how it *behaves*. Another way of saying this is that as designers, we want to minimize the gap between what the user expects, and what they experience.
+
+#### Continuity ####
+
+Continuity speaks both to the user flow and to the ‘consistency’ of the user experience. Continuity can be thought of in terms of ‘intra-continuity’ — the continuity within a scene, and ‘inter-continuity’ — the continuity within a series of scenes that make up the total user experience.
+
+#### Narrative ####
+
+Narrative is the linear progression of events in the user experience that results in a temporal/spatial framework. This can be thought of as the series of discreet moments and events that connect together throughout the user experience.
+
+#### Relationship ####
+
+Relationship refers to the spatial, temporal, and hierarchal representations between interface objects that guide user understanding and decision making.
+
+### Principles, Techniques, Properties, and Values ###
+
+[Tyler Waye](http://tylerwaye.com/learning-to-learn-principles-vs-techniques/) says it as good as any when he writes, “Principles… are the underlying premises and rules of function giving rise to any number of techniques. These elements remain consistent, no matter what is happening.” It bears repeating that Principles are agnostic of design.
+
+From there, we can imagine a hierarchy with Principles at the top, Techniques further down, Properties below that, and Values at the bottom.
+
+**Techniques** can be thought of as the various and unlimited executions of Principles and/or combination of Principles. I think of technique as akin to ‘style.’
+
+**Properties** are the specific object parameters that are being animated to create the technique. These include (and are not limited to) position, opacity, scale, rotation, anchor point, color, stroke-width, shape, etc.
+
+**Values** are the actual numeric property values that vary over time to create what we call ‘animation.’
+
+So to land the plane here (and jumping ahead a bit), we could say that a hypothetical UI animation reference is using the Obscuration Principle with a ‘blurred glass’ Technique that affects the Blur and Opacity Properties at a Value of 25px and 70% respectively.
+
+Now we have some tools to work with. And more importantly, these linguistic tools are agnostic of any specific prototyping tool.
+
+
+### The 12 Principles of UX in Motion ###
+
+Easing and Offset & Delay relate to *timing*. Parenting relates to *object relationship*. Transformation, Value Change, Masking, Overlay, and Cloning all relate to *object continuity*. Parallax relate to *temporal hierarchy*. Obscuration, Dimensionality and Dolly & Zoom both relate to *spatial continuity*.
+
+
+
+#### **Principle 1: Easing** ####
+
+*Object behavior aligns with user expectations when temporal events occur.*
+
+
+
+All interface objects exhibiting temporal behavior (whether realtime or non-realtime), ease. Easing creates and reinforces the ‘naturalism’ inherent in the seamlessness of user experiences, and creates a sense of continuity when objects behave *as users expect them to. *Incidentally*, *Disney refers to this as ‘[Slow In and Slow Out](https://en.wikipedia.org/wiki/12_basic_principles_of_animation#Slow_In_and_Slow_Out).’
+
+
+
+
+
+The example on the left has linear motion and looks ‘bad.’ The first example up top has *eased* motion and looks ‘good.’ All three above examples have the exact number of frames and take place over the exact same amount of time. The only difference is in their easing.
+
+As designers concerned with usability, we need to require of ourselves rigor and inquire, aside from aesthetics, which example supports usability more?
+
+The case I am presenting here is that a certain degree of skeuomorphism is inherent in easing. You can imagine an ‘easing gradient’ wherein behaviors that fall outside user expectations result in less usable interactions. In the case of properly eased motion, users experience the motion itself as seamless and largely invisible — which is a good thing in that it is *non-distracting*. Linear motion is visibly obvious and feels somehow off, unfinished, and jarring, and distracting.
+
+Now I’m going to completely contradict myself here and talk about the example on the right. The motion *isn’t* seamless. In fact, it has a ‘designed’ feel to it. We notice how the object lands. It feels different, yet it still feels more ‘correct’ than the example with linear motion.
+
+Can you employ easing and still have it not support (or worse case undermine) usability? The answer is yes, and there are several ways. One way is timing. If your timing is too slow (mushy to borrow from [Pasquele](https://medium.com/@pasql) ), or too fast, you can break expectation and distract user attention. Similarly, if your easing is misaligned with the brand or overall experience, this can also negatively impact expectation and seamlessness.
+
+What I want to open you to is a world of opportunity when it comes to eased motion. There are literally an infinite number of ‘easings’ that you as a designer can create and implement in your projects. All of these easings have their own expectation response they trigger in users.
+
+To summarize: when to use easing? Always.
+
+#### Principle 2: Offset & Delay ####
+
+*Defines object relationships and hierarchies when introducing new elements and scenes.*
+
+
+
+Offset & Delay is the second of only two UX in Motion Principles that is influenced by Disney’s Animation Principles, in this case from ‘[Follow Through and Overlapping Action](https://en.wikipedia.org/wiki/12_basic_principles_of_animation#Follow_Through_and_Overlapping_Action).’
+
+It is important to note, however, that the implementation, while similar in execution, differs in purpose and outcome. While Disney’s Principles result in ‘more appealing animations,’ the UI Animation Principles result in more usable experiences.
+
+The utility of this principle is that it pre-consciously sets the user up for success by ‘telling’ the user something about the nature of the objects in the interface. The narrative in the above reference is that the top two objects are united and the bottom object is separate. Perhaps the top two objects could be a non-interactive image and text, while the bottom object is a button.
+
+Even before the user registers what these objects *are, *the designer has already communicated to her—through motion — that the objects are somehow ‘separate.’ This is extremely powerful.
+
+
+
+Credit: [InVision](https://dribbble.com/InVisionApp)
+
+In the above example, the floating action button (FAB) transforms into header navigation elements comprised of three buttons. Because the buttons are ‘offset’ from each other temporally, they end up supporting usability through their ‘separateness.’ Said differently, the designer is using time itself to say — even before the user registers what the objects are — that the objects are separate. This has the effect of telling the user, completely independent from the visual design, part of the nature of the objects in the interface.
+
+To better show you how this works, I’ll show you an example that breaks the Offset & Delay Principle.
+
+
+
+Credit: [Jordi Verdu](https://dribbble.com/jordiverdu) (
+
+In the above example, the static visual design tells us that there are icons over a background. The presumption is that the icons are all separate from each other and do different things. However, the motion contradicts this.
+
+Temporally, the icons are grouped into rows and behave like a single object. Their titles are likewise grouped into rows, and also behave like a single object. The motion is telling the user something other than what her eyes see. In this case, we can say that the temporal behavior of the objects in the interface are not supporting usability.
+
+#### Principle 3: Parenting ####
+
+*Creates spatial and temporal hierarchal relationships when interacting with multiple objects.*
+
+
+
+Parenting is a powerful Principle that ‘relates’ objects in the user interface. In the above example, the ‘scale’ and ‘position’ property of the top or ‘child’ object is parented to the ‘position’ property of the bottom or ‘parent’ object.
+
+Parenting is the linking of object properties to other object properties. This creates object relationships and hierarchies in ways that support usability.
+
+Parenting also allows designers to better coordinate temporal events in the user interface, while communicating to users the nature of the object relationships.
+
+Recall that object Properties include the following — Scale, Opacity, Position, Rotation, Shape, Color, Value, etc. Any of these Properties can be linked to any other Properties to create synergistic moments in the User Experience.
+
+
+
+
+
+Credit: [Andrew J Lee](https://dribbble.com/lee_aj) , [Frank Rapacciuolo](https://dribbble.com/frankiefreesbie)
+
+In this above left example, the ‘y-axis’ Property of the ‘face’ element is the ‘child’ to the ‘x-axis’ Property of the round indicator parent. When the round indicator element moves along the horizontal axis, its ‘child’ element moves along with it horizontally *and* vertically (while being Masked — another Principle).
+
+The result is a hierarchal temporal narrative framework that occurs all at the same time. Of note is that the ‘faces’ object works as a series of ‘lockups’ in that, at each number, the ‘face’ is fully and not partially visible. The user experiences this seamlessly, though we can make the claim there is a subtle ‘usability cheat’ in this example.
+
+Parenting functions best as a ‘realtime’ interaction. As the user directly manipulates the interface objects, the designer communicates to the user — via motion — how the objects are linked, and the relationship between them.
+
+Parenting occurs in 3 forms: ‘direct parenting’ (see above two examples, ‘delayed parenting,’ and ‘inverse parenting,’ see below).
+
+
+
+
+
+Delayed Parenting (Credit: [AgenceMe](https://dribbble.com/AgenceMe) ) & Inverse Parenting (Credit: [AgenceMe](https://dribbble.com/AgenceMe) )
+
+#### Principle 4: Transformation ####
+
+*Creates a continuous state of narrative flow when object utility changes.*
+
+
+
+Much has already been written about the UX in Motion Principle ‘transformation.’ In some ways, it is the most obvious and penetrable of the animation principles.
+
+Transformation is the most discernible, largely because it stands out. A ‘submit’ button changing shape to become a radial progress bar and finally changing shape again to become a confirmation check mark is something that we notice. It grabs our attention, tells a story, and has completion.
+
+
+
+Credit: [Colin Garven](https://dribbble.com/ColinGarven)
+
+What Transformation does is seamlessly transition the user through the different UX states or ‘is’s’ (as in this *is* a button, this *is* a radial progress bar, this *is* a check mark) which eventually result in a desired outcome. The user has been drawn through these functional spaces to the ultimate destination.
+
+Transformation has the effect of ‘chunking’ cognitively separate key moments in the user experience into a seamless and continuous series of events. This seamlessness results in better user awareness, retention, and followthrough.
+
+#### Principle 5: Value change ####
+*Creates a dynamic and continuous narrative relationship when value subject changes.*
+
+
+
+Text based interface objects, that is to say numbers and text, can change their values. This is one of those ‘elusive obvious’ concepts.
+
+Text and number changes are so common that they pass us by without us bringing to them distinction and rigor to assess their role in supporting usability
+
+So what is the user experiencing when values change? In user experiences, the 12 UX in Motion Principles are opportunities to support usability. The three opportunities here are to connect the user to the *reality* behind the data, the concept of *agency,* and to the dynamic nature of the values themselves.
+
+Let’s look at the example of a user dashboard.
+
+
+
+
+
+When value based interface objects load with no ‘value change,’ what this conveys to the user is that the numbers are static objects. They’re like painted signs displaying a speed limit of 55 mph.
+
+The numbers and values are representations of things that are happening *in reality.* That reality could be time, income, game scores, business metrics, fitness tracking, etc. What we are distinguishing through motion is that the ‘value subject’ is dynamic and the values are reflecting something from that dynamic value set.
+
+Not only does this relationship get lost with static objects comprised visually of values, but another deeper opportunity is also lost.
+
+When we employ representations of dynamic systems in the form of motion based values, it activates a sort of ‘neurofeedback.’ Users, grasping the dynamic nature of their data can now be cause in altering those values and are empowered to become *agents*. When the values are static, there is less connection to the reality behind the values, and users lose their *agency*.
+
+
+
+
+
+
+
+Credit: [Barthelemy Chalvet](https://dribbble.com/BarthelemyChalvet), [Gal Shir](https://dribbble.com/galshir) , Unknown
+
+The Value Change Principle can occur both in realtime and non-realtime events. In realtime events, the user is interacting with the objects to change values. In non-realtime events, such as loaders and transitions, the values change without user input to reflect a dynamic narrative.
+
+#### Principle 6: Masking ####
+
+*Creates continuity in an interface object or object group when utility is determined by which part of the object or group is revealed or concealed.*
+
+
+
+The *act* of masking asking can be thought of as a relationship between the shape of the object and it’s utility.
+
+Because designers are familiar with ‘masking’ in the context of static design, it behooves us to bring distinction to the UX in Motion Principle ‘Masking’ as it occurs in time, *as an* act, and not as a *state*.
+
+Through the temporal use of revealing and concealing regions of an object, utility transitions in a continuous and seamless way. This also has the effect of preserving narrative flow.
+
+
+
+Credit: [Anish Chandran](https://dribbble.com/anish_chandran)
+
+In the above example, the header image changes bounding shape and position but not the content, and becomes an album. This has the effect of changing what the object *is,while preserving the content within the mask — a fairly neat trick*. *This occurs in non-realtime, as a transition, that is activated after a user takes an action.
+
+Remember, UI Animation Principles occur temporally and support usability through continuity, narrative, relationship, and expectation. In the above reference, while the object itself remains unchanged, it also has boundary and location, and these two factors determine what the object is.
+
+#### Principle 7: Overlay ####
+
+*Creates narrative and object spatial relationship in visual flatland when layered objects are location dependent.*
+
+
+
+Overlay supports usability by allowing users to utilize flatland ordering properties to overcome a lack of non-spatial hierarchies.
+
+To land the plane a bit, Overlay allows designers to use motion to communicate location dependent objects that exist behind or in front of others in non 3D space.
+
+
+
+
+
+Credit: [Bady](https://dribbble.com/bady), [Javi Pérez](https://dribbble.com/javiperez)
+
+In the example on the left, the foreground object slides to the right to reveal the location of additional background objects. In the example on the right, the entire scene slides down to reveal additional content and options (while using the Offset & Delay Principle to communicate the individuality of the photo objects).
+
+To a certain degree, as designers, the idea of ‘layers’ is so obvious as to be self-evident. We design with layers and the concept of layers are deeply internalized. However, we must be careful to distinguish between the process of ‘making’ verses ‘using.’
+
+As designers who are continually engaged in the process of ‘making,’ we are intimately familiar with all of the pieces of the object (including the hidden pieces) we are designing. As a user, however, those non visible pieces are by definition and practice, hidden both visually and cognitively.
+
+The Overlay Principle allows designers to communicate relationship between ‘z-axis’ positioned layers and in so doing, promote spatial orientation to their users.
+
+#### Principle 8: Cloning ####
+
+*Creates continuity, relationship and narrative, when new objects originate and depart.*
+
+
+
+When new objects are created in current scenes (and from current objects), it is important to narratively account for their appearance. In this manifesto, I argue forcefully for the importance of creating a narrative framework for object origin and departure. Simple opacity fades tend to not achieve this result. Masking, Cloning, and Dimensionality are three usability based approaches to produce strong narratives.
+
+
+
+
+
+
+
+Credit: [Jakub Antalík](https://dribbble.com/antalik) , [Jakub Antalík](https://dribbble.com/antalik) , Unknown
+
+In the above three examples, new objects are created from existing hero objects during the time the user’s attention is focused on those objects. This two fold approach — the directing of attention, and then leading the eye through the creation of a cloned new object — has the strong effect of communicating a clear and unambiguous chain of events: that action ‘x’ has the result ‘y’ of the creation of new child objects.
+
+#### Principle 9: Obscuration ####
+
+*Allows users to spatially orient themselves in relationship to objects or scenes not in the primary visual hierarchy.*
+
+
+
+Similar to the UX in Motion Principles of Masking, Obscuration lives as both a static and temporal phenomena.
+
+This can be confusing to designers who don’t have experience thinking temporally — that is, the moments *between* moments. Designers typically design screen to screen or task to task. Think of Obscuration as the *act* of being obscured and not the *state* of being obscured. A static design represents the state of being obscured. Introducing time gives us the *act* of an object being obscured.
+
+
+
+
+
+Credit: [Virgil Pana](https://dribbble.com/virgilpana), [Apple](http://www.apple.com/)
+
+From the above two examples, we can see that obscuration, which *looks like* transparent objects or overlays, is also a temporal interaction involving multiple properties in time.
+
+Various common techniques of this involve blur effects and a lessoning of overall object transparency. The user is made aware of an additional non primary context that she is operating in — that there is another world, as it were, ‘behind’ her primary object hierarchy.
+
+Obscuration allows designers to compensate for a single unified field of view, or ‘objective view,’ in user experiences.
+
+#### Principle 10: Parallax ####
+
+*Creates spatial hierarchy in visual flatland when users scroll.*
+
+
+
+‘Parallax,’ as a UX in Motion Principle describes different interface objects moving at different rates.
+
+Parallax allows user to focus on primary actions and content while maintaining design integrity. Background elements ‘recede’ perceptually and cognitively for the user during a Parallax event. Designers can use Parallax to separate out immediacy content from ambient or supportive content.
+
+
+
+
+
+Credit: [Austin Neill](https://dribbble.com/austinneill), [Michael Sevilla](https://dribbble.com/SVLA)
+
+The effect this has on the user, is to clearly define *for the duration of the interaction,* the various object relationships. Foreground objects, or objects that move ‘quicker’ appear to the user as ‘closer.’ Likewise, background objects or objects that move ‘slower’ are perceived as being ‘further away.’
+
+Designers can create these relationships, using time itself, to tell the user what objects in the interface are higher priority. Therefore it makes sense to push background or non-interactive elements further ‘back.’
+
+Not only does the user perceive the interface objects as now having a hierarchy beyond that which is determined in the visual design, but this hierarchy can now be leveraged into having the user grasp *the nature* of the user experience before being consciously aware of the design/content.
+
+#### Principle 11: Dimensionality ####
+
+*Provides a spatial narrative framework when new objects originate and depart.*
+
+
+
+Critical to User Experiences is the phenomenon of continuity as well as the sense of location.
+
+Dimensionality provides a powerful way to overcome the flatland non-logic of User Experiences.
+
+Humans are remarkably adept at using spatial frameworks to navigate both in the real world and in digital experiences. Providing spatial origin and departure references helps reinforce mental models of where users are in the UX.
+
+Additionally, the Dimensionality Principle overcomes the layering paradox in visual flatland wherein objects lacking depth exist on the same plane but occur as ‘in front of’ or ‘behind’ other objects.
+
+Dimensionality presents itself in three ways — Origami Dimensionality, Floating Dimensionality, and Object Dimensionality.
+
+**Origami Dimensionality** can be thought of in terms of ‘folding’ or ‘hinged’ three dimensional interface objects.
+
+
+
+
+
+Examples of Origami Dimensionality (Credit: [Eddie Lobanovskiy](https://dribbble.com/lobanovskiy) , [Virgil Pana](https://dribbble.com/virgilpana))
+
+Because multiple objects are combined into ‘origami’ structures, the hidden objects still can be said to ‘exist,’ spatially even though they are not visible. This effectively renders the User Experience as a continuous spatial event that the user navigates and creates an operating context both in the interaction model itself, and in the temporal behavior of the interface objects themselves.
+
+**Floating Dimensionality** gives interface objects a spatial origin and departure, making the interaction models intuitive and highly narrative.
+
+
+
+Example of Floating Dimensionality (Credit: [Virgil Pana](https://dribbble.com/virgilpana) )
+
+In the above example, Dimensionality is achieved through the use of 3D ‘cards.’ This provides a strong narrative framework that supports the visual design. The narrative is extended by ‘flipping’ the cards to access additional content and interactivity. Dimensionality can be a powerful way to introduce new elements in ways that minimize abruptness.
+
+**Object Dimensionality **results in dimensional objects with true depth and form.
+
+
+
+
+
+Examples of Object Dimensionality (Credit: [Issara Willenskomer](https://uxinmotion.net/) , [Creativedash](https://dribbble.com/Creativedash) )
+
+Here, multiple 2D layers are arranged in 3D space to form true dimensional objects. Their dimensionality is revealed during realtime and non-realtime transitional moments. The utility of Object Dimensionality is that users develop a keen awareness of object utility based on non-visible spatial locations.
+
+#### Principle 12: Dolly & Zoom ####
+
+*Preserves continuity and spatial narrative when navigating interface objects and spaces.*
+
+
+
+Dolly and Zoom are filmic concepts referring to the movement of objects relevant to the camera, and the size of the image itself in the frame smoothly changing from a long shot to a close up shot (or vice versa).
+
+In certain contexts, it is impossible to tell if an object is zooming, if it is moving towards the camera in 3D space, or if the camera is moving towards the object in 3D space (see below references). The below three examples illustrate the possible scenarios.
+
+![](https://cdn-images-1.medium.com/max/800/1*R9wPWQUu26wjibaTBUstqQ.gif)
+
+Is the layer dollying, zooming, or is the camera moving?
+As such, it is appropriate to treat the instances of ‘dollying’ and ‘zooming’ as separate but similar in that they involve continuous element and scenic transformations, and meet the requirements of the UX in Motion Principles: they support usability through motion.
+
+![](https://cdn-images-1.medium.com/max/400/1*I4yZ2k1zeo3qc9qrbn0LDw.gif)
+
+![](https://cdn-images-1.medium.com/max/400/1*XVtnYMrp8LhGJzcsF0Lw7Q.gif)
+
+![](https://cdn-images-1.medium.com/max/400/1*o2ellGNN8CTJbwUoJ0ts8Q.gif)
+
+The two images on the left are dollying, while the image on the right is zooming
+
+**Dolly** is a film term and applies to camera movement either toward or away from a subject (it also applies to horizontal ‘tracking’ movement as well, but is less relevant in a usability context).
+
+![](https://cdn-images-1.medium.com/max/800/1*8TYALn5P87i2OuuZfhfELg.gif)
+
+Credit: [Apple](http://www.apple.com/)
+
+Spatially in UX, this motion could refer either to a change in the viewers perspective, or to the perspective remaining static while the object changes position. The Dolly Principle supports usability through continuity and narrative, seamlessly transitioning between interface objects and destinations. Dolly can also incorporate the Dimensionality Principle, resulting in a more spatial experience, more depth, and communicating to the user additional areas or content that is ‘in front’ or ‘behind’ the current view.
+
+**Zoom** refers to events where neither the perspective nor the object is moving spatially, but rather the object itself is scaling (or our view of it is decreasing, resulting in the image enlarging). This communicates to the viewer that additional interface objects are ‘inside’ other objects or scenes.
+
+![](https://cdn-images-1.medium.com/max/800/1*I6-dXGCq9cXjAZGyVOkXrA.gif)
+
+Credit: [Apple](http://www.apple.com/)
+
+This allows for seamless transitions — both realtime and non-realtime — that support usability. This seamlessness employed in the Dolly & Zoom Principle is quite powerful when it comes to creating spatial mental models.
+
+If you made it this far, congratulations! That was a beast of a manifesto. I hope all the gifs loaded for you and didn’t kill your browser. I also really hope you got some value for yourself and some new tools and leverage in your interactive projects.
+
+I encourage you to learn more about how you can begin using motion as a design tool to support usability.
+
+Again, the final plug, if you want me to speak at your conference or lead an onsite workshop for your team on the exciting topic of motion and usability, go [here](https://uxinmotion.net/workshops-and-speaking/) . If you want to attend a class in your city go [here](https://uxinmotion.net/workshops-and-speaking/#classes) . Finally, if you want me to consult on your project, you can go [here](https://uxinmotion.net/consulting/) . To get added to my list, go [here](http://uxinmotion.net/joinnow) .
+
+This manifesto would not have been possible without the generous and patient contribution and constant feedback of [Kateryna Sitner](https://www.linkedin.com/in/katerynasitner/) from Amazon — thank you! Special thanks to [Alex Chang](https://www.linkedin.com/in/alexychang/) for his brainstorming and insistence that I land the plane, [Bryan Mamaril](http://ficuscreative.com/) at Microsoft for his eagle eye, Jeremey Hanson for his editing notes, [Eric Braff](https://www.linkedin.com/in/eric-braff-276504b) for being the insane UI motion guru he is, Rob Girling at [Artefact](http://artefactgroup.com/) for his belief in me all those years ago, [Matt Silverman](http://www.swordfish-sf.com/) for his inspiring talk on UI motion at the After Effects conference, [Bradley Munkowitz](http://gmunk.com/) for being an awesome roommate and getting me inspired in UI, [Pasquale D’Silva](https://medium.com/@pasql) for his incredible articles on motion, [Rebecca Ussai Henderson](https://medium.freecodecamp.com/@becca_u) for her awesome article on UI Choreography, [Adrian Zumbrunnen](https://medium.com/@azumbrunnen) for his awesome contributions to the topic of UI and motion, [Wayne Greenfield](http://www.seattlekombucha.com/) and [Christian Brodin](http://www.theapartmentinvestor.com/author/christian-brodin/) for being my mastermind brothers and always pushing me to level up, and all you thousands of UI Animators who keep cranking out inspiring gifs.
+
+
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 50a020d04b8ae6c32b5c5a02a0ef2b86cc16bdbf Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Fri, 7 Apr 2017 22:00:19 +0800
Subject: [PATCH 130/945] Create swift-lazy-initialization-with-closures.md
---
...swift-lazy-initialization-with-closures.md | 362 ++++++++++++++++++
1 file changed, 362 insertions(+)
create mode 100644 TODO/swift-lazy-initialization-with-closures.md
diff --git a/TODO/swift-lazy-initialization-with-closures.md b/TODO/swift-lazy-initialization-with-closures.md
new file mode 100644
index 00000000000..00d175ee6b5
--- /dev/null
+++ b/TODO/swift-lazy-initialization-with-closures.md
@@ -0,0 +1,362 @@
+> * 原文地址:[Swift Lazy Initialization with Closures](https://blog.bobthedeveloper.io/swift-lazy-initialization-with-closures-a9ef6f6312c)
+> * 原文作者:[Bob Lee](https://blog.bobthedeveloper.io/@bobthedev?source=post_header_lockup)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+
+# Swift Lazy Initialization with Closures #
+
+## Learn how to create objects with modularity and readability ##
+
+![](https://cdn-images-1.medium.com/max/2000/1*KNmIy5QAOeokXPW86TtVyA.png)
+
+Magic Keyboard 2 and Magic Mouse 2
+
+*Welcome my lovely readers. Good to see you here today. For those who are new, I’m Bob. Just for real quick, if you wish to be on my mailing list, you can sign up* [*here*](https://boblee.typeform.com/to/oR9Nt2) *and get more value for your learning with iOS development :)*
+
+### Motivation ###
+
+In the beginning of my iOS journey, I followed tutorials on YouTube. I saw a few using something like below to create UI objects.
+
+```
+let makeBox: UIView = {
+ let view = UIView()
+ return view
+}()
+```
+
+As a learner, I copied the practice and used it. One day, however, one of my readers asked me, “why do you add `{}` and why does `()` exist at the end? Is it a computed property?” I could not answer. I was a zombie.
+
+I wrote this tutorial for my younger self. Yet, some may find it useful.
+
+### Objectives ###
+
+There are three objectives. First, understand how to initialize an object using the unconventional way as shown above. Second, learn when to use `lazy var` in Swift. Last, join my mailing list.
+
+#### Prerequisites ####
+
+To fully enjoy the ride with me, I highly recommend you to be familiar with the topics below.
+
+1. [*Closures*](https://blog.bobthedeveloper.io/no-fear-closure-in-swift-3-with-bob-72a10577c564)
+2. [*Capture List and retention cycle [weak self]*](https://blog.bobthedeveloper.io/swift-retention-cycle-in-closures-and-delegate-836c469ef128)
+3. *Descent Object Oriented Programming*
+
+### Create UI Components ###
+
+Before I explain the unconventional method above, let’s look into your past. In order to create a button in Swift, you probably have done something like this,
+
+```
+// Determine Size
+let buttonSize = CGRect(x: 0, y: 0, width: 100, height: 100)
+
+// Create Instance
+let bobButton = UIButton(frame: buttonSize)
+bobButton.backgroundColor = .black
+bobButton.titleLabel?.text = "Bob"
+bobButton.titleLabel?.textColor = .white
+```
+
+This is *Okay.*
+
+Assume, you have to create three other buttons, you probably have to copy the code above and then change the name from `bobButton` to `bobbyButton`.
+
+It’s quite tedious.
+
+```
+// New Button
+let bobbyButton = UIButton(frame: buttonSize)
+bobbyButton.backgroundColor = .black
+bobbyButton.titleLabel?.text = "Bob"
+bobbyButton.titleLabel?.textColor = .white
+```
+
+To make things just a bit easier, you may
+
+![](https://cdn-images-1.medium.com/max/800/1*oDIPy0i4YzUnKVR4XYI4kg.gif)
+
+This works too with the keyboard shortcut: ctrl-cmd-e
+
+If you don’t wish to repeat yourself, you may create a function instead.
+
+```
+func createButton(enterTitle: String) -> UIButton {
+ let button = UIButton(frame: buttonSize)
+ button.backgroundColor = .black
+ button.titleLabel?.text = enterTitle
+ return button
+}
+
+createButton(enterTitle: "Yoyo") // 👍
+```
+
+However, in iOS development, it is rare that custom buttons look similar. Therefore, a function may require a lot more parameters including background color, title, border radius, shadow, and so on. You function may end up looking like,
+
+```
+func createButton(title: String, borderWidth: Double, backgrounColor, ...) -> Button
+```
+
+The code above is not ideal even if you add default parameters to the function. It decreases readability. Therefore, it’s better to stay with the tedious method above.
+
+But, is there any way we can make it less tedious and more organized? Of course. We’ve looked into your past — It’s time to step up and look into your future.
+
+### Introducing the Unconventional Way ###
+
+Before we create UI components with the unconventional way, let’s first answer the initial questions my reader asked. What does `{}` mean, and is it a `computed property`?
+
+*Nope, it’s just a* ***closure block***.
+
+First, let’s demonstrate how to create an object using a closure. We will design a struct called `Human`.
+
+```
+struct Human {
+ init() {
+ print("Born 1996")
+ }
+}
+```
+
+Now, this is how you create an object with a closure
+
+```
+let createBob = { () -> Human in
+ let human = Human()
+ return human
+}
+
+let babyBob = createBob() // "Born 1996"
+```
+
+*If the syntax above doesn’t look familiar to you, you may stop reading now, and go to* [*Fear No Closure with Bob*](https://blog.bobthedeveloper.io/no-fear-closure-in-swift-3-with-bob-72a10577c564) *, and bring some bullets.*
+
+Just to explain, `createBob` is a closure whose type is `() -> Human`. You’ve created an instance called, `babyBob` by calling `createBob()` .
+
+However, you had to create two constants: `createBob` and `babyBob`. What if you want to do everything in a single statement? Here you go.
+
+```
+let bobby = { () -> Human in
+ let human = Human()
+ return human
+}()
+```
+
+Now, the closure block executes itself through adding `()` at the end and `bobby` now has a `Human` object attached. Pretty good stuff.
+
+**You’ve learned how to initialize an object with a closure block.**
+
+Now, let’s apply to creating an UI object which should be similar to the example right above.
+
+```
+let bobView = { () -> UIView in
+ let view = UIView()
+ view.backgroundColor = .black
+ return view
+}()
+```
+
+Great, we can make it shorter. In fact, we don’t need to specify the type of the closure block. Instead, all we have to do is specify the type of the instance, `bobView`, for example.
+
+```
+let bobbyView: **UIView** = {
+ let view = UIView()
+ view.backgroundColor = .black
+ return view
+}()
+```
+
+Swift is able to infer that the closure block is `() -> UIView` based on the keyword, `return`.
+
+Now, take a look. The example right above should look identical to the “unconventional way” I feared.
+
+### Benefits of Init with Closures ###
+
+We discussed the tediousness of creating objects and the problem that arises from using a function. In your head, you must be thinking, “why should I use a closure block instead?”
+
+#### Easy to Duplicate ####
+
+I don’t like to use Storyboard, I love copy and pasting UI objects. In fact, I’ve a “library” of code in my computer. Let us assume that there is a button as shown below in the library.
+
+```
+let myButton: UIButton = {
+ let button = UIButton(frame: buttonSize)
+ button.backgroundColor = .black
+ button.titleLabel?.text = "Button"
+ button.titleLabel?.textColor = .white
+ button.layer.cornerRadius =
+ button.layer.masksToBounds = true
+return button
+}()
+```
+
+All I have to do is copy the entire lines, and then just change the name of `myButton` to `newButton` for the usage. Had I not used the closure method, I probably had to change the name of `button` to `newButton` 7–8 times. We could use the Xcode shortcut above, but why not make it just simpler.
+
+#### Look Cleaner ####
+
+Since objects are grouped together, it feels cleaner based on my eyes. Let’s compare
+
+```
+// Init with Closure
+let leftCornerButton: UIButton = {
+ let button = UIButton(frame: buttonSize)
+ button.backgroundColor = .black
+ button.titleLabel?.text = "Button"
+ button.titleLabel?.textColor = .white
+ button.layer.cornerRadius =
+ button.layer.masksToBounds = true
+return button
+}()
+
+let rightCornerButton: UIButton = {
+ let button = UIButton(frame: buttonSize)
+ button.backgroundColor = .black
+ button.titleLabel?.text = "Button"
+ button.titleLabel?.textColor = .white
+ button.layer.cornerRadius =
+ button.layer.masksToBounds = true
+return button
+}()
+```
+
+vs
+
+```
+// Init With Fingers
+let leftCornerButton = UIButton(frame: buttonSize)
+leftCornerButton.backgroundColor = .black
+leftCornerButton.titleLabel?.text = "Button"
+leftCornerButton.titleLabel?.textColor = .white
+leftCornerButton.layer.cornerRadius =
+leftCornerButton.layer.masksToBounds = true
+
+let rightCornerButton = UIButton(frame: buttonSize)
+rightCornerButton.backgroundColor = .black
+rightCornerButton.titleLabel?.text = "Button"
+rightCornerButton.titleLabel?.textColor = .white
+rightCornerButton.layer.cornerRadius =
+rightCornerButton.layer.masksToBounds = true
+```
+
+Although creating an object with the closure add a couple lines more, I feel less overwhelmed since I only have to add attributes to `button` rather than `rightCornerButton` or `leftCornerButton`.
+
+*In fact, if the name of a button gets more descriptive, often times it requires fewer lines to create an object with a closure block.*
+
+**You’ve accomplished the first objective. Congratulations**
+
+### Lazy Init Application ###
+
+You’ve come a long way. It’s time to meet the second objective of this tutorial.
+
+You might have seen something like this below
+
+```
+class IntenseMathProblem {
+ lazyvar complexNumber: Int = {
+ // imagine it requires a lot of CPU
+ 1 * 1
+ }()
+}
+```
+
+What `lazy` allows you to do is, the `complexNumber` property will be only calculated when you try to access it. For example,
+
+```
+let problem = IntenseMathProblem
+problem() // No value for complexNumber
+```
+
+Currently, there is no value for `complexNumber`. However, once you access the property,
+
+```
+problem().complexNumber // Now returns 1
+```
+
+The `lazy var` is often used to sort database and fetch data from any backend services because you definitely don’t want to calculate and sort everything when you create an object.
+
+*In fact, your phone will crash since the object is super bloated and the RAM can’t handle.*
+
+### Application ###
+
+Below is just an application of `lazy var`.
+
+#### Sorting ####
+
+```
+class SortManager {
+ lazy var sortNumberFromDatabase: [Int] = {
+ // Sorting logic
+ return [1, 2, 3, 4]
+ }()
+}
+```
+
+#### Image Compression ####
+
+```
+class CompressionManager {
+ lazy var compressedImage: UIImage = {
+ let image = UIImage()
+ // Compress the image
+ // Logic
+ return image
+ }()
+}
+```
+
+### Rules with `Lazy` ###
+
+1. You can’t use `lazy` with `let` since there is no initial value, and it is attained later when it is accessed.
+2. You can’t use it with a `computed property` since computed property is always recalculated (requires CPU) when you modify any of the variables that has a relationship with the `lazy` property.
+3. `Lazy` is only valid for members of a struct or class
+
+### Does Lazy Capture? ###
+
+So, if you’ve read the previous article on [Retention Cycle in Closures and Delegate](https://blog.bobthedeveloper.io/swift-retention-cycle-in-closures-and-delegate-836c469ef128) , you might wonder. Let’s test it out. Create a class called `BobGreet`. It has two properties: `name` whose type is `String` and `greeting` whose type is also `String` but initialized with a closure block.
+
+```
+class BobGreet {
+ var name = "Bob the Developer"
+ lazy var greeting: String = {
+ return "Hello, \(self.name)"
+ }()
+
+deinit {
+ print("I'm gone, bruh 🙆")}
+ }
+}
+```
+
+The closure block *might* have a strong reference to `BobGuest` but let’s attempt to deallocate.
+
+```
+var bobGreet: BobGreet? = BobClass()
+bobGreet?.greeting
+bobClass = nil // I'm gone, bruh 🙆
+```
+
+No need to worry about `[unowned self]` The closure block does not have a reference to the object. Instead, it just copies `self` within the closure block. If you are confused by the previous statement, feel free to read [Swift Capture Lists](https://blog.bobthedeveloper.io/swift-capture-list-in-closures-e28282c71b95) to learn more. 👍
+
+### Last Remarks ###
+
+I learned a quite a bit while preparing for this tutorial. I hope you did as well. I’d appreciate your genuine fat ❤️. But, there is just one more. As the last objective, if you wish to on my mailing list and receive greater value from me, you can sign up right [**here**](https://boblee.typeform.com/to/oR9Nt2) .
+
+As you can see by the cover photo, I recently bought Magic Keyboard and Mouse. They are pretty good and increase my productivity a lot. You can get the mouse [here](http://amzn.to/2noHxgl) or the keyboard [here](http://amzn.to/2noHxgl). I never regret despite the price. 😓
+
+> [Source Code](https://github.com/bobthedev/Blog_Lazy_Init_with_Closures)
+
+### Swift Conference I Will Join ###
+
+I will be joining my first conference @[SwiftAveir](https://twitter.com/SwiftAveiro) o from June 1–2. A friend of mine, [Joao](https://twitter.com/NSMyself) , is helping organize the conference, so I’m super excited. You can learn more about the event [here](http://swiftaveiro.xyz) !
+
+#### Recommended Articles ####
+
+> Intro to Functional Programming ([Blog](https://blog.bobthedeveloper.io/intro-to-swift-functional-programming-with-bob-9c503ca14f13))
+
+> My Favorite Xcode Shortcuts ([Blog](https://blog.bobthedeveloper.io/intro-to-swift-functional-programming-with-bob-9c503ca14f13) )
+
+### Bob the Developer ###
+
+I’m an iOS instructor from Seoul, 🇰🇷. Feel free to get to know me on [Instagram](https://instagram.com/bobthedev) . I post regular updates on [Facebook Page](https://facebook.com/bobthedeveloper) and 🖨 on Sat 8pm EST.
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 360e4e55dce24e687e9b1d80344131a2b70d5f58 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Fri, 7 Apr 2017 22:03:19 +0800
Subject: [PATCH 131/945] Update secure-web-app-http-headers.md
---
TODO/secure-web-app-http-headers.md | 6 ------
1 file changed, 6 deletions(-)
diff --git a/TODO/secure-web-app-http-headers.md b/TODO/secure-web-app-http-headers.md
index 4514e6c4c2a..e2f1066fd33 100644
--- a/TODO/secure-web-app-http-headers.md
+++ b/TODO/secure-web-app-http-headers.md
@@ -6,12 +6,6 @@
## [How To Secure Your Web App With HTTP Headers](https://www.smashingmagazine.com/2017/04/secure-web-app-http-headers/) ##
-
-Today, too many websites are still inaccessible. In our new book *Inclusive Design Patterns*, we explore how to craft **flexible front-end design patterns** and make **future-proof and accessible interfaces without extra effort**. Hardcover, 312 pages. [Get the book now!](https://shop.smashingmagazine.com/products/inclusive-design-patterns?utm_source=magazine&utm_campaign=inclusive-design-patterns&utm_medium=html-ad-content-1)
-
-[![](https://www.smashingmagazine.com/wp-content/uploads/2016/10/inclusive-design-pattners-250px.png)](https://shop.smashingmagazine.com/products/inclusive-design-patterns?utm_source=magazine&utm_campaign=inclusive-design-patterns&utm_medium=html-ad-content-1)
-
-
Web applications, be they thin websites or thick single-page apps, are notorious targets for cyber-attacks. In 2016, approximately [40% of data breaches](http://www.verizonenterprise.com/verizon-insights-lab/dbir/2016/) originated from attacks on web apps — the leading attack pattern. Indeed, these days, understanding cyber-security is not a luxury but rather **a necessity for web developers**, especially for developers who build consumer-facing applications.
HTTP response headers can be leveraged to tighten up the security of web apps, typically just by adding a few lines of code. In this article, we’ll show how web developers can use HTTP headers to build secure apps. While the code examples are for Node.js, setting HTTP response headers is supported across all major server-side-rendering platforms and is typically simple to set up.
From 3a3a9df00e43ef7a88fb8d62a78c464ce9d0d44a Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Fri, 7 Apr 2017 22:15:05 +0800
Subject: [PATCH 132/945] Create on-loser-experience-design.md
---
TODO/on-loser-experience-design.md | 70 ++++++++++++++++++++++++++++++
1 file changed, 70 insertions(+)
create mode 100644 TODO/on-loser-experience-design.md
diff --git a/TODO/on-loser-experience-design.md b/TODO/on-loser-experience-design.md
new file mode 100644
index 00000000000..733b4435aba
--- /dev/null
+++ b/TODO/on-loser-experience-design.md
@@ -0,0 +1,70 @@
+> * 原文地址:[On Loser Experience Design](https://medium.com/on-human-centric-systems/on-loser-experience-design-1916629c36fc)
+> * 原文作者:[Matt LeMay](https://medium.com/@mattlemay?source=post_header_lockup)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+# On Loser Experience Design #
+
+## Designing platforms and products that aren’t just for winners, “influencers,” and “thought leaders.” ##
+
+![](https://cdn-images-1.medium.com/max/800/1*kVOEiUv3YK8tcYEa5QKmLA.jpeg)
+
+Turntable.fm, a Great Lesson in Loser Experience Design
+
+Do you remember turntable.fm? The idea behind the product was simple and brilliant: you join a virtual “room” with friends and/or strangers, and take turns DJing for each other. Like a lot of products that experience meteoric early-stage growth, it took a real-world behavior — social listening — and created a digital proxy that could instantly connect people across geographic and cultural divides. I was super-excited when turntable.fm launched, and I was not alone.
+
+But the more I used turntable.fm, the more I started to feel like — well, a *loser*. I was primarily interested in discovering new music, not in becoming a **star virtual DJ**. But the platform seemed to offer no encouragement or incentives for being a good listener, no shiny prizes for thoughtfully engaging in conversation with my virtual roommates. My avatar remained puny and plain, while others grew mighty, adorned with jewels, and outfitted with cool animal costumes. Eventually, I just gave up.
+
+**Turntable.fm suffered from bad loser experience design: making casual users feel like unwitting competitors in a game they cannot win.**
+
+In the absence of good loser experience design, products and platforms turn into ghost towns inhabited by thirsty would-be “influencers,” howling desperately into a void that was once occupied by curious, casual users. And in a world obsessed with performance metrics and status markers, bad loser experience design is all around us.
+
+#### The Patterns of Bad Loser Experience Design ####
+
+Here are a few signs that your product may be suffering from **bad loser experience design:**
+
+![](https://cdn-images-1.medium.com/max/600/1*k_ZpnygG7JhLUteHxZOJyQ.png)
+
+Illustration by [Joan LeMay](http://joanlemay.com)
+
+- **You have a “leaderboard” or a points system.**
+
+Yes, in the short-term, people may engage with a product for an abstract reward such as “points” or “[coins](https://techcrunch.com/2015/12/09/swarm-now-lets-you-spend-those-coins-on-upgraded-stickers/) .” But watch what happens as your users see themselves fall to the bottom of that “leaderboard” or fail to get any real value out of the time they’ve invested in earning those shiny trinkets. Competing for something only to realize that it’s worthless is embarrassing, frustrating, and makes you feel *like a huge loser*. Gratuitous “gamification” is one of the most odious and lazy patterns of bad loser experience design — and in the long term, it [doesn’t work](http://www.gartner.com/newsroom/id/2251015) .
+
+- **You visually distinguish the winners from the losers.**
+
+Twitter’s verification system was designed to foster trust by “verifying” that people are who they say they are. But in its early days, “verification” felt like a [badge of honor for “important” people](http://anildash.com/2013/03/what-its-like-being-verified-on-twitter.html) and an [aspirational delineator between the haves and have-nots](http://www.xojane.com/tech/how-to-get-verified-on-twitter) . Twitter took a great step towards mitigating this bad loser experience design when they bulk-verified journalists, making it clear that there was a functional *purpose* to this distinction and conferring it on people who are not high-status celebrities.
+
+- **You over-index on popularity metrics.**
+
+This is perhaps the most widespread pattern of bad loser experience design. Trying to decide what content to surface? How about the most popular content! Trying to find some “interesting” users? Look for the ones who have the most followers! This is the “rich get richer” problem of bad loser experience design — by building platforms that reward the people who are already succeeding, you create a permanent and inaccessible overclass that tends to reward the people who are the most ruthless and aggressive about increasing their own status — even if it means gaming the system.
+
+- **You treat casual users like failed power users.**
+
+Zach Holman’s “[Don’t Give Your Users the Shit Work](https://zachholman.com/posts/shit-work/) ” does a great job of describing this bad loser experience pattern in action: a new user gets onboarded, and is immediately presented with a barrage of “power user” features that will help them get the *most* out of this product. If they neglect to use these features, they find themselves stuck in a kind of permanent limbo, nagged by reminders to use features they don’t need and unable to derive immediate value from the product. If you treat casual users like failed power users, they will start to *feel *like failures — and they will leave.
+
+#### Tips for Good Loser Experience Design ####
+
+While bad loser experience design can significantly harm a product, *good* loser experience design can help foster a broad, engaged, and self-sustaining user base. Here are a few tips for good loser experience design:
+
+- **Help people find *their* people, not the most popular people.**
+
+When platforms focus on shared interests and social bonds over “likes” and “favorites,” they help everybody find a place where they belong. Instagram has done a great job doing this with their discovery features, consistently surfacing people who are adjacent to *your* people, not people with the most likes or followers. Their new discovery features have helped me find my [guitar](https://www.instagram.com/leoleoband/) [people](https://www.instagram.com/lostincrystalcanyons/) , my [cat](https://www.instagram.com/scruffles_fatcat/)[people](https://www.instagram.com/12catslady/) , and even to discover a whole new community of [prairie](https://www.instagram.com/rinran032/) [dog](https://www.instagram.com/prairiedogpack/) [people](https://www.instagram.com/pimpa_wan/) . As I’ve used these discovery features more, I’ve started to pay *less attention to how many “follows” and “likes” I get, and more attention to the people with whom I interact* — which is a surefire signal of great loser experience design.
+
+- **Give “passive” users meaningful ways to engage.**
+
+Tumblr’s reblog feature is a great example of good loser experience design. If somebody sees something they like, they can *share it to their entire network*, rather than just leaving a comment or a “like.” This helps people engage with each other more deeply, and means that a post written by somebody with a small number of followers can find a wide audience as it makes its way through networks and communities of people. It also gives people who don’t want to “write” a post a meaningful way to contribute and engage.
+
+- **Test your product with people who are not “power users.”**
+
+Finally, and perhaps most importantly, break out of the design and testing patterns that lead to equating “power users” with “good users.” Over-reliance on internal “dogfooding,” where new products and features are tested primarily with a company’s own employees, is a one-way ticket to bad loser experience design. Dismissing user testing candidates who are not over the moon for your product is another surefire road to bad loser experience design. Think through the needs and behaviors of casual users as extensively as you think through those of “power users” — and ask yourself, “if I only use this product a few times a week, will it make me feel like a *loser*?”
+
+#### Loser Experience Design in the Wild ####
+
+Has your day-to-day usage of a product or platform been affected by bad loser experience design? Leave a response and let me know your thoughts. And be sure to hit that “like” button, so that I can feel like a high-status *winner* right here on medium dot com.
+
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From c7230b4ac0ba734d2c4062851c97149fcf7a2270 Mon Sep 17 00:00:00 2001
From: sqrtthree
Date: Fri, 7 Apr 2017 22:22:00 +0800
Subject: [PATCH 133/945] update link
---
TODO/swift-lazy-initialization-with-closures.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/swift-lazy-initialization-with-closures.md b/TODO/swift-lazy-initialization-with-closures.md
index 00d175ee6b5..051ab41f287 100644
--- a/TODO/swift-lazy-initialization-with-closures.md
+++ b/TODO/swift-lazy-initialization-with-closures.md
@@ -1,5 +1,5 @@
> * 原文地址:[Swift Lazy Initialization with Closures](https://blog.bobthedeveloper.io/swift-lazy-initialization-with-closures-a9ef6f6312c)
-> * 原文作者:[Bob Lee](https://blog.bobthedeveloper.io/@bobthedev?source=post_header_lockup)
+> * 原文作者:[Bob Lee](https://blog.bobthedeveloper.io/@bobthedev)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:
> * 校对者:
From de20c171c428fc387c46314efd0266f8710ce9bb Mon Sep 17 00:00:00 2001
From: zhuzi
Date: Sat, 8 Apr 2017 00:22:18 +0800
Subject: [PATCH 134/945] update according to proof-reading
---
TODO/beyond-browser-web-desktop-apps.md | 140 ++++++++++++------------
1 file changed, 69 insertions(+), 71 deletions(-)
diff --git a/TODO/beyond-browser-web-desktop-apps.md b/TODO/beyond-browser-web-desktop-apps.md
index 7b4f8ffcec8..9803dd3b2c6 100644
--- a/TODO/beyond-browser-web-desktop-apps.md
+++ b/TODO/beyond-browser-web-desktop-apps.md
@@ -2,31 +2,31 @@
> * 原文作者:本文已获原作者 [Adam Lynch](https://www.smashingmagazine.com/author/adamlynch/) 授权
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者: [bambooom](https://github.com/bambooom)/[imink](https://github.com/imink)
-> * 校对者:
+> * 校对者:[bambooom](https://github.com/bambooom)/[sunui](https://github.com/sunui)
## 超越浏览器:从 web 应用到桌面应用
-一开始我是个 web 开发者,现在我是个全栈开发者,但从未想过在桌面上有所作为。我热爱 web 技术,热爱这个无私的社区,热爱它对于开源的友好,尝试挑战极限。我热爱探索好看的网站和强大的应用。当我被指派做桌面应用的任务的时候,我非常忧虑和害怕,因为那看起来很难,或者至少非常不同。
+一开始我是个 web 开发者,现在我是个全栈开发者,但从未想过在桌面上有所作为。我热爱 web 技术,热爱这个无私的社区,热爱它对于开源的友好,尝试挑战极限。我热爱探索好看的网站和强大的应用。当我被指派做桌面应用任务的时候,我非常忧虑和害怕,因为那看起来很难,或者至少不一样。
这并不吸引人,对吧?你需要学一门新的语言,甚至三门?想象一下过时的工作流,古旧的工具,没有任何你喜欢的有关 web 的一切。你的职业发展会被怎样影响呢?
-别慌,深呼吸,现实情况是,作为 web 开发者,你已经拥有一切开发现代桌面应用所需的一切技能,感谢新的强大的 API,你甚至可以在桌面应用中发挥你最大的潜能。
+别慌,深呼吸,现实情况是,作为 web 开发者,你已经拥有开发现代桌面应用所需的一切技能,得益于新的强大的 API,你甚至可以在桌面应用中发挥你最大的潜能。
本文将会介绍使用 [NW.js](http://nwjs.io/) 和 [Electron](https://electron.atom.io/) 开发桌面应用,它们的优劣,使用同一套代码库给桌面、web,甚至更多。
### 为什么?
-首先,为什么会有人开发桌面应用?任何现有的 web 应用(不同于网站,如果你认为它们是不同的)都可能适合变成一个桌面应用。你可以围绕任何可以从与用户系统集中中获益的 web 应用构建桌面应用;例如本地通知,开机启动,与文件的交互等。有些用户单纯更喜欢在自己的电脑中永久保存一些 app,无论是否联网都可以访问。
+首先,为什么会有人开发桌面应用?任何现有的 web 应用(不同于网站,如果你认为它们是不同的)都可能适合变成一个桌面应用。你可以围绕任何可以从与用户系统集成中获益的 web 应用构建桌面应用;例如本地通知,开机启动,与文件的交互等。有些用户单纯更喜欢在自己的电脑中永久保存一些 app,无论是否联网都可以访问。
也许你有个想法,但只能用作桌面应用,有些事情只是在 web 应用中不可能实现(至少还有一点,但更多的是这一点)。你可能想要为公司内部创建一个独立的功能性应用程序,而不需要任何人安装除了你的 app 之外的任何内容(因为内置 Node.js )。也许你有个有关 Mac 应用商店的想法,也许只是你的一个个人兴趣的小项目。
-很难总结为什么你应该考虑开发桌面应用,因为真的有很多类型的应用你可以创建。这非常取决于你想要达到什么目的,API 是否足够有利于开发,离线使用将多大程度上增强用户体验。在我的团队,这些都是毋庸置疑的,因为我们在开发一个[聊天应用程序](https://teamwork.com/chat)。另一方面来说,一个依赖于网络而没有任何与系统集成的桌面应用应该是一个 web 应用,并且只是 web 应用。当用户并不能从桌面应用中获得比在浏览器中访问一个网址更多的价值的时候,期待用户下载你的应用(其中自带浏览器以及 Node.js)是不公平的。
+很难总结为什么你应该考虑开发桌面应用,因为真的有很多类型的应用你可以创建。这非常取决于你想要达到什么目的,API 是否足够有利于开发,离线使用将多大程度上增强用户体验。在我的团队,这些都是毋庸置疑的,因为我们在开发一个[聊天应用程序](https://teamwork.com/chat)。另一方面来说,一个依赖于网络而没有任何与系统集成的桌面应用应该做成一个 web 应用,并且只做 web 应用。当用户并不能从桌面应用中获得比在浏览器中访问一个网址更多的价值的时候,期待用户下载你的应用(其中自带浏览器以及 Node.js)是不公平的。
-比起描述你个人应该建造的桌面应用以及为什么建造,我更希望的是激发一个想法,或者只是激发你对这篇文章的兴趣。继续往下读来看看用 web 技术构造一个强大的桌面应用是多么简单,以及需要承受的困难超过(或者相同)创造一个 web 应用。
+比起描述你个人应该建造的桌面应用及其原因,我更希望的是激发一个想法,或者只是激发你对这篇文章的兴趣。继续往下读来看看用 web 技术构造一个强大的桌面应用是多么简单,以及创建完成(或者在这过程中)一个 web 应用你需要承受什么。。
### NW.js
-桌面应用已经有很长一段时间了,我知道你没有很多时间,所以我们跳过一些历史,从2011年的上海开始。来自 Intel 开源技术中心的 Roger Wang 创造了 node-webkit,一个概念验证的 Node.js 模块,这个模块可以让用户生成一个自带 WebKit 内核的浏览器窗口并直接在 `
+
+
+
+
+
+
+
+
+ You signed in with another tab or window. Reload to refresh your session.
+ You signed out in another tab or window. Reload to refresh your session.
+
+
+
+
+
+
+
+
+
+
+
+
-> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From dea537b873dcacec4bc5b2ea99a122b7767799dc Mon Sep 17 00:00:00 2001
From: Wenlin Ou
Date: Fri, 7 Apr 2017 22:02:31 -0400
Subject: [PATCH 140/945] ...
---
...ng-will-change-until-you-start-building.md | 807 +-----------------
1 file changed, 47 insertions(+), 760 deletions(-)
diff --git a/TODO/nothing-will-change-until-you-start-building.md b/TODO/nothing-will-change-until-you-start-building.md
index 46f78f0efb5..0d57984107d 100644
--- a/TODO/nothing-will-change-until-you-start-building.md
+++ b/TODO/nothing-will-change-until-you-start-building.md
@@ -1,803 +1,90 @@
+> * 原文地址:[Nothing will change until you start building.](https://medium.freecodecamp.com/nothing-will-change-until-you-start-building-2681e85e7bdc)
+> * 原文作者:[Jonathan Z. White](https://medium.freecodecamp.com/@JonathanZWhite?source=post_header_lockup)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+---
+![](https://cdn-images-1.medium.com/max/1000/1*EwHpnCZ70FtJMi-lNSl-9Q.png)
+# Nothing will change until you start building.
+This past week, I was in a Lyft. My driver was telling me about all of her ideas for side projects. She had ideas for a children’s book, an app that helps people find parking, and a more efficient way to package gifts. The problem was she was frozen by indecision. She had the ideas but she didn’t know where to start.
+This is what I told her. **Start by building. Pick one project and do whatever you have to do to ship it.** If you want to write a book, start with writing a page a day. If you want to build an app, start with some sketches. Anyone can do it.
-
-
-
-
+This advice applies to all creators. Once you start building and launching your projects, you won’t be able to stop. Building will become part of your identity. And even if your project fails, you’ll keep at it.
+The steps you take today will compound over time. Look at great product makers like [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), and [Sebastian Dobrincu](https://twitter.com/Sebyddd). **They don’t wait for the right opportunity.** They build and ship weekly.
+In an interview on [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics), [Josh Pigford](https://twitter.com/Shpigford) disclosed that before he built Baremetrics, he had iterated through a *hundred* failed ideas. Today, Baremetrics makes over 60,000 in monthly recurring revenue.
-
-
-
-
-
-
-
-
- gold-miner/nothing-will-change-until-you-start-building.md at translate · owenlyn/gold-miner
-
-
-
+![](https://cdn-images-1.medium.com/max/800/1*BzmVaqAKEzNRybUmMYOxdA.png)
-
-
+With that in mind, here are some tactics I’ve compiled from friends and acquaintances who are successful product makers.
-
-
-
-
-
-
+---
-
+### 1. Define your goal
-
-
-
+Early on, figure out the main goal of your project. If you want to break out of your usual design style, focus on the design. If you want to test a new framework for your front-end, focus on the code. If you want to acquire a hundred users, focus on selling.
-
-
+**Define your goal early so you don’t get distracted and end up not achieving what you set out to do.** Perhaps your project doesn’t become the next Snapchat, but by working towards your goal you will have honed the tools in your product-making arsenal. Little by little, you’ll be able to design, code, and sell faster and more efficiently.
+### 2. Stay on track
+Sometimes product makers fail to ship because they lose motivation. **Set small goals; meeting those goals will help you build momentum.** Small wins will compound.
+![](https://cdn-images-1.medium.com/max/800/1*ESildSVTSSOnFXGDxD-l9w.png)
-
+Another thing you can do to stay on track is to be more public about your work. Share your progress online and with your peers. People love to see things like wireframes and will often times provide valuable input.
+One trick that I find particularly motivating is making bets with friends. I tell them, if I don’t complete a task by a deadline, I’ll give them a hundred dollars. For example, I might set my task to getting twenty people to pay me within two weeks of launching a project.
-
+### 3. Solve a problem
-
-
+Start by solving a problem. If you have experienced the problem yourself, even better. Ask your friends what kind of problems they experience.
-
-
+For example, I built [YC Careers](http://jonathanzwhite.github.io/yc-careers/) and [AtomSpace](https://atomspace.co/) to address the needs of one my friends who was having trouble applying to jobs and interviewing as a product designer. Overnight, YC Careers made it to the top of [ProductHunt](https://www.producthunt.com/posts/yc-careers) and AtomSpace booked $100 dollars from strangers within 6 hours of going live.
+**The bigger the pain point you’re trying to solve, the easier it will be to find users.**
-
+Also, analyze problem-solutions from multiple perspectives. Some successful ideas might seem like a solution without a problem. For example, Instagram doesn’t seem to solve any immediate problem. But it does. Instagram solves the need for friends to stay updated on one another. If this need is not met, then it becomes a problem.
-
-
+For further reading, [John Carmack on Idea Generation](https://amasad.me/carmack) and [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) are two great essays to get you started.
-
-
-
+### 4. Ditch the “good idea / bad idea” mindset
-
-
+Once you’ve identified a problem and come up with a solution, you’ll probably ask yourself if your idea is good.
+**Plenty of seemingly bad ideas have turned out to be great businesses.** For example, no one wanted to invest in Airbnb. Brian Chesky, one of the co-founders of Airbnb, detailed their rejections in his essay [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz).
-
+![](https://cdn-images-1.medium.com/max/800/1*WpxUxMCO-7NXr-o1yo023g.png)
+The only way to know for sure if your idea will work is to formulate a product hypothesis and run an experiment to test it. Talk to people and ask if they are interested in your solution. If you can get them to pay you before you’ve built anything, even better.
-
+### 5. Ask for help
-
+When working on a project, reach out to people and ask for their help. **You will be surprised at how many people are willing to pay it forward and help a complete stranger.**
-
-
+Some of the best advice and feedback I have received came from reaching out to people with an email or a DM on Twitter.
-
+Also, when asking for help, be specific and prepare your questions. If you want feedback on a design, send the mockup. If you want feedback on your marketing strategy, detail what you have tried so far. Context is key. And if you don’t get a response, follow up politely. Sometimes the person you’re reaching out to might have genuinely missed your message.
+---
-
+Build and launch your side projects. Use this process as a way to hone your craft so that when opportunity presents itself, you’re ready to take full advantage of it. Luck always favors the prepared.
-
+So next time you come up with an idea, act on it. Identify a problem, define your goal, stay on track, and ask for help.
-
-
+What are you working on right now? How can I help you? Leave me a note here or [tweet](https://twitter.com/jonathanzwhite) them to me on Twitter.
+You can find me on Medium where I publish every week. Or you can follow me on [Twitter](https://twitter.com/JonathanZWhite), where I post non-sensical ramblings about design, front-end development, and virtual reality.
-
-
-
- You can't perform that action at this time.
-
-
-
-
-
-
-
-
-
-
-
-
- You signed in with another tab or window. Reload to refresh your session.
- You signed out in another tab or window. Reload to refresh your session.
-
-
-
-
-
-
-
-
-
-
-
-
+---
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
\ No newline at end of file
From b3a1f53af854a7ca405d5e19b9343feb4195b227 Mon Sep 17 00:00:00 2001
From: Wenlin Ou
Date: Fri, 7 Apr 2017 22:04:58 -0400
Subject: [PATCH 141/945] ready for proofreading
---
...ng-will-change-until-you-start-building.md | 70 +++++++++----------
1 file changed, 35 insertions(+), 35 deletions(-)
diff --git a/TODO/nothing-will-change-until-you-start-building.md b/TODO/nothing-will-change-until-you-start-building.md
index 0d57984107d..26cc605b437 100644
--- a/TODO/nothing-will-change-until-you-start-building.md
+++ b/TODO/nothing-will-change-until-you-start-building.md
@@ -1,90 +1,90 @@
> * 原文地址:[Nothing will change until you start building.](https://medium.freecodecamp.com/nothing-will-change-until-you-start-building-2681e85e7bdc)
> * 原文作者:[Jonathan Z. White](https://medium.freecodecamp.com/@JonathanZWhite?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[owenlyn](https://github.com/owenlyn)
> * 校对者:
---
![](https://cdn-images-1.medium.com/max/1000/1*EwHpnCZ70FtJMi-lNSl-9Q.png)
-# Nothing will change until you start building.
+# Nothing will change until you start building. 真正行动之前 你将一无所成
-This past week, I was in a Lyft. My driver was telling me about all of her ideas for side projects. She had ideas for a children’s book, an app that helps people find parking, and a more efficient way to package gifts. The problem was she was frozen by indecision. She had the ideas but she didn’t know where to start.
+This past week, I was in a Lyft. My driver was telling me about all of her ideas for side projects. She had ideas for a children’s book, an app that helps people find parking, and a more efficient way to package gifts. The problem was she was frozen by indecision. She had the ideas but she didn’t know where to start.上周我打了一辆 Lyft 出租车,司机跟我聊了很多关于她的天花乱坠的想法。她的想法里,有一个关于儿童的书籍,有帮助人们停车的APP、有一个更高效的打包礼物的方法,但问题是她总是犹豫不决:她又想法但却不知道从何开始。
-This is what I told her. **Start by building. Pick one project and do whatever you have to do to ship it.** If you want to write a book, start with writing a page a day. If you want to build an app, start with some sketches. Anyone can do it.
+This is what I told her. **Start by building. Pick one project and do whatever you have to do to ship it.** If you want to write a book, start with writing a page a day. If you want to build an app, start with some sketches. Anyone can do it. 接下来这段话是我对她说的:**开始一些实际行动吧。选一个项目然后尽可能的去完成它。** 如果你想写一本书,那么从每天写一页开始;如果你像做一个APP,那就从草稿开始。任何人都可以做到这些。
-This advice applies to all creators. Once you start building and launching your projects, you won’t be able to stop. Building will become part of your identity. And even if your project fails, you’ll keep at it.
+This advice applies to all creators. Once you start building and launching your projects, you won’t be able to stop. Building will become part of your identity. And even if your project fails, you’ll keep at it. 这些建议对所有创新者都适用。一旦你开始着手在你的项目上,你将会不由自主的继续做下去。创新将会成为你人格的一部分 —— 即使你的项目失败了。
-The steps you take today will compound over time. Look at great product makers like [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), and [Sebastian Dobrincu](https://twitter.com/Sebyddd). **They don’t wait for the right opportunity.** They build and ship weekly.
+The steps you take today will compound over time. Look at great product makers like [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), and [Sebastian Dobrincu](https://twitter.com/Sebyddd). **They don’t wait for the right opportunity.** They build and ship weekly. 你今天迈出的一小步将会是你人生的一大步。看看那些伟大的产品创造者吧,比如 [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), 和 [Sebastian Dobrincu](https://twitter.com/Sebyddd)。**他们不仅仅是等待合适的时机。** 他们每周都在更新他们的产品。
-In an interview on [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics), [Josh Pigford](https://twitter.com/Shpigford) disclosed that before he built Baremetrics, he had iterated through a *hundred* failed ideas. Today, Baremetrics makes over 60,000 in monthly recurring revenue.
+In an interview on [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics), [Josh Pigford](https://twitter.com/Shpigford) disclosed that before he built Baremetrics, he had iterated through a *hundred* failed ideas. Today, Baremetrics makes over 60,000 in monthly recurring revenue. 在一个 [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics) 的节目中, [Josh Pigford](https://twitter.com/Shpigford) 透露,在做 Baremetrics 之前,他已经尝试过 *一百个* 失败的点子。 现在, Baremetrics 每个月都会有 60,000 美金进账。
![](https://cdn-images-1.medium.com/max/800/1*BzmVaqAKEzNRybUmMYOxdA.png)
-With that in mind, here are some tactics I’ve compiled from friends and acquaintances who are successful product makers.
+With that in mind, here are some tactics I’ve compiled from friends and acquaintances who are successful product makers.记住这些,接下来的要说的,是我总结的一些来自成功的产品创作者朋友们的策略。
---
-### 1. Define your goal
+### 1. Define your goal确定目标
-Early on, figure out the main goal of your project. If you want to break out of your usual design style, focus on the design. If you want to test a new framework for your front-end, focus on the code. If you want to acquire a hundred users, focus on selling.
+Early on, figure out the main goal of your project. If you want to break out of your usual design style, focus on the design. If you want to test a new framework for your front-end, focus on the code. If you want to acquire a hundred users, focus on selling.在开始的时候,确定你的项目要实现的主要目标。如果你想打破你以往的设计风格,那就把心思花在设计上;如果你测试一个新的前端框架,那就把钻研代码;如果你想获得上百个用户(”大量用户“ 会不会好点),就把精力用在销售上。
-**Define your goal early so you don’t get distracted and end up not achieving what you set out to do.** Perhaps your project doesn’t become the next Snapchat, but by working towards your goal you will have honed the tools in your product-making arsenal. Little by little, you’ll be able to design, code, and sell faster and more efficiently.
+**Define your goal early so you don’t get distracted and end up not achieving what you set out to do.** Perhaps your project doesn’t become the next Snapchat, but by working towards your goal you will have honed the tools in your product-making arsenal. Little by little, you’ll be able to design, code, and sell faster and more efficiently. **尽早确定目标可以帮助你摆脱诱惑,防止你到最后忘记初心。** 也许你的项目并不能成为下一个 Snapchat, 但是在通往你目标的道路上,你会更加的熟练运用各种做产品的工具。渐渐地,你会拥有越来越强大的设计、代码、销售等的能力。
-### 2. Stay on track
+### 2. Stay on track执行计划
-Sometimes product makers fail to ship because they lose motivation. **Set small goals; meeting those goals will help you build momentum.** Small wins will compound.
+Sometimes product makers fail to ship because they lose motivation. **Set small goals; meeting those goals will help you build momentum.** Small wins will compound. 有一些产品没能完的原因是作者在半路失去了动力。 **设定一些小目标(比如先赚一个亿);完成这些目标会让你获得前进的动力。** 小的成就累计起来也是很了不起的。
![](https://cdn-images-1.medium.com/max/800/1*ESildSVTSSOnFXGDxD-l9w.png)
-Another thing you can do to stay on track is to be more public about your work. Share your progress online and with your peers. People love to see things like wireframes and will often times provide valuable input.
+Another thing you can do to stay on track is to be more public about your work. Share your progress online and with your peers. People love to see things like wireframes and will often times provide valuable input.另一个让你保持动力的建议就是让你的项目更加公开。在网上和你的同行们分享你的进展。人们很喜欢看到线框(wireframe 翻译成 ”框架“ 会不会好一点?)类的东西,并常常带来有价值的投入。
-One trick that I find particularly motivating is making bets with friends. I tell them, if I don’t complete a task by a deadline, I’ll give them a hundred dollars. For example, I might set my task to getting twenty people to pay me within two weeks of launching a project.
+One trick that I find particularly motivating is making bets with friends. I tell them, if I don’t complete a task by a deadline, I’ll give them a hundred dollars. For example, I might set my task to getting twenty people to pay me within two weeks of launching a project.我自己找到的一个非常有用的保持动力的小秘诀是和朋友打赌。我对他们说,如果我在截止日期前没有完成某个任务,我就给他们一百刀。比如,我会在一个项目开始的两周内找到二十个人打赌。
-### 3. Solve a problem
+### 3. Solve a problem 解决一个问题
-Start by solving a problem. If you have experienced the problem yourself, even better. Ask your friends what kind of problems they experience.
+Start by solving a problem. If you have experienced the problem yourself, even better. Ask your friends what kind of problems they experience.解决一个问题。如果你自己也经历了这个问题那就再好不过了。问问你的朋友们他们都有哪些问题需要解决。
-For example, I built [YC Careers](http://jonathanzwhite.github.io/yc-careers/) and [AtomSpace](https://atomspace.co/) to address the needs of one my friends who was having trouble applying to jobs and interviewing as a product designer. Overnight, YC Careers made it to the top of [ProductHunt](https://www.producthunt.com/posts/yc-careers) and AtomSpace booked $100 dollars from strangers within 6 hours of going live.
+For example, I built [YC Careers](http://jonathanzwhite.github.io/yc-careers/) and [AtomSpace](https://atomspace.co/) to address the needs of one my friends who was having trouble applying to jobs and interviewing as a product designer. Overnight, YC Careers made it to the top of [ProductHunt](https://www.producthunt.com/posts/yc-careers) and AtomSpace booked $100 dollars from strangers within 6 hours of going live.比如,我做了 [YC Careers](http://jonathanzwhite.github.io/yc-careers/) 和 [AtomSpace](https://atomspace.co/) 来解决一个产品设计师朋友找工作和面试的烦恼。一夜之间,YC Careers 一下子爬到了 [ProductHunt](https://www.producthunt.com/posts/yc-careers) 的榜首,AtomSpace 在上线6个小时就拿到了来自陌生人的一百美元订单。
-**The bigger the pain point you’re trying to solve, the easier it will be to find users.**
+**The bigger the pain point you’re trying to solve, the easier it will be to find users.你尝试解决多大的痛点,找用户就有多容易。**
-Also, analyze problem-solutions from multiple perspectives. Some successful ideas might seem like a solution without a problem. For example, Instagram doesn’t seem to solve any immediate problem. But it does. Instagram solves the need for friends to stay updated on one another. If this need is not met, then it becomes a problem.
+Also, analyze problem-solutions from multiple perspectives. Some successful ideas might seem like a solution without a problem. For example, Instagram doesn’t seem to solve any immediate problem. But it does. Instagram solves the need for friends to stay updated on one another. If this need is not met, then it becomes a problem. 同时,尝试从过个角度分析问题和解决方案。有些成功的点子看起来并没有解决任何问题,但其实不然。比如,Instagram 看起来没有直接解决任何问题。其实不是。Instagram 满足了朋友之间实时社交的需求。如果这个需求不能得到满足,那才是个问题。
-For further reading, [John Carmack on Idea Generation](https://amasad.me/carmack) and [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) are two great essays to get you started.
+For further reading, [John Carmack on Idea Generation](https://amasad.me/carmack) and [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) are two great essays to get you started. 想要了解更多的话,[John Carmack on Idea Generation](https://amasad.me/carmack) 和 [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) 这两篇文章是非常好的开始。
-### 4. Ditch the “good idea / bad idea” mindset
+### 4. Ditch the “good idea / bad idea” mindset 放弃”好主意/坏主意“的想法
-Once you’ve identified a problem and come up with a solution, you’ll probably ask yourself if your idea is good.
+Once you’ve identified a problem and come up with a solution, you’ll probably ask yourself if your idea is good.当你明确了你想要解决的问题并有了一个解决方案的时候,你可能会问自己这是不是一个好的点子。
-**Plenty of seemingly bad ideas have turned out to be great businesses.** For example, no one wanted to invest in Airbnb. Brian Chesky, one of the co-founders of Airbnb, detailed their rejections in his essay [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz).
+**Plenty of seemingly bad ideas have turned out to be great businesses.** For example, no one wanted to invest in Airbnb. Brian Chesky, one of the co-founders of Airbnb, detailed their rejections in his essay [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz). **相当一部分看起来很糟糕的点子最后被证明是超级棒的生意。** 比如,多年没人想投资 Airbnb。 Brian Chesky, Airbnb 的创始人之一,在他的文章 [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz) 里详细讲述了被投资人拒绝的故事。
![](https://cdn-images-1.medium.com/max/800/1*WpxUxMCO-7NXr-o1yo023g.png)
-The only way to know for sure if your idea will work is to formulate a product hypothesis and run an experiment to test it. Talk to people and ask if they are interested in your solution. If you can get them to pay you before you’ve built anything, even better.
+The only way to know for sure if your idea will work is to formulate a product hypothesis and run an experiment to test it. Talk to people and ask if they are interested in your solution. If you can get them to pay you before you’ve built anything, even better. 唯一验证你想法的办法就是做出一个产品来测试它。 找人聊聊,看看他们对你的解决方案是不是感兴趣。如果在你做出任何产品之前就有人原意为之付费那就再棒不过了。
-### 5. Ask for help
+### 5. Ask for help寻求帮助
-When working on a project, reach out to people and ask for their help. **You will be surprised at how many people are willing to pay it forward and help a complete stranger.**
+When working on a project, reach out to people and ask for their help. **You will be surprised at how many people are willing to pay it forward and help a complete stranger.** 当你在你的项目上埋头苦干的时候,记得寻求别人的帮助。 **你会很惊喜的发现竟然有这么多人原意去帮助一个陌生人(对,就是你)。**
-Some of the best advice and feedback I have received came from reaching out to people with an email or a DM on Twitter.
+Some of the best advice and feedback I have received came from reaching out to people with an email or a DM on Twitter. 我获得的最好的一部分建议就是来自于给别人发邮件或是推特(微博)私信。
-Also, when asking for help, be specific and prepare your questions. If you want feedback on a design, send the mockup. If you want feedback on your marketing strategy, detail what you have tried so far. Context is key. And if you don’t get a response, follow up politely. Sometimes the person you’re reaching out to might have genuinely missed your message.
+Also, when asking for help, be specific and prepare your questions. If you want feedback on a design, send the mockup. If you want feedback on your marketing strategy, detail what you have tried so far. Context is key. And if you don’t get a response, follow up politely. Sometimes the person you’re reaching out to might have genuinely missed your message. 在寻求帮助的时候,请提一些精确、有价值的问题。如果你需要关于设计的一些反馈,就把草稿发给别人;如果你需要营销方面的建议,就详细的列出你尝试过的方法。内容才是关键。如果别人没有回复你,请礼貌的再次询问 —— 有时候别人只是没注意到你的消息而已。
---
-Build and launch your side projects. Use this process as a way to hone your craft so that when opportunity presents itself, you’re ready to take full advantage of it. Luck always favors the prepared.
+Build and launch your side projects. Use this process as a way to hone your craft so that when opportunity presents itself, you’re ready to take full advantage of it. Luck always favors the prepared.开始你的兴趣项目吧。利用这个过程磨练自己,这样当机会来临的时候,你才能紧紧抓住。记住,机会总是青睐有准备的人。
-So next time you come up with an idea, act on it. Identify a problem, define your goal, stay on track, and ask for help.
+So next time you come up with an idea, act on it. Identify a problem, define your goal, stay on track, and ask for help.所以下次你有一个想法的时候,采取行动吧。找到一个问题,确定你的目标,坚持下去,别忘了学会寻求帮助。
-What are you working on right now? How can I help you? Leave me a note here or [tweet](https://twitter.com/jonathanzwhite) them to me on Twitter.
+What are you working on right now? How can I help you? Leave me a note here or [tweet](https://twitter.com/jonathanzwhite) them to me on Twitter.你现在在做神马项目呢?有什么是我能帮到你的吗?可以在这里或者我的 [tweet](https://twitter.com/jonathanzwhite) 下面留言。
-You can find me on Medium where I publish every week. Or you can follow me on [Twitter](https://twitter.com/JonathanZWhite), where I post non-sensical ramblings about design, front-end development, and virtual reality.
+You can find me on Medium where I publish every week. Or you can follow me on [Twitter](https://twitter.com/JonathanZWhite), where I post non-sensical ramblings about design, front-end development, and virtual reality. 我在 Medium 上每周都会发文。也可以关注我的 [Twitter](https://twitter.com/JonathanZWhite),我会发一些关于设计、前端开发和虚拟现实的杂想。
-*P.S. If you enjoyed this article, it would mean a lot if you click the 💚 and share with friends.*
+*P.S. If you enjoyed this article, it would mean a lot if you click the 💚 and share with friends.*(可删?)
---
-> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
\ No newline at end of file
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 784732d642d3393d58ce677312cb0e1f3854a8d8 Mon Sep 17 00:00:00 2001
From: Wenlin Ou
Date: Fri, 7 Apr 2017 22:06:28 -0400
Subject: [PATCH 142/945] ...
---
...ng-will-change-until-you-start-building.md | 780 ++++++++++++++++--
1 file changed, 733 insertions(+), 47 deletions(-)
diff --git a/TODO/nothing-will-change-until-you-start-building.md b/TODO/nothing-will-change-until-you-start-building.md
index 26cc605b437..df8d0872aa7 100644
--- a/TODO/nothing-will-change-until-you-start-building.md
+++ b/TODO/nothing-will-change-until-you-start-building.md
@@ -1,90 +1,776 @@
-> * 原文地址:[Nothing will change until you start building.](https://medium.freecodecamp.com/nothing-will-change-until-you-start-building-2681e85e7bdc)
-> * 原文作者:[Jonathan Z. White](https://medium.freecodecamp.com/@JonathanZWhite?source=post_header_lockup)
-> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:[owenlyn](https://github.com/owenlyn)
-> * 校对者:
----
-![](https://cdn-images-1.medium.com/max/1000/1*EwHpnCZ70FtJMi-lNSl-9Q.png)
-# Nothing will change until you start building. 真正行动之前 你将一无所成
-This past week, I was in a Lyft. My driver was telling me about all of her ideas for side projects. She had ideas for a children’s book, an app that helps people find parking, and a more efficient way to package gifts. The problem was she was frozen by indecision. She had the ideas but she didn’t know where to start.上周我打了一辆 Lyft 出租车,司机跟我聊了很多关于她的天花乱坠的想法。她的想法里,有一个关于儿童的书籍,有帮助人们停车的APP、有一个更高效的打包礼物的方法,但问题是她总是犹豫不决:她又想法但却不知道从何开始。
-This is what I told her. **Start by building. Pick one project and do whatever you have to do to ship it.** If you want to write a book, start with writing a page a day. If you want to build an app, start with some sketches. Anyone can do it. 接下来这段话是我对她说的:**开始一些实际行动吧。选一个项目然后尽可能的去完成它。** 如果你想写一本书,那么从每天写一页开始;如果你像做一个APP,那就从草稿开始。任何人都可以做到这些。
-This advice applies to all creators. Once you start building and launching your projects, you won’t be able to stop. Building will become part of your identity. And even if your project fails, you’ll keep at it. 这些建议对所有创新者都适用。一旦你开始着手在你的项目上,你将会不由自主的继续做下去。创新将会成为你人格的一部分 —— 即使你的项目失败了。
+
+
+
+
-The steps you take today will compound over time. Look at great product makers like [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), and [Sebastian Dobrincu](https://twitter.com/Sebyddd). **They don’t wait for the right opportunity.** They build and ship weekly. 你今天迈出的一小步将会是你人生的一大步。看看那些伟大的产品创造者吧,比如 [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), 和 [Sebastian Dobrincu](https://twitter.com/Sebyddd)。**他们不仅仅是等待合适的时机。** 他们每周都在更新他们的产品。
-In an interview on [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics), [Josh Pigford](https://twitter.com/Shpigford) disclosed that before he built Baremetrics, he had iterated through a *hundred* failed ideas. Today, Baremetrics makes over 60,000 in monthly recurring revenue. 在一个 [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics) 的节目中, [Josh Pigford](https://twitter.com/Shpigford) 透露,在做 Baremetrics 之前,他已经尝试过 *一百个* 失败的点子。 现在, Baremetrics 每个月都会有 60,000 美金进账。
+
+
+
+
+
+
-![](https://cdn-images-1.medium.com/max/800/1*BzmVaqAKEzNRybUmMYOxdA.png)
+
+
+ gold-miner/nothing-will-change-until-you-start-building.md at master · xitu/gold-miner
+
+
+
-With that in mind, here are some tactics I’ve compiled from friends and acquaintances who are successful product makers.记住这些,接下来的要说的,是我总结的一些来自成功的产品创作者朋友们的策略。
+
+
----
+
+
+
+
+
+
-### 1. Define your goal确定目标
+
-Early on, figure out the main goal of your project. If you want to break out of your usual design style, focus on the design. If you want to test a new framework for your front-end, focus on the code. If you want to acquire a hundred users, focus on selling.在开始的时候,确定你的项目要实现的主要目标。如果你想打破你以往的设计风格,那就把心思花在设计上;如果你测试一个新的前端框架,那就把钻研代码;如果你想获得上百个用户(”大量用户“ 会不会好点),就把精力用在销售上。
+
+
+
-**Define your goal early so you don’t get distracted and end up not achieving what you set out to do.** Perhaps your project doesn’t become the next Snapchat, but by working towards your goal you will have honed the tools in your product-making arsenal. Little by little, you’ll be able to design, code, and sell faster and more efficiently. **尽早确定目标可以帮助你摆脱诱惑,防止你到最后忘记初心。** 也许你的项目并不能成为下一个 Snapchat, 但是在通往你目标的道路上,你会更加的熟练运用各种做产品的工具。渐渐地,你会拥有越来越强大的设计、代码、销售等的能力。
+
+
-### 2. Stay on track执行计划
-Sometimes product makers fail to ship because they lose motivation. **Set small goals; meeting those goals will help you build momentum.** Small wins will compound. 有一些产品没能完的原因是作者在半路失去了动力。 **设定一些小目标(比如先赚一个亿);完成这些目标会让你获得前进的动力。** 小的成就累计起来也是很了不起的。
-![](https://cdn-images-1.medium.com/max/800/1*ESildSVTSSOnFXGDxD-l9w.png)
-Another thing you can do to stay on track is to be more public about your work. Share your progress online and with your peers. People love to see things like wireframes and will often times provide valuable input.另一个让你保持动力的建议就是让你的项目更加公开。在网上和你的同行们分享你的进展。人们很喜欢看到线框(wireframe 翻译成 ”框架“ 会不会好一点?)类的东西,并常常带来有价值的投入。
+
-One trick that I find particularly motivating is making bets with friends. I tell them, if I don’t complete a task by a deadline, I’ll give them a hundred dollars. For example, I might set my task to getting twenty people to pay me within two weeks of launching a project.我自己找到的一个非常有用的保持动力的小秘诀是和朋友打赌。我对他们说,如果我在截止日期前没有完成某个任务,我就给他们一百刀。比如,我会在一个项目开始的两周内找到二十个人打赌。
-### 3. Solve a problem 解决一个问题
+
-Start by solving a problem. If you have experienced the problem yourself, even better. Ask your friends what kind of problems they experience.解决一个问题。如果你自己也经历了这个问题那就再好不过了。问问你的朋友们他们都有哪些问题需要解决。
+
+
-For example, I built [YC Careers](http://jonathanzwhite.github.io/yc-careers/) and [AtomSpace](https://atomspace.co/) to address the needs of one my friends who was having trouble applying to jobs and interviewing as a product designer. Overnight, YC Careers made it to the top of [ProductHunt](https://www.producthunt.com/posts/yc-careers) and AtomSpace booked $100 dollars from strangers within 6 hours of going live.比如,我做了 [YC Careers](http://jonathanzwhite.github.io/yc-careers/) 和 [AtomSpace](https://atomspace.co/) 来解决一个产品设计师朋友找工作和面试的烦恼。一夜之间,YC Careers 一下子爬到了 [ProductHunt](https://www.producthunt.com/posts/yc-careers) 的榜首,AtomSpace 在上线6个小时就拿到了来自陌生人的一百美元订单。
+
+
-**The bigger the pain point you’re trying to solve, the easier it will be to find users.你尝试解决多大的痛点,找用户就有多容易。**
-Also, analyze problem-solutions from multiple perspectives. Some successful ideas might seem like a solution without a problem. For example, Instagram doesn’t seem to solve any immediate problem. But it does. Instagram solves the need for friends to stay updated on one another. If this need is not met, then it becomes a problem. 同时,尝试从过个角度分析问题和解决方案。有些成功的点子看起来并没有解决任何问题,但其实不然。比如,Instagram 看起来没有直接解决任何问题。其实不是。Instagram 满足了朋友之间实时社交的需求。如果这个需求不能得到满足,那才是个问题。
+
-For further reading, [John Carmack on Idea Generation](https://amasad.me/carmack) and [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) are two great essays to get you started. 想要了解更多的话,[John Carmack on Idea Generation](https://amasad.me/carmack) 和 [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) 这两篇文章是非常好的开始。
+
+
-### 4. Ditch the “good idea / bad idea” mindset 放弃”好主意/坏主意“的想法
+
+
+
-Once you’ve identified a problem and come up with a solution, you’ll probably ask yourself if your idea is good.当你明确了你想要解决的问题并有了一个解决方案的时候,你可能会问自己这是不是一个好的点子。
+
+
-**Plenty of seemingly bad ideas have turned out to be great businesses.** For example, no one wanted to invest in Airbnb. Brian Chesky, one of the co-founders of Airbnb, detailed their rejections in his essay [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz). **相当一部分看起来很糟糕的点子最后被证明是超级棒的生意。** 比如,多年没人想投资 Airbnb。 Brian Chesky, Airbnb 的创始人之一,在他的文章 [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz) 里详细讲述了被投资人拒绝的故事。
-![](https://cdn-images-1.medium.com/max/800/1*WpxUxMCO-7NXr-o1yo023g.png)
+
-The only way to know for sure if your idea will work is to formulate a product hypothesis and run an experiment to test it. Talk to people and ask if they are interested in your solution. If you can get them to pay you before you’ve built anything, even better. 唯一验证你想法的办法就是做出一个产品来测试它。 找人聊聊,看看他们对你的解决方案是不是感兴趣。如果在你做出任何产品之前就有人原意为之付费那就再棒不过了。
-### 5. Ask for help寻求帮助
+
-When working on a project, reach out to people and ask for their help. **You will be surprised at how many people are willing to pay it forward and help a complete stranger.** 当你在你的项目上埋头苦干的时候,记得寻求别人的帮助。 **你会很惊喜的发现竟然有这么多人原意去帮助一个陌生人(对,就是你)。**
+
-Some of the best advice and feedback I have received came from reaching out to people with an email or a DM on Twitter. 我获得的最好的一部分建议就是来自于给别人发邮件或是推特(微博)私信。
+
+
-Also, when asking for help, be specific and prepare your questions. If you want feedback on a design, send the mockup. If you want feedback on your marketing strategy, detail what you have tried so far. Context is key. And if you don’t get a response, follow up politely. Sometimes the person you’re reaching out to might have genuinely missed your message. 在寻求帮助的时候,请提一些精确、有价值的问题。如果你需要关于设计的一些反馈,就把草稿发给别人;如果你需要营销方面的建议,就详细的列出你尝试过的方法。内容才是关键。如果别人没有回复你,请礼貌的再次询问 —— 有时候别人只是没注意到你的消息而已。
+
----
-Build and launch your side projects. Use this process as a way to hone your craft so that when opportunity presents itself, you’re ready to take full advantage of it. Luck always favors the prepared.开始你的兴趣项目吧。利用这个过程磨练自己,这样当机会来临的时候,你才能紧紧抓住。记住,机会总是青睐有准备的人。
+
-So next time you come up with an idea, act on it. Identify a problem, define your goal, stay on track, and ask for help.所以下次你有一个想法的时候,采取行动吧。找到一个问题,确定你的目标,坚持下去,别忘了学会寻求帮助。
+
-What are you working on right now? How can I help you? Leave me a note here or [tweet](https://twitter.com/jonathanzwhite) them to me on Twitter.你现在在做神马项目呢?有什么是我能帮到你的吗?可以在这里或者我的 [tweet](https://twitter.com/jonathanzwhite) 下面留言。
+
+
-You can find me on Medium where I publish every week. Or you can follow me on [Twitter](https://twitter.com/JonathanZWhite), where I post non-sensical ramblings about design, front-end development, and virtual reality. 我在 Medium 上每周都会发文。也可以关注我的 [Twitter](https://twitter.com/JonathanZWhite),我会发一些关于设计、前端开发和虚拟现实的杂想。
-*P.S. If you enjoyed this article, it would mean a lot if you click the 💚 and share with friends.*(可删?)
+
This past week, I was in a Lyft. My driver was telling me about all of her ideas for side projects. She had ideas for a children’s book, an app that helps people find parking, and a more efficient way to package gifts. The problem was she was frozen by indecision. She had the ideas but she didn’t know where to start.
+
This is what I told her. Start by building. Pick one project and do whatever you have to do to ship it. If you want to write a book, start with writing a page a day. If you want to build an app, start with some sketches. Anyone can do it.
+
This advice applies to all creators. Once you start building and launching your projects, you won’t be able to stop. Building will become part of your identity. And even if your project fails, you’ll keep at it.
+
The steps you take today will compound over time. Look at great product makers like Drew Wilson, Pieter Levels, and Sebastian Dobrincu. They don’t wait for the right opportunity. They build and ship weekly.
+
In an interview on IndieHackers, Josh Pigford disclosed that before he built Baremetrics, he had iterated through a hundred failed ideas. Today, Baremetrics makes over 60,000 in monthly recurring revenue.
+
+
With that in mind, here are some tactics I’ve compiled from friends and acquaintances who are successful product makers.
+
+
1. Define your goal
+
Early on, figure out the main goal of your project. If you want to break out of your usual design style, focus on the design. If you want to test a new framework for your front-end, focus on the code. If you want to acquire a hundred users, focus on selling.
+
Define your goal early so you don’t get distracted and end up not achieving what you set out to do. Perhaps your project doesn’t become the next Snapchat, but by working towards your goal you will have honed the tools in your product-making arsenal. Little by little, you’ll be able to design, code, and sell faster and more efficiently.
+
2. Stay on track
+
Sometimes product makers fail to ship because they lose motivation. Set small goals; meeting those goals will help you build momentum. Small wins will compound.
+
+
Another thing you can do to stay on track is to be more public about your work. Share your progress online and with your peers. People love to see things like wireframes and will often times provide valuable input.
+
One trick that I find particularly motivating is making bets with friends. I tell them, if I don’t complete a task by a deadline, I’ll give them a hundred dollars. For example, I might set my task to getting twenty people to pay me within two weeks of launching a project.
+
3. Solve a problem
+
Start by solving a problem. If you have experienced the problem yourself, even better. Ask your friends what kind of problems they experience.
+
For example, I built YC Careers and AtomSpace to address the needs of one my friends who was having trouble applying to jobs and interviewing as a product designer. Overnight, YC Careers made it to the top of ProductHunt and AtomSpace booked $100 dollars from strangers within 6 hours of going live.
+
The bigger the pain point you’re trying to solve, the easier it will be to find users.
+
Also, analyze problem-solutions from multiple perspectives. Some successful ideas might seem like a solution without a problem. For example, Instagram doesn’t seem to solve any immediate problem. But it does. Instagram solves the need for friends to stay updated on one another. If this need is not met, then it becomes a problem.
Once you’ve identified a problem and come up with a solution, you’ll probably ask yourself if your idea is good.
+
Plenty of seemingly bad ideas have turned out to be great businesses. For example, no one wanted to invest in Airbnb. Brian Chesky, one of the co-founders of Airbnb, detailed their rejections in his essay 7 Rejections.
+
+
The only way to know for sure if your idea will work is to formulate a product hypothesis and run an experiment to test it. Talk to people and ask if they are interested in your solution. If you can get them to pay you before you’ve built anything, even better.
+
5. Ask for help
+
When working on a project, reach out to people and ask for their help. You will be surprised at how many people are willing to pay it forward and help a complete stranger.
+
Some of the best advice and feedback I have received came from reaching out to people with an email or a DM on Twitter.
+
Also, when asking for help, be specific and prepare your questions. If you want feedback on a design, send the mockup. If you want feedback on your marketing strategy, detail what you have tried so far. Context is key. And if you don’t get a response, follow up politely. Sometimes the person you’re reaching out to might have genuinely missed your message.
+
+
Build and launch your side projects. Use this process as a way to hone your craft so that when opportunity presents itself, you’re ready to take full advantage of it. Luck always favors the prepared.
+
So next time you come up with an idea, act on it. Identify a problem, define your goal, stay on track, and ask for help.
+
What are you working on right now? How can I help you? Leave me a note here or tweet them to me on Twitter.
+
You can find me on Medium where I publish every week. Or you can follow me on Twitter, where I post non-sensical ramblings about design, front-end development, and virtual reality.
+
P.S. If you enjoyed this article, it would mean a lot if you click the 💚 and share with friends.
+
+
+ You can't perform that action at this time.
+
+
+
+
+
+
+
+
+
+
+
+
+ You signed in with another tab or window. Reload to refresh your session.
+ You signed out in another tab or window. Reload to refresh your session.
+
+
+
+
+
+
+
+
+
+
+
+
-> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From c61a1606f3499108f7170f24c3937f5428968021 Mon Sep 17 00:00:00 2001
From: Wenlin Ou
Date: Fri, 7 Apr 2017 22:07:53 -0400
Subject: [PATCH 143/945] ...
---
...ng-will-change-until-you-start-building.md | 780 ++----------------
1 file changed, 47 insertions(+), 733 deletions(-)
diff --git a/TODO/nothing-will-change-until-you-start-building.md b/TODO/nothing-will-change-until-you-start-building.md
index df8d0872aa7..0d57984107d 100644
--- a/TODO/nothing-will-change-until-you-start-building.md
+++ b/TODO/nothing-will-change-until-you-start-building.md
@@ -1,776 +1,90 @@
+> * 原文地址:[Nothing will change until you start building.](https://medium.freecodecamp.com/nothing-will-change-until-you-start-building-2681e85e7bdc)
+> * 原文作者:[Jonathan Z. White](https://medium.freecodecamp.com/@JonathanZWhite?source=post_header_lockup)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+---
+![](https://cdn-images-1.medium.com/max/1000/1*EwHpnCZ70FtJMi-lNSl-9Q.png)
+# Nothing will change until you start building.
+This past week, I was in a Lyft. My driver was telling me about all of her ideas for side projects. She had ideas for a children’s book, an app that helps people find parking, and a more efficient way to package gifts. The problem was she was frozen by indecision. She had the ideas but she didn’t know where to start.
+This is what I told her. **Start by building. Pick one project and do whatever you have to do to ship it.** If you want to write a book, start with writing a page a day. If you want to build an app, start with some sketches. Anyone can do it.
-
-
-
-
+This advice applies to all creators. Once you start building and launching your projects, you won’t be able to stop. Building will become part of your identity. And even if your project fails, you’ll keep at it.
+The steps you take today will compound over time. Look at great product makers like [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), and [Sebastian Dobrincu](https://twitter.com/Sebyddd). **They don’t wait for the right opportunity.** They build and ship weekly.
+In an interview on [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics), [Josh Pigford](https://twitter.com/Shpigford) disclosed that before he built Baremetrics, he had iterated through a *hundred* failed ideas. Today, Baremetrics makes over 60,000 in monthly recurring revenue.
-
-
-
-
-
-
-
-
- gold-miner/nothing-will-change-until-you-start-building.md at master · xitu/gold-miner
-
-
-
+![](https://cdn-images-1.medium.com/max/800/1*BzmVaqAKEzNRybUmMYOxdA.png)
-
-
+With that in mind, here are some tactics I’ve compiled from friends and acquaintances who are successful product makers.
-
-
-
-
-
-
+---
-
+### 1. Define your goal
-
-
-
+Early on, figure out the main goal of your project. If you want to break out of your usual design style, focus on the design. If you want to test a new framework for your front-end, focus on the code. If you want to acquire a hundred users, focus on selling.
-
-
+**Define your goal early so you don’t get distracted and end up not achieving what you set out to do.** Perhaps your project doesn’t become the next Snapchat, but by working towards your goal you will have honed the tools in your product-making arsenal. Little by little, you’ll be able to design, code, and sell faster and more efficiently.
+### 2. Stay on track
+Sometimes product makers fail to ship because they lose motivation. **Set small goals; meeting those goals will help you build momentum.** Small wins will compound.
+![](https://cdn-images-1.medium.com/max/800/1*ESildSVTSSOnFXGDxD-l9w.png)
-
+Another thing you can do to stay on track is to be more public about your work. Share your progress online and with your peers. People love to see things like wireframes and will often times provide valuable input.
+One trick that I find particularly motivating is making bets with friends. I tell them, if I don’t complete a task by a deadline, I’ll give them a hundred dollars. For example, I might set my task to getting twenty people to pay me within two weeks of launching a project.
-
+### 3. Solve a problem
-
-
+Start by solving a problem. If you have experienced the problem yourself, even better. Ask your friends what kind of problems they experience.
-
-
+For example, I built [YC Careers](http://jonathanzwhite.github.io/yc-careers/) and [AtomSpace](https://atomspace.co/) to address the needs of one my friends who was having trouble applying to jobs and interviewing as a product designer. Overnight, YC Careers made it to the top of [ProductHunt](https://www.producthunt.com/posts/yc-careers) and AtomSpace booked $100 dollars from strangers within 6 hours of going live.
+**The bigger the pain point you’re trying to solve, the easier it will be to find users.**
-
+Also, analyze problem-solutions from multiple perspectives. Some successful ideas might seem like a solution without a problem. For example, Instagram doesn’t seem to solve any immediate problem. But it does. Instagram solves the need for friends to stay updated on one another. If this need is not met, then it becomes a problem.
-
-
+For further reading, [John Carmack on Idea Generation](https://amasad.me/carmack) and [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) are two great essays to get you started.
-
-
-
+### 4. Ditch the “good idea / bad idea” mindset
-
-
+Once you’ve identified a problem and come up with a solution, you’ll probably ask yourself if your idea is good.
+**Plenty of seemingly bad ideas have turned out to be great businesses.** For example, no one wanted to invest in Airbnb. Brian Chesky, one of the co-founders of Airbnb, detailed their rejections in his essay [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz).
-
+![](https://cdn-images-1.medium.com/max/800/1*WpxUxMCO-7NXr-o1yo023g.png)
+The only way to know for sure if your idea will work is to formulate a product hypothesis and run an experiment to test it. Talk to people and ask if they are interested in your solution. If you can get them to pay you before you’ve built anything, even better.
-
+### 5. Ask for help
-
+When working on a project, reach out to people and ask for their help. **You will be surprised at how many people are willing to pay it forward and help a complete stranger.**
-
-
+Some of the best advice and feedback I have received came from reaching out to people with an email or a DM on Twitter.
-
+Also, when asking for help, be specific and prepare your questions. If you want feedback on a design, send the mockup. If you want feedback on your marketing strategy, detail what you have tried so far. Context is key. And if you don’t get a response, follow up politely. Sometimes the person you’re reaching out to might have genuinely missed your message.
+---
-
+Build and launch your side projects. Use this process as a way to hone your craft so that when opportunity presents itself, you’re ready to take full advantage of it. Luck always favors the prepared.
-
+So next time you come up with an idea, act on it. Identify a problem, define your goal, stay on track, and ask for help.
-
-
+What are you working on right now? How can I help you? Leave me a note here or [tweet](https://twitter.com/jonathanzwhite) them to me on Twitter.
+You can find me on Medium where I publish every week. Or you can follow me on [Twitter](https://twitter.com/JonathanZWhite), where I post non-sensical ramblings about design, front-end development, and virtual reality.
-
This past week, I was in a Lyft. My driver was telling me about all of her ideas for side projects. She had ideas for a children’s book, an app that helps people find parking, and a more efficient way to package gifts. The problem was she was frozen by indecision. She had the ideas but she didn’t know where to start.
-
This is what I told her. Start by building. Pick one project and do whatever you have to do to ship it. If you want to write a book, start with writing a page a day. If you want to build an app, start with some sketches. Anyone can do it.
-
This advice applies to all creators. Once you start building and launching your projects, you won’t be able to stop. Building will become part of your identity. And even if your project fails, you’ll keep at it.
-
The steps you take today will compound over time. Look at great product makers like Drew Wilson, Pieter Levels, and Sebastian Dobrincu. They don’t wait for the right opportunity. They build and ship weekly.
-
In an interview on IndieHackers, Josh Pigford disclosed that before he built Baremetrics, he had iterated through a hundred failed ideas. Today, Baremetrics makes over 60,000 in monthly recurring revenue.
-
-
With that in mind, here are some tactics I’ve compiled from friends and acquaintances who are successful product makers.
-
-
1. Define your goal
-
Early on, figure out the main goal of your project. If you want to break out of your usual design style, focus on the design. If you want to test a new framework for your front-end, focus on the code. If you want to acquire a hundred users, focus on selling.
-
Define your goal early so you don’t get distracted and end up not achieving what you set out to do. Perhaps your project doesn’t become the next Snapchat, but by working towards your goal you will have honed the tools in your product-making arsenal. Little by little, you’ll be able to design, code, and sell faster and more efficiently.
-
2. Stay on track
-
Sometimes product makers fail to ship because they lose motivation. Set small goals; meeting those goals will help you build momentum. Small wins will compound.
-
-
Another thing you can do to stay on track is to be more public about your work. Share your progress online and with your peers. People love to see things like wireframes and will often times provide valuable input.
-
One trick that I find particularly motivating is making bets with friends. I tell them, if I don’t complete a task by a deadline, I’ll give them a hundred dollars. For example, I might set my task to getting twenty people to pay me within two weeks of launching a project.
-
3. Solve a problem
-
Start by solving a problem. If you have experienced the problem yourself, even better. Ask your friends what kind of problems they experience.
-
For example, I built YC Careers and AtomSpace to address the needs of one my friends who was having trouble applying to jobs and interviewing as a product designer. Overnight, YC Careers made it to the top of ProductHunt and AtomSpace booked $100 dollars from strangers within 6 hours of going live.
-
The bigger the pain point you’re trying to solve, the easier it will be to find users.
-
Also, analyze problem-solutions from multiple perspectives. Some successful ideas might seem like a solution without a problem. For example, Instagram doesn’t seem to solve any immediate problem. But it does. Instagram solves the need for friends to stay updated on one another. If this need is not met, then it becomes a problem.
Once you’ve identified a problem and come up with a solution, you’ll probably ask yourself if your idea is good.
-
Plenty of seemingly bad ideas have turned out to be great businesses. For example, no one wanted to invest in Airbnb. Brian Chesky, one of the co-founders of Airbnb, detailed their rejections in his essay 7 Rejections.
-
-
The only way to know for sure if your idea will work is to formulate a product hypothesis and run an experiment to test it. Talk to people and ask if they are interested in your solution. If you can get them to pay you before you’ve built anything, even better.
-
5. Ask for help
-
When working on a project, reach out to people and ask for their help. You will be surprised at how many people are willing to pay it forward and help a complete stranger.
-
Some of the best advice and feedback I have received came from reaching out to people with an email or a DM on Twitter.
-
Also, when asking for help, be specific and prepare your questions. If you want feedback on a design, send the mockup. If you want feedback on your marketing strategy, detail what you have tried so far. Context is key. And if you don’t get a response, follow up politely. Sometimes the person you’re reaching out to might have genuinely missed your message.
-
-
Build and launch your side projects. Use this process as a way to hone your craft so that when opportunity presents itself, you’re ready to take full advantage of it. Luck always favors the prepared.
-
So next time you come up with an idea, act on it. Identify a problem, define your goal, stay on track, and ask for help.
-
What are you working on right now? How can I help you? Leave me a note here or tweet them to me on Twitter.
-
You can find me on Medium where I publish every week. Or you can follow me on Twitter, where I post non-sensical ramblings about design, front-end development, and virtual reality.
-
P.S. If you enjoyed this article, it would mean a lot if you click the 💚 and share with friends.
-
-
- You can't perform that action at this time.
-
-
-
-
-
-
-
-
-
-
-
-
- You signed in with another tab or window. Reload to refresh your session.
- You signed out in another tab or window. Reload to refresh your session.
-
-
-
-
-
-
-
-
-
-
-
-
+---
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
\ No newline at end of file
From f4857dd5c82e2387af2c9ffbd88d767f974a0866 Mon Sep 17 00:00:00 2001
From: Wenlin Ou
Date: Fri, 7 Apr 2017 22:15:49 -0400
Subject: [PATCH 144/945] get ready for proofreading
---
...ng-will-change-until-you-start-building.md | 70 +++++++++----------
1 file changed, 35 insertions(+), 35 deletions(-)
diff --git a/TODO/nothing-will-change-until-you-start-building.md b/TODO/nothing-will-change-until-you-start-building.md
index 0d57984107d..7d9880eb6ba 100644
--- a/TODO/nothing-will-change-until-you-start-building.md
+++ b/TODO/nothing-will-change-until-you-start-building.md
@@ -1,90 +1,90 @@
> * 原文地址:[Nothing will change until you start building.](https://medium.freecodecamp.com/nothing-will-change-until-you-start-building-2681e85e7bdc)
> * 原文作者:[Jonathan Z. White](https://medium.freecodecamp.com/@JonathanZWhite?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[owenlyn](https://github.com/owenlyn)
> * 校对者:
---
![](https://cdn-images-1.medium.com/max/1000/1*EwHpnCZ70FtJMi-lNSl-9Q.png)
-# Nothing will change until you start building.
+# 真正行动之前 你将一无所成
-This past week, I was in a Lyft. My driver was telling me about all of her ideas for side projects. She had ideas for a children’s book, an app that helps people find parking, and a more efficient way to package gifts. The problem was she was frozen by indecision. She had the ideas but she didn’t know where to start.
+就在上周我打了一辆 Lyft 出租车,司机跟我聊了很多关于她的天花乱坠的想法。她的想法里,有一个关于儿童的书籍,有帮助人们停车的APP、有一个更高效的打包礼物的方法,但问题是她总是犹豫不决:她又想法但却不知道从何开始。
-This is what I told her. **Start by building. Pick one project and do whatever you have to do to ship it.** If you want to write a book, start with writing a page a day. If you want to build an app, start with some sketches. Anyone can do it.
+接下来这段话是我对她说的:**开始一些实际行动吧。选一个项目然后尽可能的去完成它。** 如果你想写一本书,那么从每天写一页开始;如果你像做一个APP,那就从草稿开始。任何人都可以做到这些。
-This advice applies to all creators. Once you start building and launching your projects, you won’t be able to stop. Building will become part of your identity. And even if your project fails, you’ll keep at it.
+这些建议对所有创新者都适用。一旦你开始着手在你的项目上,你将会不由自主的继续做下去。创新将会成为你人格的一部分 —— 即使你的项目失败了。
-The steps you take today will compound over time. Look at great product makers like [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), and [Sebastian Dobrincu](https://twitter.com/Sebyddd). **They don’t wait for the right opportunity.** They build and ship weekly.
+你今天迈出的一小步将会是你人生的一大步。看看那些伟大的产品创造者吧,比如 [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), 和 [Sebastian Dobrincu](https://twitter.com/Sebyddd)。**他们不仅仅是等待合适的时机。** 他们每周都在更新他们的产品。
-In an interview on [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics), [Josh Pigford](https://twitter.com/Shpigford) disclosed that before he built Baremetrics, he had iterated through a *hundred* failed ideas. Today, Baremetrics makes over 60,000 in monthly recurring revenue.
+在一个 [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics) 的节目中, [Josh Pigford](https://twitter.com/Shpigford) 透露,在做 Baremetrics 之前,他已经尝试过 *一百个* 失败的点子。 现在, Baremetrics 每个月都会有 60,000 美金进账。
![](https://cdn-images-1.medium.com/max/800/1*BzmVaqAKEzNRybUmMYOxdA.png)
-With that in mind, here are some tactics I’ve compiled from friends and acquaintances who are successful product makers.
+记住这些,接下来的要说的,是我总结的一些来自成功的产品创作者朋友们的策略。
---
-### 1. Define your goal
+### 1. 确定目标
-Early on, figure out the main goal of your project. If you want to break out of your usual design style, focus on the design. If you want to test a new framework for your front-end, focus on the code. If you want to acquire a hundred users, focus on selling.
+在开始的时候,确定你的项目要实现的主要目标。如果你想打破你以往的设计风格,那就把心思花在设计上;如果你测试一个新的前端框架,那就把钻研代码;如果你想获得上百个用户(”大量用户“ 会不会好点),就把精力用在销售上。
-**Define your goal early so you don’t get distracted and end up not achieving what you set out to do.** Perhaps your project doesn’t become the next Snapchat, but by working towards your goal you will have honed the tools in your product-making arsenal. Little by little, you’ll be able to design, code, and sell faster and more efficiently.
+**尽早确定目标可以帮助你摆脱诱惑,防止你到最后忘记初心。** 也许你的项目并不能成为下一个 Snapchat, 但是在通往你目标的道路上,你会更加的熟练运用各种做产品的工具。渐渐地,你会拥有越来越强大的设计、代码、销售等的能力。
-### 2. Stay on track
+### 2. 执行计划
-Sometimes product makers fail to ship because they lose motivation. **Set small goals; meeting those goals will help you build momentum.** Small wins will compound.
+有一些产品没能完的原因是作者在半路失去了动力。 **设定一些小目标(比如先赚一个亿);完成这些目标会让你获得前进的动力。** 小的成就累计起来也是很了不起的。
![](https://cdn-images-1.medium.com/max/800/1*ESildSVTSSOnFXGDxD-l9w.png)
-Another thing you can do to stay on track is to be more public about your work. Share your progress online and with your peers. People love to see things like wireframes and will often times provide valuable input.
+另一个让你保持动力的建议就是让你的项目更加公开。在网上和你的同行们分享你的进展。人们很喜欢看到线框(wireframe 翻译成 ”框架“ 会不会好一点?)类的东西,并常常带来有价值的投入。
-One trick that I find particularly motivating is making bets with friends. I tell them, if I don’t complete a task by a deadline, I’ll give them a hundred dollars. For example, I might set my task to getting twenty people to pay me within two weeks of launching a project.
+我自己找到的一个非常有用的保持动力的小秘诀是和朋友打赌。我对他们说,如果我在截止日期前没有完成某个任务,我就给他们一百刀。比如,我会在一个项目开始的两周内找到二十个人打赌。
-### 3. Solve a problem
+### 3. 解决一个问题
-Start by solving a problem. If you have experienced the problem yourself, even better. Ask your friends what kind of problems they experience.
+解决一个问题。如果你自己也经历了这个问题那就再好不过了。问问你的朋友们他们都有哪些问题需要解决。
-For example, I built [YC Careers](http://jonathanzwhite.github.io/yc-careers/) and [AtomSpace](https://atomspace.co/) to address the needs of one my friends who was having trouble applying to jobs and interviewing as a product designer. Overnight, YC Careers made it to the top of [ProductHunt](https://www.producthunt.com/posts/yc-careers) and AtomSpace booked $100 dollars from strangers within 6 hours of going live.
+比如,我做了 [YC Careers](http://jonathanzwhite.github.io/yc-careers/) 和 [AtomSpace](https://atomspace.co/) 来解决一个产品设计师朋友找工作和面试的烦恼。一夜之间,YC Careers 一下子爬到了 [ProductHunt](https://www.producthunt.com/posts/yc-careers) 的榜首,AtomSpace 在上线6个小时就拿到了来自陌生人的一百美元订单。
-**The bigger the pain point you’re trying to solve, the easier it will be to find users.**
+**你解决越大的痛点,就越容易找到用户。**
-Also, analyze problem-solutions from multiple perspectives. Some successful ideas might seem like a solution without a problem. For example, Instagram doesn’t seem to solve any immediate problem. But it does. Instagram solves the need for friends to stay updated on one another. If this need is not met, then it becomes a problem.
+同时,尝试从过个角度分析问题和解决方案。有些成功的点子看起来并没有解决任何问题,但其实不然。比如,Instagram 看起来没有直接解决任何问题。其实不是。Instagram 满足了朋友之间实时社交的需求。如果这个需求不能得到满足,那才是个问题。
-For further reading, [John Carmack on Idea Generation](https://amasad.me/carmack) and [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) are two great essays to get you started.
+想要了解更多的话,[John Carmack on Idea Generation](https://amasad.me/carmack) 和 [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) 这两篇文章是非常好的开始。
-### 4. Ditch the “good idea / bad idea” mindset
+### 4. 放弃”好主意/坏主意“的想法
-Once you’ve identified a problem and come up with a solution, you’ll probably ask yourself if your idea is good.
+当你明确了你想要解决的问题并有了一个解决方案的时候,你可能会问自己这是不是一个好的点子。
-**Plenty of seemingly bad ideas have turned out to be great businesses.** For example, no one wanted to invest in Airbnb. Brian Chesky, one of the co-founders of Airbnb, detailed their rejections in his essay [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz).
+**相当一部分看起来很糟糕的点子最后被证明是超级棒的生意。** 比如,多年没人想投资 Airbnb。 Brian Chesky, Airbnb 的创始人之一,在他的文章 [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz) 里详细讲述了被投资人拒绝的故事。
![](https://cdn-images-1.medium.com/max/800/1*WpxUxMCO-7NXr-o1yo023g.png)
-The only way to know for sure if your idea will work is to formulate a product hypothesis and run an experiment to test it. Talk to people and ask if they are interested in your solution. If you can get them to pay you before you’ve built anything, even better.
+唯一验证你想法的办法就是做出一个产品来测试它。 找人聊聊,看看他们对你的解决方案是不是感兴趣。如果在你做出任何产品之前就有人原意为之付费那就再棒不过了。
-### 5. Ask for help
+### 5. 寻求帮助
-When working on a project, reach out to people and ask for their help. **You will be surprised at how many people are willing to pay it forward and help a complete stranger.**
+当你在你的项目上埋头苦干的时候,记得寻求别人的帮助。 **你会很惊喜的发现竟然有这么多人原意去帮助一个陌生人(对,就是你)。**
-Some of the best advice and feedback I have received came from reaching out to people with an email or a DM on Twitter.
+我获得的最好的一部分建议就是来自于给别人发邮件或是推特(微博)私信。
-Also, when asking for help, be specific and prepare your questions. If you want feedback on a design, send the mockup. If you want feedback on your marketing strategy, detail what you have tried so far. Context is key. And if you don’t get a response, follow up politely. Sometimes the person you’re reaching out to might have genuinely missed your message.
+在寻求帮助的时候,请提一些精确、有价值的问题。如果你需要关于设计的一些反馈,就把草稿发给别人;如果你需要营销方面的建议,就详细的列出你尝试过的方法。内容才是关键。如果别人没有回复你,请礼貌的再次询问 —— 有时候别人只是没注意到你的消息而已。
---
-Build and launch your side projects. Use this process as a way to hone your craft so that when opportunity presents itself, you’re ready to take full advantage of it. Luck always favors the prepared.
+开始你的兴趣项目吧。利用这个过程磨练自己,这样当机会来临的时候,你才能紧紧抓住。记住,机会总是青睐有准备的人。
-So next time you come up with an idea, act on it. Identify a problem, define your goal, stay on track, and ask for help.
+所以下次你有一个想法的时候,采取行动吧。找到一个问题,确定你的目标,坚持下去,别忘了学会寻求帮助。
-What are you working on right now? How can I help you? Leave me a note here or [tweet](https://twitter.com/jonathanzwhite) them to me on Twitter.
+你现在在做什么项目呢?有什么是我能帮到你的吗?可以在这里或者我的 [tweet](https://twitter.com/jonathanzwhite) 下面留言。
-You can find me on Medium where I publish every week. Or you can follow me on [Twitter](https://twitter.com/JonathanZWhite), where I post non-sensical ramblings about design, front-end development, and virtual reality.
+我在 Medium 上每周都会发文。也可以关注我的 [Twitter](https://twitter.com/JonathanZWhite),我会发一些关于设计、前端开发和虚拟现实的杂想。
-*P.S. If you enjoyed this article, it would mean a lot if you click the 💚 and share with friends.*
+*P.S. If you enjoyed this article, it would mean a lot if you click the 💚 and share with friends.*(可删?)
---
-> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
\ No newline at end of file
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 9df8dac2357ba3070b08880809c050cd61dad0f4 Mon Sep 17 00:00:00 2001
From: Wenlin Ou
Date: Fri, 7 Apr 2017 22:25:45 -0400
Subject: [PATCH 145/945] typo fix
---
...ing-will-change-until-you-start-building.md | 18 +++++++++---------
1 file changed, 9 insertions(+), 9 deletions(-)
diff --git a/TODO/nothing-will-change-until-you-start-building.md b/TODO/nothing-will-change-until-you-start-building.md
index 7d9880eb6ba..111ebd58e1e 100644
--- a/TODO/nothing-will-change-until-you-start-building.md
+++ b/TODO/nothing-will-change-until-you-start-building.md
@@ -10,15 +10,15 @@
# 真正行动之前 你将一无所成
-就在上周我打了一辆 Lyft 出租车,司机跟我聊了很多关于她的天花乱坠的想法。她的想法里,有一个关于儿童的书籍,有帮助人们停车的APP、有一个更高效的打包礼物的方法,但问题是她总是犹豫不决:她又想法但却不知道从何开始。
+就在上周我打了一辆 Lyft 出租车,司机跟我聊了很多关于她的天花乱坠的想法。她的想法里,有一个关于儿童书籍,有帮助人们停车的APP、有一个更高效的打包礼物的方法,但问题是她总是犹豫不决:她很有想法但却不知道从何开始。
-接下来这段话是我对她说的:**开始一些实际行动吧。选一个项目然后尽可能的去完成它。** 如果你想写一本书,那么从每天写一页开始;如果你像做一个APP,那就从草稿开始。任何人都可以做到这些。
+接下来这段话是我对她说的:**开始一些实际行动吧。选一个项目然后尽可能的去完成它。** 如果你想写一本书,那么从每天写一页开始;如果你想做一个APP,那就从草稿开始。任何人都可以做到这些。
这些建议对所有创新者都适用。一旦你开始着手在你的项目上,你将会不由自主的继续做下去。创新将会成为你人格的一部分 —— 即使你的项目失败了。
-你今天迈出的一小步将会是你人生的一大步。看看那些伟大的产品创造者吧,比如 [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), 和 [Sebastian Dobrincu](https://twitter.com/Sebyddd)。**他们不仅仅是等待合适的时机。** 他们每周都在更新他们的产品。
+你今天迈出的一小步将会是你人生的一大步。看看那些伟大的产品缔造者吧,比如 [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), 和 [Sebastian Dobrincu](https://twitter.com/Sebyddd)。**他们不仅仅是等待合适的时机。** 他们每周都在更新他们的产品。
-在一个 [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics) 的节目中, [Josh Pigford](https://twitter.com/Shpigford) 透露,在做 Baremetrics 之前,他已经尝试过 *一百个* 失败的点子。 现在, Baremetrics 每个月都会有 60,000 美金进账。
+在一个 [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics) 的节目中, [Josh Pigford](https://twitter.com/Shpigford) 透露,在做 Baremetrics 之前,他已经尝试过 *一百个* 失败的点子。 然而现在, Baremetrics 每个月都会有 60,000 美金进账。
![](https://cdn-images-1.medium.com/max/800/1*BzmVaqAKEzNRybUmMYOxdA.png)
@@ -31,11 +31,11 @@
在开始的时候,确定你的项目要实现的主要目标。如果你想打破你以往的设计风格,那就把心思花在设计上;如果你测试一个新的前端框架,那就把钻研代码;如果你想获得上百个用户(”大量用户“ 会不会好点),就把精力用在销售上。
-**尽早确定目标可以帮助你摆脱诱惑,防止你到最后忘记初心。** 也许你的项目并不能成为下一个 Snapchat, 但是在通往你目标的道路上,你会更加的熟练运用各种做产品的工具。渐渐地,你会拥有越来越强大的设计、代码、销售等的能力。
+**尽早确定目标可以帮助你摆脱诱惑,防止你到最后忘记初心。** 也许你的项目并不能成为下一个 Snapchat, 但是在通往你目标的道路上,你会更加的熟练运用各种做产品的工具。渐渐地,你会拥有越来越强大的设计、代码、销售等能力。
### 2. 执行计划
-有一些产品没能完的原因是作者在半路失去了动力。 **设定一些小目标(比如先赚一个亿);完成这些目标会让你获得前进的动力。** 小的成就累计起来也是很了不起的。
+有一些产品没能完成的原因是作者在半路失去了动力。 **设定一些小目标(比如先赚一个亿);完成这些目标会让你获得前进的动力。** 小的成就累计起来也是很了不起的。
![](https://cdn-images-1.medium.com/max/800/1*ESildSVTSSOnFXGDxD-l9w.png)
@@ -47,19 +47,19 @@
解决一个问题。如果你自己也经历了这个问题那就再好不过了。问问你的朋友们他们都有哪些问题需要解决。
-比如,我做了 [YC Careers](http://jonathanzwhite.github.io/yc-careers/) 和 [AtomSpace](https://atomspace.co/) 来解决一个产品设计师朋友找工作和面试的烦恼。一夜之间,YC Careers 一下子爬到了 [ProductHunt](https://www.producthunt.com/posts/yc-careers) 的榜首,AtomSpace 在上线6个小时就拿到了来自陌生人的一百美元订单。
+比如,我做了 [YC Careers](http://jonathanzwhite.github.io/yc-careers/) 和 [AtomSpace](https://atomspace.co/) 来解决一个产品设计师朋友找工作和面试的烦恼。一夜之间,YC Careers 占领了 [ProductHunt](https://www.producthunt.com/posts/yc-careers) 的榜首,AtomSpace 在上线6个小时之内就拿到了来自陌生人的一百美元订单。
**你解决越大的痛点,就越容易找到用户。**
同时,尝试从过个角度分析问题和解决方案。有些成功的点子看起来并没有解决任何问题,但其实不然。比如,Instagram 看起来没有直接解决任何问题。其实不是。Instagram 满足了朋友之间实时社交的需求。如果这个需求不能得到满足,那才是个问题。
-想要了解更多的话,[John Carmack on Idea Generation](https://amasad.me/carmack) 和 [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) 这两篇文章是非常好的开始。
+想要了解更多关于这方面内容的话,[John Carmack on Idea Generation](https://amasad.me/carmack) 和 [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) 这两篇文章是非常好的起点。
### 4. 放弃”好主意/坏主意“的想法
当你明确了你想要解决的问题并有了一个解决方案的时候,你可能会问自己这是不是一个好的点子。
-**相当一部分看起来很糟糕的点子最后被证明是超级棒的生意。** 比如,多年没人想投资 Airbnb。 Brian Chesky, Airbnb 的创始人之一,在他的文章 [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz) 里详细讲述了被投资人拒绝的故事。
+**相当一部分看起来很糟糕的点子最后被证明是超级棒的生意。** 比如,当年没人想投资 Airbnb。 Brian Chesky, Airbnb 的创始人之一,在他的文章 [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz) 里详细讲述了被投资人拒绝的故事。
![](https://cdn-images-1.medium.com/max/800/1*WpxUxMCO-7NXr-o1yo023g.png)
From 40c655106da805551994c8c586db94b6267adb63 Mon Sep 17 00:00:00 2001
From: sqrtthree
Date: Sat, 8 Apr 2017 11:33:32 +0800
Subject: [PATCH 146/945] fix typos
---
TODO/preload-prefetch-and-priorities-in-chrome.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/preload-prefetch-and-priorities-in-chrome.md b/TODO/preload-prefetch-and-priorities-in-chrome.md
index 150d1401a6a..5f13ea73d2b 100644
--- a/TODO/preload-prefetch-and-priorities-in-chrome.md
+++ b/TODO/preload-prefetch-and-priorities-in-chrome.md
@@ -107,7 +107,7 @@ preload 使用 “as” 属性加载的资源将会获得与资源 “type”
### 这将会浪费用户的带宽吗?
-**用“preload”和“prefetch”情况下,如果资源不能被缓存,那么都有可能浪费一部分带宽。
+**用“preload”和“prefetch”情况下,如果资源不能被缓存,那么都有可能浪费一部分带宽。**
没有用到的 preload 资源在 Chrome 的 console 里会在 *onload* 事件 3s 后发生警告。
From d9add60f9fca9a897e37485caead9529c889380c Mon Sep 17 00:00:00 2001
From: ZhangFe
Date: Sat, 8 Apr 2017 15:46:41 +0800
Subject: [PATCH 147/945] fix **
---
TODO/css-is-fine-its-just-really-hard.md | 12 ++++++------
1 file changed, 6 insertions(+), 6 deletions(-)
diff --git a/TODO/css-is-fine-its-just-really-hard.md b/TODO/css-is-fine-its-just-really-hard.md
index 692e3cca856..3160c7e1bab 100644
--- a/TODO/css-is-fine-its-just-really-hard.md
+++ b/TODO/css-is-fine-its-just-really-hard.md
@@ -18,7 +18,7 @@ CSS 很棒。它只是太难了,所以我对它进行了一些编译。
![](https://cdn-images-1.medium.com/max/1600/1*ioYNZ-FgsSpoos6b3iKblg.png)
-很长一段时间以来, 我在电脑和手机上画了很多很多的矩形。我写了很多糟糕的 CSS,成千上万行差劲的 Less 代码和大量可怕的 Sass。他们已经融入我的血液了。
+很长一段时间以来,我在电脑和手机上画了很多很多的矩形。我写了很多糟糕的 CSS,成千上万行差劲的 Less 代码和大量可怕的 Sass。他们已经融入我的血液了。
但是我也写过很多很棒的 CSS!我使用 borders 画过三角形,使用 CSS transforms 绘制贝塞尔曲线,制作 60fps 的滚动动画,以及会让你大吃一惊的工具提示。
@@ -30,12 +30,12 @@ CSS性能极高。长久以来,很多人都在为了让 CSS 更快速,可调
—
-当我年轻的时候,我发现每当我想让边框和超链接文本是同一个颜色的时候我都需要在两三个不同的地方去修改,这真是太可怕了。然后我发现了 LESS。现在我可以定义一个 `@wonderfulBlue` 并且在任何地方去使用它。*喂,Jordan,现在的 CSS 也有变量了...*
+当我年轻的时候,我发现每当我想让边框和超链接文本是同一个颜色的时候我都需要在两三个不同的地方去修改,这真是太可怕了。然后我发现了 LESS。现在我可以定义一个 `@wonderfulBlue` 并且在任何地方去使用它。**喂,Jordan,现在的 CSS 也有变量了...**
-然后我开始考虑为什么要为解释 `#left-section` 的宽度是546px(250 * 2 + 23 * 2)留下很长的注释?我开始使用 Less 写我的数学表达式:`2 * @sectionWidth + 2 * @sectionPadding`。*我猜测你不熟悉 calc(),因为它浏览器兼容性不好*
+然后我开始考虑为什么要为解释 `#left-section` 的宽度是546px(250 * 2 + 23 * 2)留下很长的注释?我开始使用 Less 写我的数学表达式:`2 * @sectionWidth + 2 * @sectionPadding`。**我猜测你不熟悉 calc(),因为它浏览器兼容性不好**
-当年 `border-radius` 需要被 polyfill 时,我在所有使用到的地方添加前缀。后来我使用了 `border-radius()` mixin,这样只要在我需要使用的时候把代码添加上就可以了。*好吧如果你只用到了组件分类呢 —*。伙计你能停一停么?让我完成我的文章先。*我错了—*。没事,别担心,继续听下去。
+当年 `border-radius` 需要被 polyfill 时,我在所有使用到的地方添加前缀。后来我使用了 `border-radius()` mixin,这样只要在我需要使用的时候把代码添加上就可以了。**好吧如果你只用到了组件分类呢 — **。伙计你能停一停么?让我完成我的文章先。**我错了 — **。没事,别担心,继续听下去。
当 CSS 解决不了我的问题时,我开始写 Less。它们会被编译成 CSS,并且它在我的用户的设备上工作的非常棒。只是我比原来忙了 10 倍,我无法单独去编写它了。
@@ -45,7 +45,7 @@ CSS性能极高。长久以来,很多人都在为了让 CSS 更快速,可调
他们中的某些页面已经很庞大了,因此通常我们会将我们的 CSS (好吧,Less)和 JavaScript 分割成独立的文件,这样用户就不必下载练习页面的代码来观看视频。
-有些时候,我们移除了很多代码后样式看起来就不对了。因为我们的主页菜单可能希望有一个 `.left-arrow` 类,但是现在这个 class 的样式在 `exercise.css` 文件里。通常我们注意不到这点,因为导航条被鼠标点击几次后 `.left-arrow` 会被整齐地卷起来。*这么看来你应该有截图测试或更严格的 QA 过程*,我刚才说了什么来着?
+有些时候,我们移除了很多代码后样式看起来就不对了。因为我们的主页菜单可能希望有一个 `.left-arrow` 类,但是现在这个 class 的样式在 `exercise.css` 文件里。通常我们注意不到这点,因为导航条被鼠标点击几次后 `.left-arrow` 会被整齐地卷起来。**这么看来你应该有截图测试或更严格的 QA 过程**,我刚才说了什么来着?
唉,这是很辛苦的工作!但是代码就是偶尔会出 bug,修复它们并且继续前进,这是件很酷的事。
@@ -68,7 +68,7 @@ CSS性能极高。长久以来,很多人都在为了让 CSS 更快速,可调
CSS很好,速度很快,它已经发展了有20多年了,并且适用于各种应用程序。
-但是我真的不喜欢写 CSS。很多人也不喜欢,所以我们开发了这些很棒的模式去写 CSS。但是我也不喜欢以这些模式去写,我有更好的事情要去做。并且 JavaScript 也很酷。*实际上 JavaScript 有更多的可能性*。[所以我用 JavaScript 去编写我的 CSS](https://github.com/khan/aphrodite).
+但是我真的不喜欢写 CSS。很多人也不喜欢,所以我们开发了这些很棒的模式去写 CSS。但是我也不喜欢以这些模式去写,我有更好的事情要去做。并且 JavaScript 也很酷。**实际上 JavaScript 有更多的可能性**。[所以我用 JavaScript 去编写我的 CSS](https://github.com/khan/aphrodite).
把这样的代码:
From 01ef70394f6f662541c7f6f9eac0996902a9b094 Mon Sep 17 00:00:00 2001
From: ZhangFe
Date: Sat, 8 Apr 2017 15:53:25 +0800
Subject: [PATCH 148/945] =?UTF-8?q?fix=E8=BD=AC=E6=8D=A2=E6=A0=BC=E5=BC=8F?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/css-is-fine-its-just-really-hard.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/css-is-fine-its-just-really-hard.md b/TODO/css-is-fine-its-just-really-hard.md
index 3160c7e1bab..423250fcfb8 100644
--- a/TODO/css-is-fine-its-just-really-hard.md
+++ b/TODO/css-is-fine-its-just-really-hard.md
@@ -35,7 +35,7 @@ CSS性能极高。长久以来,很多人都在为了让 CSS 更快速,可调
然后我开始考虑为什么要为解释 `#left-section` 的宽度是546px(250 * 2 + 23 * 2)留下很长的注释?我开始使用 Less 写我的数学表达式:`2 * @sectionWidth + 2 * @sectionPadding`。**我猜测你不熟悉 calc(),因为它浏览器兼容性不好**
-当年 `border-radius` 需要被 polyfill 时,我在所有使用到的地方添加前缀。后来我使用了 `border-radius()` mixin,这样只要在我需要使用的时候把代码添加上就可以了。**好吧如果你只用到了组件分类呢 — **。伙计你能停一停么?让我完成我的文章先。**我错了 — **。没事,别担心,继续听下去。
+当年 `border-radius` 需要被 polyfill 时,我在所有使用到的地方添加前缀。后来我使用了 `border-radius()` mixin,这样只要在我需要使用的时候把代码添加上就可以了。**好吧如果你只用到了组件分类呢 —**。伙计你能停一停么?让我完成我的文章先。**我错了 —**。没事,别担心,继续听下去。
当 CSS 解决不了我的问题时,我开始写 Less。它们会被编译成 CSS,并且它在我的用户的设备上工作的非常棒。只是我比原来忙了 10 倍,我无法单独去编写它了。
From 9886d460d18b7e1f4d32fd86d87814861034d315 Mon Sep 17 00:00:00 2001
From: reid
Date: Sat, 8 Apr 2017 17:26:48 +0800
Subject: [PATCH 149/945] =?UTF-8?q?=E4=BA=8C=E6=A0=A1?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...k-bits-getting-the-most-out-of-the-commonschunkplugin.md | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md b/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
index 1d88f590d61..cd003ab1e9f 100644
--- a/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
+++ b/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
@@ -91,7 +91,7 @@ webpack 核心团队隔三差五地就会在 Twitter 上作一些寓教于乐的
filename: "commonlazy.js"
});
-类似地 —— webpack 会扫描所有 chunks 并检查公共模块。由于设置了 `async: true`,只有代码拆分的 bundles 会被扫描。因为我们并没有指明 `minChunks` 的值,所有 webpack 会取其默认值 3。综上,上述代码的含义是:
+类似地 —— webpack 会扫描所有 chunks 并检查公共模块。由于设置了 `async: true`,只有代码拆分的 bundles 会被扫描。因为我们并没有指明 `minChunks` 的值,所以 webpack 会取其默认值 3。综上,上述代码的含义是:
> **嘿 webpack,请检查所有的普通(即懒加载的)chunks,如果某个模块出现在了 3 个或 3 个以上的 chunks 中,就将其分离到一个独立的异步公共 chunk 中去。**
@@ -99,7 +99,7 @@ webpack 核心团队隔三差五地就会在 Twitter 上作一些寓教于乐的
![Markdown](http://i4.buimg.com/1949/626cbab70072f442.png)
-现在异步 chunks 都非常的小,并且所有代码都被聚合到 `commonlazy.js` 文件中去了。因为这些 bundles 本来就很小了, 首次访问可能都察觉不到代码体积的变化。现在,每一个代码拆分的 bundle 所需携带的数据更少了;而且,通过将这些公共模块放到一个独立可缓存的 chunk 中,我们节省了用户加载时间,降低了数据消费量(data consumption)。
+现在异步 chunks 都非常的小,并且所有代码都被聚合到 `commonlazy.js` 文件中去了。因为这些 bundles 本来就很小了, 首次访问可能都察觉不到代码体积的变化。现在,每一个代码拆分的 bundle 所需携带的数据更少了;而且,通过将这些公共模块放到一个独立可缓存的 chunk 中,我们节省了用户加载时间,减少了需要传输的数据量(data consumption)。
#### 更多控制:minChunks 函数 ####
@@ -129,7 +129,7 @@ webpack 核心团队隔三差五地就会在 Twitter 上作一些寓教于乐的
function lodashMomentModuleFilter(module, count) {
return module.resource && /lodash|moment/.test(module.resource) && count >= 2;
}
-
+
function immutableReactModuleFilter(module, count) {
return module.resource && /immutable|react/.test(module.resource) && count >=4
}
From 719e846484632c4e1d2ca7b654644ee4d1987a76 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E9=94=85=E7=82=89=E5=B7=A5?=
Date: Sat, 8 Apr 2017 18:14:32 +0800
Subject: [PATCH 150/945] modified according to @Vivienmm's revision
---
...-java-2-java-8-stream-android-rxjava2-hell-part4.md | 10 +++++-----
1 file changed, 5 insertions(+), 5 deletions(-)
diff --git a/TODO/war-learning-curve-rx-java-2-java-8-stream-android-rxjava2-hell-part4.md b/TODO/war-learning-curve-rx-java-2-java-8-stream-android-rxjava2-hell-part4.md
index bdc42075a07..b0b63eeb7c5 100644
--- a/TODO/war-learning-curve-rx-java-2-java-8-stream-android-rxjava2-hell-part4.md
+++ b/TODO/war-learning-curve-rx-java-2-java-8-stream-android-rxjava2-hell-part4.md
@@ -11,13 +11,13 @@
又是新的一天,如果学点新东西,这一天一定会很酷炫。
-小伙伴们一切顺利啊,这是我们的 RxJava2 Android 系列的第四部分 [ [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md), [第二部分](http://www.uwanttolearn.com/android/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2/), [第三部分](https://github.com/xitu/gold-miner/blob/master/TODO/functional-interfaces-functional-programming-and-lambda-expressions-reactive-programming-android-rxjava2-what-the-hell-is-this-part3.md) ]。 好消息是我们已经满足了一些先决条件,已经准备好用 Rx 来工作了。在使用 RxJava2 Android Observable 之前,我会先用 Java8 的 Stream 来做响应式编程。我认为我们应该了解 Java8,我还感觉通过使用 Java8 的流式 API 会让 RxJava2 Android 的学习曲线更简单。
+小伙伴们一切顺利啊,这是我们的 RxJava2 Android 系列的第四部分 [ [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md), [第二部分](http://www.uwanttolearn.com/android/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2/), [第三部分](https://github.com/xitu/gold-miner/blob/master/TODO/functional-interfaces-functional-programming-and-lambda-expressions-reactive-programming-android-rxjava2-what-the-hell-is-this-part3.md) ]。 好消息是我们已经做好准备,可以开始使用 Rx 了。在使用 RxJava2 Android Observable 之前,我会先用 Java8 的 Stream 来做响应式编程。我认为我们应该了解 Java8,我还感觉通过使用 Java8 的流式 API 会让 RxJava2 Android 的学习曲线更简单。
**动机:**
动机跟我在 [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md) 和大家分享过的一样。在我开始学习 RxJava2 Android 的时候,我并不知道我该怎样使用、在什么地方使用它。
-现在我们知道学习这个需要一些先决条件,但当初我对此一无所知。在我刚刚开始的时候,该如何从任何数据或对象上创建一个 Observable 呢。然后我知道当 Observable 上的数据发生了一些事件的时候,有接口或者可以叫做“回调”会被调用。这在理论上很棒,但是当我付诸实践的时候,GG了。我发现很多理论上应该成立的模式在我去用的时候完全不起作用。对我来说最大的问题,是不能用响应或者函数式响应的思维思考问题。我有命令式编程和面向对象编程的背景,由于先入为主,所以对我来说理解会有些难。我一直在问这些问题:我该在哪里实现?我应该怎么实现?如果你跟着这篇文章看下来,我可以 100% 保证你会知道怎样把命令式代码转换到 Rx 代码,虽然写出来的 Rx 代码可能不是很好,但是你起码知道怎么开始。
+现在我们已经学会了一些预备知识,但当时我什么都不知道。我那时从学习如何从数据或对象创建 Observable 开始。然后学会当 Observable 数据上发生变化时,去调用哪些接口(或者可以叫做“回调”)。这在理论上很棒,但是当我付诸实践的时候,GG了。我发现很多理论上应该成立的模式在我去用的时候完全不起作用。对我来说最大的问题,是不能用响应或者函数式响应的思维思考问题。我有命令式编程和面向对象编程的背景,由于先入为主,所以对我来说理解会有些难。我一直在问这些问题:我该在哪里实现?我应该怎么实现?如果你跟着这篇文章看下来,我可以 100% 保证你会知道怎样把命令式代码转换到 Rx 代码,虽然写出来的 Rx 代码可能不是很好,但是你起码知道怎么开始。
**回顾:**
@@ -50,7 +50,7 @@
Stream:
- 用来支持在元素形成的流上进行函数式操作(比如在集合上进行的 map-reduce 变换)的类(*docs.oracle*)。
+支持在元素形成的流上进行函数式操作(比如在集合上进行的 map-reduce 变换)的类(*docs.oracle*)。
第一个问题:在英语中 Stream 是什么意思?
@@ -156,7 +156,7 @@ public static void main(String[] args) {
同志们这里需要注意下!
-在 Java8 Stream 中有个叫做 Predicate(谓词,可以判断真假,详情见线性代数中的相关定义——译注)的函数式接口。所以,如果我想进行过滤的话需要把这个函数式接口送到流的过滤器函数里面。现在,我给大家展示在我们的代码中如何创建“大石子过滤器”。
+在 Java8 Stream 中有个叫做 Predicate(谓词,可以判断真假,详情见离散数学中的相关定义——译注)的函数式接口。所以,如果我想进行过滤的话,可以用这个函数式接口实现流的过滤功能。现在,我给大家展示在我们的代码中如何创建“大石子过滤器”。
```
private static Predicate BigStoneFilter = new Predicate() {
@@ -767,7 +767,7 @@ Observable.range(0, 10)
**总结:**
-同志们干得好!今天我们学 Rx Android 学得很开心。我们从图画开始,然后使用了 Java8 的流(Stream)。之后将 Java8 的流转换到 RxJava 2 Android 的 Observable。再之后,我们看到了真实项目中的示例并且展示了在现有的项目中如何开始使用 Rx,把 if 换成 filter,用 map 进行数据转化,用 flatmap 代替嵌套的循环。下篇文章: [Dialogue between Rx Observable and a Developer (Me) [ Android RxJava2 ] ( What the hell is this ) Part5](http://www.uwanttolearn.com/android/dialogue-rx-observable-developer-android-rxjava2-hell-part5/).
+同志们干得好!今天我们学 Rx Android 学得很开心。我们从图画开始,然后使用了 Java8 的流(Stream)。之后将 Java8 的流转换到 RxJava 2 Android 的 Observable。再之后,我们看到了真实项目中的示例并且展示了在现有的项目中如何开始使用 Rx。最后,我展示了一些转换到 Rx 的技巧:把循环用 forEach 替换,把 if 换成 filter,用 map 进行数据转化,用 flatmap 代替嵌套的循环。下篇文章: [Dialogue between Rx Observable and a Developer (Me) [ Android RxJava2 ] ( What the hell is this ) Part5](http://www.uwanttolearn.com/android/dialogue-rx-observable-developer-android-rxjava2-hell-part5/).
希望你们开心,同志们再见!
From a57c45737bc90135aac4b4343880fc81159f1b0e Mon Sep 17 00:00:00 2001
From: zhuzi
Date: Sat, 8 Apr 2017 20:12:00 +0800
Subject: [PATCH 151/945] add proofreader info
---
TODO/using-devtools-tweak-designs-browser.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/using-devtools-tweak-designs-browser.md b/TODO/using-devtools-tweak-designs-browser.md
index fe280545985..9fc7266025c 100644
--- a/TODO/using-devtools-tweak-designs-browser.md
+++ b/TODO/using-devtools-tweak-designs-browser.md
@@ -2,7 +2,7 @@
> * 原文作者:[AHMAD SHADEED](https://css-tricks.com/author/shadeed9/)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[bambooom](https://github.com/bambooom)
-> * 校对者:
+> * 校对者:[gy134340](https://github.com/gy134340) / [avocadowang](https://github.com/avocadowang)
# 使用开发者工具在浏览器中调整设计
From 411b25901dc1d966ea0c32cfecfc26b40d05c9eb Mon Sep 17 00:00:00 2001
From: gy134340
Date: Sat, 8 Apr 2017 22:57:16 +0800
Subject: [PATCH 152/945] =?UTF-8?q?=E8=B7=8C=E5=AE=95=E8=B5=B7=E4=BC=8F?=
=?UTF-8?q?=E7=9A=84=E5=87=BD=E6=95=B0=E5=BC=8F=E7=BC=96=E7=A8=8B=EF=BC=88?=
=?UTF-8?q?=E7=BB=84=E6=88=90=E5=8C=96=E8=BD=AF=E4=BB=B6=EF=BC=89?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...ctional-programming-composable-software.md | 124 +++++++++---------
1 file changed, 62 insertions(+), 62 deletions(-)
diff --git a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
index fda67ce9863..463b3cbdc66 100644
--- a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
+++ b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
@@ -4,117 +4,117 @@
> * 译者:
> * 校对者:
-# The Rise and Fall and Rise of Functional Programming (Composing Software)
+# 跌宕起伏的函数式编程(组成化软件)
-Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0)
+烟雾的方块艺术 —MattysFlicks —(CC BY 2.0)
-> Note: This is part 1 of the “Composing Software” series on learning functional programming and compositional software techniques in JavaScript ES6+ from the ground up. Stay tuned. There’s a lot more of this to come!
+> 注意:这是从基础学习函数式编程和使用 JavaScript ES6+ 组成软件的第一部分。保持关注,接下来还有很多!
-When I was about 6 years old, I spent a lot of time playing computer games with my best friend. His family had a room full of computers. To me, they were irresistible. Magic. I spent many hours exploring all the games. One day I asked my friend, “how do we make a game?”
+当我 6 岁时,我花了很多时间跟我的小伙伴玩电脑游戏,他家里有一整房间的电脑。对于我说,那是及其不可抗拒的魔力。我花了很多时间探索所有的游戏。一天我问他,“我们怎样做一个游戏?”
-He didn’t know, so we asked his dad, who reached up on a high shelf and pulled down a book of games written in Basic. So began my journey with programming. By the time public school got around to teaching algebra, I already knew the topic well, because programming is basically algebra. It can be, anyway.
+他不知道,所以我们问了他的老爸,他的老爸爬上一个很高的架子拿下来一本使用 Basic 编写的游戏书籍。就那样开始了我的编程之路。当公立学校开始教授代数时,我已经熟稔其中的概念了,因为编程基本上是代数。无论如何,它都是。
-### The Rise of Composable Software
+### 软件组成化的兴起
-In the beginning of computer science, before most of computer science was actually done on computers, there lived two great computer scientists: Alonzo Church, and Alan Turing. They produced two different, but equivalent universal models of computation. Both models could compute anything that can be computed (hence, “universal”).
+在计算机科学的起步阶段,在大多数的计算机科学在电脑上完成之前,有两位伟大的计算机科学家:阿隆佐·邱奇和艾伦·图灵。他们发明了两种不同、但是普遍通用的计算模型。两种都可以计算所有可被计算的东西(因此,“普遍”)。
-Alonzo Church invented lambda calculus. Lambda calculus is a universal model of computation based on function application. Alan Turing is known for the turing machine. A turing machine is a universal model of computation that defines a theoretical device that manipulates symbols on a strip of tape.
+阿隆佐·邱奇发明了 lambda 表达式。lambda 表达式是基于函数应用的通用计算模型。艾伦·图灵因为图灵机而知名。图灵机使用定义一个在磁带上操作符号的理论装置来计算的通用模型。
-Together, they collaborated to show that lambda calculus and the turing machine are functionally equivalent.
+总的说,他们共同说明了 lambda 表达式和图灵机功能上是相等的。
-Lambda calculus is all about function composition. Thinking in terms of function composition is a remarkably expressive and eloquent way to compose software. In this text, we’re going to discuss the importance of function composition in software design.
+lambda 表达式全是函数组成,依靠函数来编写软件是非常高效和有意义的。本文中,我们将会讨论软件设计中函数的组成的重要性。
-There are three important points that make lambda calculus special:
+有三点造就了 lambda 表达式的特别之处:
-1. Functions are always anonymous. In JavaScript, the right side of `const sum = (x, y) => x + y` is the *anonymous* function expression `(x, y) => x + y`.
-2. Functions in lambda calculus only accept a single input. They’re unary. If you need more than one parameter, the function will take one input and return a new function that takes the next, and so on. The n-ary function `(x, y) => x + y` can be expressed as a unary function like: `x => y => x + y`. This transformation from an n-ary function to a unary function is known as currying.
-3. Functions are first-class, meaning that functions can be used as inputs to other functions, and functions can return functions.
+1. 函数都是匿名的,在 JavaScript 中,右边的 `const sum = (x, y) => x + y` 等同于匿名函数 `(x, y) => x + y`.
+2. lambda 表达式中的函数只接收一个参数。他们是一元的,如果你需要多个参数,函数将会接受一个输入返回一个调用下一个函数的函数,然后继续这样。非一元函数 `(x, y) => x + y` 可以被表示为一个像 `x => y => x + y` 的一元函数。这个把多元函数转换成一元函数的过程叫做柯里化。
+3. 函数是一级的,意味着函数可以作为参数传递给其他函数,同时函数可以返回函数。
-Together, these features form a simple, yet expressive vocabulary for composing software using functions as the primary building block. In JavaScript, anonymous functions and curried functions are optional features. While JavaScript supports important features of lambda calculus, it does not enforce them.
+总的说来,这些特性形成使用函数作为初始模块的一个简单且具有表达性的方法来构造软件。在 JavaScript 中,函数的匿名和柯里化都是可选的特性。虽然 JavaScript 支持这些 lambda 表达式的重要属性,它却并不强制使用这些。
-The classic function composition takes the output from one function and uses it as the input for another function. For example, the composition:
+这些经典的函数组成方法用一个函数的输出来作为另一个函数的输入,例如,对于组合:
- f . g
+ f . g
+
+也可以写做:
-Can be written as:
+ compose2 = f => g => x => f(g(x))
+
+这里是你使用它的方法:
+
+ double = n => n * 2
+ inc = n => n + 1
+
+ compose2(double)(inc)(3)
- compose2 = f => g => x => f(g(x))
+`compose2()` 函数使用 `double` 函数作为第一个参数,使用 `inc` 函数作为第二个参数,同时对于两个函数的组合传入参数 `3`。再看一下 `compose2()` 函数所写的,`f` 是 `double()`,`g` 是 `inc()`,同时 `x` 是 `3`。函数 `compose2(double)(inc)(3)` 的调用,实际上是三个不同函数的调用:
-Here’s how you’d use it:
+1. 首先传入 `double` 同时返回一个新的函数。
+2. 返回的函数传入 `inc` 同时再返回一个新的函数。
+3. 再返回的函数传入 `3` 同时计算 `f(g(x))`,最后实际上是 `double(inc(3))`。
+4. `x` 等于 `3` 同时传给 `inc()`。
+5. `inc(3)` 等于 `4`。
+6. `double(4)` 等于 `8`。
+7. 函数返回 `8`。
- double = n => n * 2
- inc = n => n + 1
+组成软件时,可以被看作一个由函数组成的图。看一下下面:
+
+ append = s1 => s2 => s1 + s2
+ append('Hello, ')('world!')
- compose2(double)(inc)(3)
-
-The `compose2()` function takes the `double` function as the first argument, the `inc` function as the second, and then applies the composition of those two functions to the argument `3`. Looking at the signature of `compose2()` again, `f` is `double()`, `g` is `inc()`, and `x` is `3`. The function call, `compose2(double)(inc)(3)`, is actually 3 different function invocations:
-
-1. The first passes `double` and returns a new function.
-2. The returned function takes `inc` and returns a new function.
-3. The next returned function takes `3` and evaluates `f(g(x))`, which is now `double(inc(3))`.
-4. `x` evaluates to `3` and gets passed into `inc()`.
-5. `inc(3)` evaluates to `4`.
-6. `double(4)` evaluates to `8`.
-7. `8` gets returned from the function.
-
-When software is composed, it can be represented by a graph of function compositions. Consider the following:
-
- append = s1 => s2 => s1 + s2
- append('Hello, ')('world!')
-
-You could represent it visually:
+你可以想象成这样:
-Lambda calculus was hugely influential on software design, and prior to about 1980, many very influential icons of computer science were building software using function composition. Lisp was created in 1958, and was heavily influenced by lambda calculus. Today, Lisp is the second-oldest language that’s still in popular use.
+lambda 表达式对软件设计产生了很大的影响,在 1980 年之前,计算机科学领域很多有影响的东西使用函数来构造软件。Lisp 在 1958 年被创作出来,很大程度上受到了 lambda 表达式的影响。如今,Lisp 是广泛使用的第二老的语言了。
-I was introduced to it through AutoLISP: the scripting language used in the most popular Computer Aided Design (CAD) software: AutoCAD. AutoCAD is so popular, virtually every other CAD application supports AutoLISP so that they can be compatible. Lisp is also a popular teaching language in computer science curriculum for three reasons:
+我通过 AutoLISP:一个作为脚本语言被用于最流行的计算机辅助设计(CAD)软件:AutoCAD,接触到它。AutoCAD 很流行,实际上所有其他的 CAD 软件都兼容支持 AutoLISP。Lisp 因为以下三点原因被广泛作为计算机科学的课程:
-1. Its simplicity makes it easy to learn the basic syntax and semantics of Lisp in about a day.
-2. Lisp is all about function composition, and function composition is an elegant way to structure applications.
-3. The best computer science text book I know of uses Lisp: [Structure and Interpretation of Computer Programs](https://www.amazon.com/Structure-Interpretation-Computer-Programs-Engineering/dp/0262510871/ref=as_li_ss_tl?ie=UTF8&linkCode=ll1&tag=eejs-20&linkId=4896ed63eee8657b6379c2acd99dd3f3).
+1. 可以很容易的在一天左右学习 Lisp 基础的词法和语法。
+2. Lisp 全是由函数组成,函数组合是构造应用非常优雅的方式。
+3. 我知道的使用 Lisp 的最棒的计算机科学书籍:[计算机程序的结构与解释](https://www.amazon.com/Structure-Interpretation-Computer-Programs-Engineering/dp/0262510871/ref=as_li_ss_tl?ie=UTF8&linkCode=ll1&tag=eejs-20&linkId=4896ed63eee8657b6379c2acd99dd3f3)。
-### The Fall of Composable Software
+### 组合型软件的衰落
-Somewhere between 1970 and 1980, the way that software was created drifted away from simple compositions, and became a list of linear instructions for the computer to follow. Then came object-oriented programming — a great idea about component encapsulation and message passing that got distorted by popular languages into a horrible idea about inheritance hierarchies and **is-a** relationships for feature reuse.
+在 1970 到 1980 中间的某段时间,软件的构造开始偏离简单的组合,成为一串线性的让计算机执行的指令。然后面向对象编程 — 一个伟大的关于组件的封装和信息传递的思想被流行的编程语言扭曲成为了特性的重用而采取的糟糕的继承层次和 *is-a* 关系。
-Functional programming was relegated to the sidelines and academia: The blissful obsession of the geekiest of programming geeks, professors in their ivy league towers, and some lucky students who escaped the Java force-feeding obsession of the 1990’s — 2010's.
+函数式编程语言退居二线:只有编程极客的痴迷、常春藤盟校的教授和一些幸运的学生可以在 1990 — 2010 年间逃离 Java 的强迫性学习。
-For most of us, creating software was a bit of a nightmare for 30 years. Dark times.
+对于我们的大多数人来说,已经经历了大约 30 年的软件编写噩梦和黑暗时期。
-### The Rise of Composable Software
+### 组合型软件的兴起
-Around 2010, something great began to happen: JavaScript exploded. Before about 2006, JavaScript was widely considered a toy language used to make cute animations happen in web browsers, but it had some powerful features hidden in it. Namely, the most important features of lambda calculus. People started whispering in the shadows about this cool new thing called “functional programming”.
+在 2010 年左右,一些有趣的事情发生了:JavaScript 的崛起。在 2006 年左右,JavaScript 被广泛的看作玩具语言和被用制作浏览器中好玩的动画,但是它有一些隐藏饿及其强大的属性。即 lambda 表达式中最重要的特性。人们开始暗中讨论一个叫做 “函数式编程的” 酷东西。
[![](https://i.embed.ly/1/display/resize?url=https%3A%2F%2Fpbs.twimg.com%2Fprofile_images%2F3722564505%2Fb8030b8f3990875e8f38ed877fdf8d25_bigger.png&key=4fce0568f2ce49e8b54624ef71a8a5bd&width=40)](https://video.twimg.com/tweet_video/CmectVVVUAAsvpo.mp4)
-I keep telling you [#JavaScript](https://twitter.com/hashtag/JavaScript?src=hash) is not a toy language. Now I have to show you. [#StarWars](https://twitter.com/hashtag/StarWars?src=hash) [@BrendanEich](https://twitter.com/BrendanEich)
+我一直在告诉大家 [JavaScript] 并不是一门玩具语言。现在我需要展示它。[#StarWars](https://twitter.com/hashtag/StarWars?src=hash) [@BrendanEich](https://twitter.com/BrendanEich)
-By 2015, the idea of building software with function composition was popular again. To make it simpler, the JavaScript specification got its first major upgrade of the decade and added arrow functions, which made it easier to create and read functions, currying, and lambda expressions.
+在 2015 年,使用函数的组合来编写软件又开始流行起来。为了更简单化,JavaScript 规范获得的数十年来第一次主要的更新并且添加了箭头函数,为了更简单的编写和读取函数、柯里化,和 lambda 语句。
-Arrow functions were like rocket fuel for the functional programming explosion in JavaScript. Today it’s rare to see a large application which doesn’t use a lot of functional programming techniques.
+箭头函数像是 JavaScript 函数式编程飞升的燃料。现在很少看见不使用很多函数式编程技术的大型应用了。
-Composition is a simple, elegant, and expressive way to clearly model the behavior of software. The process of composing small, deterministic functions to create larger software components and functionality produces software that is easier to organize, understand, debug, extend, test, and maintain.
+组合型可以简单、优雅的表达软件的模型和行为。通过把小的、确定的函数组成稍大的组件并构成软件的过程,可以更为简单的组织、理解、调试、扩展、测试和掌控。
-As you read the following text, please experiment with the examples. Remember what it was like as a child to pick things apart, explore, and play while you learn. Rediscover the childhood joy of discovery. Let’s make some magic.
+你在阅读下一部分时,可以使用实例实验,记住要把你自己当孩子一样把其他的思想扔在一边在学习中去探索和玩耍。重新发现孩童时发现新事物的欣喜。让我们来做一些魔术吧。
-[Continued in part 2: “Why Learn Functional Programming in JavaScript?”](https://medium.com/javascript-scene/why-learn-functional-programming-in-javascript-composing-software-ea13afc7a257#.i6vf0q8uy)
+[接下来的第二部分:“为什么要用 JavaScript 学习函数式编程?”]
-### Next Steps
+### 下一步
-Want to learn more about functional programming in JavaScript?
+想更多的学习 JavaScript 的函数式编程?
-[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/). If you’re not a member, you’re missing out!
+[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/),什么,你还不是其中之一,out 了!
[![](https://cdn-images-1.medium.com/freeze/max/30/1*3njisYUeHOdyLCGZ8czt_w.jpeg?q=20)![](https://cdn-images-1.medium.com/max/800/1*3njisYUeHOdyLCGZ8czt_w.jpeg)](https://ericelliottjs.com/product/lifetime-access-pass/)
-***Eric Elliott*** is the author of [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly), and [*“Learn JavaScript with Eric Elliott”*](http://ericelliottjs.com/product/lifetime-access-pass/). He has contributed to software experiences for **Adobe Systems, Zumba Fitness, he Wall Street Journal, ESPN, BBC, and top recording artists including Usher, Frank Ocean, Metallica**, and many more.
+*Eric Elliott* 是 [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly) 和 “Learn JavaScript with Eric Elliott” 的作者。他曾效力于 *Adobe Systems, Zumba Fitness, he Wall Street Journal, ESPN, BBC, and top recording artists including Usher, Frank Ocean, Metallica* 和其他一些公司。
-*He spends most of his time in the San Francisco Bay Area with the most beautiful woman in the world.*
+*他和她的老婆(很漂亮)大部分时间都在旧金山湾区里。*
---
From 4479517a81c541ddd708327ce6e127ca1190575c Mon Sep 17 00:00:00 2001
From: Wenlin Ou
Date: Sun, 9 Apr 2017 00:45:08 -0400
Subject: [PATCH 153/945] fix errors according proofreader
---
...ng-will-change-until-you-start-building.md | 20 +++++++++----------
1 file changed, 10 insertions(+), 10 deletions(-)
diff --git a/TODO/nothing-will-change-until-you-start-building.md b/TODO/nothing-will-change-until-you-start-building.md
index 111ebd58e1e..034e8ae1484 100644
--- a/TODO/nothing-will-change-until-you-start-building.md
+++ b/TODO/nothing-will-change-until-you-start-building.md
@@ -10,11 +10,11 @@
# 真正行动之前 你将一无所成
-就在上周我打了一辆 Lyft 出租车,司机跟我聊了很多关于她的天花乱坠的想法。她的想法里,有一个关于儿童书籍,有帮助人们停车的APP、有一个更高效的打包礼物的方法,但问题是她总是犹豫不决:她很有想法但却不知道从何开始。
+就在上周我打了一辆 Lyft 出租车,司机跟我聊了很多关于她的天花乱坠的想法。她的想法里,她有写一本儿童书籍的想法,有开发一个帮助人们找到停车位置的软件的想法,有寻找到一种更高效打包礼物的想法,但问题是她总是犹豫不决:她有很多想法但却不知道从何开始。
-接下来这段话是我对她说的:**开始一些实际行动吧。选一个项目然后尽可能的去完成它。** 如果你想写一本书,那么从每天写一页开始;如果你想做一个APP,那就从草稿开始。任何人都可以做到这些。
+接下来这段话是我对她说的:**开始一些实际行动吧。选一个项目然后尽可能的去完成它。** 如果你想写一本书,那么从每天写一页开始;如果你想做一个APP,那就从构建草图开始。任何人都可以做到这些。
-这些建议对所有创新者都适用。一旦你开始着手在你的项目上,你将会不由自主的继续做下去。创新将会成为你人格的一部分 —— 即使你的项目失败了。
+这个建议适用于所有的创作者。一旦你开始着手在你的项目上,你将会不由自主的继续做下去。创新将会成为你人格的一部分 —— 即使你的项目失败了。
你今天迈出的一小步将会是你人生的一大步。看看那些伟大的产品缔造者吧,比如 [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), 和 [Sebastian Dobrincu](https://twitter.com/Sebyddd)。**他们不仅仅是等待合适的时机。** 他们每周都在更新他们的产品。
@@ -29,11 +29,11 @@
### 1. 确定目标
-在开始的时候,确定你的项目要实现的主要目标。如果你想打破你以往的设计风格,那就把心思花在设计上;如果你测试一个新的前端框架,那就把钻研代码;如果你想获得上百个用户(”大量用户“ 会不会好点),就把精力用在销售上。
+在开始的时候,确定你的项目要实现的主要目标。如果你想打破你以往的设计风格,那就把心思花在设计上;如果你测试一个新的前端框架,那就钻研代码;如果你想获得大量用户,就把精力用在销售上。
-**尽早确定目标可以帮助你摆脱诱惑,防止你到最后忘记初心。** 也许你的项目并不能成为下一个 Snapchat, 但是在通往你目标的道路上,你会更加的熟练运用各种做产品的工具。渐渐地,你会拥有越来越强大的设计、代码、销售等能力。
+**尽早确定目标可以帮助你摆脱诱惑,防止你到最后忘记初心。** 也许你的项目并不能成为下一个 Snapchat, 但是在通往你目标的道路上,你会更加的熟练运用各种做产品的工具。渐渐地,你会拥有越来越强大的设计、编码、销售等能力。
-### 2. 执行计划
+### 2. 保持进展
有一些产品没能完成的原因是作者在半路失去了动力。 **设定一些小目标(比如先赚一个亿);完成这些目标会让你获得前进的动力。** 小的成就累计起来也是很了不起的。
@@ -49,9 +49,9 @@
比如,我做了 [YC Careers](http://jonathanzwhite.github.io/yc-careers/) 和 [AtomSpace](https://atomspace.co/) 来解决一个产品设计师朋友找工作和面试的烦恼。一夜之间,YC Careers 占领了 [ProductHunt](https://www.producthunt.com/posts/yc-careers) 的榜首,AtomSpace 在上线6个小时之内就拿到了来自陌生人的一百美元订单。
-**你解决越大的痛点,就越容易找到用户。**
+**你解决的痛点越大,就越容易找到用户。**
-同时,尝试从过个角度分析问题和解决方案。有些成功的点子看起来并没有解决任何问题,但其实不然。比如,Instagram 看起来没有直接解决任何问题。其实不是。Instagram 满足了朋友之间实时社交的需求。如果这个需求不能得到满足,那才是个问题。
+同时,尝试从各个不同的角度寻找解决方案。有些成功的点子看起来并没有解决任何问题,但其实不然。比如,Instagram 看起来没有直接解决任何问题。其实不是。Instagram 满足了朋友之间实时社交的需求。如果这个需求不能得到满足,那才是个问题。
想要了解更多关于这方面内容的话,[John Carmack on Idea Generation](https://amasad.me/carmack) 和 [How to Get Startup Ideas](http://paulgraham.com/startupideas.html) 这两篇文章是非常好的起点。
@@ -67,11 +67,11 @@
### 5. 寻求帮助
-当你在你的项目上埋头苦干的时候,记得寻求别人的帮助。 **你会很惊喜的发现竟然有这么多人原意去帮助一个陌生人(对,就是你)。**
+当你在你的项目上埋头苦干的时候,记得寻求别人的帮助。 **你会很惊喜的发现竟然有这么多人愿意去帮助一个陌生人(对,就是你)。**
我获得的最好的一部分建议就是来自于给别人发邮件或是推特(微博)私信。
-在寻求帮助的时候,请提一些精确、有价值的问题。如果你需要关于设计的一些反馈,就把草稿发给别人;如果你需要营销方面的建议,就详细的列出你尝试过的方法。内容才是关键。如果别人没有回复你,请礼貌的再次询问 —— 有时候别人只是没注意到你的消息而已。
+在寻求帮助的时候,请准备好的你的问题,并尽可能具体。如果你需要关于设计的一些反馈,就把草稿发给别人;如果你需要营销方面的建议,就详细的列出你尝试过的方法。内容才是关键。如果别人没有回复你,请礼貌的再次询问 —— 有时候别人只是没注意到你的消息而已。
---
From b29686077845eda6b54570ae44c68eb629006026 Mon Sep 17 00:00:00 2001
From: xiaoyusilen
Date: Sun, 9 Apr 2017 13:03:50 +0800
Subject: [PATCH 154/945] Modify translation
---
TODO/anatomy-of-a-function-call-in-go.md | 22 +++++++++++-----------
TODO/go-function-calls-redux.md | 6 +++---
2 files changed, 14 insertions(+), 14 deletions(-)
diff --git a/TODO/anatomy-of-a-function-call-in-go.md b/TODO/anatomy-of-a-function-call-in-go.md
index f6141663c05..baca272576d 100644
--- a/TODO/anatomy-of-a-function-call-in-go.md
+++ b/TODO/anatomy-of-a-function-call-in-go.md
@@ -20,7 +20,7 @@ func add(a, b int) int {
}
```
-我们编译的时候需要关闭优化这样方便跟踪每个部分。我们用 `go build -gcflags 'N -l'` 这个命令来完成上述操作。然后我们可以用 `go tool objdump -s main.add func` 输出我们函数的具体细节(这里的 func 是我们的包名,也就是我们刚刚用 go build 编译出的可执行文件)。
+我们编译的时候需要关闭优化,这样方便理解每个部分。我们用 `go build -gcflags 'N -l'` 这个命令来完成上述操作。然后我们可以用 `go tool objdump -s main.add func` 输出我们函数的具体细节(这里的 func 是我们的包名,也就是我们刚刚用 go build 编译出的可执行文件)。
如果你以前从来没有看过汇编语言,那么恭喜啦,你今天看到的内容对你来说是全新的。我将在 Mac 上完成这些事情,所以配置是 Intel 64位。
@@ -36,18 +36,18 @@ func add(a, b int) int {
我们看到了什么?每一行如下所示被分为了4部分:
- 源文件的名称和行号(main.go:15)。这行的源代码会被转换为标有代码行号的说明。Go 的一行可能被转换成多行程序集。
-- 对象的偏移量(例如 0x22C0)。
+- 目标文件中的偏移量(例如 0x22C0)。
- 机器码(例如 48c744241800000000)。这是 CPU 实际执行的二进制机器码。我们不需要看这个,几乎没有人看这玩意。
- 机器码的汇编表示形式,这也是我们想要理解的部分。
让我们将注意力集中在最后一部分,汇编语言。
-- MOVQ,ADDQ 和 RET 是说明。它们告诉 CPU 需要执行的操作。后面的参数告诉 CPU 对什么执行该操作。
+- MOVQ,ADDQ 和 RET 是指令。它们告诉 CPU 需要执行的操作。后面的参数告诉 CPU 对什么执行该操作。
- SP,AX 和 CX 是 CPU 寄存器。寄存器是 CPU 用于存储值的地方,CPU 有多个寄存器可以使用。
- SP 是一个专用寄存器,用于存储当前堆栈指针。堆栈是记录局部变量,参数和函数调用的寄存器。每个 goroutine 都有一个堆栈。当一个函数调用另一个函数,然后另一个函数再调用其他函数,每个函数在堆栈上获得自己的存储区域。在函数调用期间创建存储区域,将 SP 的大小中减去所需的存储大小。
-- 0x8(SP)是指超过 SP 指向的内存位置的 8 个字节的内存位置。
+- 0x8(SP)是指超过 SP 指向的存储单元的 8 个字节的存储单元。
-因此,我们的工作的内容包含存储器位置,CPU 寄存器,用于在存储器和寄存器之间移动值的指令以及寄存器上的操作。 这几乎就是一个 CPU 所完成的事情了。
+因此,我们的工作的内容包含存储单元,CPU 寄存器,用于在存储器和寄存器之间移动值的指令以及寄存器上的操作。 这几乎就是一个 CPU 所完成的事情了。
现在让我们从第一条指令开始看每一条内容。别忘了我们需要从内存中加载两个参数 `a` 和 `b`,把它们相加,然后返回至调用函数。
@@ -67,9 +67,9 @@ func add(a, b int) int {
所以我们从中明白了什么:
- 好,看起来参数都存在堆栈中,第一个参数存储在 SP+0x8 中,另一个在更高编号的地址中。
-- 并且看上去返回的结果存储在参数后边,一个更高编号的已知地址中。
+- 并且看上去返回的结果存储在参数后边,一个更高编号的地址中。
-现在让我们看另一个函数。这个函数有一个全局变量,不过我们依然会让它看起来很简单。
+现在让我们看另一个函数。这个函数有一个局部变量,不过我们依然会让它看起来很简单。
```
func add3(a int) int {
@@ -108,7 +108,7 @@ func add3(a int) int {
这一行代码似乎没有做什么。所以这可能是一种声明函数的方法。让我们分析一下。
-- `SUBQ $0x10, SP` 从 SP 减去 0x10=16。这个操作为我们分类了 16 字节的堆栈空间。
+- `SUBQ $0x10, SP` 从 SP 减去 0x10=16。这个操作为我们释放了 16 字节的堆栈空间
- `MOVQ BP, 0x8(SP)` 将寄存器 BP 中的值存储至 SP+8 中,然后 `LEAQ 0x8(SP), BP` 将地址 SP+8 中的内容加载到 BP 中。现在我们已经有空间可以存储 BP 中之前所存的内容,然后将 BP 中的内容存储至刚刚分配的存储空间中,这有助于建立堆栈区域链(或者堆栈框架)。这有点神秘,不过在这篇文章中我们恐怕不会解决这个问题。
- 在这一部分的最后是 `MOVQ $ 0x0, 0x20 (SP)`,它和我们刚刚分析的最后一句类似,就是将返回值初始化为0。
@@ -126,7 +126,7 @@ func add3(a int) int {
- 调用函数在堆栈中为返回值和参数分配空间。返回值的存储地址比参数的存储地址高。
- 如果被调用函数有局部变量,则通过减少堆栈指针 SP 的值为它们分配空间。它也和寄存器 BP 做了一些神秘的事情。
-- 当函数返回任何 SP&BP 的操作都会相反。
+- 当函数返回任何对 SP 和 BP 的操作都会相反。
让我们看看堆栈在 add3() 方法中如何使用:
@@ -145,8 +145,8 @@ SP+0x08: the old value of BP
SP+0x0: the local variable b
```
-如果你觉得文章中没有提到 SP+0x10,所以不知道这是干什么用的。我可以告诉你,这是存储返回地址的地方。这是为了让 `RET` 指令知道返回到哪里去。
+如果你觉得文章中没有提到 SP+0x10,所以不*知道*这是干什么用的。我可以告诉你,这是存储返回地址的地方。这是为了让 `RET` 指令知道返回到哪里去。
-这篇文章已经足够了。 希望如果以前你不知道这些东西如何工作,但是现在你觉得你已经有了一些了解,或者如果你被汇编吓倒了,那么它可能不那么清晰。 如果你想了解有关汇编的更多信息,请在评论中告诉我,我会考虑在之后的文章中写出来。
+这篇文章已经足够了。 希望如果以前你不知道这些东西如何工作,但是现在你觉得你已经有了一些了解,或者如果你被汇编吓倒了,那么也许它不那么晦涩难懂了。 如果你想了解有关汇编的更多信息,请在评论中告诉我,我会考虑在之后的文章中写出来。
既然你已经看到这儿了,如果喜欢我的这篇文章或者可以从中学到一点什么的话,那么请给我点个赞这样这篇文章就可以被更多人看到了。
\ No newline at end of file
diff --git a/TODO/go-function-calls-redux.md b/TODO/go-function-calls-redux.md
index 2e8b6efa515..0fefbdda247 100644
--- a/TODO/go-function-calls-redux.md
+++ b/TODO/go-function-calls-redux.md
@@ -10,7 +10,7 @@
什么是调用堆栈?它是一个用于保存局部变量和调用参数的内存区域,并且跟踪每个函数应该返回到哪里去。每个 goroutine 都有它自己的堆栈。你甚至可以说每个 goroutine 就是它自己的堆栈。
-下面是我用于演示堆栈的代码。就是一系列简单的函数调用,main() 函数调用 [f1(0xdeadbeef)](https://en.wikipedia.org/wiki/Hexspeak),然后调用 `f2(0xabad1dea)`,再调用 `f3(0xbaddcafe)`。然后 f3() 将其中一个作为它的参数,并且将它存储在名为 `local` 的本地变量中。然后获取 `local` 的内存地址并且从那里开始输出。因为 `loacl` 在栈内,所以输出的就是栈。
+下面是我用于演示堆栈的代码。就是一系列简单的函数调用,main() 函数调用 [f1(0xdeadbeef)](https://en.wikipedia.org/wiki/Hexspeak),然后调用 `f2(0xabad1dea)`,再调用 `f3(0xbaddcafe)`。然后 `f3()` 将其中一个作为它的参数,并且将它存储在名为 `local` 的本地变量中。然后获取 `local` 的内存地址并且从那里开始输出。因为 `local` 在栈内,所以输出的就是栈。
```go
package main
@@ -59,7 +59,7 @@ func showFunc(at uintptr) {
}
```
-下面是上述代码的输出结果。它是从 `local` 的地址开始的内存转储,是以十六进制形式展示的8字节列表。左边是每个整数的存储地址,右边是地址内存储的整数。
+下面是上述代码的输出结果。它是从 `local` 的地址开始的内存转储,是以十六进制形式展示的 8 字节列表。左边是每个整数的存储地址,右边是地址内存储的整数。
我们知道 `local` 应该等于 0xBADDCAFE + 1,或者 0xBADDCAFF,这确实是我们转储开始时看到的。
@@ -118,7 +118,7 @@ C42003FFC0: C4200001A0
通过这些我们可以看出:
- 首先,堆栈从高地址开始,堆栈地址随着函数调用变小。
-- 当进行函数调用时,调用者将参数放入栈内,然后返回地址(调用函数中的下一条指令的地址),接着指向堆栈中较高的指针。
+- 当进行函数调用时,调用者将参数放入栈内,然后是返回地址(调用函数中的下一条指令的地址),接着是指向堆栈中较高的指针。
- 当调用返回时,这个指针用于在堆栈中查找先前调用的函数。
- 局部变量存储在堆栈指针之后。
From 2668d72d4ad6e182aebb2ba64c3d21c11f69a186 Mon Sep 17 00:00:00 2001
From: xiaoyusilen
Date: Sun, 9 Apr 2017 13:07:56 +0800
Subject: [PATCH 155/945] add msg
---
TODO/anatomy-of-a-function-call-in-go.md | 2 +-
TODO/go-function-calls-redux.md | 2 +-
2 files changed, 2 insertions(+), 2 deletions(-)
diff --git a/TODO/anatomy-of-a-function-call-in-go.md b/TODO/anatomy-of-a-function-call-in-go.md
index baca272576d..4fa70afdb03 100644
--- a/TODO/anatomy-of-a-function-call-in-go.md
+++ b/TODO/anatomy-of-a-function-call-in-go.md
@@ -2,7 +2,7 @@
> * 原文作者:[Phil Pearl](https://syslog.ravelin.com/@philpearl?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[xiaoyusilen](http://xiaoyu.world)
-> * 校对者:
+> * 校对者:[1992chenlu](https://github.com/1992chenlu)
# 解析 Go 中的函数调用 #
diff --git a/TODO/go-function-calls-redux.md b/TODO/go-function-calls-redux.md
index 0fefbdda247..b8717411ff1 100644
--- a/TODO/go-function-calls-redux.md
+++ b/TODO/go-function-calls-redux.md
@@ -2,7 +2,7 @@
> * 原文作者:[Phil Pearl](https://hackernoon.com/@philpearl?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[xiaoyusilen](http://xiaoyu.world)
-> * 校对者:
+> * 校对者:[1992chenlu](https://github.com/1992chenlu)
# Go 函数调用 Redux #
From 297bd3774a9a46e74b85a6806a36117486b9bf37 Mon Sep 17 00:00:00 2001
From: Tuccuay
Date: Sun, 9 Apr 2017 01:56:00 +0800
Subject: [PATCH 156/945] translated mastering swift essential details about
string
---
...g-swift-essential-details-about-strings.md | 548 +++++++++---------
1 file changed, 277 insertions(+), 271 deletions(-)
diff --git a/TODO/mastering-swift-essential-details-about-strings.md b/TODO/mastering-swift-essential-details-about-strings.md
index b223e0726ef..322df4619dc 100644
--- a/TODO/mastering-swift-essential-details-about-strings.md
+++ b/TODO/mastering-swift-essential-details-about-strings.md
@@ -1,270 +1,271 @@
> * 原文地址:[Mastering Swift: essential details about strings](https://rainsoft.io/mastering-swift-essential-details-about-strings/)
* 原文作者:[Dmitri Pavlutin](https://rainsoft.io/author/dmitri-pavlutin/)
* 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-* 译者:
+* 译者:[Tuccuay](https://www.tuccuay.com)
* 校对者:
-# Mastering Swift: essential details about strings #
+# 掌握 Swift 的字符串细节
-String type is an important component of any programming language. The most useful information that user reads from the window of an iOS application is pure text.
+String 类型在任何编程语言中都是一个重要的组成部分。而用户从 iOS 应用的屏幕上能读取到最有效的信息也来自文本。
-To reach a higher number of users, the iOS application must be internationalised and support a lot of modern languages. The Unicode standard solves this problem, but creates additional complexity when working with strings.
+为了触及更多的用户,iOS 应用必须国际化以支持大量现代语言。Unicode 标准解决了这个问题,不过这也给我们使用 string 类型带来了额外的挑战性。
-On one hand, the language should provide a good balance between the Unicode complexity and the performance when processing strings. On the other hand, it should provide developer with comfortable structures to handle strings.
+从一方面来说,编程语言应该在平衡处理字符串时应该在 Unicode 复杂性和性能之间取得平衡。而另一方面,它需要为开发者提供一个舒适的结构来处理字符串。
-In my opinion, Swift does a great job on both hands.
+而在我看来,Swift 在这两方面都做的不错。
-Fortunately Swift's string is not a simple sequence of UTF-16 code units, like in JavaScript or Java.
+幸运的是 Swift 的 string 类型并不是像 JavaScript 或者 Java 那样简单的 UTF-16 序列。
-In case of a sequence of UTF-16 code units it's a pain to do Unicode-aware string manipulations: you might break a surrogate pair or combining character sequence.
+对一个 UTF-16 码单元序列执行 Unicode 感知的字符串操作是很痛苦的:你可能会打破代理对或组合字符序列。
-Swift implements a better approach. The string itself is not a collection, instead it provides views over the string content that may be applied according to situation. And one particular view, `String.CharacterView`, is fully Unicode-aware.
+Swift 对此有着更好的实现方式。字符串本身不再是集合,而是能够根据不同情况为内容提供不同的 view。其中一个特殊的 view: `String.CharacterView` 则是完全支持 Unicode 的。
-For `let myStr = "Hello, world"` you can access the following string views:
+对于 `let myStr = "Hello, world"` 来说,你可以访问到下面这些 view:
-- `myStr.characters` is `String.CharacterView`. Valuable to access graphemes, that visually are rendered as a single symbol. The most used view.
-- `myStr.unicodeScalars` is `String.UnicodeScalarView`. Valuable to access the Unicode code point numbers as 21-bit integers
-- `myStr.utf16` is `String.UTF16View`. Useful to access the code unit values encoded in UTF16
-- `myStr.utf8` is `String.UTF8View`. Valuable to access the code unit values encoded in UTF8
+- `myStr.characters` 即 `String.CharacterView`。可以获取字形的值,视觉上呈现为单一的符号。是最常用的试图
+- `myStr.unicodeScalars` 即 `String.UnicodeScalarView`。可以获取 21 整数表示的 Unicode 码位。
+- `myStr.utf16` 即 `String.UTF16View`。用于获取 UTF16 编码的代码单元。
+- `myStr.utf8` 即 `String.UTF8View`。能够获取 UTF8 编码的代码单一。
-![CharacterView, UnicodeScalarView, UTF16View, UTF8View of strings in Swift](https://rainsoft.io/content/images/2016/10/Swift-strings--3-.png)
+![Swift 中的 CharacterView, UnicodeScalarView, UTF16View 和 UTF8View](https://rainsoft.io/content/images/2016/10/Swift-strings--3-.png)
-Most of the time developer deals with simple string characters, without diving into details like encoding or code units.
+在大多数时候开发者都在处理简单的字符串字符,而不是深入到编码或者码位这样的细节中。
-`CharacterView` works nice for most of the string related tasks: iteration over the characters, counting the number of characters, verify substring existence, access by index, different manipulations and so on.
+`CharacterView` 能很好的完成大多数任务:迭代字符串、字符计数、验证是否包含字符串、通过索引访问和比较操作等。
-Let's see in more details how these tasks are accomplished in Swift.
+让我们看看如何用 Swift 来完成这些任务。
-# 1. Character and CharacterView structures #
+# 1. Character 和 CharterView 的结构
-`String.CharacterView` structure is a view over string content that is a collection of `Character`.
+`String.CharacterView` 的结构是一个字符内容的视图,它是 `Character` 的集合。
-To access the view from a string, use `characters` string property:
+要从字符串访问视图,使用字符的 `characters` 属性:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57ff7e018ef62b25bcea2ab1)
-```
-let message ="Hello, world"
-let characters = message.characters
-print(type(of: characters))// => "CharacterView"
+```swift
+let message = "Hello, world"
+let characters = message.characters
+print(type(of: characters))// => "CharacterView"
```
-`message.characters` returns the `CharacterView` structure.
+`message.characters` 返回了 `CharacterView` 结构.
-The character view is a collection of `Character` structures. For example, let's access the first character in a string view:
+字符视图是 `Character` 结构的集合。例如我们可以这样来访问字符视图里的第一个字符:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57ff7e188ef62b25bcea2ab2)
-```
-let message = "Hello, world"
-let firstCharacter = message.characters.first!
-print(firstCharacter) // => "H"
+```swift
+let message = "Hello, world"
+let firstCharacter = message.characters.first!
+print(firstCharacter) // => "H"
print(type(of: firstCharacter)) // => "Character"
-let capitalHCharacter: Character = "H"
+let capitalHCharacter: Character = "H"
print(capitalHCharacter == firstCharacter) // => true
```
-`message.characters.first` returns an optional that is the first character `"H"`.
+`message.characters.first` 返回了一个可选类型,内容是它的第一个字符 `"H"`.
-The character instance represents a single symbol `H`.
+这个字符实例代表了单个符号 `H`。
-In Unicode terms `H` is *Latin Capital letter H*, `U+0048` code point.
+在 Unicode 标准中,`H` 代表 *Latin Capital letter H*,码位是 `U+0048`。
-Let's go beyond ASCII and see how Swift handles composite symbols. Such characters are rendered as a single visual symbol, but are composed from a sequence of two or more [Unicode scalars](http://unicode.org/glossary/#unicode_scalar_value). Strictly such characters are named **grapheme clusters**.
+让我们掠过 ASCII 看看 Swift 如何处理更复杂的符号。这些字符被渲染成单个视觉符号,但实际上是由两个或更多个 Unicode 标量](http://unicode.org/glossary/#unicode_scalar_value) 组成。严格来说这些字符被称为 **字形簇**
-*Important*: `CharacterView` is a collection of grapheme clusters of the string.
+*Important*: `CharacterView` is a collection of grapheme clusters of the string.
+**重点**: `CharacterView` 是字符串的字形簇集合。
-Let's take a closer look at `ç` grapheme. It may be represented in two ways:
+让我们看看 `ç` 的字形。他可以有两种表现形式:
-- Using `U+00E7` *LATIN SMALL LETTER C WITH CEDILLA*: rendered as `ç`
-- Or using a combining character sequence: `U+0063`*LATIN SMALL LETTER C* plus the combining mark `U+0327` *COMBINING CEDILLA*. The grapheme is composite: `c` + `◌̧` = `ç`
+- 使用 `U+00E7` *LATIN SMALL LETTER C WITH CEDILLA*:被渲染为 `ç`
+- 或者使用组合字符序列:`U+0063`*LATIN SMALL LETTER C* 加上 组合标记 `U + 0327` *COMBINING CEDILLA* 组成复合字形:`c` + `◌̧` = `ç`
-Let's pick the second option and see how Swift handles it:
+我们看看在第二个选项中 Swift 是如何处理它的:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f3466012fc531b0551b918)
-```
-let message = "c\u{0327}a va bien" // => "ça va bien"
-let firstCharacter = message.characters.first!
+```swift
+let message = "c\u{0327}a va bien" // => "ça va bien"
+let firstCharacter = message.characters.first!
print(firstCharacter) // => "ç"
-let combiningCharacter: Character = "c\u{0327}"
-print(combiningCharacter == firstCharacter) // => true
+let combiningCharacter: Character = "c\u{0327}"
+print(combiningCharacter == firstCharacter) // => true
```
-`firstCharacter` contains a single grapheme `ç` that is rendered using two Unicode scalars `U+0063` and `U+0327`.
+`firstCharacter` 包含了一个字形 `ç`,它是由两个 Unicode 标量 `U+0063` and `U+0327` 组合渲染出来的。
-`Character` structure accepts multiple Unicode scalars as long as they create a single grapheme. If you try to add more graphemes into a single `Character`, Swift triggers an error:
+`Character` 结构接受多个 Unicode 标量来创建一个单一的字形。如果你尝试在单个 `Character` 中添加更多的字形,Swift 将会出发错误:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f3510212fc531b0551b91c)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f3510212fc531b0551b91c)
-```
-let singleGrapheme: Character = "c\u{0327}\u{0301}" // Works
+```swift
+let singleGrapheme: Character = "c\u{0327}\u{0301}" // Works
print(singleGrapheme) // => "ḉ"
-let multipleGraphemes: Character = "ab" // Error!
+let multipleGraphemes: Character = "ab" // Error!
```
-Even if `singleGrapheme` is composed of 3 Unicode scalars, it creates a single grapheme `ḉ`.
-`multipleGraphemes` tries to create a `Character` from 2 Unicode scalars. This creates 2 separated graphemes `a` and `b` in a single `Character` structure, which is not allowed.
+即使 `singleGrapheme` 由 3 个 Unicode 标量组成,它创建了一个字形 `ḉ`。
+而 `multipleGraphemes` 则是从两个 Unicode 标量创建一个 `Character`,这将在单个 `Character` 结构中创建两个分离的字母 `a` 和 `b`,这不是被允许的操作。
-# 2. Iterating over characters in a string #
+# 2. 迭代字符串中的字符
-`CharacterView` collection conforms to `Sequence` protocol. This allows to iterate over the view characters in a `for-in` loop:
+`CharacterView` 集合遵循了 `Sequence` 协议。这将允许在 `for-in` 循环中便利字符视图:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bc8f27a61152fe7c7410)
-```
+```swift
let weather ="rain"for char in weather.characters {print(char)}// => "r" // => "a" // => "i" // => "n"
```
-Each character from `weather.characters` is accessed using `for-in` loop. On every iteration `char` variable is assigned with a character from `weather` string: `"r"`, `"a"`, `"i"` and `"n"`.
+我们可以在 `for-in` 循环中访问到 `weather.characters` 中的每个字符。`char` 变量将会在迭代中依次分配给 `weather` 中的 `"r"`, `"a"`, `"i"` 和 `"n"` 字符。
-As an alternative, you can iterate over the characters using `forEach(_:)` method, indicating a closure as the first argument:
+当然你也可以用 `forEach(_:)` 方法来迭代字符,指定一个闭包作为第一个参数:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bca927a61152fe7c7411)
-```
-let weather = "rain"
-for char in weather.characters {
+```swift
+let weather = "rain"
+for char in weather.characters {
print(char)
}
-// => "r"
-// => "a"
-// => "i"
+// => "r"
+// => "a"
+// => "i"
// => "n"
```
-The iteration using `forEach(_:)` method is almost the same as `for-in`, only that you cannot use `continue` or `break` statements.
+使用 `forEach(_:)` 的方式与 `for-in` 相似,唯一的不同是你不能使用 `continue` 或者 `break` 语句。
-To access the index of the current character in the loop, `CharacterView` provides the `enumerated()` method. The method returns a sequence of tuples `(index, character)`:
+要在循环中访问当前字符串的索引可以通过 `CharacterView` 提供的 `enumerated()` 方法。这个方法将会返回一个元组序列 `(index, character)`:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bcd127a61152fe7c7412)
-```
-let weather = "rain"
-for (index, char) in weather.characters.enumerated() {
+```swift
+let weather = "rain"
+for (index, char) in weather.characters.enumerated() {
print("index: \(index), char: \(char)")
}
-// => "index: 0, char: r"
-// => "index: 1, char: a"
-// => "index: 2, char: i"
-// => "index: 3, char: n"
+// => "index: 0, char: r"
+// => "index: 1, char: a"
+// => "index: 2, char: i"
+// => "index: 3, char: n"
```
-`enumerated()` method on each iteration returns tuples `(index, char)`.
-`index` variable contains the character index at the current loop step. Correspondingly `char` variable contains the character.
+`enumerated()` 方法在每次迭代时返回元组 `(index, char)`。
+`index` 变量即为循环中当前字符的索引,而 `char` 变量则是循环中当前的字符。
-# 3. Counting characters #
+# 3. 统计字符
-Simply use `count` property of the `CharacterView` to get the number of characters:
+只需要访问 `CharacterView` 的 `counter` 属性就可以获得字符串中字符的个数:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bcf327a61152fe7c7413)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bcf327a61152fe7c7413)
-```
-let weather ="sunny"print(weather.characters.count)// => 5
+```swift
+let weather ="sunny"print(weather.characters.count)// => 5
```
-`weather.characters.count` contains the number of characters in the string.
+`weather.characters.count` 是字符串中字符的个数。
-Each character in the view holds a grapheme. When an adjacent character (for example a [combining mark](http://unicode.org/glossary/#combining_character)) is appended to string, you may find that `count` property is not increased.
+视图中的每一个字符都拥有一个字形。当相邻字符(比如 [组合标记](http://unicode.org/glossary/#combining_character) )被添加到字符串时,你可能发现 `count` 属性没有没有变大。
-It happens because an adjacent character does not create a new grapheme in the string, instead it modifies an existing [base Unicode character](http://unicode.org/glossary/#base_character). Let's see an example:
+这是因为相邻字符并没有在字符串中创建一个新的字形,而是附加到了已经存在的 [基本 Unicode 字形](http://unicode.org/glossary/#base_character) 中。让我们看一个例子:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd0927a61152fe7c7414)
-```
-var drink = "cafe"
-print(drink.characters.count) // => 4
-drink += "\u{0301}"
-print(drink) // => "café"
-print(drink.characters.count) // => 4
+```swift
+var drink = "cafe"
+print(drink.characters.count) // => 4
+drink += "\u{0301}"
+print(drink) // => "café"
+print(drink.characters.count) // => 4
```
-Initially `drink` has 4 characters.
+一开始 `drink` 含有四个字符。
-When the combining mark `U+0301`*COMBINING ACUTE ACCENT* is appended to string, it modifies the previous base character `e` and creates a new grapheme `é`. The property `count` is not increased, because the number of graphemes is still the same.
+当组合标记 `U+0301`*COMBINING ACUTE ACCENT* 被添加到字符串中,它改变了上一个基本字符 `e` 并创建了新的字形 `é`。这时属性 `count` 并没有变大,因为字形数量仍然相同。
-# 4. Accessing character by index #
+# 4. 按索引访问字符
-Swift doesn't know about the characters count in the string view until it actually evaluates the graphemes in it. As result a subscript that allows to access the character by an integer index directly does not exist.
+因为 Swift 直到它实际评估字符视图中的字形之前都不知道字符串中的字符个数。结果就造成了无法通过通过下标的方式访问字符串索引。
-You can access the characters by a special type `String.Index`.
+你可以通过特殊的类型 `String.Index` 访问字符。
-If you need to access the first or last characters in the string, the character view structure has `first` and `last` properties:
+如果你需要访问字符串中的第一个或者最好一个字符,字符视图结构提供了 `first` 和 `last` 属性:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd2027a61152fe7c7415)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd2027a61152fe7c7415)
-```
-let season = "summer"
-print(season.characters.first!) // => "s"
-print(season.characters.last!) // => "r"
-let empty = ""
-print(empty.characters.first == nil) // => true
-print(empty.characters.last == nil) // => true
+```swift
+let season = "summer"
+print(season.characters.first!) // => "s"
+print(season.characters.last!) // => "r"
+let empty = ""
+print(empty.characters.first == nil) // => true
+print(empty.characters.last == nil) // => true
```
-Notice that `first` and `last` properties are optional type `Character?`.
+注意 `first` 和 `last` 属性将会返回可选类型 `Character?`。
-In the empty string `empty` these properties are `nil`.
+在空字符串 `empty` 这些属性将会是 `nil`。
![String indexes in Swift](https://rainsoft.io/content/images/2016/10/Swift-strings--2--1.png)
-To get a character at specific position, you have to use `String.Index` type (actually an alias of `String.CharacterView.Index`). String offers a subscript that accepts `String.Index` to access the character, as well as pre-defined indexes `myString.startIndex` and `myString.endIndex`.
+要获取特定位置的字符,你必须使用 `String.Index` 类型(实际上是 `String.CharacterView.Index`的别名)。字符提供了一个接受 `String.Index` 下标访问字符的方法,以及预定义的索引 `myString.startIndex` 和 `myString.endIndex`。
-Using string index type, let's access the first and last characters:
+让我们使用字符串索引来访问第一个和最后一个字符:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd3627a61152fe7c7416)
-```
-let color = "green"
-let startIndex = color.startIndex
-let beforeEndIndex = color.index(before: color.endIndex)
-print(color[startIndex]) // => "g"
-print(color[beforeEndIndex]) // => "n"
+```swift
+let color = "green"
+let startIndex = color.startIndex
+let beforeEndIndex = color.index(before: color.endIndex)
+print(color[startIndex]) // => "g"
+print(color[beforeEndIndex]) // => "n"
```
-`color.startIndex` is the first character index, so `color[startIndex]` evaluates to `g`.
-`color.endIndex` indicates the *past the end* position, or simply the position one greater than the last valid subscript argument. To access the last character, you must calculate the index right before string's end index: `color.index(before: color.endIndex)`.
+`color.startIndex` 是第一个字符的索引,所以 `color[startIndex]` 表示为 `g`。
+`color.endIndex` 表示**结束**位置,或者简单的说是比最后一个有效小标参数大的位置。要访问最后一个字符,你必须计算它的前一个索引:`color.index(before: color.endIndex)`
-To access characters at position by an offset, use the `offsetBy` argument of `index(theIndex, offsetBy: theOffset)` method:
+要通过偏移访问字符的位置, 在 `index(theIndex, offsetBy: theOffset)` 方法中使用 `offsetBy` 参数:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd4d27a61152fe7c7417)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd4d27a61152fe7c7417)
-```
-let color = "green"
-let secondCharIndex = color.index(color.startIndex, offsetBy: 1)
-let thirdCharIndex = color.index(color.startIndex, offsetBy: 2)
-print(color[secondCharIndex]) // => "r"
-print(color[thirdCharIndex]) // => "e"
+```swift
+let color = "green"
+let secondCharIndex = color.index(color.startIndex, offsetBy: 1)
+let thirdCharIndex = color.index(color.startIndex, offsetBy: 2)
+print(color[secondCharIndex]) // => "r"
+print(color[thirdCharIndex]) // => "e"
```
-Indicating the `offsetBy` argument, you can access the character at specific offset.
+指定 `offsetBy` 参数,你将可以放特定偏移量位置的字符。
-Of course `offsetBy` argument is jumping over string graphemes, i.e. the offset applies over `Character` instances of string's `CharacterView`.
+当然,`offsetBy` 参数是的步进是字符串的字形。即偏移量适用于 `ChacterView` 中的 `Chacter` 实例。
-If the index is out of range, Swift generates an error:
+如果索引超出范围,Swift 会触发错误。
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd7227a61152fe7c7418)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd7227a61152fe7c7418)
-```
+```swift
let color ="green"
-let oops = color.index(color.startIndex, offsetBy:100)// Error!
+let oops = color.index(color.startIndex, offsetBy:100) // Error!
```
-To prevent such situations, indicate an additional argument `limitedBy` to limit the offset: `index(theIndex, offsetBy: theOffset, limitedBy: theLimit)`. The function returns an optional, which is `nil` for out of bounds index:
+为了防止这种情况,可以指定一个 `limitedBy` 参数来限制最大偏移量:`index(theIndex, offsetBy: theOffset, limitedBy: theLimit)`。这个函数将会返回一个可选类型,当索引超出范围时将会返回 `nil`:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd8d27a61152fe7c7419)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd8d27a61152fe7c7419)
-```
-let color = "green"
-let oops = color.index(color.startIndex, offsetBy: 100,
+```swift
+let color = "green"
+let oops = color.index(color.startIndex, offsetBy: 100,
limitedBy: color.endIndex)
-if let charIndex = oops {
+if let charIndex = oops {
print("Correct index")
} else {
print("Incorrect index")
@@ -272,269 +273,274 @@ if let charIndex = oops {
// => "Incorrect index"
```
-`oops` is an optional `String.Index?`. The optional unwrap verifies whether the index didn't jump out of the string.
+`oops` 是一个可选类型 `String.Index?`。展开可选类型可以验证索引是否超出了字符串的范围。
-# 5. Checking substring existence #
+# 5. 检查子串是否存在
-The simplest way to verify the substring existence is to call `contains(_ other: String)` string method:
+验证子串是否存在的最简单方法是调用 `contains(_ other: String)` 方法:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bda427a61152fe7c741a)
-```
-importFoundationlet animal ="white rabbit"print(animal.contains("rabbit"))// => true print(animal.contains("cat"))// => false
+```swift
+import Foundation
+let animal = "white rabbit"
+print(animal.contains("rabbit")) // => true
+print(animal.contains("cat")) // => false
```
-`animal.contains("rabbit")` returns `true` because `animal` contains `"rabbit"` substring.
+`animal.contains("rabbit")` 将返回 `true` 因为 `animal` 包含了 `"rabbit"` 字符串。
-Correspondingly `animal.contains("cat")` evaluates to `false` for a non-existing substring.
+那么当子字串不存在的时候 `animal.contains("cat")` 的值将为 `false`。
-To verify whether the string has specific prefix or suffix, the methods `hasPrefix(_:)` and `hasSuffix(_:)` are available. Let's use them in an example:
+要验证字符串是否具有特定的前缀或后缀,可以使用 `hasPrefix(_:)` 和 `hasSuffix(_:)` 方法。我们来看一个例子:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bdb627a61152fe7c741b)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bdb627a61152fe7c741b)
-```
-importFoundationlet
-animal ="white rabbit"
-print(animal.hasPrefix("white"))// => true print(animal.hasSuffix("rabbit"))// => true
+```swift
+importFoundationlet
+animal = "white rabbit"
+print(animal.hasPrefix("white")) // => true
+print(animal.hasSuffix("rabbit")) // => true
```
-`"white"` is a prefix and `"rabbit"` is a suffix of `"white rabbit"`. So the corresponding method calls `animal.hasPrefix("white")` and `animal.hasSuffix("rabbit")` return `true`.
+`"white rabbit"` 以 `"white"` 开头并以 `"rabbit"` 结尾。所以我们调用 `animal.hasPrefix("white")` 和 `animal.hasSuffix("rabbit")` 方法都将返回 `true`。
-When you need to search for a particular character, it makes sense to query directly the character view. For example:
+当你想搜索字符串时,直接查询字符视图是就可以了。比如:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bdc827a61152fe7c741c)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bdc827a61152fe7c741c)
-```
-let animal = "white rabbit"
-let aChar: Character = "a"
-let bChar: Character = "b"
-print(animal.characters.contains(aChar)) // => true
-print(animal.characters.contains {
+```swift
+let animal = "white rabbit"
+let aChar: Character = "a"
+let bChar: Character = "b"
+print(animal.characters.contains(aChar)) // => true
+print(animal.characters.contains {
$0 == aChar || $0 == bChar
}) // => true
```
-`contains(_:)` verifies whether the character view has a particular character.
+`contains(_:)` 将验证字符视图是否包含指定视图。
-The second function form accepts a closure: `contains(where predicate: (Character) -> Bool)` and performs the same verification.
+而第二个函数 `contains(where predicate: (Character) -> Bool)` 则是接受一个闭包并执行验证。
-# 6. String manipulation #
+# 6. 字符串操作
-The string in Swift is a *value type*. Whether you pass a string as an argument on function call, assign it to a variable or constant - every time a *copy* of the original string is created.
+字符串在 Swift 中是 *value type*(值类型)。无论你是将它作为参数进行函数调用还是将它分配给一个变量或者常量——每次复制都将会创建一个全新的**拷贝**。
-A mutating method call changes the string in place.
+所有的可变方法都是在空间内将字符串改变。
-This chapter covers the common manipulations over strings.
+本节涵盖了对字符串的常见操作。
-#### Append to string a character or another string ####
+#### 附加字符串到另一个字符串
-The simplest way to append to string is `+=` operator. You can append an entire string to original one:
+附加字符串较为简便的方法是直接使用 `+=` 操作符。你可以直接将整个字符串附加到原始字符串:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bddf27a61152fe7c741d)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bddf27a61152fe7c741d)
-```
-var bird ="pigeon"
+```swift
+var bird ="pigeon"
bird +=" sparrow"
-print(bird)// => "pigeon sparrow"
+print(bird) // => "pigeon sparrow"
```
-String structure provides a mutating method `append()`. The method accepts a string, a character or even a sequence of characters, and appends it to the original string. For instance:
+字符串结构提供了一个可变方法 `append()`。该方法接受字符串、字符甚至字符序列,并将其附加到原始字符串。例如
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bdff27a61152fe7c741e)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bdff27a61152fe7c741e)
-```
-var bird = "pigeon"
-let sChar: Character = "s"
-bird.append(sChar)
-print(bird) // => "pigeons"
-bird.append(" and sparrows")
-print(bird) // => "pigeons and sparrows"
-bird.append(contentsOf: " fly".characters)
-print(bird) // => "pigeons and sparrows fly"
+```swift
+var bird = "pigeon"
+let sChar: Character = "s"
+bird.append(sChar)
+print(bird) // => "pigeons"
+bird.append(" and sparrows")
+print(bird) // => "pigeons and sparrows"
+bird.append(contentsOf: " fly".characters)
+print(bird) // => "pigeons and sparrows fly"
```
-#### Extract a substring from string ####
+#### 从字符串中截取字符串
-The method `substring()` allows to extract substrings:
+使用 `substring()` 方法可以截取字符串:
-- from a specific index up to the end of string
-- from the the start up to a specific index
-- or based on a range of indexes.
+- 从特定索引到字符串的末尾
+- 从开头到特定索引
+- 或者基于一个索引区间
-Let's see how it works:
+让我们来看看它是如何工作的
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4be1527a61152fe7c741f)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4be1527a61152fe7c741f)
-```
-
-let plant = "red flower"
-let strIndex = plant.index(plant.startIndex, offsetBy: 4)
-print(plant.substring(from: strIndex)) // => "flower"
+```swift
+let plant = "red flower"
+let strIndex = plant.index(plant.startIndex, offsetBy: 4)
+print(plant.substring(from: strIndex)) // => "flower"
print(plant.substring(to: strIndex)) // => "red "
-if let index = plant.characters.index(of: "f") {
+if let index = plant.characters.index(of: "f") {
let flowerRange = index.. "flower"
+ print(plant.substring(with: flowerRange)) // => "flower"
}
```
-The string subscript accepts a range or closed range of string indexes. This helps extracting substrings based on ranges of indexes:
+字符串下标接受一个区间或者封闭区间作为字符索引。这有助于根据范围截取子串:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4be3127a61152fe7c7420) (target=undefined)
-```
-let plant ="green tree"let excludeFirstRange =
+```swift
+let plant ="green tree"let excludeFirstRange =
plant.index(plant.startIndex, offsetBy:1).. "reen tree" let lastTwoRange =
- plant.index(plant.endIndex, offsetBy:-2).. "ee"
+print(plant[excludeFirstRange]) // => "reen tree"
+let lastTwoRange = plant.index(plant.endIndex, offsetBy:-2).. "ee"
```
-#### Insert into string ####
+#### 插入字符串
-The string type provides the mutating method `insert()`. The method allows to insert a character or a sequence of characters at specific index.
+字符串类型提供了可变方法 `insert()`。此方法可以在特定索引处插入一个字符或者一个字符序列。
-The new character or sequence is inserted before the element currently at the specified index.
+新的字符将被插入到指定索引的元素之前。
-See the following sample:
+来看一个例子:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4be4a27a61152fe7c7421)
-```
-var plant = "green tree"
-plant.insert("s", at: plant.endIndex)
-print(plant) // => "green trees"
-plant.insert(contentsOf: "nice ".characters, at: plant.startIndex)
-print(plant) // => "nice green trees"
+```swift
+var plant = "green tree"
+plant.insert("s", at: plant.endIndex)
+print(plant) // => "green trees"
+plant.insert(contentsOf: "nice ".characters, at: plant.startIndex)
+print(plant) // => "nice green trees"
```
-#### Remove from string ####
+#### 移除字符
-The mutating method `remove(at:)` removes the character at an index:
+可变方法 `remove(at:)` 可以删除指定索引处的字符:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4be6527a61152fe7c7422)
-```
-var weather = "sunny day"
-if let index = weather.characters.index(of: " ") {
+```swift
+var weather = "sunny day"
+if let index = weather.characters.index(of: " ") {
weather.remove(at: index)
print(weather) // => "sunnyday"
}
```
-You can remove characters in the string that are in a range of indexes using `removeSubrange(_:)`:
+你也可以使用 `removeSubrange(_:)` 来从字符串中移除一个索引区间内的全部字符:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4be7b27a61152fe7c7423)
-```
-var weather = "sunny day"
-let index = weather.index(weather.startIndex, offsetBy: 6)
-let range = index.. "sunny"
+```swift
+var weather = "sunny day"
+let index = weather.index(weather.startIndex, offsetBy: 6)
+let range = index.. "sunny"
```
-#### Replace in string ####
+#### 替换字符串
-The method `replaceSubrange(_:with:)` accepts a range of indexes that should be replaced with a particular string. The method is mutating the string.
+`replaceSubrange(_:with:)` 方法接受一个索引区间并可以将区间内的字符串替换为特定字符串。这是字符串的一个可变方法。
-Let's see a sample:
+一个简单的例子:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4be9327a61152fe7c7424)
-```
-var weather = "sunny day"
-if let index = weather.characters.index(of: " ") {
+```swift
+var weather = "sunny day"
+if let index = weather.characters.index(of: " ") {
let range = weather.startIndex.. "rainy day"
}
```
-#### The character view mutation alternative ####
+#### 另一些关于字符串的可变操作
-Many of string manipulations described above may be applied directly on string's character view.
+上面描述的许多字符串操作都是直接应用于字符串中的字符视图。
-It is a good alternative if you find more comfortable to work directly with a collection of characters.
+而更方便的直接使用一个字符序列可能是更好的选择。
-For example you can remove characters at specific index, or directly the first or last characters:
+比如你可以删除特定索引出的字符,或者直接删除第一个或者最好一个字符:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bea927a61152fe7c7425)
-```
-var fruit = "apple"
-fruit.characters.remove(at: fruit.startIndex)
-print(fruit) // => "pple"
-fruit.characters.removeFirst()
-print(fruit) // => "ple"
-fruit.characters.removeLast()
-print(fruit) // => "pl"
+```swift
+var fruit = "apple"
+fruit.characters.remove(at: fruit.startIndex)
+print(fruit) // => "pple"
+fruit.characters.removeFirst()
+print(fruit) // => "ple"
+fruit.characters.removeLast()
+print(fruit) // => "pl"
```
-To reverse a word use `reversed()` method of the character view:
+使用字符视图中的 `reversed()` 方法来翻转字符视图:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bebf27a61152fe7c7426)
-```
+
+```swift
var fruit ="peach"
var reversed =String(fruit.characters.reversed())
-print(reversed)// => "hcaep"
+print(reversed)// => "hcaep"
```
-You can easily filter the string:
+ 你可以很简单得过滤字符串:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4beea27a61152fe7c7427)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4beea27a61152fe7c7427)
-```
-let fruit = "or*an*ge"
-let filtered = fruit.characters.filter { char in
+```swift
+let fruit = "or*an*ge"
+let filtered = fruit.characters.filter { char in
return char != "*"
}
-print(String(filtered)) // => "orange"
+print(String(filtered)) // => "orange"
```
-Map the string content by applying a transformer closure:
+Map 可以接受一个闭包来对字符串进行变换:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4befd27a61152fe7c7428)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4befd27a61152fe7c7428)
-```
-let fruit = "or*an*ge"
-let mapped = fruit.characters.map { char -> Character in
+```swift
+let fruit = "or*an*ge"
+let mapped = fruit.characters.map { char -> Character in
if char == "*" {
return "+"
}
return char
}
-print(String(mapped)) // => "or+an+ge"
+print(String(mapped)) // => "or+an+ge"
```
-Or reduce the string content to an accumulator value:
+或者使用 reduce 来对字符串来进行一些累加操作:
-[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bf1d27a61152fe7c7429) (target=undefined)
+[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bf1d27a61152fe7c7429)
-```
-let fruit = "or*an*ge"
-let numberOfStars = fruit.characters.reduce(0) { countStars, char in
+```swift
+let fruit = "or*an*ge"
+let numberOfStars = fruit.characters.reduce(0) { countStars, char in
if (char == "*") {
return countStarts + 1
}
return countStars
-}
-print(numberOfStars) // => 2
+}
+print(numberOfStars) // => 2
```
-# 7. Final words #
+# 7. 说在最后
-At first sight, the idea of different types of views over string's content may seem overcomplicated.
+首先要说,大家对于字符串内容持有的不同观点看起来似乎过于复杂。
-In my opinion it is a great implementation. Strings can be viewed in different angles: as a collection of graphemes, UTF-8 or UTF-16 code units or simple Unicode scalars.
+而在我看来这是一个很好的实现。字符串可以从不同的角度来例假:昨晚字形集合、UTF-8 或 UTF-16 码位和简单是 Unicode 标量。
-Just pick the view depending on your task. In most of the cases it is `CharacterView`.
+根据你的任务来选择合适的视图。在大多数情况下,`CharacterView` 都很合适。
-The character view deals with graphemes that may be compound from one or more Unicode scalars. As result the string cannot be integer indexed (like arrays). Instead a special type of index is applicable: `String.Index`.
+因为字符视图中可能包含来自一个或多个 Unicode 标量组成的字形。因此字符串并不能像数组那样直接被整数索引。不过可以用特殊的 `String.Index` 来索引字符串。
-Special index type adds a bit of complexity when accessing individual characters or manipulating strings. I agree to pay this price, because having truly Unicode-aware operations on strings is awesome!
+虽然特殊的索引类型导致在访问单个字符串或者操作字符串时增加了一些难度。我接受这个成本,因为在字符串上进行诊治的 Unicode 感知操作真的很棒!
*Do you find string views comfortable to use? Write a comment bellow and let's discuss!*
+*您是否发现使用舒适的字符串视图? 写下评论,让我们讨论一下!
+**对于字符操作你有没有找到更舒适的方法?写下评论我们一起来讨论一些吧!**
-**P.S.** You might be interested to read my [detailed overview of array and dictionary literals in Swift](https://rainsoft.io/concise-initialization-of-collections-in-swift/).
-
+**P.S.** 不知道你有没有兴趣阅读我的另一篇文章:[detailed overview of array and dictionary literals in Swift](https://rainsoft.io/concise-initialization-of-collections-in-swift/)
\ No newline at end of file
From fff51d3ef8a59fb7d27b3141addd5df9f0801668 Mon Sep 17 00:00:00 2001
From: Tuccuay
Date: Sun, 9 Apr 2017 15:16:14 +0800
Subject: [PATCH 157/945] fix typo
---
TODO/mastering-swift-essential-details-about-strings.md | 4 +---
1 file changed, 1 insertion(+), 3 deletions(-)
diff --git a/TODO/mastering-swift-essential-details-about-strings.md b/TODO/mastering-swift-essential-details-about-strings.md
index 322df4619dc..325223b5681 100644
--- a/TODO/mastering-swift-essential-details-about-strings.md
+++ b/TODO/mastering-swift-essential-details-about-strings.md
@@ -537,10 +537,8 @@ print(numberOfStars) // => 2
因为字符视图中可能包含来自一个或多个 Unicode 标量组成的字形。因此字符串并不能像数组那样直接被整数索引。不过可以用特殊的 `String.Index` 来索引字符串。
-虽然特殊的索引类型导致在访问单个字符串或者操作字符串时增加了一些难度。我接受这个成本,因为在字符串上进行诊治的 Unicode 感知操作真的很棒!
+虽然特殊的索引类型导致在访问单个字符串或者操作字符串时增加了一些难度。我接受这个成本,因为在字符串上进行真正的 Unicode 感知操作真的很棒!
-*Do you find string views comfortable to use? Write a comment bellow and let's discuss!*
-*您是否发现使用舒适的字符串视图? 写下评论,让我们讨论一下!
**对于字符操作你有没有找到更舒适的方法?写下评论我们一起来讨论一些吧!**
**P.S.** 不知道你有没有兴趣阅读我的另一篇文章:[detailed overview of array and dictionary literals in Swift](https://rainsoft.io/concise-initialization-of-collections-in-swift/)
\ No newline at end of file
From fc3e9aa67f16aee91eade3e94fc777ce289cd439 Mon Sep 17 00:00:00 2001
From: lsvih
Date: Sun, 9 Apr 2017 18:04:08 +0800
Subject: [PATCH 158/945] Webpack and Rollup: the same but different
---
...bpack-and-rollup-the-same-but-different.md | 50 +++++++++----------
1 file changed, 25 insertions(+), 25 deletions(-)
diff --git a/TODO/webpack-and-rollup-the-same-but-different.md b/TODO/webpack-and-rollup-the-same-but-different.md
index 7bc3f470f12..4e286f35689 100644
--- a/TODO/webpack-and-rollup-the-same-but-different.md
+++ b/TODO/webpack-and-rollup-the-same-but-different.md
@@ -1,57 +1,57 @@
> * 原文地址:[Webpack and Rollup: the same but different](https://medium.com/webpack/webpack-and-rollup-the-same-but-different-a41ad427058c)
> * 原文作者:[Rich Harris](https://medium.com/@Rich_Harris?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[lsvih](https://github.com/lsvih)
> * 校对者:
-# Webpack and Rollup: the same but different #
+# 同中有异的 Webpack 与 Rollup #
![](https://cdn-images-1.medium.com/max/1000/1*rtjClMZ8sq3cLFT9Aq8Xyg.png)
-This week, Facebook merged a [monster pull request](https://github.com/facebook/react/pull/9327) into React that replaced its existing build process with one based on [Rollup](https://rollupjs.org/) , [prompting](https://twitter.com/stanlemon/status/849366789825994752) [several](https://twitter.com/MrMohtas/status/849362334988595201) [people](https://twitter.com/kyleholzinger/status/849683292760797184) to ask ‘why did you choose Rollup over webpack’?
+本周,Facebook 将一个[非常大的 pull request](https://github.com/facebook/react/pull/9327) 合并到了 React 主分支。这个 PR 将 React 以前使用的构建工具替换成了 [Rollup](https://rollupjs.org/)。这让许多人感到不解,纷纷在推特上提问:“为什么你们选择 Rollup 而不选择 Webpack 呢?”[1](https://twitter.com/stanlemon/status/849366789825994752)[2](https://twitter.com/MrMohtas/status/849362334988595201)[3](https://twitter.com/kyleholzinger/status/849683292760797184)
-Which is a completely reasonable question. [Webpack](https://webpack.js.org/) is one of the modern JavaScript community’s greatest success stories, with millions of downloads every month powering tens of thousands of websites and applications. It has a large ecosystem, dozens of contributors, and — unusually for a community open source project — [meaningful financial support](https://opencollective.com/webpack) .
+有人问这个问题是很正常的。[Webpack](https://webpack.js.org/) 是现在 JavaScript 社区中最伟大的成功传奇之一,它有着数百万/月的下载量,驱动了成千上万的网站与应用。它有着巨大的生态系统、众多的贡献者,并且它与一般的社区开源项目不同——它有着[意义非凡的经济支持](https://opencollective.com/webpack)。
-By comparison, Rollup is a minnow. But React isn’t alone — Vue, Ember, Preact, D3, Three.js, Moment, and dozens of other well-known libraries also use Rollup. So what’s going on? Why can’t we have just one JavaScript module bundler that everyone agrees on?
+相比之下,Rollup 是那么的微不足道。但是,除了 React 之外,Vue、Ember、Preact、D3、Three.js、Moment 等众多知名项目都使用了 Rollup。为什么会这样呢?为什么这些项目不使用大家一致认可的 JavaScript 模块打包工具呢?
-### A tale of two bundlers ###
+### 这两个打包工具的优缺点 ###
-webpack was started in 2012 by [Tobias Koppers](https://medium.com/@sokra) to solve a hard problem that existing tools didn’t address: building complex single-page applications (SPAs). Two features in particular changed everything:
+Webpack 由 [Tobias Koppers](https://medium.com/@sokra) 在 2012 年创建,用于解决当时的工具不能处理的问题:构建复杂的单页应用(SPA)。它的两个特点改变了一切:
-1. **Code-splitting** makes it possible to break your app apart into manageable chunks that can be loaded on-demand, meaning your users get an interactive site much faster than if they had to wait for the whole application to download and parse. You *can* do this manually, but, well… good luck.
-2. **Static assets** such as images and CSS can be imported into your app and treated as just another node in the dependency graph. No more carefully placing your files in the right folders and hacked-together scripts for adding hashes to file URLs — webpack can take care of it for you.
+1. **代码分割**可以将你的 app 分割成许多个容易管理的分块,这些分块能够在用户使用你的 app 时按需加载。这意味着你的网站可以比那些没有使用此技术的网站要快上很多。因为访问那些网站必须要等待整个应用都被下载并解析完成。当然,你**也可以**自己手动去进行代码分割,但是……总之,祝你好运。
+2. **静态资源**的导入:图片、CSS 等静态资源可以直接导入到你的 app 中,就和其它的模块、节点一样能够进行依赖管理。因此,我们再也不用小心翼翼地将各个静态文件放在特定的文件夹中,然后再去用脚本给文件 URL 加上哈希串了。Webpack 已经帮你完成了这一切。
-Rollup was created for a different reason: to build flat distributables of JavaScript libraries as efficiently as possible, taking advantage of the ingenious design of ES2015 modules. Other module bundlers — webpack included — work by wrapping each module in a function, putting them in a bundle with a browser-friendly implementation of `require`, and evaluating them one-by-one. That’s great if you need things like on-demand loading, but otherwise it’s a bit of a waste, and it [gets worse if you have lots of modules](https://nolanlawson.com/2016/08/15/the-cost-of-small-modules/).
+而 Rollup 的开发理念则不同:它利用 ES2015 模块的巧妙设计,尽可能高效地构建精简且易分发的 JavaScript 库。而其它的模块打包器(包括 Webpack在内)都是通过将模块分别封装进函数中,然将这些函数通过能在浏览器中实现的 `require` 方法打包,最后依次处理这些函数。在你需要实现按需加载的时候,这种做法非常的方便,但是这样做引入了很多无关代码,比较浪费资源。当[你有很多模块要打包的时候,这种情况会变得更糟糕](https://nolanlawson.com/2016/08/15/the-cost-of-small-modules/)。
-ES2015 modules enable a different approach, which Rollup uses. All your code is put in the same place and evaluates in one go, resulting in leaner, simpler code that starts up faster. You can [see it for yourself with the Rollup REPL](https://rollupjs.org/repl).
+ES2015 模块则启用了一种不同的实现方法,Rollup 用的也就是这种方法。所有代码都将被放置在同一个地方,并且会在一起进行处理。因此得到的最终代码相较而言会更加的精简,运行起来自然也就更快。你可以[点击这儿亲自试试 Rollup 交互式解释器(REPL)](https://rollupjs.org/repl)。
-But there’s a trade-off: code-splitting is a much hairier problem, and at the time of writing Rollup doesn’t support it. Similarly, Rollup doesn’t do hot module replacement (HMR). And perhaps the biggest pain point for people coming to Rollup — while it handles most CommonJS files (via a [plugin](https://github.com/rollup/rollup-plugin-commonjs) ), some things just don’t translate to ES2015, whereas webpack handles everything you throw at it with aplomb.
+但这儿也存在一些需要权衡的点:代码分割是一个很棘手的问题,而 Rollup 并不能做到这一点。同样的,Rollup 也不支持模块热替换(HMR)。而且对于打算使用 Rollup 的人来说,还有一个最大的痛点:它通过[插件](https://github.com/rollup/rollup-plugin-commonjs)处理大多数 CommonJS 文件的时候,一些代码将无法被翻译回 ES2015。而与之相反,你可以把这一切的事全部放心交给 Webpack 去处理。
-### So which should I use? ###
+### 那么我到底应该选用哪一个呢? ###
-By now, hopefully it’s clear why both tools coexist and support each other — they serve different purposes. The tl;dr is this:
+回答这个问题之前,我们已经了解了这两个工具各自的优缺点,它们同时存在并相互支持的原因正是因为它们解决的问题不同。那么,现在这个问题的答案简单来说就是:
-> Use webpack for apps, and Rollup for libraries
+> 在开发应用时使用 Webpack,开发库时使用 Rollup
-That’s not a hard and fast rule — lots of sites and apps are built with Rollup, and lots of libraries are built with webpack. But it’s a good rule of thumb.
+当然这不是什么严格的规定——有很多的网站和 app 一样是使用 Rollup 构建的,同时也有很多的库使用 Webpack。不过,这是个很值得参考的经验之谈。
-If you need code-splitting, or you have lots of static assets, or you’re building something with lots of CommonJS dependencies, Webpack is a better choice. If your codebase is ES2015 modules and you’re making something to be used by other people, you probably want Rollup.
+如果你需要进行代码分割,或者你有很多的静态资源,再或者你做的东西深度依赖 CommonJS,毫无疑问 Webpack 是你的最佳选择。如果你的代码基于 ES2015 模块编写,并且你做的东西是准备给他人使用的,你应该更适合用 Rollup。
-### Package authors: use `pkg.module!` ###
+### 对于包作者的建议:请使用 `pkg.module`! ###
-For a long time, using JavaScript libraries was a bit of a crapshoot, because you and the library author effectively had to agree on a module system. If you were using Browserify but she preferred AMD, you would have to duct tape things together before you could actually build anything. The [Universal Module Definition](https://github.com/umdjs/umd) (UMD) format *sort of* fixed that, but because it wasn’t enforced anywhere you never knew quite what you were going to get.
+在很长一段时间里,使用 JavaScript 库是一件相当有风险的事,因为这意味着你必须和库的作者在模块系统上的意见保持一致。如果你使用 Browserify 而他更喜欢 AMD,你就不得不在 build 之前先强行将两者粘起来。[通用模块定义(UMD)](https://github.com/umdjs/umd)格式对这个问题进行了 *部分* 的修复,但是它没有强制要求所有的人都用它,最后反而会让你更加不知所措。
-ES2015 changes all that, because `import` and `export` are part of the language. In the future, there’ll be no ambiguity, and things will work a lot more seamlessly. Unfortunately, because browsers (mostly) and Node don’t yet support `import` and `export`, we still need to ship UMD files (or CommonJS, if you’re building something Node-only).
+ES2015 改变了这一切,因为 `import` 与 `export` 就是语言规范本身的一部分。在未来,不再会有现在这种模棱两可的情况,所有东西都将更加无缝地配合工作。不幸的是,由于大多数浏览器和 Node 还不支持 `import` 和 `export`,我们仍然需要依靠 UMD 规范(如果你只写 Node 的话也可以用 CommonJS)。
-By adding a `"module": "dist/my-library.es.js"` entry to your library’s package.json file (aka `pkg.module`), it’s possible to serve UMD and ES2015 at the same time, right now. **That’s important because Webpack and Rollup can both use** `pkg.module` **to generate the most efficient code possible** — in some cases, they can even both [tree-shake](https://webpack.js.org/guides/tree-shaking/) unused parts of your library away.
+现在给你的库的 package.json 文件增加一个 `"module": "dist/my-library.es.js"` 入口,可以让你的库同时支持 UMD 与 ES2015。**这很重要,因为 Webpack 和 Rollup 都使用了 ** `pkg.module` ** 来尽可能的生成效率更高的代码**——在一些情况下,它们都能使用 [tree-shake](https://webpack.js.org/guides/tree-shaking/) 来精简掉你的库中未使用的部分。
-*Learn more about`pkg.module` on the [*Rollup wiki*](https://github.com/rollup/rollup/wiki/pkg.module) .*
+*了解更多有关 `pkg.module` 的内容请访问 [Rollup wiki](https://github.com/rollup/rollup/wiki/pkg.module) 。*
-Hopefully this article makes the relationship between the two projects a bit clearer. If you still have questions, find us on Twitter at [rich_harris](https://twitter.com/rich_harris)/[rollupjs](https://twitter.com/rollupjs) and [thelarkinn](https://twitter.com/thelarkinn) . Happy bundling!
+希望这篇文章能让你理清这两个开源项目之间的关系。如果你还有问题,可以在推特联系[rich_harris](https://twitter.com/rich_harris)、[rollupjs](https://twitter.com/rollupjs)、[thelarkinn](https://twitter.com/thelarkinn)。祝你打包快乐!
-Our thanks to Rich Harris for writing this article. We believe that collaboration in open source is incredibly vital to ensure we push technology and the web forward together.
+感谢 Rich Harris 写了这篇文章。我们坚信开源协作是共同促进 web 技术前进的重要动力。
-No time to help contribute? Want to give back in other ways? Become a Backer or Sponsor to webpack by [donating to our open collective](https://opencollective.com/webpack). Open Collective not only helps support the Core Team, but also supports contributors who have spent significant time improving our organization on their free time! ❤
+没有时间为开源项目做贡献?想要以其它方式回馈吗?欢迎通过 [Open Collective 进行捐赠](https://opencollective.com/webpack),成为 Webpack 的支持者或赞助商。Open Collective 不仅会资助核心团队,而且还会资助那些贡献出空闲时间帮助我们改进项目的贡献者们。
---
From 117db55d80428317dd5efa2f50cb612546f5a361 Mon Sep 17 00:00:00 2001
From: gy134340
Date: Sun, 9 Apr 2017 18:05:38 +0800
Subject: [PATCH 159/945] =?UTF-8?q?=E4=B8=BA=E4=BB=80=E4=B9=88=E7=94=A8=20?=
=?UTF-8?q?JavaScript=20=E5=AD=A6=E4=B9=A0=E5=87=BD=E6=95=B0=E5=BC=8F?=
=?UTF-8?q?=E7=BC=96=E7=A8=8B=EF=BC=9F=EF=BC=88=E7=BB=84=E6=88=90=E5=8C=96?=
=?UTF-8?q?=E8=BD=AF=E4=BB=B6=EF=BC=89=EF=BC=88=E7=AC=AC=E4=BA=8C=E9=83=A8?=
=?UTF-8?q?=E5=88=86=EF=BC=89?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...amming-in-javascript-composing-software.md | 157 +++++++++---------
1 file changed, 78 insertions(+), 79 deletions(-)
diff --git a/TODO/why-learn-functional-programming-in-javascript-composing-software.md b/TODO/why-learn-functional-programming-in-javascript-composing-software.md
index dd82cf42a4a..3203cee2ce5 100644
--- a/TODO/why-learn-functional-programming-in-javascript-composing-software.md
+++ b/TODO/why-learn-functional-programming-in-javascript-composing-software.md
@@ -4,152 +4,151 @@
> * 译者:
> * 校对者:
-# Why Learn Functional Programming in JavaScript? (Composing Software)
+# 为什么用 JavaScript 学习函数式编程?(组成化软件)(第二部分)
-Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0)
-> Note: This is part of the “Composing Software” series on learning functional programming and compositional software techniques in JavaScript ES6+ from the ground up. Stay tuned. There’s a lot more of this to come!
-> [Start over at Part 1](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [Next >](https://medium.com/javascript-scene/a-functional-programmers-introduction-to-javascript-composing-software-d670d14ede30#.2e4youss2)
+烟雾的方块艺术 —MattysFlicks —(CC BY 2.0)
+> 注意:这是从基础学习函数式编程和使用 JavaScript ES6+ 组成软件的第一部分。保持关注,接下来还有很多!
+> [从第一部分开始](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [接下来的 >](https://medium.com/javascript-scene/a-functional-programmers-introduction-to-javascript-composing-software-d670d14ede30#.2e4youss2)
-Forget whatever you think you know about JavaScript, and approach this material with a beginner’s mind. To help you do that, we’re going to review the JavaScript basics from the ground up, as if you’ve never seen JavaScript before. If you’re a beginner, you’re in luck. Finally something exploring ES6 and functional programming from scratch! Hopefully all the new concepts are explained along the way — but don’t count on too much pampering.
+忘了你认为知道的关于 JavaScript 的一切,用初学者的眼光去看待它。为了帮助你做到这一点,我们将会从头复习一下 JavaScript 的基础,就像你与其尚未谋面一样。如果你是初学者,那你就很幸运了。最终从零开始探索 ES6 和函数式编程!希望所有的概念都被解释清楚 — 但不要太依赖于此。
-If you’re a seasoned developer already familiar with JavaScript, or a pure functional language, maybe you’re thinking that JavaScript is a funny choice for an exploration of functional programming. Set those thoughts aside, and try to approach the material with an open mind. You may find that there is another level to JavaScript programming. One you never knew existed.
+如果你是已经熟悉 JavaScript 或者纯函数式语言的老开发者了,也许你会认为 JavaScript 是探索函数式编程有趣的选择。把这些想法放在一边,用更开放的思想接触它,你会发现 JavaScript 编程更高层次的东西。一些你从来不知道的东西。
-Since this text is called “Composing Software”, and functional programming is the obvious way to compose software (using function composition, higher order functions, etc…), you may be wondering why I’m not talking about Haskell, ClojureScript, or Elm, instead of JavaScript.
+由于这个被称为“组成式软件”,同时函数式编程是明显的组成软件的方法(使用函数组合,高阶函数等等),你也许想知道为什么我不用 Haskell, ClojureScript,或者 Elm,而是 JavaScript。
-JavaScript has the most important features needed for functional programming:
+JavaScript 有函数式编程所需要的最重要的特性:
-1. **First class functions:** The ability to use functions as data values: pass functions as arguments, return functions, and assign functions to variables and object properties. This property allows for higher order functions, which enable partial application, currying, and composition.
-2. **Anonymous functions and concise lambda syntax:**`x => x * 2` is a valid function expression in JavaScript. Concise lambdas make it easier to work with higher-order functions.
-3. **Closures:** A closure is the bundling of a function with its lexical environment. Closures are created at function creation time. When a function is defined inside another function, it has access to the variable bindings in the outer function, even after the outer function exits. Closures are how partial applications get their fixed arguments. A fixed argument is an argument bound in the closure scope of a returned function. In `add2(1)(2)`, `1` is a fixed argument in the function returned by `add2(1)`.
+1. **一级函数:**使用函数作为数据值的能力:用函数传参,返回函数,用函数做变量和对象属性。这个属性允许更高级别的函数,使局部应用、柯里化和组合成为可能。
+2. **匿名函数和简介的 lambda 语法:**`x => x * 2` 是 JavaScript 中有效的函数表达式。简介的 lambda 语法使得它更好的跟高阶函数合作。
+3. **闭包:**闭包是一个有着自己独立作用域的捆绑函数。闭包在函数被创建时被创建。当一个函数在另一个函数内部被创建,它可以访问外部函数的变量,即使在外部函数退出后。闭包时使局部应用或者固定的参数。固定的参数时绑定在返回函数的作用域范围内的参数。在 `add2(1)(2)` 中,`1` 是 `add2(1)` 返回的函数中的固定参数。
-### What JavaScript is Missing
+### JavaScript 缺少了什么
-JavaScript is a multi-paradigm language, meaning that it supports programming in many different styles. Other styles supported by JavaScript include procedural (imperative) programming (like C), where functions represent a subroutine of instructions that can be called repeatedly for reuse and organization, object-oriented programming, where objects — not functions — are the primary building blocks, and of course, functional programming. The disadvantage of a multi-paradigm language is that imperative and object-oriented programming tend to imply that almost everything needs to be mutable.
+JavaScript 是多范式语言,意味着它支持多种风格的编程。其他被 JavaScript 支持的风格包括过程式的(命令式)编程(比如 C),把函数看作可以被重复调用和组织的子程序指令;面向对象编程,对象 — 而不是函数 — 作为初始构造块;当然,还有函数式编程。多范式编程语言的劣性在于命令式和面向对象往往意味着所有东西都是可变的。
-Mutation is a change to data structure that happens in-place. For example:
+可变性指的是数据结构上的变化。比如:
+
+ const foo = {
+ bar: 'baz'
+ };
- const foo = {
- bar: 'baz'
- };
+ foo.bar = 'qux'; // mutation
- foo.bar = 'qux'; // mutation
+对象通常需要可变性以便于被方法更新值,在命令式的语言中,大部分的数据结构可变以便于数组和对象的高效操作。
-Objects usually need to be mutable so that their properties can be updated by methods. In imperative programming, most data structures are mutable to enable efficient in-place manipulation of objects and arrays.
+下面是一些函数式语言拥有但是 JavaScript 没有的特性:
-Here are some features that some functional languages have, that JavaScript does not have:
+1. **纯粹性:**在一些函数式语言中,纯粹性是强制的,有副作用的表达式是不被允许的。
+2. **不可变性:**一些函数式语言不允许转变,采用表达式来产生新的数据结构来代替更改一个已存的数据结构,比如说数组或者对象。这样看起来可能不够搞笑,但是大多数函数式语言在引擎下使用 trie 数据结构,具有结构共享的特点:意味着旧的对象和新的对象是对相同数据的引用。
+3. **递归:**递归是函数引用自身来进行迭代的能力。在大多数函数式语言中,递归是迭代的唯一方式,它们没有像 `for` 、`while`、`do` 这类循环语句。
-1. **Purity:** In some FP languages, purity is enforced by the language. Expressions with side-effects are not allowed.
-2. **Immutability:** Some FP languages disable mutations. Instead of mutating an existing data structure, such as an array or object, expressions evaluate to new data structures. This may sound inefficient, but most functional languages use trie data structures under the hood, which feature structural sharing: meaning that the old object and new object share references to the data that is the same.
-3. **Recursion:** Recursion is the ability for a function to reference itself for the purpose of iteration. In many FP languages, recursion is the only way to iterate. There are no loop statements like `for`, `while`, or `do` loops.
+**纯粹性:**在 JavaScript 中,纯粹性由约定来达成,如果你不是使用纯函数来构成你的大多数应用,那么你就不是在进行函数式风格的编程。很不幸,在 JavaScript 中,你很容易就会不小心创建和使用一些不纯的函数。
-**Purity:** In JavaScript, purity must be achieved by convention. If you’re not building most of your application by composing pure functions, you’re not programming using the functional style. It’s unfortunately easy in JavaScript to get off track by accidentally creating and using impure functions.
+**不可变性:**在纯函数式语言中,可变性通常是强制的,JavaScript 缺少函数式语言中高效的、基于 trie 树的数据结构,但是又一些你可以使用的库,包括 [Immutable.js](https://facebook.github.io/immutable-js/) 和 [Mori](https://github.com/swannodette/mori),真是期望未来的 ECMAScript 规范版本可以拥抱不可变数据结构。
-**Immutability:** In pure functional languages, immutability is often enforced. JavaScript lacks efficient, immutable trie-based data structures used by most functional languages, but there are libraries that help, including [Immutable.js](https://facebook.github.io/immutable-js/) and [Mori](https://github.com/swannodette/mori). I’m hoping that future versions of the ECMAScript spec will embrace immutable data structures.
+有一些未来的迹象,比如说在 ES6 中添加了 `const` 关键字,`const` 声明的变量不能被重新赋值,知道 `const` 并不实际代表不可改变的值也很重要。
-There are signs that offer hope, like the addition of the `const` keyword in ES6. A name binding defined with `const` can't be reassigned to refer to a different value. It's important to understand that `const` does not represent an immutable *value.*
+`const` 声明的对象不能被重新声明为新的对象,但是对象的属性却是可变的,JavaScript 有 `freeze()` 对象的能力,但是这些对象只能在根实例上被冻结,意味着嵌套着的对象还是可以改变它的属性。换句话说,在 JavaScript 规范中看到真正的不可变还有很多路要走。
-A `const` object can't be reassigned to refer to a completely different object, but the object it refers to *can have its properties mutated*. JavaScript also has the ability to `freeze()` objects, but those objects are only frozen at the root level, meaning that a nested object can still have properties of its properties mutated. In other words, there's still a long road ahead before we see true composite immutables in the JavaScript specification.
+**递归:**JavaScript 支持递归,但是大多数函数式语言都有尾部调用优化的特性,尾部调用优化是一个允许递归的函数重用堆栈帧来递归调用的特性。
-**Recursion:** JavaScript technically supports recursion, but most functional languages have a feature called tail call optimization. Tail call optimization is a feature which allows recursive functions to reuse stack frames for recursive calls.
+没有尾部调用优化,一个调用的栈很可能没有边界导致堆栈溢出。JavaScript 在 ES6 规范中有一个有限的尾部调用优化。不幸的是,只有一个主要的浏览器引擎支持它,这个优化被部分应用随后从 Babel(最流行的 JavaScript 编译器,在旧的浏览器中被用来把 ES6 编译到 ES5)。
-Without tail call optimization, a call stack can grow without bounds and cause a stack overflow. JavaScript technically got a limited form of tail call optimization in the ES6 specification. Unfortunately, only one of the major browser engines implemented it, and the optimization was partially implemented and then subsequently removed from Babel (the most popular standard JavaScript compiler, used to compile ES6 to ES5 for use in older browsers).
+最后一行:现在使用递归来作为大的迭代还不是很安全 — 即使你很小心的调用尾部的函数。
-Bottom line: It still isn’t safe to use recursion for large iterations — even if you’re careful to call the function in the tail position.
+### 什么又是 JavaScript 所有但是纯函数式语言没有的
-### What JavaScript Has that Pure Functional Languages Lack
+一个纯粹主义者会告诉你 JavaScript 的可变性是它的重要缺点,这是事实。但是,引起的副作用和改变有时候很有用。事实上,不可能在规避所有副作用的情况下开发有用的现代应用。纯函数式语言比如说 Haskell 使用副作用,但是使用 monads 包来伪装纯函数,从而使程序保持纯净,尽管用 monads 所带来的副作用是不纯净的。
-A purist will tell you that JavaScript’s mutability is its major disadvantage, which is true. However, side effects and mutation are sometimes beneficial. In fact, it’s impossible to create most useful modern applications without side effects. Pure functional languages like Haskell use side-effects, but camouflage them from pure functions using boxes called monads, allowing the program to remain pure even though the side effects represented by the monads are impure.
+monads 的问题是,尽管它的使用很简单,但是对一个不是很熟悉它的人解释清楚它有点像对牛谈琴。
-The trouble with monads is that, even though their use is quite simple, explaining what a monad is to somebody unfamiliar with lots of examples is a bit like explaining what the color “blue” looks like to a blind person.
+> “monad 是 endofunctor 范畴的幺半群,有什么问题?” ~James Iry 所引用 Philip Wadler的话,解释一个 Saunders Mac Lane 说过的名言。[*“A Brief, Incomplete, and Mostly Wrong History of Programming Languages”*](http://james-iry.blogspot.com/2009/05/brief-incomplete-and-mostly-wrong.html)
-> “A monad is a monoid in the category of endofunctors, what’s the problem?” ~ James Iry, fictionally quoting Philip Wadler, paraphrasing a real quote by Saunders Mac Lane.[*“A Brief, Incomplete, and Mostly Wrong History of Programming Languages”*](http://james-iry.blogspot.com/2009/05/brief-incomplete-and-mostly-wrong.html)
+典型的,这是在调侃这有趣的一点。在上面的引用中,关于 monads 的解释相比最初的有了很大的简化,原来是下面这样:
-Typically, parody exaggerates things to make a funny point funnier. In the quote above, the explanation of monads is actually *simplified* from the original quote, which goes like this:
+> “`X` 中的 monad 是其 endofunctor 范畴的幺半群,生成 endofunctor 和被 endofunctor 单位 set 组合所代替的 `X` ” ~ Saunders Mac Lane。 [*"Categories for the Working Mathematician"*](https://www.amazon.com/Categories-Working-Mathematician-Graduate-Mathematics/dp/0387984038//ref=as_li_ss_tl?ie=UTF8&linkCode=ll1&tag=eejs-20&linkId=de6f23899da4b5892f562413173be4f0)
-> “A monad in `X` is just a monoid in the category of endofunctors of `X`, with product `×` replaced by composition of endofunctors and unit set by the identity endofunctor." ~ Saunders Mac Lane. [*"Categories for the Working Mathematician"*](https://www.amazon.com/Categories-Working-Mathematician-Graduate-Mathematics/dp/0387984038//ref=as_li_ss_tl?ie=UTF8&linkCode=ll1&tag=eejs-20&linkId=de6f23899da4b5892f562413173be4f0)
+尽管这样,在我的观点看来,害怕 monads 是没有必要的,学习 monads 最好的方法不是去读关于它的一堆书和博客,而是立刻去使用它。对于大部分的函数式编程语言来说,晦涩的学术词汇比它实际概念难的多,相信我,你不必通过了解 Saunders Mac Lane 来了解函数式编程。
-Even so, in my opinion, fear of monads is weak reasoning. The best way to learn monads is not to read a bunch of books and blog posts on the subject, but to jump in and start using them. As with most things in functional programming, the impenetrable academic vocabulary is much harder to understand than the concepts. Trust me, you don’t have to understand Saunders Mac Lane to understand functional programming.
+尽管它不是对所有的编程风格都完全的理想,JavaScript 无疑是作为适应各种编程风格和背景的人的通用编程语言被设计出来的。
-While it may not be absolutely ideal for every programmming style, JavaScript is unapologetically a general-purpose language designed to be usable by various people with various programming styles and backgrounds.
+根据 [Brendan Eric](https://brendaneich.com/2008/04/popularity/) 所言,在一开始的时候,网景公司就有意适应两类开发者:
-[According to Brendan Eich](https://brendaneich.com/2008/04/popularity/), this was intentional from the beginning. Netscape had to support two kinds of programmers:
+> “...写组件的,比如说 C++ 或者 Java;写脚本的、业余的和爱好者,比如直接写嵌在 HTML 里的代码的。”
-> *“…the component authors, who wrote in C++ or (we hoped) Java; and the ‘scripters’, amateur or pro, who would write code directly embedded in HTML.”*
+最初,网景公司的意向是支持两种不同的语言,同时脚本语言大致要像 Scheme (一个 Lisp 的方言),而且,Brendan Eich:
-Originally, the intent was that Netscape would support two different languages, and the scripting language would probably resemble Scheme (a dialect of Lisp). Again, Brendan Eich:
+> “我被招聘到网景公司,目的是在浏览器中 **做一些 Scheme**”。
-> *“I was recruited to Netscape with the promise of ‘doing Scheme’ in the browser.”*
+JavaScript 应当是一门新的语言:
-JavaScript had to be a new language:
+> “上面工程管理的命令是这门语言**应当像 Java**,这就排除了 Perl,Python,和 Tcl,以及 Scheme。”
-> *“The *diktat* from upper engineering management was that the language must ‘look like Java’. That ruled out Perl, Python, and Tcl, along with Scheme.”*
+所以,Brendan Eich 最初脑子里的想法是:
-So, the ideas in Brendan Eich’s head from the beginning were:
+1. 浏览器中的 Scheme。
+2. 看起来像 Java。
-1. Scheme in the browser.
-2. Look like Java.
+最终的结果稍微有点复杂:
-It ended up being even more of a mish-mash:
+>“我不骄傲,但我很高兴我选择了 Scheme 的一类函数和 Self(尽管奇怪)的原型作为主要的元素。”由于 Java 的影响,特别是 y2k 的 Date 问题以及对象的区别(比如string 和 String),就不幸了。”
-> *“I’m not proud, but I’m happy that I chose Scheme-ish first-class functions and Self-ish (albeit singular) prototypes as the main ingredients. The Java influences, especially y2k Date bugs but also the primitive vs. object distinction (e.g., string vs. String), were unfortunate.”*
+我列出了这些 “不好的” 的类 Java 特性,最后转换成 JavaScript:
-I’d add to the list of “unfortunate” Java-like features that eventually made their way into JavaScript:
+* 构造函数和 `new` 关键子,跟工厂函数有着不同的调用和使用语义。
+* `class` 的关键字和单一父类 `extends` 作为最初的继承机制。
+* 用户更习惯于把 `class` 看作是它的静态类型(实际并非如此)。
-- Constructor functions and the `new` keyword, with different calling and usage semantics from factory functions.
-- A `class` keyword with single-ancestor `extends` as the primary inheritance mechanism.
-- The user’s tendency to think of a `class` as if it's a static type (it's not).
+我的意见:永远避免使用这些东西。
-My advice: Avoid those whenever you can.
+很幸运 JavaScript 成为了这样厉害的语言,因为事实上证明脚本的方式赢了那些建立在“组件”上的方式(现在,Java、Flash、和 ActiveX 扩展已经不被大部分安装的浏览器支持)。
-We’re lucky that JavaScript ended up being such a capable language, because it turns out that the scripting approach won over the “component” approach (today, Java, Flash, and ActiveX extensions are unsupported in huge numbers of installed browsers).
+我们最终创作了一个直接被浏览器支持的语言:JavaScript。
-What we eventually ended up with was one language directly supported by the browser: JavaScript.
+那意味着浏览器可以减少臃肿和问题,因为它们现在只需要支持一种语言:JavaScript。你也许认为 WebAssembly 是异常,但是 WebAssembly 设计之初的目的是使用兼容的抽象语法树来共享JavaScript的语言绑定(AST)。事实上,最早的把 WebAssembly 编译成 JavaScript 的子集的示范是 ASM.js。
-That means that browsers are less bloated and less buggy, because they only need to support a single set of language bindings: JavaScript’s. You might be thinking that WebAssembly is an exception, but one of the design goals of WebAssembly is to share JavaScript’s language bindings using a compatible Abstract Syntax Tree (AST). In fact, the first demonstrations compiled WebAssembly to a subset of JavaScript known as ASM.js.
+作为 web 平台唯一的通用标准编程语言,JavaScript 在软件历史潮流中乘风直上:
-The position as the only standard general purpose programming language for the web platform allowed JavaScript to ride the biggest language popularity wave in the history of software:
+App 吞食世界, web 吞食 app, 同时 JavaScript 吞食 web。
-Apps ate the world, the web ate apps, and JavaScript ate the web.
+根据[各方](http://redmonk.com/sogrady/2016/07/20/language-rankings-6-16/)[调查](http://stackoverflow.com/research/developer-survey-2016),[JavaScript](https://octoverse.github.com/)是目前世界上最流行的语言。
-By [multiple](http://redmonk.com/sogrady/2016/07/20/language-rankings-6-16/)[measures](http://stackoverflow.com/research/developer-survey-2016), [JavaScript](https://octoverse.github.com/) is now the most popular programming language in the world.
+JavaScript 并不是函数式编程的理想化工具,但是它却是为大型的分布式的团队开发大型应用的好工具,因为不同的团队也许有不同建立应用的想法。
-JavaScript is not the ideal tool for functional programming, but it’s a great tool for building large applications on very large, distributed teams, where different teams may have different ideas about how to build an application.
+一些团队致力于脚本化,那么命令式的编程就特别有用,另外一些更精于抽象架构,那么一点保留的面向对象方法也许不失为坏。还有一些拥抱函数式编程,使用纯函数来确保稳定性、可测试性和项目状态管理以便减少用户的反馈。团队里的这些人可以使用相同的语言,意味着他们可以更好的交换想法,互相学习和在其他人的基础上更进一步的开发。
-Some teams may concentrate on scripting glue, where imperative programming is particularly useful. Others may concentrate on building architectural abstractions, where a bit of (restrained, careful) OO thinking may not be a bad idea. Still others may embrace functional programming, reducing over user actions using pure functions for deterministic, testable management of application state. Members on these teams are all using the same language, meaning that they can more easily exchange ideas, learn from each other, and build on each other’s work.
+在 JavaScript 中,所有这些想法可以共存,这样就让更多的人开始拥抱 JavaScript,然后就产生了[世界上最大的开源包管理器](http://www.modulecounts.com/) (2017 年 2 月),[npm](https://www.npmjs.com/)。
-In JavaScript, all of these ideas can co-exist, which allows more people to embrace JavaScript, which has led to the [largest open-source package registry in the world](http://www.modulecounts.com/) (as of February, 2017), [npm](https://www.npmjs.com/).
+JavaScript 的真正优势在于其生态系统中的思想和用户的多样性。它也许不是纯函数式编程的理想语言,但它是你可以想象的工作在不同平台的人共同合作的理想语言,比如说 Java、Lisp 或者 C。JavaScript 也许对这些有这些背景的用户完全友好,但是这些人很乐意学习这门语言并迅速投入生产。
-The true strength of JavaScript is diversity of thought and users in the ecosystem. It may not be absolutely the ideal language for functional programming purists, but it may be the ideal language for working together using one language that works on just about every platform you can imagine — familiar to people coming from other popular languages such as Java, Lisp, or C. JavaScript won’t feel ideally comfortable to users with any of those backgrounds, but they may feel *comfortable enough* to learn the language and become productive quickly.
+我同意 JavaScript 并不是函数式编程着最好的语言。但是,没有任何其他语言可以声称他们可以被所有人使用,同时正如 ES6 所述:JavaScript 可以满足到更与喜欢函数式编程的人的需要,同时也越来越好。相比于抛弃 JavaScript 和它不可思议的被世界上所有公司使用的生态系统,为什么不拥抱它,把它变成一个更适合软件组成化的语言?
-I agree that JavaScript is not the best language for functional programmers. However, no other functional language can claim that it is a language that everybody can use and embrace, and as demonstrated by ES6: JavaScript can and does get better at serving the needs of users interested in functional programming. Instead of abandoning JavaScript and its incredible ecosystem used by virtually every company in the world, why not embrace it, and make it a better language for software composition incrementally?
+现在,JavaScript 已经是一门足够优秀的函数式编程语言,意味着人们可以使用 JavaScript 的函数式编程方法来构造所有有趣的和有用的东西。Netflix(和其他使用 Angular 2+ 的应用)使用基于 RxJS 的函数式功能。[Facebook](https://github.com/facebook/react/wiki/sites-using-react)在 React 中使用纯函数、高阶函数和高级组件来开发 Facebook 和 Instagram,[PayPal, KhanAcademy, and Flipkart](https://github.com/reactjs/redux/issues/310)使用 Redux 来进行状态管理。
-As-is, JavaScript is already a *good enough* functional programming language, meaning that people are building all kinds of useful and interesting things in JavaScript, using functional programming techniques. Netflix (and every app built with Angular 2+) uses functional utilities based on RxJS. [Facebook](https://github.com/facebook/react/wiki/sites-using-react) uses the concepts of pure functions, higher-order functions, and higher order components in React to build Facebook and Instagram. [PayPal, KhanAcademy, and Flipkart](https://github.com/reactjs/redux/issues/310) use Redux for state management.
+它们并不孤单:Angular、React、Redux 和 Lodash 是 JavaScript 生态系统中主要的框架和库,同时它们都被函数式编程很深的影响到 — 或者在 Lodash 和 Redux中,明确的表达是为了在实际的 JavaScript 应用中使用函数式编程模式。
-They’re not alone: Angular, React, Redux, and Lodash are the leading frameworks and libraries in the JavaScript application ecosystem, and all of them are heavily influenced by functional programming — or in the cases of Lodash and Redux, built for the express purpose of enabling functional programming patterns in real JavaScript applications.
+“为什么是 JavaScript?”因为实际上所有的软件呢公司用来开发软件的语言。无论如何,JavaScript 从 Lisp 这个数十年来的标志头上偷取了 “最受欢迎的函数式编程语言” 的头衔。事实上,Haskell 更适合当今函数式编程概念的标准,但是人们并不使用它来开发实际应用。
-“Why JavaScript?” Because JavaScript is the language that most real companies are using to build real software. Love it or hate it, JavaScript has stolen the title of “most popular functional programming language” from Lisp, which was the standard bearer for decades. True, Haskell is a much more suitable standard bearer for functional programming concepts today, but people just aren’t building as many real applications in Haskell.
+在任何时候,在美国都有一百万的 JavaScript 工作需求,世界其他地方也有数百万的量。学习 Haskell 可以帮助你很好的学习函数式编程,但学习 JavaScript 将会教会你在实际工作中开发应用。
-At any given moment, there are close to a hundred thousand JavaScript job openings in the United States, and hundreds of thousands more world-wide. Learning Haskell will teach you a lot about functional programming, but learning JavaScript will teach you a lot about building production apps for real jobs.
+App 正在吞食世界, web 正在吞食 app, 同时 JavaScript 正在吞食 web。
-Apps ate the world, the web ate apps, and JavaScript ate the web.
+[**接下来的第三部分: 函数式开发者的 JavScript 介绍…**](https://medium.com/javascript-scene/a-functional-programmers-introduction-to-javascript-composing-software-d670d14ede30#.zdpw16p65)
-[**Continued in Part 3: A Functional Programmer’s Introduction to JavaScript…**](https://medium.com/javascript-scene/a-functional-programmers-introduction-to-javascript-composing-software-d670d14ede30#.zdpw16p65)
+### 下一步
-### Next Steps
+想更多的学习 JavaScript 的函数式编程?
-Want to learn more about functional programming in JavaScript?
+[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/),什么,你还不是其中之一,out 了!
-[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/). If you’re not a member, you’re missing out!
+[![](https://cdn-images-1.medium.com/freeze/max/30/1*3njisYUeHOdyLCGZ8czt_w.jpeg?q=20)![](https://cdn-images-1.medium.com/max/800/1*3njisYUeHOdyLCGZ8czt_w.jpeg)](https://ericelliottjs.com/product/lifetime-access-pass/)
-[
-](https://ericelliottjs.com/product/lifetime-access-pass/)
+*Eric Elliott* 是 [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly) 和 “Learn JavaScript with Eric Elliott” 的作者。他曾效力于 *Adobe Systems, Zumba Fitness, he Wall Street Journal, ESPN, BBC, and top recording artists including Usher, Frank Ocean, Metallica* 和其他一些公司。
-***Eric Elliott*** is the author of [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly), and [*“Learn JavaScript with Eric Elliott”*](http://ericelliottjs.com/product/lifetime-access-pass/). He has contributed to software experiences for **Adobe Systems, Zumba Fitness, The Wall Street Journal, ESPN, BBC, and top recording artists including Usher, Frank Ocean**, Metallica, and many more.
-
-*He spends most of his time in the San Francisco Bay Area with the most beautiful woman in the world.*
+*他和她的老婆(很漂亮)大部分时间都在旧金山湾区里。*
---
From b029545ce7db113488a20a0722eb40027cff9f44 Mon Sep 17 00:00:00 2001
From: sunxinlei
Date: Sun, 9 Apr 2017 18:41:57 +0800
Subject: [PATCH 160/945] =?UTF-8?q?=E6=A0=B9=E6=8D=AE=20sqrthree=20?=
=?UTF-8?q?=E6=A0=A1=E5=AF=B9=E6=84=8F=E8=A7=81=E6=9B=B4=E6=94=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/modern-javascript-for-ancient-web-developers.md | 12 ++++++------
1 file changed, 6 insertions(+), 6 deletions(-)
diff --git a/TODO/modern-javascript-for-ancient-web-developers.md b/TODO/modern-javascript-for-ancient-web-developers.md
index d62d526e6a9..f4cbf50cb77 100644
--- a/TODO/modern-javascript-for-ancient-web-developers.md
+++ b/TODO/modern-javascript-for-ancient-web-developers.md
@@ -26,13 +26,13 @@ Hi.我是一个正在学习现代 JavaScript 的“老派” web 开发者。我
这种情况下,就有必要向你身边的 JavaScript 工程师朋友伸手求助了,和他们聊一聊你的技术路线。我很荣幸在 Postlight 得到了工程师朋友(特别是 [Jeremy Mack](https://medium.com/@mutewinter))的精湛指导,感谢他们容忍我无穷无尽的问题。
-我要说的是,学习现代的 JavaScript 需要人为干预。事物还在不断发展变化,各种教程尚未成熟和定型,所谓最佳实践也未形成正式规范。如果你身边没有大牛,那么至少也得检查 Medium 上文章或教程的日期,或 Github 仓库的最近一次提交时间。如果时间超过了一年,基本上可以确定已经过时。
+我要说的是,学习现代的 JavaScript 需要人为干预。事物还在不断发展变化,各种教程尚未成熟和定型,所谓最佳实践也未形成正式规范。如果你身边没有大牛,那么至少也得检查 Medium 上文章或教程的日期,或 GitHub 仓库的最近一次提交时间。如果时间超过了一年,基本上可以确定已经过时。
### 新的问题,而不是已经确定的解决方案 ###
走类似这样的路线:当你在学习现代 JavaScript 时,你遇到的问题的解决方案还在渐渐得到解决,这正是一个好机会。事实上,很可能仅仅差一次 code review,你在使用这个包时就可以修复问题。
-当你在使用一种像 PHP 这样的古老的语言的时候,你可以 Google 一个提问或者问题,几乎百分之百能找到一个5年前的 Stack Overflow 回答来解决它,或者你能在(详尽的,大量评论的,无与伦比的)[文档](http://docs.php.net/docs.php)里找到整个描述。
+当你在使用一种像 PHP 这样的古老的语言的时候,你可以 Google 一个提问或者问题,几乎百分之百能找到一个 5 年前的 Stack Overflow 回答来解决它,或者你能在(详尽的、大量评论的、无与伦比的)[文档](http://docs.php.net/docs.php)里找到整个描述。
现代 JavaScript 就并非如此了。 我曾经徜徉在 GitHub issues 和源码的时候不止一次找到的都是一些过时的文档。剖析 GitHub 版本库是学习和使用各种包的一部分,而且对于我这样的“老派人”,差之毫厘的学习总是令人迷惑。
@@ -42,7 +42,7 @@ Hi.我是一个正在学习现代 JavaScript 的“老派” web 开发者。我
[![Markdown](http://i4.buimg.com/1949/adafb30475d3d36a.png)](https://twitter.com/capndesign/status/832638513048850433/photo/1)
-**不要因此止步不前**。我不得不放手去做,从起步到正确配置,允许自己的不完美甚至一些业余,只为舒适地使用自己的工具。(我不会告诉你我曾用[nodemon](https://nodemon.io/)做代码检查)随后我会找到更好的方法并且在每个新项目中纳入进来。
+**不要因此止步不前**。我不得不放手去做,从起步到正确配置,允许自己的不完美甚至一些业余,只为舒适地使用自己的工具。(我不会告诉你我曾用 [nodemon](https://nodemon.io/) 做代码检查)随后我会找到更好的方法并且在每个新项目中纳入进来。
这方面 JS 还有大量的工作要做。现代 JavaScript 领域依然是不断变化的,但我一个现代 JS 工程师亲友告诉我,[这份来自 Jonathan Verrecchia 的教程](https://github.com/verekia/js-stack-from-scratch)是目前构建一个当代 JavaScript 栈的不二之选。对,就是现在。
@@ -54,17 +54,17 @@ Hi.我是一个正在学习现代 JavaScript 的“老派” web 开发者。我
这是迄今为止我在这个过程中经历过的一些研讨会和教程的不完整列表。
-- [HOW-TO-NPM](https://github.com/workshopper/how-to-npm) —— npm 是 JavaScript 的包管理器。即使在学习这个教程之前我已经敲打过上千次 “npm install”,但是知道学完这个我才知道 npm 做的所有事情。(在很多项目中我已经转移使用[yarn](https://github.com/yarnpkg/yarn),而不是 npm,但所有的概念都是相通的)
+- [HOW-TO-NPM](https://github.com/workshopper/how-to-npm) —— npm 是 JavaScript 的包管理器。即使在学习这个教程之前我已经敲打过上千次 “npm install”,但是知道学完这个我才知道 npm 做的所有事情。(在很多项目中我已经转移使用[yarn](https://github.com/yarnpkg/yarn),而不是 npm,但所有的概念都是相通的)
`npm i -g how-to-npm`
-- [learnyounode](https://github.com/workshopper/learnyounode)——我打算专注于服务端 JavaScript,因为那有令我安逸的东西,那就是 Node.js。Learnyounode 是一个交互式教程,结构上类似 how-to-npm。
+- [learnyounode](https://github.com/workshopper/learnyounode)——我打算专注于服务端 JavaScript,因为那有令我安逸的东西,那就是 Node.js。Learnyounode 是一个交互式教程,结构上类似 how-to-npm。
- [expressworks](https://github.com/azat-co/expressworks) —— 和前面两个项目类似,Expressworks 是 Express.js 的介绍,一个 Node.js 的 web 框架。在 Postlight 公司 Express 没有得到广泛使用,但对于初学者,它值得学习去上手构建一个简单的 web 应用。
- 现在是时候做点真东西了。我发现 Tomomi Imura 的一篇教程 [Creating a Slack Command Bot from Scratch with Node.js](http://www.girliemac.com/blog/2016/10/24/slack-command-bot-nodejs/) 已经可以学到足够的 Node 和 Express 的新技能来应对工作。因为我专注于后端,使用 Slack 创建一个 “/” 命令是一个很好的开始,因为没有前端演示(Slack 帮你做好了)
-- 在构建这个命令的过程中,我不使用演练中所推荐的 ngrok 或者 Heroku,而是使用 [Zeit Now](https://zeit.co/now),这是任何人可用的,创建快速一次性的 JS 应用的宝贵工具。
+- 在构建这个命令的过程中,我不使用演练中所推荐的 ngrok 或者 Heroku,而是使用 [Zeit Now](https://zeit.co/now),这是任何人可用的、创建快速一次性的 JS 应用的宝贵工具。
- 一旦开始写真正意义的代码,我也开始掉下工具无底洞了,安装 Sublime 插件,获取正确的 [Node 版本](https://github.com/postlight/lux/blob/master/CONTRIBUTING.md#nodejs-version-requirements),配置 ESLint,使用 [Airbnb 的代码规范 (Postlight 公司的偏好)](https://github.com/airbnb/javascript) —— 这些事情拖了我的后退,但也都是有价值的初始化投资。对于这方面我还在坑里,例如 Webpack 对我来说依然美妙又神秘,不过[这个视频是个很不错的介绍](https://www.youtube.com/watch?v=WQue1AN93YU)*.*
- 某些时候 JS 的异步执行(特别是[回调地狱](http://callbackhell.com/))开始困扰我,[Promise It Won’t Hurt](https://github.com/stevekane/promise-it-wont-hurt) 是另一个教你怎样使用 Promise 书写优雅异步逻辑的教程。Promise 是用于解决异步执行的 JS 新概念。说实话 Promise 令我耳目一新,他们是巧妙的范式转变。感谢 [Mariko Kosaka](http://kosamari.com/notes/the-promise-of-a-burger-party),现在我每次买汉堡的时候都能想起这些。
From be832cc766613663cdf57b56822961647907101b Mon Sep 17 00:00:00 2001
From: sunxinlei
Date: Sun, 9 Apr 2017 18:47:15 +0800
Subject: [PATCH 161/945] =?UTF-8?q?=E4=B8=A4=E5=A4=84=E8=8B=B1=E6=96=87?=
=?UTF-8?q?=E7=A9=BA=E6=A0=BC?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/modern-javascript-for-ancient-web-developers.md | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/TODO/modern-javascript-for-ancient-web-developers.md b/TODO/modern-javascript-for-ancient-web-developers.md
index f4cbf50cb77..844e8f62cdf 100644
--- a/TODO/modern-javascript-for-ancient-web-developers.md
+++ b/TODO/modern-javascript-for-ancient-web-developers.md
@@ -22,7 +22,7 @@ Hi.我是一个正在学习现代 JavaScript 的“老派” web 开发者。我
### 转移目标 (.jS)
-现代 JS 的特点就是朝气蓬勃和发展迅速,所以很容易就选择了过时的框架、模板引擎、构建工具、 教程或者已经不是最佳实践的技术。( 如果真有一个被广泛接受的最佳实践的概念的话)
+现代 JS 的特点就是朝气蓬勃和发展迅速,所以很容易就选择了过时的框架、模板引擎、构建工具、 教程或者已经不是最佳实践的技术。(如果真有一个被广泛接受的最佳实践的概念的话)
这种情况下,就有必要向你身边的 JavaScript 工程师朋友伸手求助了,和他们聊一聊你的技术路线。我很荣幸在 Postlight 得到了工程师朋友(特别是 [Jeremy Mack](https://medium.com/@mutewinter))的精湛指导,感谢他们容忍我无穷无尽的问题。
@@ -54,7 +54,7 @@ Hi.我是一个正在学习现代 JavaScript 的“老派” web 开发者。我
这是迄今为止我在这个过程中经历过的一些研讨会和教程的不完整列表。
-- [HOW-TO-NPM](https://github.com/workshopper/how-to-npm) —— npm 是 JavaScript 的包管理器。即使在学习这个教程之前我已经敲打过上千次 “npm install”,但是知道学完这个我才知道 npm 做的所有事情。(在很多项目中我已经转移使用[yarn](https://github.com/yarnpkg/yarn),而不是 npm,但所有的概念都是相通的)
+- [HOW-TO-NPM](https://github.com/workshopper/how-to-npm) —— npm 是 JavaScript 的包管理器。即使在学习这个教程之前我已经敲打过上千次 “npm install”,但是知道学完这个我才知道 npm 做的所有事情。(在很多项目中我已经转移使用 [yarn](https://github.com/yarnpkg/yarn),而不是 npm,但所有的概念都是相通的)
From 3fc96b46eccefccbe0c2f147247c63484c942f48 Mon Sep 17 00:00:00 2001
From: lsvih
Date: Sun, 9 Apr 2017 20:51:55 +0800
Subject: [PATCH 162/945] =?UTF-8?q?=E4=BA=8C=E6=A0=A1=E6=84=8F=E8=A7=81?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/modules-vs-microservices.md | 10 +++++-----
1 file changed, 5 insertions(+), 5 deletions(-)
diff --git a/TODO/modules-vs-microservices.md b/TODO/modules-vs-microservices.md
index 743f181583e..fcac5d43e3b 100644
--- a/TODO/modules-vs-microservices.md
+++ b/TODO/modules-vs-microservices.md
@@ -6,11 +6,11 @@
# 模块化 vs. 微服务
-在避免微服务复杂性的同时,应用模块化系统设计设计原则。
+使用模块化系统设计原则来避免微服务的复杂性。
![](https://d3tdunqjn7n0wj.cloudfront.net/360x240/container-227877_1920-0db52b796e6b80d98f6df2d01a6ee4fb.jpg)
-从单体式应用向微服务架构迁移已经是老生常谈的话题了。除了过过嘴瘾,似乎真的动手将单体式应用拆分成微服务也不是什么很困难的事。但是这种做法真的是你们团队的最佳选择吗?维护一个凌乱的单体式应用的确很伤脑筋,但是还有另一种强有力但常常被人忽视的替代方案:模块化应用开发。本文将探讨这种替代方案,并展现其与构建微服务的关系。
+从单体式应用向微服务架构迁移已经是老生常谈的话题了。除了过过嘴瘾,似乎真的动手将单体式应用拆分成微服务也不是什么很困难的事。但是这种做法真的是你们团队的最佳选择吗?维护一个凌乱的单体式应用的确很伤脑筋,但是还有另一种优秀但常常被人忽视的替代方案:模块化应用开发。本文将探讨这种替代方案,并展现其与构建微服务的关系。
## 模块化微服务
@@ -40,9 +40,9 @@
例如在 Java 中,有几个可以帮助你构建应用的模块系统。OSGi 是其中最著名的一个,不过随着 Java 9 的发布,Java 平台将加入一个原生的模块系统。现在模块作为一等结构(first-class construct),成为了语言和平台的一部分。Java 模块可以表明对其它模块的依赖,以及在强封装实现类的时候公开暴露接口。甚至 Java 平台本身(一个庞大的代码库)已经使用了新的 Java 模块系统进行模块化。你可以在我即将出版的书[Java 9 Modularity](https://www.safaribooksonline.com/library/view/java-9-modularity/9781491954157/?utm_source=newsite&utm_medium=content&utm_campaign=lgen&utm_content=modules-vs-microservices-inline)中了解有关 Java 9 模块化开发的更多信息。(现早期版本已经发布)
-其它的语言也提供了类似的机制。例如,JavaScript 在 ES2015 规范中提供了一个[模块系统](http://exploringjs.com/es6/ch_modules.html)。在此之前,Node.js 也为 JavaScript 后端提供了一个非标准的模块系统。然而 JavaScript 作为一种动态语言,对于强制接口(类型)与模块封装的支持还是较弱。你可以考虑在 JavaScript 的基础上使用 TypeScript 来重新获得这些优点。微软的 .Net 框架与 Java 一样都有着强类型,但就强封装以及程序集(Assemblies)间的显式依赖而言,它与 Java 即将推出的模块系统并不相同。尽管如此,你可以通过使用 [.Net Core](https://msdn.microsoft.com/en-us/magazine/mt707534.aspx) 中标准化的反转控制模式(IOC)以及创建逻辑相关的程序集来实现良好的模块化架构。即使是 C++ 也在以后的版本中[考虑添加](http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2016/n4610.pdf)模块系统。许多语言现在都对模块化表现出了欣赏之情,这本身就是一个显著的进步。
+其它的语言也提供了类似的机制。例如,JavaScript 在 ES2015 规范中提供了一个[模块系统](http://exploringjs.com/es6/ch_modules.html)。在此之前,Node.js 也为 JavaScript 后端提供了一个非标准的模块系统。然而 JavaScript 作为一种动态语言,对于强制接口(类型)与模块封装的支持还是较弱。你可以考虑在 JavaScript 的基础上使用 TypeScript 来重新获得这些优点。微软的 .Net 框架与 Java 一样都有着强类型,但就强封装以及程序集(Assemblies)间的显式依赖而言,它与 Java 即将推出的模块系统并不相同。尽管如此,你可以通过使用 [.Net Core](https://msdn.microsoft.com/en-us/magazine/mt707534.aspx) 中标准化的反转控制模式(IOC)以及创建逻辑相关的程序集来实现良好的模块化架构。即使是 C++ 也在以后的版本中[考虑添加](http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2016/n4610.pdf)模块系统。许多语言都在向模块化靠近,这本身就是一个显著的进步。
-当你有意识地使用你的开发平台的模块化特性时,你就可以实现之前提及的微服务的模块化优势。基本上模块系统越好,你在开发过程中获得的帮助就越多。只要在不同团队间的接触点定义好明确的接口,不同的团队也可以独立进行不同部分的工作。当然,在部署时还是要将模块在一个单独的部署单元中组合起来。这样可以防止过大的复杂度,以及迁移到微服务所需要的开发与管理成本。诚然,这也意味着你不能使用不同的技术栈来构建不同的模块,但你的团队应该不会真的这么做吧?
+当你有意识地使用你的开发平台的模块化特性时,你就可以实现之前提及的微服务的模块化优势。基本上模块系统越好,你在开发过程中获得的帮助就越多。只要在不同团队间的接触点定义好明确的接口,不同的团队也可以独立进行不同部分的工作。当然,在部署时还是要将模块在一个单独的部署单元中组合起来。这样可以防止过于复杂,以及减少迁移到微服务所需要的开发与管理成本。诚然,这也意味着你不能使用不同的技术栈来构建不同的模块,但你的团队应该不会真的这么做吧?
## 模块设计
@@ -68,7 +68,7 @@
## 总结
-总之,最好的方案就是找到一个折中的点。这两种方案都有可取之处,需要根据实际环境、组织和应用本身进行选择。既然你可以在之后迁移成微服务架构,那为什么最开始不直接使用模块化应用呢?如果你之前就已经划分好了模块边界,那也就不需要再去拆分你的单体式应用了。甚至你还可以在模块内部搭建微服务架构。那么问题就变成了:为什么微服务一定要是”微“的呢?
+总之,最好的方案就是找到一个折中的点。这两种方案都有可取之处,需要根据实际环境、组织和应用本身进行选择。既然你可以在之后迁移成微服务架构,那为什么最开始不直接使用模块化应用呢?如果你之前就已经划分好了模块边界,那也就不需要再去拆分你的单体式应用了。甚至你还可以在模块内部搭建微服务架构。那么问题就变成了:为什么微服务一定要是“微”的呢?
即使你的应用刚从模块化应用转成微服务架构,服务也不必非得很“微”才具备可维护性。在服务中应用模块化原则能让它们在复杂度的可扩展性上超越通常的微服务。现在这份蓝图中既有微服务也有模块,减少架构中的服务的数量可以节约成本;而其中的模块可以像构建单体式应用一样,构建和扩展服务。
From b20456939ffc66de053d0857be6fd3294e961dcc Mon Sep 17 00:00:00 2001
From: ivyxuan
Date: Sun, 9 Apr 2017 20:59:48 +0800
Subject: [PATCH 163/945] =?UTF-8?q?=E6=A0=B9=E6=8D=AE=20ylq167=20=E5=92=8C?=
=?UTF-8?q?=20gaozp=20=E7=9A=84=E6=84=8F=E8=A7=81=E8=BF=9B=E8=A1=8C?=
=?UTF-8?q?=E4=BF=AE=E6=94=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/the-details-that-matter.md | 90 ++++++---------------------------
1 file changed, 15 insertions(+), 75 deletions(-)
diff --git a/TODO/the-details-that-matter.md b/TODO/the-details-that-matter.md
index 401d203b9d2..fce6047f6df 100644
--- a/TODO/the-details-that-matter.md
+++ b/TODO/the-details-that-matter.md
@@ -1,215 +1,155 @@
> * 原文地址:[The Details That Matter](https://uxplanet.org/the-details-that-matter-8b962ca58b49#.ypigeruoq)
> * 原文作者:[Nick Babich](https://uxplanet.org/@101?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
-> * 校对者:
+> * 译者:[ivyxuan](https://github.com/iloveivyxuan)
+> * 校对者:[ylq167](http://www.11167.xyz)、[gaozp](https://github.com/gaozp)
-# The Details That Matter #
# 细节是产品设计的重中之重 #
-Your product’s success is based on a combination of factors, but the overall user experience tops them all. When it comes down to designing a new app or a site, sticking to best practices is a solid way to go, but during the creation of the big picture, it’s fairly easy to skimp over design elements that feel like nice to have but not necessary. However, the difference between good and bad experiences often comes down to how thoughtful we can design these small details.
一个产品的成功是由各种因素共同造就的,而其中最重要的因素,就是整体的用户体验。在设计一款新的应用或是网站的时候,坚持最佳的实践规范是一个可靠的方法,但是在创造宏伟蓝图的时候,人们很容易就会省略掉那些能让人有更好的体验但却并非必要的设计元素。然而,设计的优劣往往在于我们能设计出多么体贴的细节。
-In this article, I’ll focus on *visual feedback*, *microcopy* and *whitespace* and you’ll see why these *little big details* are just as important as the more obvious elements of your design, and how they help determine the success of your product.
-在这篇文章中,我将会关注可视化反馈、小的提示信息还有留白这几个方面,你会发现为什么这些不起眼的细节和那些显眼的设计元素相比同样重要,而这些细节又是怎样决定你产品成败的。
+在这篇文章中,我将会重点关注**可视化反馈**、**小的文字信息**还有**留白**这几个方面,你将会发现为什么这些不起眼的细节和那些显眼的设计元素相比同样重要,而这些细节又是怎样决定你产品成败的。
-### Visual Feedback ###
### 可视化反馈 ###
-Visual feedback might be easily overlooked in the greater design scheme, but it actually hold the entire experience together. When there’s no feedback there’s no proper interaction. Imagine talking to someone who doesn’t respond in any way — you can’t communicate at all. Same goes for your app.
可视化反馈在较大的设计方案里很容易就会被忽视掉,但它实际上贯穿整体的用户体验流程。可以说,如果没有反馈就没有所谓的交互,你能想象和一个人聊天,可他一点反应也不给你吗 —— 你根本就聊不下去。而对于你的应用也是同样的道理。
-> *Lack of visual feedback is able to confuse users.*
> **缺乏可视化的反馈会让用户感到困惑。**
-You must ensure that there is always some feedback for user actions, because it makes users feel in control. Visual feedback
-你必须要确保对于用户的每个动作都有相应的反应,因为这会让用户会感觉他在控制着一切。可视化反馈
+你必须要确保对于用户的每个动作都有相应的反馈,因为这会让用户感觉应用运行一切正常。可视化反馈
-- Acknowledges that the app has received a user’s action.
-- Communicates the results of interaction, making it both *visible* and *understandable*. Gives the user a signal that they (or the app) have succeeded or failed at performing a task.
-- 首先表明这个应用了解了用户的操作。
+- 首先表明这个应用接受到了用户的操作。
- 然后它通过一种可视化而且易于理解的方式告诉用户这次交互的结果是什么,通过给用户一个信号,来告诉用户自己对于这个任务的执行是成功还是失败。
-#### Make buttons and other controls tangible ####
-#### 让按钮或是其他开关看起来是可以碰的 ####
+#### 让按钮或是其他开关看起来是可触摸的 ####
-In real life, buttons, controls and objects respond to our interaction, and this is how people expect things to work. People expect a similar level of responsiveness from app elements.
在现实生活中,按钮、开关还有其他东西都会对我们的动作有所回应,人们觉得世界就是这样运转的。而同样,人们也会期待应用里的元素能有类似的回应。
-Image credits: [Ramotion](https://dribbble.com/shots/1749645-Contact-Sync)
图片来源:[Ramotion](https://dribbble.com/shots/1749645-Contact-Sync)
-#### Result of operation ####
#### 操作的结果 ####
-Visual feedback is also helpful when you need to inform users about results of an operation. You can use existing elements to deliver a feedback.
-当你需要告诉用户他的操作结果是什么的时候,可视化反馈就很有用了。你可以利用现有的元素去传递反馈信息。
+当你需要告诉用户他的操作结果是什么的时候,可视化反馈就很有用了,你可以利用现有的元素去传递反馈信息。
-Image credits: [Colin Garven](https://dribbble.com/ColinGarven)
图片来源:[Colin Garven](https://dribbble.com/ColinGarven)
-#### **System should tell users its state** ####
-#### **系统需要告诉用户他的状态是什么** ####
+#### **系统需要告诉用户他当前的状态是什么** ####
-Users want to know their current context in a system at any given time and apps shouldn’t keep them guessing — they should tell the user what’s happening via appropriate visual feedback. For frequent and minor actions, the response can be modest, while for infrequent and major actions, the response should be more substantial.
-在系统中,用户任何时候都会想知道他此刻的状态是什么,而这不应该让用户自己去猜 —— 所以系统应该通过恰当的可视化反馈告诉用户此刻正在发生什么。对于一些常见而且次要的操作,适当的反馈就可以了,而对于不寻常而且重要的操作,反馈就需要更明显一点。
+在系统中,用户任何时候都会想知道他此刻的状态是什么,而这不应该让用户自己去猜 —— 所以系统应该通过恰当的可视化反馈告诉用户此刻正在发生什么。对于一些常见而且次要的操作,简单的反馈就可以了,而对于不寻常而且重要的操作,反馈就需要更明显一点。
-- [Animated notification](https://uxplanet.org/3-key-uses-for-animation-in-mobile-ui-design-4d7c482dd84b#.x07lyyazb) makes it possible for a user to quickly understand the current status.
- [动态提醒](https://uxplanet.org/3-key-uses-for-animation-in-mobile-ui-design-4d7c482dd84b#.x07lyyazb)可以让用户立刻明白此时的状态。
-Image credits: [Eddy Gann](https://dribbble.com/SMSeddy)
图片来源:[Eddy Gann](https://dribbble.com/SMSeddy)
-- [Loading animations](https://uxplanet.org/progress-indicators-in-mobile-ux-design-a141e22f3ea0#.etoavwmbw) provides real-time notification of app’s process status, enabling the user to quickly understand what is going on.
- [加载动画](https://uxplanet.org/progress-indicators-in-mobile-ux-design-a141e22f3ea0#.etoavwmbw)是对于应用进程的实时提醒,可以让用户立刻明白现在加载到哪里了。
-A loading bar engages the user and prevents confusion when app is busy loading data. Image credits: [Mark](https://dribbble.com/milkycookie)
-图片来源:[Mark](https://dribbble.com/milkycookie)
-应用无法快速加载信息的时候,进度条可以避免让用户感到困惑。图片来源:[Mark](https://dribbble.com/milkycookie)
+应用加载信息的时候,进度条可以避免让用户感到困惑。图片来源:[Mark](https://dribbble.com/milkycookie)
-### Microcopy ###
### 少量的文字信息 ###
-Microcopy is the little bits of text that guide users through an experience. Microcopy examples are error messages, button labels, hint text. At a glance, these tiny clusters of words seem insignificant when compared to the overall app design. But surprisingly, they have a huge impact on conversions.
少量的文字信息是一些用来是指导用户行为的一点点文字。举一些例子就是,错误信息、按钮对应的标签、提示信息。乍看之下,这么少的文字和整个应用设计比起来一点也不重要,但出人意料的是,它们对转化率有着极为重要的影响。
-> Writing good microcopy in your app is just as important as having the app work correctly and the user interface being easy and efficient to use.
> 在应用里写出好的文字信息,和让应用正常运行、用户界面易于使用一样重要。
-#### Show that you’re human ####
#### 让应用看起来像一个有血有肉的人 ####
-A quick way to make your UI warmer and less mechanical is a human tone in the copy. If your product sounds human, it’s easier for people to trust you.
有一个快速的方法能让你的 UI 变得温暖而不呆板,就是用人说话的口吻去描述内容。如果你的产品听起来好像是一个人,用户就会更加的信任你。
-Yelp shows they have real humans in charge.
Yelp 表现得好像他们是真人在负责这件事情。
-Airbnb sound human and conversational
Airbnb 的提示听起来像人说的话而且语气还很随和。
-#### Use friendly and helpful copy in a moment of failure ####
#### 报错的方式要友好而且有效 ####
-How errors are communicated can have a huge impact on the way someone experiences your product. Often overlooked, an ill-constructed error message can fill users with frustration.
-表达错误信息的方式会严重影响产品的用户体验。通常来说,省略错误信息或是没有正确描述错误信息都会让用户感到失望。
+表达错误信息的方式会严重影响产品的用户体验。通常来说,省略错误信息或是没有正确描述错误信息都会让用户受挫。
-*An alert message such as “An error occurred” is mystifying to all users and is likely to annoy experienced users.*
**像“出错啦”这种警告对所有的用户都会造成困扰,而且还会惹恼专家级用户。**
-A well-crafted error message, on the other hand, can turn a moment of frustration into a moment of delight. Thus, make error messages human, not technical, and suited to your audience.
但是,一个精心设计过的错误信息,会顿时化失望为欣喜。所以,把报错变得人性化、不用技术性的语言并且适合你的用户群体。
-Error states must incorporate concise, friendly, and instructive copy as to what to do next.
错误状态一定要具体、友好而且有用,要告诉用户下一步怎么做。
-#### Alleviate users concerns ####
#### 减少用户的担忧 ####
-Microcopy is extremely contextual. That’s why it’s so valuable. It answers a very specific question people have and speaks to their concerns right on the spot. For example, microcopy can be fundamental in reassuring your users at the point of subscribing or sharing details. Whilst ‘not to spam/auto-tweet’ might be taken for granted by good marketers when asking for email address/access to the social network account connections, the user is less than sure. Thus, when people add their emails or connect their Twitter accounts, say “we hate spam as much as you do.”
这些少量的文字信息是很情景化的,这也是为什么它很重要的一个原因。它可以解答用户具体情况下的问题,并针对他们所担忧的事情进行直接地交流。举例来说,当用户选择订阅或是提供了具体信息的时候,一些文字信息对于消除用户担忧会起到相当关键的作用。对于优秀的营销人员来说,“不会有群发消息或是自动关注”是理所当然的事情,但用户自己会存疑。因此,当用户添加了他们的邮箱地址或者绑定了 Twitter 账号的时候,一定要明确表态“我们和你一样讨厌垃圾消息”。
-This microcopy covers all the potential user concerns in one tight little sentence.
这只是一小段紧凑的文字,却涵盖了用户所有潜在的担忧。
-### Whitespace ###
### 留白 ###
-Whitespace (or negative space) is the areas of a design where there is no element placed by a designer. Elements of whitespace are space around images, margins, paddings, line-spacing and letter-spacing within text content. Although many may consider it a waste of valuable screen estate, whitespace is an essential element in user interface design.
-留白(或者说是负空间)是设计师没有摆放设计元素的地方。而设计元素间的留白是指处在图片间、内填充(margin)、外填充(padding)和行间、字间的空白。虽然很多人觉得这些空白浪费了宝贵的界面位置,但其实,留白是用户界面设计的一个重要元素。
+留白(或者说是负空间)是设计师没有摆放设计元素的地方。而设计元素间的留白是指处在图片间距、内边距、外边距、行间距和字间距的空白。虽然很多人觉得这些空白浪费了宝贵的界面位置,但其实,留白是用户界面设计的一个重要元素。
-#### Improve UI comprehension ####
#### 让用户界面更容易理解 ####
-*Clutter is bad.* Cluttering your interface overloads your user with too much information: every added button, image, and line of text make the screen more complicated. If you don’t think any part of your design should be intentionally blank, take a look at example below and you’ll see what happens when too many objects competing for your attention.
**杂乱的堆砌是很糟糕的一件事情。**在界面上杂乱堆砌元素会给用户带来过多的信息:每一个被添加的按钮、图片和文字都会让界面显得更加复杂。如果你不想你的设计有任何刻意的留白的话,下面这个例子就能很明白的告诉你,有太多东西一起吸引你的注意力是多么可怕的事情。
-A cluttered UI is unattractive and doesn’t make users want to scan it, especially when there’s no visual hierarchy within the view.
-杂乱堆砌的 UI 尤其是没有视觉层次的 UI 会让用户没有拥有一点想要审视的欲望。
+杂乱堆砌的 UI 尤其是没有视觉层次的 UI 会让用户没有一点想要审视的欲望。
-The power of white space comes from the limits of human attention and memory. Our short-term memory can hold a small amount of information (typically [around 7 items or even less](http://www.human-memory.net/types_short.html)) in mind in an active, readily-available state for a short period of time (typically from 10 to 15 seconds, or sometimes up to a minute).
-留白之所以很重要,是因为用户的注意力还有记忆里是有限的。我们的短期记忆只可以在短时间内(通常来说是 10 到 15 秒钟、或者是 1 分钟以内)记住一点有效的既有信息(通常来说[是 7 个事物或者比这个更少](http://www.human-memory.net/types_short.html))。
+留白之所以很重要,是因为用户的注意力还有记忆是很有限的。我们的短期记忆只可以在短时间内(通常来说是 10 到 15 秒钟、或者是 1 分钟以内)记住一点有效的既有信息(通常来说[是 7 个事物或者比这个更少](http://www.human-memory.net/types_short.html))。
-> User attention is a precious resource, and should be allocated accordingly.
> 用户的注意力是很珍贵的资源,所以必须合理的分配。
-If cluttering your interface overloads your user with too much information, then reducing the clutter will improve comprehension. Generous whitespace can make some of the messiest interfaces look inviting and simple. Whitespace reduces the amount of elements users see all at once and makes scanning much easier. The skill of using whitespace lies in providing your users with a digestible amount of content ([chunks of content](https://uxplanet.org/best-practices-for-cards-fa45e3ad94dd#.by8pzk56q)), then stripping away extraneous details.
如果因为你界面上的胡乱堆砌使得用户接收过多的信息,那么减少一些杂物就能改善用户的理解。大方地使用留白可以让凌乱的界面变得简单而有吸引力,留白削减了用户乍看之下接收到的元素数量,这使得浏览信息变得更加容易。留白的使用技巧在于只给用户提供能让他消化的数量的内容([一定数量的内容](https://uxplanet.org/best-practices-for-cards-fa45e3ad94dd#.by8pzk56q)),然后去掉不必要的细节。
-Medium is a great example of using whitespace to improve content and UI comprehension.
Medium 是一个典型的例子,它善于使用留白来改善用户对内容和 UI 的理解。
-#### Draw attention to elements ####
#### 让元素更具有吸引力 ####
-Whitespace creates the spaces around elements in the design to help them stand out or separate from the other elements. It helps communicate what’s most important and needs attention.
留白在设计中是通过在元素周围留出空白,以让元素更加突出或是和其他元素以进行区分。它可以告诉用户什么是最重要而且是需要格外注意的。
-> The more space you provide around the element, the more focused it becomes.
> 元素周围的留白越多,元素就会越引人注目。
-Google Search homepage is a great example of using whitespace. The layout immediately facilitates the user goal by placing primary interaction element (search box) in front and center, with plenty of white space on either side to add emphasis.
谷歌搜索的首页就是一个使用留白的典型例子,它通过在正中央摆放其最重要的交互元素(搜索框),并且在周围留出足够的空白以凸显其重要性的布局,直接实现了用户目标。
-The lack of other elements will make existing elements stand out more.
去掉其他的元素可以更加凸显留下来的元素。
-#### Clarifying relationships ####
#### 明确关系 ####
-The [law of proximity](https://www.interaction-design.org/literature/article/laws-of-proximity-uniform-connectedness-and-continuation-gestalt-principles-2) describes how the human eye perceives connections between visual elements. It states that objects near to each other appear similar. We can use whitespace to build perceived relationships between different elements. Take a look at this picture. Almost everyone sees two groups of dots, rather than simply 16 dots.
[接近法则](https://www.interaction-design.org/literature/article/laws-of-proximity-uniform-connectedness-and-continuation-gestalt-principles-2)描述了人的眼睛是如何划分视觉元素的,它阐述说距离更近的物体看起来更相似。我们可以利用留白,在不同的元素间产生视觉联系。你看下面这幅图片,几乎所有人都会说看到了两组点,而不是16个点。
-Breaking the information up into appropriate groups can help make it feel more scannable and readable. In the form on the right, categorising the 15 fields into three groups makes the process feel easier. The amount of content is the same, but the impression on users is much different.
将信息拆分成合适的组别可以让信息更好的被识别和阅读。右边的表单将 15 栏拆分成 3 组,这样填写表单也变得相对容易了。虽然内容的数量仍然相同,但是给用户的感觉却完全不同。
-Image credits: NNGroup
图片来源:NNGroup
-### Conclusion ###
### 总结 ###
-*Design with care.* Each minor detail in your app’s UI deserves close attention because UX is the sum of all details working harmoniously:
**用心设计,应用界面上的每一个小的细节都值得细心揣摩,因为用户体验就是由这些小的细节相互协调作用而成的:
-> “The details are not the details. They make the design.” ―Charles Eames
-> “细节不只是细节,细节成就了设计。”
+> “细节不只是细节,细节成就了设计。” —— Charles Eames
-Thank you!
谢谢!
-*Follow UX Planet:* [*Twitter*](https://twitter.com/101babich)[*Facebook*](https://www.facebook.com/uxplanet/)
**关注 UX Planet:** [*Twitter*](https://twitter.com/101babich)[*Facebook*](https://www.facebook.com/uxplanet/)
---
From 1004922aa7b7e87efced13f6b71a75d06f353a8b Mon Sep 17 00:00:00 2001
From: ivyxuan
Date: Sun, 9 Apr 2017 21:14:26 +0800
Subject: [PATCH 164/945] =?UTF-8?q?=E4=BF=AE=E6=94=B9=E4=B8=80=E7=82=B9?=
=?UTF-8?q?=E9=94=99=E8=AF=AF?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/the-details-that-matter.md | 10 ++++------
1 file changed, 4 insertions(+), 6 deletions(-)
diff --git a/TODO/the-details-that-matter.md b/TODO/the-details-that-matter.md
index fce6047f6df..b4ccfb5d9f8 100644
--- a/TODO/the-details-that-matter.md
+++ b/TODO/the-details-that-matter.md
@@ -62,7 +62,7 @@
> 在应用里写出好的文字信息,和让应用正常运行、用户界面易于使用一样重要。
-#### 让应用看起来像一个有血有肉的人 ####
+#### 让应用看起来像是一个人 ####
有一个快速的方法能让你的 UI 变得温暖而不呆板,就是用人说话的口吻去描述内容。如果你的产品听起来好像是一个人,用户就会更加的信任你。
@@ -80,9 +80,7 @@ Airbnb 的提示听起来像人说的话而且语气还很随和。
-**像“出错啦”这种警告对所有的用户都会造成困扰,而且还会惹恼专家级用户。**
-
-但是,一个精心设计过的错误信息,会顿时化失望为欣喜。所以,把报错变得人性化、不用技术性的语言并且适合你的用户群体。
+**像“出错啦”这种警告对所有的用户都会造成困扰,而且还会惹恼专家级用户。**但是,一个精心设计过的错误信息,会顿时化失望为欣喜。所以,把报错变得人性化、不用技术性的语言并且适合你的用户群体。
@@ -144,13 +142,13 @@ Medium 是一个典型的例子,它善于使用留白来改善用户对内容
### 总结 ###
-**用心设计,应用界面上的每一个小的细节都值得细心揣摩,因为用户体验就是由这些小的细节相互协调作用而成的:
+**用心设计**,应用界面上的每一个小的细节都值得细心揣摩,因为用户体验就是由这些小的细节相互协调作用而成的:
> “细节不只是细节,细节成就了设计。” —— Charles Eames
谢谢!
-**关注 UX Planet:** [*Twitter*](https://twitter.com/101babich)[*Facebook*](https://www.facebook.com/uxplanet/)
+**关注 UX Planet:** [*Twitter*](https://twitter.com/101babich) | [*Facebook*](https://www.facebook.com/uxplanet/)
---
> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From d8e709c9c21142931a6e1e91b87c1971f8d4ccaa Mon Sep 17 00:00:00 2001
From: Ruixi
Date: Sun, 9 Apr 2017 22:47:28 +0800
Subject: [PATCH 165/945] =?UTF-8?q?=E7=BF=BB=E8=AF=91=E5=88=9D=E7=A8=BF040?=
=?UTF-8?q?9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...ion-our-thoughts-on-design-are-changing.md | 66 +++++++++----------
1 file changed, 33 insertions(+), 33 deletions(-)
diff --git a/TODO/from-form-to-function-our-thoughts-on-design-are-changing.md b/TODO/from-form-to-function-our-thoughts-on-design-are-changing.md
index f2bce0a9d4d..3ed924e8fd9 100644
--- a/TODO/from-form-to-function-our-thoughts-on-design-are-changing.md
+++ b/TODO/from-form-to-function-our-thoughts-on-design-are-changing.md
@@ -1,96 +1,96 @@
> * 原文地址:[From Form to Function, Our Thoughts On Design Are Changing](https://medium.com/thinking-design/from-form-to-function-our-thoughts-on-design-are-changing-ed556d8f2b58)
> * 原文作者:[Adobe Creative Cloud](https://medium.com/@creativecloud)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[Ruixi](https://github.com/ruixi)
> * 校对者:
-# From Form to Function, Our Thoughts On Design Are Changing
+# 从形式到功能,设计思维的改变
![](https://cdn-images-1.medium.com/max/1600/1*bImmCpF6MPs9JslB21eAVQ.jpeg)
-Design is more important to business than ever before, because it’s an essential driver of user engagement. Looking at the recent evolution of user interface design might give you a sense of how user expectations have been changing and what’s coming next. But before we dive into details we need to find an answer to one important question — *what is design?*
+设计作为用户参与的基本驱动力,对商业而言比以往任何时候都更加重要。留意一下最近的用户界面设计趋势,这可能让你感受到用户的期待以往的变化,以及未来的趋势。但在我们深入讨论之前,有个问题需要回答—— **设计是什么?**
-### What is Design?
+### 设计是什么?
-Most people (even some designers) perceive design as visual elements that are added to a product after it’s done; a process that comes at the end of a product’s development, and is treated like a decoration that the designers slapped onto the real work of the engineers. While design is visual aesthetics, it is also much more. As Steve Jobs once said, “Design is not just what it looks like and feels like. *Design is how it works.*” Design is both how product ***looks*** and how it ***works*** .
+大多数人(甚至包括一些设计师)都将设计视作在产品完成之后所添加的视觉点缀,在产品开发结束之后的工艺流程,就像是设计师们强加在工程师的真实工作之上的。设计的确是视觉美学,但不仅如此。就像史蒂夫·乔布斯曾经说过的:“设计不只关乎视觉和感官。**设计关乎(产品)如何运作**。(Design is not just what it looks like and feels like. Design is how it works.)” 设计既包括产品的**视觉体验**,也包括产品的**运作原理**。
-### The Evolution of Graphical User Interface (GUI) Design
+### 图形用户界面(GUI)设计的演化
-Computers and humans don’t speak the same language. To make interaction possible, designers rely on *graphical* user interfaces. The recent evolution of GUI makes it clear that *design trends are evolving for users.* In order to prove this point let’s examine the GUI changes in the last decade.
+计算机和人类说着不同的语言。设计师们依靠**图形**用户界面使交互成为可能。最近的 GUI 设计演化清晰地表明:**设计趋势就是用户的不断变化。** 为了证明这一点,我们来看看最近十年的 GUI 设计变化。
-#### From Complexity to Simplicity
+#### 从复杂到简单
-Towards the end of the 2000’s GUI design started to change significantly due to the rise in popularity of mobile devices. This huge shift in device preference led to designers having to rethink interfaces from scratch, which in turn led to global changes in GUI design.
+由于移动设备的普及,2000 年底的 GUI 设计开始产生了明显的变化。设备配置的极大变化迫使设计师们不得不界面进行重新思考,而这又反过来引起了全球的 GUI 设计的变化。
-Looking at the history of the web, we can see that a decade ago websites were rudimentary in terms of design. But visual appearance wasn’t the only problem in this approach to design. Websites tried to provide *as many options as possible* : all the information a site contained seemed to be available; everything included on a site was ‘equally’ important. Designers thought that it would make websites more valuable to users. Unfortunately this often led to a cluttered interface. In the example below, you can see how distracting a cluttered interface can be from a usability standpoint.
+看看 Web 的历史,我们就会发现十年前的网站设计是不成熟的。但视觉并不是这种设计方式唯一的问题所在。网站试图提供**尽可能多的选择**:一个站点所包含的所有信息似乎都是可用的,页面上的每个元素都“同等”重要。(那时的)设计师们觉得这样会让网站对用户来说更有价值。不幸的是,这往往导致页面的杂乱无章。在下方的案例中,你可以看出,从可用性的角度出发,一个乱七八糟的界面能凌乱到什么程度。
![](https://cdn-images-1.medium.com/max/1600/1*MgAzj4RVV2zFTQKQCmfSRw.jpeg)
-With the rise of mobile devices designers began to realize that user attention is a precious resource and should be treated accordingly. This lead to the *highly focused* and *very prioritized interfaces*. Such interfaces have just the right amount of information available at the exactly right time users need it.
+随着移动设备的兴起,设计师们开始意识到,用户的注意力是一种需要被合理利用的珍贵资源。这就促使了**高度集中**和**按优先级处理界面**的出现。这样的界面能在用户最需要的时候提供最恰当数量的可用信息。
![](https://cdn-images-1.medium.com/max/1600/1*IMwAqnMVH2peTtdNtwPPhQ.jpeg)
-#### From Skeuomorphism to Flat Design
+#### 从拟物化到扁平化
-Do you remember when all touch screen apps looked very physical? At one point in time, almost all apps used a skeuomorphic design style which was based on symbols borrowed from the real world.
+你还记得触屏上的所有应用程序看起来都像真的一样的时候吗?那个时候,几乎所有应用程序都在使用拟物化设计风格,而这种风格又是对现实世界中符号的借用。
![](https://cdn-images-1.medium.com/max/1600/1*Mrt--PeX7t5qPmnmwzDUAg.png)
-Skeuomorphism wasn’t just a pure design trend, it played an important role in usability. When touch screen devices were fairly new to many users, designers had to make sure users would understand how apps worked. Skeuomorphic designs help users understand how to use a new interface works by making the design familiar. That’s why the iOS Newsstand app in the example above looks like a physical bookshelf. As users progressively became more familiar with touch screens, such design metaphors weren’t needed and this style went away.
+拟物化并不是一种纯粹的设计趋势,它在可用性方面也扮演了重要的角色。当很多用户都对触屏设备知之甚少的时候,设计师们必须确保用户能够明白这些应用程序是做什么的。拟物化设计通过让设计更加贴近(现实生活)来帮助人们理解新页面的运作方式。这就是上方的 iOS 报刊亭应用看上去就像是一个真实的书架(的原因)。随着用户渐渐的熟悉了触屏,这种设计隐喻显得鸡肋,而这种设计风格也逐渐退出了历史舞台。
![](https://cdn-images-1.medium.com/max/1600/1*TxE-vVFUv_61nYtdml78Kw.jpeg)
-With the quick adoption of new technologies, there was a movement towards a pure digital look, called flat design. This new style relies mainly on flat textures and icons, typography, spacing, and color to bring order to the digital canvas.
+随着新技术的使用,一种纯粹的数字化外观出现了:扁平化。这种新的设计风格主要依靠平面质感、图标、排版、间距和色彩来营造数字界面的秩序。
-#### From a Single Channel to Seamless Experience Across All Channels
+#### 从单一终端到跨终端
-Ten years ago a major design challenge was to make sure designs worked in every browser. Today, the major design challenge is to *make sure your designworks on the devices your users use*. There’s no such thing as mobile user and desktop user any more. There are users who may want to use your product in the same way, regardless of the device. That’s why *continuity across multiple devices* — creating a seamless experience across mobile, desktop, tablet and wearables — is so important. The goal is to put users at the center of your multi-channel design, [offering an omnichannel approach](https://uxmag.com/articles/5-elements-of-omni-channel-user-experiences) that lets users efficiently use product regardless of the device.
+十年前,一个主要的设计挑战就是保证设计能够在每一个浏览器下正常运行。今天,主要的设计挑战是**保证你的设计在用户所使用的设备上正常运行**。不再有移动用户和桌面端用户这种说法了。只有无论在什么设备上都可能想要照常使用你的产品的用户。这就是为什么**跨终端体验**(在移动端、桌面端、平板和可穿戴设备上的无缝体验)如此重要。目标就是将用户放在你的设计(包括多种终端)之中,[提供一个全方位的解决方案](https://uxmag.com/articles/5-elements-of-omni-channel-user-experiences) ,能够让用户无论在何种设备上都能够高效地使用产品。
![](https://cdn-images-1.medium.com/max/1600/1*j5kqBjTpLFS5e1J3wkmvgw.jpeg)
-### From Pixels to People
+### 从像素到人
-Modern apps and sites are more than just the graphical representation of a solution — they are complex systems focused on solving user problems and generating valuable outcome. Despite all advantages, these systems have a serious natural barrier — graphical user interface. No matter how good a GUI is, people still have to learn to use it. In order to solve this problem modern UX goes beyond on-screen design and into a world of UI-less interactions.
+现代应用程序和网站不只是解决方案的视觉表现,更是专注于解决用户问题并提供有价值结果的复杂系统。尽管这些系统有种种优点,但它还是有一个严重的天然障碍——图形用户界面。无论 GUI 如何优秀,人们都不得不去学习如何使用它。为了解决这个问题,现代 UX 走得更远——屏幕设计,以及融入没有用户界面的交互之中。
-#### Time-Saving Design
+#### 节省时间的设计
-Today, users expect more user-devoted, frictionless experiences from their interactions with technology. They want to use products designed to save their time. Since time-saving design is all about respecting a user’s time, it’s clear why it’s on the rise. Modern apps strive to follow this trend by:
+今天,用户期望从他们与科技的交互中的用户投入更多,以及经验的无需磨合。他们想要使用被设计来节省他们的时间的产品。节约时间的设计完全是对用户时间的慎重对待,这也清楚地解释了它们崛起的原因。当代应用程序设计力求追随如下趋势:
-- **Anticipating user needs** Take Dark Sky weather app for instance. Some users might still prefer to open a weather app to check the forecast, but the most useful thing the weather app can do is to alert user about suddenly changed weather conditions (e.g. notify user that it’ll be snowing soon).
+- **预见用户需求。** 这里用 Dark Sky weather 举个例子。一些用户可能仍然喜欢打开天气应用来查看天气预报,但天气应用最有用的举动是提醒用户突然变化的天气状况(比如通知用户很快就要下雪了)。
![](https://cdn-images-1.medium.com/max/1600/1*79Wbi92BeDyVaDPEEQjMLQ.jpeg)
-- **Interpreting user actions and goals.** When you open the Uber app for Apple Watch, it goes straight to a screen showing how long it’ll be until a car can come get you — no pulling out your phone to drop pins required.
+- **理解用户行为和目标。** 当你打开 Apple Watch 的优步应用时,它会直接在屏幕上显示车辆能够到达的时间——不需要拿出手机选择定位。
![](https://cdn-images-1.medium.com/max/1600/0*0ouzEuTHaORKAryP.jpg)
-#### Self-learning systems (SLS)
+#### 自学习系统(Self-learning systems ,SLS)
-Self learning systems (SLS) powered software anticipates tasks that need to be done and simply auto-completes them for the user, or at least gets the user several steps closer to finishing the tasks. Software that functions more autonomously has a major benefit for the users — it requires *much less attention*. The basic building block of self-learning software is the ability for a system to learn based on experience, analyse incoming data, and take action in response to new events. The challenge with SLS is to design behaviours based on the fewest possible interactions, while focusing on people’s behavior. Why is it a challenge? Because you need to find a balance between saving your user’s time and providing just enough options so users feel that they have control over a system.
+自学习系统控制软件预见一些简单的、需要被自动代替用户完成的任务,或者至少让用户更接近任务完成的步骤。功能更为自主的软件对用户来说有个很重大的益处——不需要**太多注意力**。 自学习软件的基本构件是基于经验的系统学习能力,分析传入数据,以及对新的事件作出反应。自学习系统面临的挑战是在控制交互行为数量的基础上设计行为,关注人的行为。为什么这是个挑战?因为你需要在节约用户时间和为用户提供足够多的选项(让用户感受到自己对系统的控制)找到一个平衡点
![](https://cdn-images-1.medium.com/max/1600/1*_yiHy0NAU1xANCf6HYxMDA.jpeg)
-[Nest](https://nest.com/) is great example of SLS. It’s a semi-intelligent thermostat that can program itself around user’s life. Each time a user changes the settings, Nest remembers temperature adjustments, and after a few days users will be adjusting Nest less, because it pulls all it has learned into a schedule for the home. Yes, Nest has a lot of downsides (the most critical one is that [the system often lives its own way](https://www.nngroup.com/articles/emotional-design-fail/)), but still it’s a great example of the next wave of product. Looking ahead, self-learning software will be the one thing that distinguishes legacy apps from modern ones.
+[Nest](https://nest.com/) 就是个很好的自学习系统的例子。它是一个可以围绕用户的生活来制定计划的半智能恒温器。每当用户更改系统设置的时候,Nest 都会记得调整温度,并且在几天之后用户对 Nest 的调整会减少,因为它把它学到的一切都变成家庭的时间表。是的,Nest 有很多缺点(最严重的是[系统总是采用自己的方式](https://www.nngroup.com/articles/emotional-design-fail/)),但它依然是下一代产品的绝好示例。展望未来,自学习软件将会是区别传统应用和现代应用的一个重要因素。
-#### Conversational Interfaces
+#### 会话界面
-With the advent of iPhone Messages, Slack or WhatsApp, the way we exchange information changed irreversibly. Text messages have become an extremely natural way of communicating.
+随着 iPhone 消息, Slack 以及 WhatsApp 的出现,我们交换信息的方式被完全改变。 短信已经成为一种极其自然的交流方式。
![](https://cdn-images-1.medium.com/max/1600/1*ZMmEYOSW_mZttqHXIEqDpw.png)
-This trend led to the popularization of conversational interfaces. Essentially, a conversational interface is any user interface that mimics chatting with a real human. “Chatbot” is one of the hottest terms in our industry right now. More and more apps are leaving behind the GUI in favor of personal chat. Why? Because conversation feels natural for us and this property makes the use of chatbots much more intuitive than tapping on a bunch of buttons in traditional user interfaces. Another benefit of conversational interfaces is in detalization: GUI can only have a finite amount of options in practice, but speaking to a chatbot can in theory (if designed well) allow open ended discovery and interaction.
+这一趋势导致了会话界面的普及。从本质上讲,会话界面就是是模仿真实人类聊天的用户界面。“聊天机器人(Chatbot)”是目前我们行业最热门的术语之一。越来越多的应用程序的 GUI 支持个人聊天。为什么?因为交谈对我们来说感觉很自然,这种特性使聊天机器人的使用比在传统用户界面中敲击一堆按钮更加直观。另一个会话界面的优势在于细节程度(这个词我不太确定——译者注):GUI 在实际中只有有限的选项,但和聊天机器人交谈在理论上(如果设计得好)允许开放式的发现和交互。
![](https://cdn-images-1.medium.com/max/1600/1*odt0dGAYbEau2LXZo8EF3A.gif)
-Last but not least, teens and millennials — who represent the bulk of tomorrow’s market — spend more time on messaging apps than on any other apps or sites, creating a huge opportunity for businesses who want to reach this audience.
+最后但同样重要,青少年和千禧一代——未来市场的主要代表——在短信应用程序中花的时间比任何其他应用程序或网站都要多,为想要接触到这个群体的(人/企业)创造了一个巨大的商业机会。
-But chatbots aren’t a final step of conversational interface evolution. Voice interfaces will be a natural next step for chatbots. In the not too distant future, voice interaction will make up a large part of how we interact with the technology around us. The experience of using voice commands to control computers has already been transformed by a new generation of voice-interaction systems such as Apple’s Siri, Google Now and Amazon Echo. The latter doesn’t use a traditional GUI as a means of interaction. But the biggest challenge is to understand how people will interact with voice interfaces. This requires a better understanding of humans — not only the topics they are interested in talking about, but *how* they are going to talk about the topics.
+但聊天机器人不是会话界面进化的最终形态。语音界面将会是聊天机器人的下一步。在不远的将来,语音交互将会在我们与技术的互动中占据很大部分。使用语音命令来控制计算机已经由新一代语音交互软件,比如 Siri,Google Now 和 Amazon Echo 实现了。它们不使用传统的 GUI 作为交互手段。但最大的挑战是了解人们将如何与语音界面进行交互。这需要更好地了解人类——不仅是他们感兴趣的话题,还包括他们将**如何**谈论这些话题。
![](https://cdn-images-1.medium.com/max/1600/1*coZE_xgldZTEQyI7SCdkvg.jpeg)
-### Conclusion
+### 结语
-With technology continuously evolving, we are on a path that could make interaction with digital services more intuitive, more accessible, and more efficient. Next generation platforms continue to develop much more like human-to-human conversation. The interface of the future might not always be made of pixels.
+随着技术的不断发展,我们正在使互动与数字服务更直观,更便捷,更高效的道路上不断前行。下一代平台继续发展得更接近人与人之间的对话。未来的界面不一定是由像素组成的。
---
From 0075d9a6d2d3a1f1cfad38b54f0f4f7a51015e3e Mon Sep 17 00:00:00 2001
From: ace
Date: Mon, 10 Apr 2017 10:06:48 +0800
Subject: [PATCH 166/945] =?UTF-8?q?update:=20=E4=B8=89=E6=A0=A1?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/react-is-slow-react-is-fast.md | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/TODO/react-is-slow-react-is-fast.md b/TODO/react-is-slow-react-is-fast.md
index f2c44574c00..2188b60bd4d 100644
--- a/TODO/react-is-slow-react-is-fast.md
+++ b/TODO/react-is-slow-react-is-fast.md
@@ -171,7 +171,7 @@ class DatagridBody extends Component {
export default DatagridBody;
```
-**小提示**:相比手工实现 `shouldComponentUpdate()` 方法,我可以继承 React 的 `PureComponent` 而不是 `Component`。这个组件会用严格对等(`===`)对比所有的 props,并且仅当 **任一** props 变更时重绘。但是我知道在例子的上下文中 `resource` 和 `children` 不不会变更,所以无需检查他们的对等性。
+**小提示**:相比手工实现 `shouldComponentUpdate()` 方法,我可以继承 React 的 `PureComponent` 而不是 `Component`。这个组件会用严格对等(`===`)对比所有的 props,并且仅当 **任一** props 变更时重绘。但是我知道在例子的上下文中 `resource` 和 `children` 不会变更,所以无需检查他们的对等性。
有了这一优化,点击表头后,`` 组件的重绘会跳过表体及其全部 231 个组件。这会将 500ms 的更新时间减少到 60ms。网络性能提高超过 400ms!
@@ -488,7 +488,7 @@ export default onlyUpdateForKeys(['basePath', 'refresh'])(Toolbar);
## 结论
-还有许多可以使 React 应用更快的方法(使用 keys,懒加载重路由,`react-addons-perf` 包,使用 ServiceWorkers 缓存应用状态,使用同构等等),但正确实现 `shouldComponentUpdate` 是第一步 - 也是最有用的。
+还有许多可以使 React 应用更快的方法(使用 keys、懒加载重路由、`react-addons-perf` 包、使用 ServiceWorkers 缓存应用状态、使用同构等等),但正确实现 `shouldComponentUpdate` 是第一步 - 也是最有用的。
React 默认是不快的,但是无论是什么规模的应用,它都提供了许多工具来加速。这也许是违反直觉的,尤其自从许多框架提供了 React 的替代品,它们声称比 React 快 n 倍。但 React 把开发者的体验放在了性能之前。这也是为什么用 React 开发大型应用是个愉快的体验,没有惊吓,只有不变的实现速度。
From 41b103a78d20e01e81aa9013b816cab4180785c0 Mon Sep 17 00:00:00 2001
From: lsvih
Date: Mon, 10 Apr 2017 13:44:43 +0800
Subject: [PATCH 167/945] =?UTF-8?q?=E4=B8=80=E6=A0=A1=E4=BF=AE=E6=94=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/webpack-and-rollup-the-same-but-different.md | 14 +++++++-------
1 file changed, 7 insertions(+), 7 deletions(-)
diff --git a/TODO/webpack-and-rollup-the-same-but-different.md b/TODO/webpack-and-rollup-the-same-but-different.md
index 4e286f35689..b80a7886547 100644
--- a/TODO/webpack-and-rollup-the-same-but-different.md
+++ b/TODO/webpack-and-rollup-the-same-but-different.md
@@ -8,7 +8,7 @@
![](https://cdn-images-1.medium.com/max/1000/1*rtjClMZ8sq3cLFT9Aq8Xyg.png)
-本周,Facebook 将一个[非常大的 pull request](https://github.com/facebook/react/pull/9327) 合并到了 React 主分支。这个 PR 将 React 以前使用的构建工具替换成了 [Rollup](https://rollupjs.org/)。这让许多人感到不解,纷纷在推特上提问:“为什么你们选择 Rollup 而不选择 Webpack 呢?”[1](https://twitter.com/stanlemon/status/849366789825994752)[2](https://twitter.com/MrMohtas/status/849362334988595201)[3](https://twitter.com/kyleholzinger/status/849683292760797184)
+本周,Facebook 将一个[非常大的 pull request](https://github.com/facebook/react/pull/9327) 合并到了 React 主分支。这个 PR 将 React 当前使用的构建工具替换成了 [Rollup](https://rollupjs.org/)。这让许多人感到不解,纷纷在推特上提问:“为什么你们选择 Rollup 而不选择 Webpack 呢?”[1](https://twitter.com/stanlemon/status/849366789825994752)[2](https://twitter.com/MrMohtas/status/849362334988595201)[3](https://twitter.com/kyleholzinger/status/849683292760797184)
有人问这个问题是很正常的。[Webpack](https://webpack.js.org/) 是现在 JavaScript 社区中最伟大的成功传奇之一,它有着数百万/月的下载量,驱动了成千上万的网站与应用。它有着巨大的生态系统、众多的贡献者,并且它与一般的社区开源项目不同——它有着[意义非凡的经济支持](https://opencollective.com/webpack)。
@@ -16,9 +16,9 @@
### 这两个打包工具的优缺点 ###
-Webpack 由 [Tobias Koppers](https://medium.com/@sokra) 在 2012 年创建,用于解决当时的工具不能处理的问题:构建复杂的单页应用(SPA)。它的两个特点改变了一切:
+Webpack 由 [Tobias Koppers](https://medium.com/@sokra) 在 2012 年创建,用于解决当时的工具不能处理的问题:构建复杂的单页应用(SPA)。尤其是它的两个特点改变了一切:
-1. **代码分割**可以将你的 app 分割成许多个容易管理的分块,这些分块能够在用户使用你的 app 时按需加载。这意味着你的网站可以比那些没有使用此技术的网站要快上很多。因为访问那些网站必须要等待整个应用都被下载并解析完成。当然,你**也可以**自己手动去进行代码分割,但是……总之,祝你好运。
+1. **代码分割**可以将你的 app 分割成许多个容易管理的分块,这些分块能够在用户使用你的 app 时按需加载。这意味着你的用户可以有更快的交互体验。因为访问那些网站必须要等待整个应用都被下载并解析完成。当然,你**也可以**自己手动去进行代码分割,但是……总之,祝你好运。
2. **静态资源**的导入:图片、CSS 等静态资源可以直接导入到你的 app 中,就和其它的模块、节点一样能够进行依赖管理。因此,我们再也不用小心翼翼地将各个静态文件放在特定的文件夹中,然后再去用脚本给文件 URL 加上哈希串了。Webpack 已经帮你完成了这一切。
而 Rollup 的开发理念则不同:它利用 ES2015 模块的巧妙设计,尽可能高效地构建精简且易分发的 JavaScript 库。而其它的模块打包器(包括 Webpack在内)都是通过将模块分别封装进函数中,然将这些函数通过能在浏览器中实现的 `require` 方法打包,最后依次处理这些函数。在你需要实现按需加载的时候,这种做法非常的方便,但是这样做引入了很多无关代码,比较浪费资源。当[你有很多模块要打包的时候,这种情况会变得更糟糕](https://nolanlawson.com/2016/08/15/the-cost-of-small-modules/)。
@@ -29,21 +29,21 @@ ES2015 模块则启用了一种不同的实现方法,Rollup 用的也就是这
### 那么我到底应该选用哪一个呢? ###
-回答这个问题之前,我们已经了解了这两个工具各自的优缺点,它们同时存在并相互支持的原因正是因为它们解决的问题不同。那么,现在这个问题的答案简单来说就是:
+到目前为止,我们已经清晰地了解了这两个工具共存并且相互支撑的原因 — 它们应用于不同的场景。那么,现在这个问题的答案简单来说就是:
> 在开发应用时使用 Webpack,开发库时使用 Rollup
当然这不是什么严格的规定——有很多的网站和 app 一样是使用 Rollup 构建的,同时也有很多的库使用 Webpack。不过,这是个很值得参考的经验之谈。
-如果你需要进行代码分割,或者你有很多的静态资源,再或者你做的东西深度依赖 CommonJS,毫无疑问 Webpack 是你的最佳选择。如果你的代码基于 ES2015 模块编写,并且你做的东西是准备给他人使用的,你应该更适合用 Rollup。
+如果你需要进行代码分割,或者你有很多的静态资源,再或者你做的东西深度依赖 CommonJS,毫无疑问 Webpack 是你的最佳选择。如果你的代码基于 ES2015 模块编写,并且你做的东西是准备给他人使用的,你或许可以考虑使用 Rollup。
### 对于包作者的建议:请使用 `pkg.module`! ###
-在很长一段时间里,使用 JavaScript 库是一件相当有风险的事,因为这意味着你必须和库的作者在模块系统上的意见保持一致。如果你使用 Browserify 而他更喜欢 AMD,你就不得不在 build 之前先强行将两者粘起来。[通用模块定义(UMD)](https://github.com/umdjs/umd)格式对这个问题进行了 *部分* 的修复,但是它没有强制要求所有的人都用它,最后反而会让你更加不知所措。
+在很长一段时间里,使用 JavaScript 库是一件有点风险的事,因为这意味着你必须和库的作者在模块系统上的意见保持一致。如果你使用 Browserify 而他更喜欢 AMD,你就不得不在 build 之前先强行将两者粘起来。[通用模块定义(UMD)](https://github.com/umdjs/umd)格式对这个问题进行了 **部分** 的修复,但是它没有强制要求在任何场景下都使用它,因此你无法预料你将会遇到什么坑。
ES2015 改变了这一切,因为 `import` 与 `export` 就是语言规范本身的一部分。在未来,不再会有现在这种模棱两可的情况,所有东西都将更加无缝地配合工作。不幸的是,由于大多数浏览器和 Node 还不支持 `import` 和 `export`,我们仍然需要依靠 UMD 规范(如果你只写 Node 的话也可以用 CommonJS)。
-现在给你的库的 package.json 文件增加一个 `"module": "dist/my-library.es.js"` 入口,可以让你的库同时支持 UMD 与 ES2015。**这很重要,因为 Webpack 和 Rollup 都使用了 ** `pkg.module` ** 来尽可能的生成效率更高的代码**——在一些情况下,它们都能使用 [tree-shake](https://webpack.js.org/guides/tree-shaking/) 来精简掉你的库中未使用的部分。
+现在给你的库的 package.json 文件增加一个 `"module": "dist/my-library.es.js"` 入口,可以让你的库同时支持 UMD 与 ES2015。**这很重要,因为 Webpack 和 Rollup 都使用了 `pkg.module` 来尽可能的生成效率更高的代码**——在一些情况下,它们都能使用 [tree-shake](https://webpack.js.org/guides/tree-shaking/) 来精简掉你的库中未使用的部分。
*了解更多有关 `pkg.module` 的内容请访问 [Rollup wiki](https://github.com/rollup/rollup/wiki/pkg.module) 。*
From b4d215686cd9c950fd952a64b27468bc6a75d4b8 Mon Sep 17 00:00:00 2001
From: zhouzihanntu
Date: Mon, 10 Apr 2017 15:09:31 +0800
Subject: [PATCH 168/945] =?UTF-8?q?=E6=A0=B9=E6=8D=AE=E7=AC=AC=E4=BA=8C?=
=?UTF-8?q?=E4=BD=8D=E6=A0=A1=E5=AF=B9=E8=80=85=E6=84=8F=E8=A7=81=E4=BF=AE?=
=?UTF-8?q?=E6=94=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...modules-in-node-js-everything-you-need-to-know.md | 12 ++++++------
1 file changed, 6 insertions(+), 6 deletions(-)
diff --git a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
index d48fe1071ac..afdf951dfad 100644
--- a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
+++ b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
@@ -15,9 +15,9 @@ Node 提供了两个核心模块来管理模块依赖:
- `require` 模块在全局范围内可用,不需要写 `require('require')`.
- `module` 模块同样在全局范围内可用,不需要写 `require('module')`.
-你可以将 `require` 模块理解为命令,将 `module` 模块理解为所有必要模块的组织者。
+你可以将 `require` 模块理解为命令,将 `module` 模块理解为所有引入模块的组织者。
-在 Node 中引入一个模块其实并没有想象中的那么复杂。
+在 Node 中引入一个模块其实并不是个多么复杂的概念。
```
const config = require('/path/to/file');
@@ -29,7 +29,7 @@ const config = require('/path/to/file');
- **加载**:确定文件内容的类型。
- **打包**:为文件划分私有作用域,这样 `require` 和 `module` 两个对象对于我们要引入的每个模块来说就都是本地的。
- **评估**:最后由虚拟机对加载得到的代码做评估。
-- **缓存**:当再次引用该文件时,无需再重复以上所有步骤。
+- **缓存**:当再次引用该文件时,无需再重复以上步骤。
在本文中,我将尝试举例说明这些不同阶段的工作原理,以及它们是如何影响我们在 Node 中编写模块的方式的。
@@ -62,7 +62,7 @@ Module {
Node 模块与文件系统中的文件有着一对一的关系。我们通过加载模块对应的文件内容到内存中来实现模块引用。
-然而,由于 Node 允许使用许多方式引入文件(例如,使用相对路径或预先配置的路径),我们需要在将文件的内容加载到内存前找到该文件的绝对位置。
+然而,由于 Node 允许使用多种方式引入文件(例如,使用相对路径或预先配置的路径),我们需要在将文件的内容加载到内存前找到该文件的绝对位置。
例如,我们不声明路径,直接引入一个 `'find-me'` 模块时:
@@ -389,7 +389,7 @@ fs.readFile('/etc/passwd', (err, data) => {
我们现在来回答关于 Node 中循环依赖的重要问题:当我们在模块1中引用模块2,在模块2中引用模块1时会发生什么?
-为了找到答案,我们在 `lib/` 下创建 `module1.js` 和 `module.js` 两个文件并让它们互相引用:
+为了找到答案,我们在 `lib/` 下创建 `module1.js` 和 `module2.js` 两个文件并让它们互相引用:
```
// lib/module1.js
@@ -643,7 +643,7 @@ if (require.main === module) {
}
```
-所以我们可以使用这个条件来满足以上的使用需求,通过不同的方式调用 printInFrame 函数。
+所以我们可以使用该条件判断来满足上述使用需求,通过不同的方式调用 printInFrame 函数。
```
// 在 print-in-frame.js 中
From 63a5a48054b1f5fcaf5cce5c283d554209bd2b19 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Mon, 10 Apr 2017 15:15:42 +0800
Subject: [PATCH 169/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E6=8E=88=E6=9D=83?=
=?UTF-8?q?=E4=BF=A1=E6=81=AF?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...bpack-bits-getting-the-most-out-of-the-commonschunkplugin.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md b/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
index cd003ab1e9f..21e2cff04aa 100644
--- a/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
+++ b/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
@@ -1,5 +1,5 @@
> * 原文地址:[webpack bits: Getting the most out of the CommonsChunkPlugin()](https://medium.com/webpack/webpack-bits-getting-the-most-out-of-the-commonschunkplugin-ab389e5f318#.hn8v7ul1f)
-> * 原文作者:[Sean T. Larkin](https://medium.com/@TheLarkInn?source=post_header_lockup)
+> * 原文作者:本文已获原作者 [Sean T. Larkin](https://medium.com/@TheLarkInn) 授权
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[reid3290](https://github.com/reid3290)
> * 校对者:
From 2421f9334d1b28b4b0232bca842b3a35542d943b Mon Sep 17 00:00:00 2001
From: zhouzihanntu
Date: Mon, 10 Apr 2017 15:37:42 +0800
Subject: [PATCH 170/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E6=A0=A1=E5=AF=B9?=
=?UTF-8?q?=E8=80=85=E4=BF=A1=E6=81=AF?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../requiring-modules-in-node-js-everything-you-need-to-know.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
index afdf951dfad..d160796cd96 100644
--- a/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
+++ b/TODO/requiring-modules-in-node-js-everything-you-need-to-know.md
@@ -2,7 +2,7 @@
> * 原文作者:[Samer Buna](https://medium.freecodecamp.com/@samerbuna?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[zhouzihanntu](https://github.com/zhouzihanntu)
-> * 校对者:[lsvih](https://github.com/lsvih)
+> * 校对者:[lsvih](https://github.com/lsvih), [reid3290](https://github.com/reid3290)
# 关于在 Node.js 中引用模块,知道这些就够了 #
From 9fb579dbd77e063247690279625b8b49a0b3367a Mon Sep 17 00:00:00 2001
From: DeepMissea <398752853@qq.com>
Date: Mon, 10 Apr 2017 15:53:10 +0700
Subject: [PATCH 171/945] =?UTF-8?q?=E6=A0=B9=E6=8D=AE=E4=B8=A4=E4=BD=8D?=
=?UTF-8?q?=E6=A0=A1=E5=AF=B9=E8=80=85=E7=9A=84=E6=84=8F=E8=A7=81=E4=BF=AE?=
=?UTF-8?q?=E6=94=B9=E3=80=82?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/mvvmc-with-swift.md | 38 +++++++++++++++++++-------------------
1 file changed, 19 insertions(+), 19 deletions(-)
diff --git a/TODO/mvvmc-with-swift.md b/TODO/mvvmc-with-swift.md
index 0e68011078c..01134ab233a 100644
--- a/TODO/mvvmc-with-swift.md
+++ b/TODO/mvvmc-with-swift.md
@@ -12,18 +12,18 @@
# 简介
-现今,iOS 开发者面临的最大挑战是构建一个健壮应用程序,它必须易于维护、测试和扩展。
+现今,iOS 开发者面临的最大挑战是构建一个健壮的应用程序,它必须易于维护、测试和扩展。
在这篇文章里,你会学到一种可靠的方法来达到目的。
-首先,需要简介一下你即将学习的内容:
+首先,简要介绍下你即将学习的内容:
**架构模式**.
# 架构模式
## 它是什么
-> 架构模式是给定上下文中软件体系结构中常见的,可重用的解决方案。架构与软件设计模式相似,但作用范围更广。架构解决了软件工程中的各种问题,如计算机硬件性能限制,高可用性和最小化业务风险。一些架构模式已经在软件框架内实现。
+> 架构模式是给定上下文中软件体系结构中常见的,可重用的解决方案。架构与软件设计模式相似,但涉及的范围更广。架构解决了软件工程中的各种问题,如计算机硬件性能限制,高可用性和最小化业务风险。一些架构模式已经在软件框架内实现。
摘自 [Wikipedia](https://en.wikipedia.org/wiki/Architectural_pattern)。
@@ -31,31 +31,31 @@
## 主要的模式
-在项目中,有几种可用的架构模式,并且你可以在项目中使用多个,因为每个模式都能高好的适应特定的场景。
+在项目中,有几种可用的架构模式,并且你可以在项目中使用多个,因为每个模式都能更好地适应特定的场景。
-当你阅读关于这几种模式时,主要会遇到这种:
+当你阅读这几种模式时,主要会遇到:
### [Model-View-Controller](https://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93controller)
![](https://marcosantadev.com/wp-content/uploads/mvc_2.jpg)
-这是最常见的,也许在你的第一个 iOS 应用中已经使用过。不幸地是,这也是最糟糕的模式,因为 `Controller` 不得不管理每一个依赖(API、数据库等等),包括你应用的业务逻辑,而且与 `UIKit` 的耦合度很高,这意味着测试异常的艰难。
+这是最常见的,也许在你的第一个 iOS 应用中已经使用过。不幸地是,这也是最糟糕的模式,因为 `Controller` 不得不管理每一个依赖(API、数据库等等),包括你应用的业务逻辑,而且与 `UIKit` 的耦合度很高,这意味着很难去测试。
-你通常会避免这种模式,用下面的某种来代替它。
+你应该避免这种模式,用下面的某种来代替它。
### [Model-View-Presenter](https://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93presenter)
![](https://marcosantadev.com/wp-content/uploads/mvp.jpg)
-这是第一个 MVC 的替代方案之一,一次对 `Controller` 和 `View` 之间解耦的很好的尝试。
+这是第一个 MVC 模式的备选方案之一,一次对 `Controller` 和 `View` 之间解耦的很好的尝试。
-在 MVP 中,你有一层叫做 `Presenter` 的新结构来处理业务逻辑。而 `View` —— 你的 `UIViewController` 以及任何 `UIKit` 组件,都是一个笨的对象,他们只通过 `Presenter` 更新,并在触发 UI 事件的时候,负责通知 `Presenter`。由于 `Presenter` 没有任何 `UIKit` 的引用,所以非常容易测试。
+在 MVP 中,你有一层叫做 `Presenter` 的新结构来处理业务逻辑。而 `View` —— 你的 `UIViewController` 以及任何 `UIKit` 组件,都是一个笨的对象,他们只通过 `Presenter` 更新,并在 UI 事件被触发的时候,负责通知 `Presenter`。由于 `Presenter` 没有任何 `UIKit` 的引用,所以非常容易测试。
### [Viper](https://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93presenter)
[![](https://www.objc.io/images/issue-13/2014-06-07-viper-intro-0a53d9f8.jpg)](https://www.objc.io/issues/13-architecture/viper/)
-这是 [Bob 叔叔清晰架构](https://8thlight.com/blog/uncle-bob/2012/08/13/the-clean-architecture.html)的代表。
+这是 [Bob 叔叔的清晰架构](https://8thlight.com/blog/uncle-bob/2012/08/13/the-clean-architecture.html)的代表。
这种模式的强大之处在于,它合理分配了不同层次之间的职责。通过这种方式,你的每个层次做的的事变得很少,易于测试,并且具备单一职责。这种模式的问题是,在大多数场合里,它过于复杂。你需要管理很多层,这会让你感到混乱,难于管理。
@@ -65,7 +65,7 @@
![](https://marcosantadev.com/wp-content/uploads/mvvm.jpg)
-最后但也是最重要的,MVVM 是一个类似于 MVP 的框架,因为层次看几乎相同。你可以认为 MVVM 是 MVP 版本的一个进化,而这得益于 UI 绑定。
+最后但也是最重要的,MVVM 是一个类似于 MVP 的框架,因为层级结构几乎相同。你可以认为 MVVM 是 MVP 版本的一个进化,而这得益于 UI 绑定。
UI 绑定是在 `View` 和 `ViewModel` 之间建立一座单向或双向的桥梁,并且两者之间以一种非常透明地方式进行沟通。
@@ -73,7 +73,7 @@ UI 绑定是在 `View` 和 `ViewModel` 之间建立一座单向或双向的桥
在 Swift 里有多种方式实现 UI 绑定:
-#### RxSwift (or ReactiveCocoa)
+#### RxSwift (或 ReactiveCocoa)
[RxSwift](https://github.com/ReactiveX/RxSwift) 是 [ReactiveX](http://reactivex.io/) 家族的一个 Swift 版本的实现。一旦你掌握了它,你就能很轻松地切换到 RxJava、RxJavascript 等等。
@@ -97,7 +97,7 @@ class ViewController: UIViewController {
我不会解释如何彻底地使用 RxSwift,因为这超出本文的目标,它自己会有文章来解释。
-FRP 让你学习到了一种新的方式来开发,你可能对它或爱或恨。如果你没用过 FRP 开发,那你不得不花费几小时的时间来使用,并理解如何正确的使用,因为它是一个完全不同的编程概念。
+FRP 让你学习到了一种新的方式来开发,你可能对它或爱或恨。如果你没用过 FRP 开发,那你需要花费几个小时来熟悉和理解如何正确地使用它,因为它是一个完全不同的编程概念。
另一个类似于 RxSwift 的框架是 [ReactiveCocoa](https://github.com/ReactiveCocoa/ReactiveCocoa),如果你想了解他们之间主要的区别的话,你可以看看[这篇文章](https://www.raywenderlich.com/126522/reactivecocoa-vs-rxswift)。
@@ -211,7 +211,7 @@ class ViewModel {
## 抉择: MVVM-C
-在你不得不选择一个架构模式时,你需要理解哪一种更适合你的需求。在这些模式里,MVVM 是最好的选择,因为它强大的同时,也易于使用。
+在你不得不选择一个架构模式时,你需要理解哪一种更适合你的需求。在这些模式里,MVVM 是最好的选择之一,因为它强大的同时,也易于使用。
不幸地是这种模式并不完美,主要的缺陷是 MVVM 没有路由管理。
@@ -225,7 +225,7 @@ class ViewModel {
你可以在[这里](https://github.com/MarcoSantarossa/MVVM-C_with_Swift)下载项目源码。
-这些类被简化了,以便于你可以专注于 MVVM-C 是如何工作的,因此 GitHub 上的类可能会有轻微出入。
+这个例子被简化了,以便于你可以专注于 MVVM-C 是如何工作的,因此 GitHub 上的类可能会有轻微出入。
示例应用是一个普通的仪表盘应用,它从公共 API 获取数据,一旦数据准备就绪,用户就可以通过 ID 查找实体,如下面的截图:
@@ -294,7 +294,7 @@ final class DashboardContainerCoordinator: Coordinator {
}
```
-你一定能注意到在 `Coordinator` 里,一个父类 `UIViewController` 对象或者子类对象,类似于 `UINavigationController`,被注入到构造器之中。因为 `Coordinator` 有责任添加 `View` 到视图层级之中,它必须知道那个父类添加了 `View`。
+你一定能注意到在 `Coordinator` 里,一个父类 `UIViewController` 对象或者子类对象,比如 `UINavigationController`,被注入到构造器之中。因为 `Coordinator` 有责任添加 `View` 到视图层级之中,它必须知道那个父类添加了 `View`。
在上面的例子里,`DashboardContainerCoordinator` 实现了协议 `Coordinator`:
@@ -337,7 +337,7 @@ class AppDelegate: UIResponder, UIApplicationDelegate {
在 `AppDelegate` 里,我们实例化一个新的 `DashboardContainerCoordinator`,通过 `start` 方法,我们把新的视图推入 `navigationController` 里。
-**你可以看到 GitHub 项目如何注入一个 `UINavigationController` 类型的对象,并去除 `UIKit` 和 `Coordinator` 之间的耦合。**
+**你可以看到在 GitHub 上的项目是如何注入一个 `UINavigationController` 类型的对象,并去除 `UIKit` 和 `Coordinator` 之间的耦合。**
### Model
@@ -450,7 +450,7 @@ final class UsersViewModel {
MVVM-C 有很多优点,可以提高应用程序的质量。你应该注意使用哪种方式来进行 UI 绑定,因为 RxSwift 不容易掌握,而且如果你不明白你做的是什么,调试和测试有时可能会有点棘手。
-我的建议是一点点地开始使用这种架构模式,这样你可以对不同的层次得到使用,并且能保证层次之间的良好的分离,易于测试。
+我的建议是一点点地开始使用这种架构模式,这样你可以学习不同层次的使用,并且能保证层次之间的良好的分离,易于测试。
# FAQ
@@ -466,7 +466,7 @@ MVVM-C 有很多优点,可以提高应用程序的质量。你应该注意使
这取决于你要开新项目,还是要维护旧代码。在有遗留代码的项目中,你可能无法使用 RxSwift,因为你需要重构很多的类。如果你有时间和资源来做,我建议你新开一项目一点一点的做,否则还是尝试其他的方法来解决 UI 绑定的问题。
-需要考虑的一个重要事情是,RxSwift 会一直是你项目中的另一个依赖,你可能会因为 RxSwift 的突破性改动而导致浪费时间的风险,或者缺少要在边缘案例中实现功能的文档。
+需要考虑的一个重要事情是,RxSwift 最终会成为你项目中的另一个依赖,你可能会因为 RxSwift 的破坏性改动而导致浪费时间的风险,或者缺少要在边缘案例中实现功能的文档。
---
From 575abaa19e6fc5bf58452a64699a00e0d62bf6fa Mon Sep 17 00:00:00 2001
From: DeepMissea <398752853@qq.com>
Date: Mon, 10 Apr 2017 15:55:20 +0700
Subject: [PATCH 172/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E6=A0=A1=E5=AF=B9?=
=?UTF-8?q?=E8=80=85=E3=80=82?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/mvvmc-with-swift.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/mvvmc-with-swift.md b/TODO/mvvmc-with-swift.md
index 01134ab233a..eb77db7fd87 100644
--- a/TODO/mvvmc-with-swift.md
+++ b/TODO/mvvmc-with-swift.md
@@ -2,7 +2,7 @@
> * 原文作者:[Marco Santarossa](https://marcosantadev.com/)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[Deepmissea](http://deepmissea.blue)
-> * 校对者:
+> * 校对者:[atuooo](http://atuo.xyz),[1992chenlu](https://github.com/1992chenlu)
---
From 59b306ee920a7a4ffa66bb66396b88012751398f Mon Sep 17 00:00:00 2001
From: reid
Date: Mon, 10 Apr 2017 17:45:38 +0800
Subject: [PATCH 173/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E6=A0=A1=E5=AF=B9?=
=?UTF-8?q?=E8=80=85=E4=BF=A1=E6=81=AF?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...bpack-bits-getting-the-most-out-of-the-commonschunkplugin.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md b/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
index cd003ab1e9f..e9b092a47c2 100644
--- a/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
+++ b/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
@@ -2,7 +2,7 @@
> * 原文作者:[Sean T. Larkin](https://medium.com/@TheLarkInn?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[reid3290](https://github.com/reid3290)
-> * 校对者:
+> * 校对者:[avocadowang](https://github.com/avocadowang),[Aladdin-ADD](https://github.com/Aladdin-ADD)
# webpack 拾翠:充分利用 CommonsChunkPlugin() #
From b5bc496940ed9f61b97ec13d0c21fd9844045b84 Mon Sep 17 00:00:00 2001
From: Tuccuay
Date: Mon, 10 Apr 2017 22:31:50 +0800
Subject: [PATCH 174/945] quote
---
TODO/mastering-swift-essential-details-about-strings.md | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
diff --git a/TODO/mastering-swift-essential-details-about-strings.md b/TODO/mastering-swift-essential-details-about-strings.md
index 325223b5681..1abd697d093 100644
--- a/TODO/mastering-swift-essential-details-about-strings.md
+++ b/TODO/mastering-swift-essential-details-about-strings.md
@@ -541,4 +541,6 @@ print(numberOfStars) // => 2
**对于字符操作你有没有找到更舒适的方法?写下评论我们一起来讨论一些吧!**
-**P.S.** 不知道你有没有兴趣阅读我的另一篇文章:[detailed overview of array and dictionary literals in Swift](https://rainsoft.io/concise-initialization-of-collections-in-swift/)
\ No newline at end of file
+**P.S.** 不知道你有没有兴趣阅读我的另一篇文章:[detailed overview of array and dictionary literals in Swift](https://rainsoft.io/concise-initialization-of-collections-in-swift/)
+
+>[掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
\ No newline at end of file
From 6b3ffdc1a0481ea0e9aa9cde2298129ac52041cf Mon Sep 17 00:00:00 2001
From: zhuzi
Date: Mon, 10 Apr 2017 22:50:34 +0800
Subject: [PATCH 175/945] update according to proof-reading
---
TODO/beyond-browser-web-desktop-apps.md | 12 ++++++------
1 file changed, 6 insertions(+), 6 deletions(-)
diff --git a/TODO/beyond-browser-web-desktop-apps.md b/TODO/beyond-browser-web-desktop-apps.md
index 9803dd3b2c6..1899ee8f1d9 100644
--- a/TODO/beyond-browser-web-desktop-apps.md
+++ b/TODO/beyond-browser-web-desktop-apps.md
@@ -2,7 +2,7 @@
> * 原文作者:本文已获原作者 [Adam Lynch](https://www.smashingmagazine.com/author/adamlynch/) 授权
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者: [bambooom](https://github.com/bambooom)/[imink](https://github.com/imink)
-> * 校对者:[bambooom](https://github.com/bambooom)/[sunui](https://github.com/sunui)
+> * 校对者:[bambooom](https://github.com/bambooom)/[imink](https://github.com/imink)/[sunui](https://github.com/sunui)
## 超越浏览器:从 web 应用到桌面应用
@@ -12,7 +12,7 @@
别慌,深呼吸,现实情况是,作为 web 开发者,你已经拥有开发现代桌面应用所需的一切技能,得益于新的强大的 API,你甚至可以在桌面应用中发挥你最大的潜能。
-本文将会介绍使用 [NW.js](http://nwjs.io/) 和 [Electron](https://electron.atom.io/) 开发桌面应用,它们的优劣,使用同一套代码库给桌面、web,甚至更多。
+本文将会介绍使用 [NW.js](http://nwjs.io/) 和 [Electron](https://electron.atom.io/) 开发桌面应用,包括它们的优劣,以及如何使用同一套代码库来开发桌面、web 应用,甚至更多。
### 为什么?
@@ -22,7 +22,7 @@
很难总结为什么你应该考虑开发桌面应用,因为真的有很多类型的应用你可以创建。这非常取决于你想要达到什么目的,API 是否足够有利于开发,离线使用将多大程度上增强用户体验。在我的团队,这些都是毋庸置疑的,因为我们在开发一个[聊天应用程序](https://teamwork.com/chat)。另一方面来说,一个依赖于网络而没有任何与系统集成的桌面应用应该做成一个 web 应用,并且只做 web 应用。当用户并不能从桌面应用中获得比在浏览器中访问一个网址更多的价值的时候,期待用户下载你的应用(其中自带浏览器以及 Node.js)是不公平的。
-比起描述你个人应该建造的桌面应用及其原因,我更希望的是激发一个想法,或者只是激发你对这篇文章的兴趣。继续往下读来看看用 web 技术构造一个强大的桌面应用是多么简单,以及创建完成(或者在这过程中)一个 web 应用你需要承受什么。。
+比起描述你个人应该建造的桌面应用及其原因,我更希望的是激发一个想法,或者只是激发你对这篇文章的兴趣。继续往下读来看看用 web 技术构造一个强大的桌面应用是多么简单,以及在创建过程中你应该付出什么。
### NW.js
@@ -81,7 +81,7 @@ Chromium 有一个主要的后台进程,每个标签页也会有自己的进
就是这么简单。NW.js 初始化了第一个窗口,加载了你的 HTML 文件,虽然这看起来并没有什么,但接下来就是你来添加标签及样式了,就和在 web 应用中一样。
-你可以凭自己喜好去掉窗口栏,构建自己的框架模板。你可以有半透明或全透明的窗口,可以有隐藏窗口或者更多。我最近试了一下用 NW.js 复活了[回形针](http://engineroom.teamwork.com/resurrecting-clippy/) Office 助手(一般昵称 Clippy)。能看到它同时在 macOS 或者 Windows 10 中复活有种奇妙的满足感。
+你可以凭自己喜好去掉窗口栏,构建自己的框架模板。你可以有半透明或全透明的窗口,可以有隐藏窗口或者更多。我最近尝试使用 NW.js 做了[Clippy](http://engineroom.teamwork.com/resurrecting-clippy/)(Office 助手)。能在 macOS 和 Windows 10 上看到它有种奇妙的满足感。
![Screenshot of clippy.desktop on macOS](https://www.smashingmagazine.com/wp-content/uploads/2017/01/clippy-preview-opt.png)
@@ -217,11 +217,11 @@ NW.js 和 Electron 都支持很多平台,包括 Windows,Mac 和 Linux。Elec
Electron 甚至支持 ARM 版本,所以你的 app 可以在 Chromebook 或者树莓派上运行,最终,Google 可能会[逐步淘汰 Chrome 封装应用 (Packaged App)](https://blog.chromium.org/2016/08/from-chrome-apps-to-web.html),但是 NW.js 仍然支持将应用程序移植到 NW.js 应用,并且仍然可以访问相同的 Chromium API。
-虽然 32 位和 64 位的版本都支持,你也可以使用 64 位的 Mac 和 Windows 应用。但是,为了兼容,32 位和 64 位 Linux 应用程序是都需要的。
+虽然 32 位和 64 位的版本都支持,所以你完全可以使用 64 位的 Mac 和 Windows 应用。但是,为了兼容,32 位和 64 位 Linux 应用程序是都需要的。
假如 Electron 胜出,你想发行一个 Electron 应用。有一个很不错的 Node.js 包叫 [electron-packager](https://github.com/electron-userland/electron-packager) 可以帮你将 app 打包成一个 `.app` 或者 `.exe` 文件。也有其他几个类似的项目,包括交互式的一步一步告诉你该怎么做。不过,你应该用 [electron-builder](https://github.com/electron-userland/electron-builder),它以 electron-packager 为基础,添加了其他几个相关的模块,生成的是 `.dmg` 文件和 Windows 安装包,并且为你处理好了代码签名的问题。这很重要,如果没有这一步,你的应用将会被操作系统认为是不可信的,你的应用程序可能会触发防毒软件的运行,Microsoft SmartScreen 可能会尝试阻止用户启动你的应用。
-关于代码签名的令人讨厌的事情是,你必须在 Mac 上为 Mac 和 Windows 上为 Windows 签署你的应用程序。因此,如果是认真要发行桌面应用的话,就需要为每个发行版本给多种机器构建。
+关于代码签名的令人讨厌的事情是,你必须单独为某个平台签名你的应用程序,比如在 Mac 上签名 Mac 应用,在 Windows 签名 Windows 应用。因此,如果你很在乎发行桌面应用的话,就必须为每个发行版本分别构建适用于不同平台的应用(以及分别签名)。
这可能会感到不够自动化很繁琐,特别是如果你习惯于在 web 上创建。幸运的是,electron-builder 被创造出来完成这些自动化工作。我说的是持续集成工具例如 [Jenkins](https://jenkins.io/),[CodeShip](http://codeship.com/),[Travis-CI](https://travis-ci.org/),[AppVeyor](https://www.appveyor.com/)(Windows 集成)等。这些工具可以让你按一个按钮或者每次更新代码到 GitHub 时重新构建你的桌面应用。
From 0e05d37cf28db7bc0282e66922aa56d1588ce5c7 Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Mon, 10 Apr 2017 23:04:02 +0800
Subject: [PATCH 176/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20Preload=EF=BC=8CPr?=
=?UTF-8?q?efetch=20=E5=92=8C=E5=AE=83=E4=BB=AC=E5=9C=A8=20Chrome=20?=
=?UTF-8?q?=E4=B9=8B=E4=B8=AD=E7=9A=84=E4=BC=98=E5=85=88=E7=BA=A7=20?=
=?UTF-8?q?=E7=9A=84=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 3 ++-
front-end.md | 1 +
2 files changed, 3 insertions(+), 1 deletion(-)
diff --git a/README.md b/README.md
index 066b76a852c..7e89841009a 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [448](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [449](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -37,6 +37,7 @@
## 前端
+* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪](https://juejin.im/post/58ddeb1a570c3500579016ef?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
* [如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象](https://juejin.im/post/58dc9da661ff4b0061547ca0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
diff --git a/front-end.md b/front-end.md
index e38e54951be..3a2df9cf268 100644
--- a/front-end.md
+++ b/front-end.md
@@ -1,3 +1,4 @@
+* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪](https://juejin.im/post/58ddeb1a570c3500579016ef?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
* [如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象](https://juejin.im/post/58dc9da661ff4b0061547ca0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
From 41cb0f72b78d66bde60349afd8257b160f40b9c6 Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Mon, 10 Apr 2017 23:57:52 +0800
Subject: [PATCH 177/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20CSS=E5=BE=88?=
=?UTF-8?q?=E6=A3=92=EF=BC=8C=E5=8F=AA=E6=98=AF=E7=9C=9F=E7=9A=84=E5=A4=AA?=
=?UTF-8?q?=E9=9A=BE=E4=BA=86=20=E7=9A=84=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 3 ++-
front-end.md | 1 +
2 files changed, 3 insertions(+), 1 deletion(-)
diff --git a/README.md b/README.md
index 7e89841009a..c7c8629cecf 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [449](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [450](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -37,6 +37,7 @@
## 前端
+* [CSS很棒,只是真的太难了](https://juejin.im/entry/58eae24a61ff4b0061a6a102?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([ZhangFe](https://github.com/ZhangFe) 翻译)
* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪](https://juejin.im/post/58ddeb1a570c3500579016ef?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
diff --git a/front-end.md b/front-end.md
index 3a2df9cf268..52e31749ac5 100644
--- a/front-end.md
+++ b/front-end.md
@@ -1,3 +1,4 @@
+* [CSS很棒,只是真的太难了](https://juejin.im/entry/58eae24a61ff4b0061a6a102?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([ZhangFe](https://github.com/ZhangFe) 翻译)
* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪](https://juejin.im/post/58ddeb1a570c3500579016ef?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
From 9b8d36b6d710e3bc3411c12a2f9601c984112811 Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Tue, 11 Apr 2017 00:16:25 +0800
Subject: [PATCH 178/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20Android=20?=
=?UTF-8?q?=E5=A6=82=E4=BD=95=E5=AE=9E=E7=8E=B0=E6=B0=94=E6=B3=A1=E9=80=89?=
=?UTF-8?q?=E6=8B=A9=E5=8A=A8=E7=94=BB=20=E7=9A=84=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 7 +++----
android.md | 1 +
2 files changed, 4 insertions(+), 4 deletions(-)
diff --git a/README.md b/README.md
index c7c8629cecf..dae41fd13fa 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [450](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [451](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -20,10 +20,10 @@
## Android
+* [Android 如何实现气泡选择动画](https://juejin.im/post/58e5ec838d6d8100616d82e2/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([skyar2009](https://github.com/skyar2009) 翻译)
* [一个人的 Android 开发](https://juejin.im/entry/58dca515b123db00603887fd/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([BoilerYao](https://github.com/BoilerYao) 翻译)
* [我是如何做到在 5 分钟之内将应用大小减少 60% 的](https://juejin.im/post/58d9b6a1a22b9d0064719f9e/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([jifaxu](https://github.com/jifaxu) 翻译)
* [拉模式和推模式,命令式和响应式 – 响应式编程 [Android RxJava2](这到底是什么):第二部分](https://juejin.im/entry/58d78547a22b9d006465ca57/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([XHShirley](https://github.com/XHShirley) 翻译)
-* [离线支持:不再『稍后重试』](https://juejin.im/post/58d491a8128fe1006cb6e750/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([skyar2009](https://github.com/skyar2009) 翻译)
* [所有 Android 译文>>](https://github.com/xitu/gold-miner/blob/master/android.md)
@@ -41,8 +41,7 @@
* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪](https://juejin.im/post/58ddeb1a570c3500579016ef?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
-* [如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象](https://juejin.im/post/58dc9da661ff4b0061547ca0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
-* [生活在 JavaScript 之中:学习第二门语言的好处](https://juejin.im/post/58d908deac502e0058db544b/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
+* [如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象](https://juejin.im/post/58dc9da661ff4b0061547ca0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github)
* [所有前端译文>>](https://github.com/xitu/gold-miner/blob/master/front-end.md)
diff --git a/android.md b/android.md
index 2c1b495bf9d..8b02a079727 100644
--- a/android.md
+++ b/android.md
@@ -1,3 +1,4 @@
+* [Android 如何实现气泡选择动画](https://juejin.im/post/58e5ec838d6d8100616d82e2/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([skyar2009](https://github.com/skyar2009) 翻译)
* [一个人的 Android 开发](https://juejin.im/entry/58dca515b123db00603887fd/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([BoilerYao](https://github.com/BoilerYao) 翻译)
* [我是如何做到在 5 分钟之内将应用大小减少 60% 的](https://juejin.im/post/58d9b6a1a22b9d0064719f9e/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([jifaxu](https://github.com/jifaxu) 翻译)
* [拉模式和推模式,命令式和响应式 – 响应式编程 [Android RxJava2](这到底是什么):第二部分](https://juejin.im/entry/58d78547a22b9d006465ca57/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([XHShirley](https://github.com/XHShirley) 翻译)
From 771a38fa640cf24e5ad6d75ee86e182e6476175b Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Tue, 11 Apr 2017 00:46:50 +0800
Subject: [PATCH 179/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20=E5=9C=A8=20Apache?=
=?UTF-8?q?=20=E5=92=8C=20Nginx=20=E6=97=A5=E5=BF=97=E9=87=8C=E6=A3=80?=
=?UTF-8?q?=E6=B5=8B=E7=88=AC=E8=99=AB=E6=9C=BA=E5=99=A8=E4=BA=BA=20?=
=?UTF-8?q?=E7=9A=84=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 4 ++--
backend.md | 1 +
2 files changed, 3 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index dae41fd13fa..62a56a84c1f 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [451](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [452](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -56,10 +56,10 @@
## 后端
+* [在 Apache 和 Nginx 日志里检测爬虫机器人](https://juejin.im/post/58ea5758ac502e4957c78808/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([luoyaqifei](https://github.com/luoyaqifei) 翻译)
* [如何在 ChromeOS 下用 Go 搭建 Web 服务](https://juejin.im/post/58d9e1711b69e6006bc38b1a/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([xiaoyusilen](https://github.com/xiaoyusilen) 翻译)
* [在你沉迷于包的海洋之前,还是了解一下运行时 Node.js 的本身](https://juejin.im/post/58cf4a3144d90400690b7be7/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([fghpdf](https://github.com/fghpdf) 翻译)
* [Pull request review 的十大错误](https://juejin.im/post/58ce3b3e61ff4b006c988f63/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([luoyaqifei](https://github.com/luoyaqifei) 翻译)
-* [震惊!RxJava 5 个不为人知的小秘密](https://juejin.im/post/58cb833b8ac247218c2632e5/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([skyar2009](https://github.com/skyar2009) 翻译)
* [所有后端译文>>](https://github.com/xitu/gold-miner/blob/master/backend.md)
## 教程
diff --git a/backend.md b/backend.md
index 2e7926a7cba..8b96be0ba3f 100644
--- a/backend.md
+++ b/backend.md
@@ -1,3 +1,4 @@
+* [在 Apache 和 Nginx 日志里检测爬虫机器人](https://juejin.im/post/58ea5758ac502e4957c78808/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([luoyaqifei](https://github.com/luoyaqifei) 翻译)
* [如何在 ChromeOS 下用 Go 搭建 Web 服务](https://juejin.im/post/58d9e1711b69e6006bc38b1a/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([xiaoyusilen](https://github.com/xiaoyusilen) 翻译)
* [在你沉迷于包的海洋之前,还是了解一下运行时 Node.js 的本身](https://juejin.im/post/58cf4a3144d90400690b7be7/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([fghpdf](https://github.com/fghpdf) 翻译)
* [Pull request review 的十大错误](https://juejin.im/post/58ce3b3e61ff4b006c988f63/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([luoyaqifei](https://github.com/luoyaqifei) 翻译)
From 6ea9fa02939c766668d981cb3682c6915e650786 Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Tue, 11 Apr 2017 01:00:14 +0800
Subject: [PATCH 180/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20=E6=A8=A1=E5=9D=97?=
=?UTF-8?q?=E5=8C=96=20vs.=20=E5=BE=AE=E6=9C=8D=E5=8A=A1=20=E7=9A=84?=
=?UTF-8?q?=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 4 ++--
backend.md | 1 +
2 files changed, 3 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index 62a56a84c1f..2df4c508b86 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [452](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [453](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -56,10 +56,10 @@
## 后端
+* [模块化 vs. 微服务](https://juejin.im/post/58eb2627da2f60005f0b2d60/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([lsvih](https://github.com/lsvih) 翻译)
* [在 Apache 和 Nginx 日志里检测爬虫机器人](https://juejin.im/post/58ea5758ac502e4957c78808/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([luoyaqifei](https://github.com/luoyaqifei) 翻译)
* [如何在 ChromeOS 下用 Go 搭建 Web 服务](https://juejin.im/post/58d9e1711b69e6006bc38b1a/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([xiaoyusilen](https://github.com/xiaoyusilen) 翻译)
* [在你沉迷于包的海洋之前,还是了解一下运行时 Node.js 的本身](https://juejin.im/post/58cf4a3144d90400690b7be7/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([fghpdf](https://github.com/fghpdf) 翻译)
-* [Pull request review 的十大错误](https://juejin.im/post/58ce3b3e61ff4b006c988f63/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([luoyaqifei](https://github.com/luoyaqifei) 翻译)
* [所有后端译文>>](https://github.com/xitu/gold-miner/blob/master/backend.md)
## 教程
diff --git a/backend.md b/backend.md
index 8b96be0ba3f..909e6435309 100644
--- a/backend.md
+++ b/backend.md
@@ -1,3 +1,4 @@
+* [模块化 vs. 微服务](https://juejin.im/post/58eb2627da2f60005f0b2d60/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([lsvih](https://github.com/lsvih) 翻译)
* [在 Apache 和 Nginx 日志里检测爬虫机器人](https://juejin.im/post/58ea5758ac502e4957c78808/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([luoyaqifei](https://github.com/luoyaqifei) 翻译)
* [如何在 ChromeOS 下用 Go 搭建 Web 服务](https://juejin.im/post/58d9e1711b69e6006bc38b1a/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([xiaoyusilen](https://github.com/xiaoyusilen) 翻译)
* [在你沉迷于包的海洋之前,还是了解一下运行时 Node.js 的本身](https://juejin.im/post/58cf4a3144d90400690b7be7/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([fghpdf](https://github.com/fghpdf) 翻译)
From 8a4cb26b6b86066a9584e81025c2fe30f17553ef Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Tue, 11 Apr 2017 10:12:45 +0800
Subject: [PATCH 181/945] Fix typos.
---
README.md | 1 -
1 file changed, 1 deletion(-)
diff --git a/README.md b/README.md
index 2df4c508b86..d83a7c5e1b9 100644
--- a/README.md
+++ b/README.md
@@ -41,7 +41,6 @@
* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪](https://juejin.im/post/58ddeb1a570c3500579016ef?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
-* [如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象](https://juejin.im/post/58dc9da661ff4b0061547ca0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github)
* [所有前端译文>>](https://github.com/xitu/gold-miner/blob/master/front-end.md)
From d57c25645060c19dc1e98f953c2ed33b8656db70 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Tue, 11 Apr 2017 10:13:31 +0800
Subject: [PATCH 182/945] =?UTF-8?q?=E6=9B=B4=E6=96=B0=E8=AF=91=E8=80=85?=
=?UTF-8?q?=E4=BA=BA=E6=95=B0?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/README.md b/README.md
index d83a7c5e1b9..445c5394d6d 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [453](#近期文章列表) 篇文章,共有 [270](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [453](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
From f995c21c858bb1e6a988995537f3583cf6db71ce Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Tue, 11 Apr 2017 10:46:48 +0800
Subject: [PATCH 183/945] Update preload-prefetch-and-priorities-in-chrome.md
---
TODO/preload-prefetch-and-priorities-in-chrome.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/preload-prefetch-and-priorities-in-chrome.md b/TODO/preload-prefetch-and-priorities-in-chrome.md
index 5f13ea73d2b..6fc8912e434 100644
--- a/TODO/preload-prefetch-and-priorities-in-chrome.md
+++ b/TODO/preload-prefetch-and-priorities-in-chrome.md
@@ -1,5 +1,5 @@
> * 原文地址:[Preload, Prefetch And Priorities in Chrome](https://medium.com/reloading/preload-prefetch-and-priorities-in-chrome-776165961bbf)
-> * 原文作者:[Addy Osmani](https://medium.com/@addyosmani?source=post_header_lockup)
+> * 原文作者:本文已获原作者 [Addy Osmani](https://medium.com/@addyosmani) 授权
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[gy134340](https://github.com/gy134340)
> * 校对者:[IridescentMia](https://github.com/IridescentMia),[vuuihc](https://github.com/vuuihc)
From 48e36d994a282a1d23321bd0a5564386a6871889 Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Tue, 11 Apr 2017 22:55:28 +0800
Subject: [PATCH 184/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E4=BD=9C=E8=80=85?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/README.md b/README.md
index 2df4c508b86..b00acba8e90 100644
--- a/README.md
+++ b/README.md
@@ -41,7 +41,7 @@
* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪](https://juejin.im/post/58ddeb1a570c3500579016ef?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
-* [如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象](https://juejin.im/post/58dc9da661ff4b0061547ca0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github)
+* [如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象](https://juejin.im/post/58dc9da661ff4b0061547ca0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github)([IridescentMia](https://github.com/IridescentMia) 翻译)
* [所有前端译文>>](https://github.com/xitu/gold-miner/blob/master/front-end.md)
From 422f548bef95305ec1f87d3f48958113f675ff5e Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Tue, 11 Apr 2017 23:04:46 +0800
Subject: [PATCH 185/945] =?UTF-8?q?=E4=BF=AE=E6=94=B9=E9=94=99=E8=AF=AF?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 3 ---
1 file changed, 3 deletions(-)
diff --git a/README.md b/README.md
index b494634b4e5..54648b75c5d 100644
--- a/README.md
+++ b/README.md
@@ -41,10 +41,7 @@
* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪](https://juejin.im/post/58ddeb1a570c3500579016ef?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
-<<<<<<< HEAD
* [如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象](https://juejin.im/post/58dc9da661ff4b0061547ca0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github)([IridescentMia](https://github.com/IridescentMia) 翻译)
-=======
->>>>>>> 2ffa1a092c355fc606b7cfc27f14af24d385d679
* [所有前端译文>>](https://github.com/xitu/gold-miner/blob/master/front-end.md)
From 7d63b83df1a22c2082cab44926079a3f08063d2a Mon Sep 17 00:00:00 2001
From: Tuccuay
Date: Tue, 11 Apr 2017 23:47:00 +0800
Subject: [PATCH 186/945] revision for @atuooo
---
...g-swift-essential-details-about-strings.md | 34 +++++++++----------
1 file changed, 17 insertions(+), 17 deletions(-)
diff --git a/TODO/mastering-swift-essential-details-about-strings.md b/TODO/mastering-swift-essential-details-about-strings.md
index 1abd697d093..14cd30aa1a8 100644
--- a/TODO/mastering-swift-essential-details-about-strings.md
+++ b/TODO/mastering-swift-essential-details-about-strings.md
@@ -2,7 +2,7 @@
* 原文作者:[Dmitri Pavlutin](https://rainsoft.io/author/dmitri-pavlutin/)
* 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
* 译者:[Tuccuay](https://www.tuccuay.com)
-* 校对者:
+* 校对者:[oOatuo](https://github.com/atuooo)
# 掌握 Swift 的字符串细节
@@ -11,7 +11,7 @@ String 类型在任何编程语言中都是一个重要的组成部分。而用
为了触及更多的用户,iOS 应用必须国际化以支持大量现代语言。Unicode 标准解决了这个问题,不过这也给我们使用 string 类型带来了额外的挑战性。
-从一方面来说,编程语言应该在平衡处理字符串时应该在 Unicode 复杂性和性能之间取得平衡。而另一方面,它需要为开发者提供一个舒适的结构来处理字符串。
+从一方面来说,编程语言在处理字符串时应该在 Unicode 复杂性和性能之间取得平衡。而另一方面,它需要为开发者提供一个舒适的结构来处理字符串。
而在我看来,Swift 在这两方面都做的不错。
@@ -23,16 +23,16 @@ Swift 对此有着更好的实现方式。字符串本身不再是集合,而
对于 `let myStr = "Hello, world"` 来说,你可以访问到下面这些 view:
-- `myStr.characters` 即 `String.CharacterView`。可以获取字形的值,视觉上呈现为单一的符号。是最常用的试图
+- `myStr.characters` 即 `String.CharacterView`。可以获取字形的值,视觉上呈现为单一的符号,是最常用的视图。
- `myStr.unicodeScalars` 即 `String.UnicodeScalarView`。可以获取 21 整数表示的 Unicode 码位。
- `myStr.utf16` 即 `String.UTF16View`。用于获取 UTF16 编码的代码单元。
-- `myStr.utf8` 即 `String.UTF8View`。能够获取 UTF8 编码的代码单一。
+- `myStr.utf8` 即 `String.UTF8View`。能够获取 UTF8 编码的代码单元。
![Swift 中的 CharacterView, UnicodeScalarView, UTF16View 和 UTF8View](https://rainsoft.io/content/images/2016/10/Swift-strings--3-.png)
在大多数时候开发者都在处理简单的字符串字符,而不是深入到编码或者码位这样的细节中。
-`CharacterView` 能很好的完成大多数任务:迭代字符串、字符计数、验证是否包含字符串、通过索引访问和比较操作等。
+`CharacterView` 能很好地完成大多数任务:迭代字符串、字符计数、验证是否包含字符串、通过索引访问和比较操作等。
让我们看看如何用 Swift 来完成这些任务。
@@ -52,7 +52,7 @@ print(type(of: characters))// => "CharacterView"
`message.characters` 返回了 `CharacterView` 结构.
-字符视图是 `Character` 结构的集合。例如我们可以这样来访问字符视图里的第一个字符:
+字符视图是 `Character` 结构的集合。例如,我们可以这样来访问字符视图里的第一个字符:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57ff7e188ef62b25bcea2ab2)
@@ -70,7 +70,7 @@ print(capitalHCharacter == firstCharacter) // => true
这个字符实例代表了单个符号 `H`。
-在 Unicode 标准中,`H` 代表 *Latin Capital letter H*,码位是 `U+0048`。
+在 Unicode 标准中,`H` 代表 *Latin Capital letter H* (拉丁文大写字母 H),码位是 `U+0048`。
让我们掠过 ASCII 看看 Swift 如何处理更复杂的符号。这些字符被渲染成单个视觉符号,但实际上是由两个或更多个 Unicode 标量](http://unicode.org/glossary/#unicode_scalar_value) 组成。严格来说这些字符被称为 **字形簇**
@@ -79,7 +79,7 @@ print(capitalHCharacter == firstCharacter) // => true
让我们看看 `ç` 的字形。他可以有两种表现形式:
-- 使用 `U+00E7` *LATIN SMALL LETTER C WITH CEDILLA*:被渲染为 `ç`
+- 使用 `U+00E7` *LATIN SMALL LETTER C WITH CEDILLA* (拉丁文小写变音字母 C):被渲染为 `ç`
- 或者使用组合字符序列:`U+0063`*LATIN SMALL LETTER C* 加上 组合标记 `U + 0327` *COMBINING CEDILLA* 组成复合字形:`c` + `◌̧` = `ç`
我们看看在第二个选项中 Swift 是如何处理它的:
@@ -111,9 +111,9 @@ let multipleGraphemes: Character = "ab" // Error!
即使 `singleGrapheme` 由 3 个 Unicode 标量组成,它创建了一个字形 `ḉ`。
而 `multipleGraphemes` 则是从两个 Unicode 标量创建一个 `Character`,这将在单个 `Character` 结构中创建两个分离的字母 `a` 和 `b`,这不是被允许的操作。
-# 2. 迭代字符串中的字符
+# 2. 遍历字符串中的字符
-`CharacterView` 集合遵循了 `Sequence` 协议。这将允许在 `for-in` 循环中便利字符视图:
+`CharacterView` 集合遵循了 `Sequence` 协议。这将允许在 `for-in` 循环中遍历字符视图:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bc8f27a61152fe7c7410)
@@ -161,7 +161,7 @@ for (index, char) in weather.characters.enumerated() {
# 3. 统计字符
-只需要访问 `CharacterView` 的 `counter` 属性就可以获得字符串中字符的个数:
+只需要访问 `CharacterView` 的 `count` 属性就可以获得字符串中字符的个数:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bcf327a61152fe7c7413)
@@ -191,11 +191,11 @@ print(drink.characters.count) // => 4
# 4. 按索引访问字符
-因为 Swift 直到它实际评估字符视图中的字形之前都不知道字符串中的字符个数。结果就造成了无法通过通过下标的方式访问字符串索引。
+因为 Swift 直到它实际评估字符视图中的字形之前都不知道字符串中的字符个数,所以无法通过下标的方式访问字符串索引。
你可以通过特殊的类型 `String.Index` 访问字符。
-如果你需要访问字符串中的第一个或者最好一个字符,字符视图结构提供了 `first` 和 `last` 属性:
+如果你需要访问字符串中的第一个或者最后一个字符,字符视图结构提供了 `first` 和 `last` 属性:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bd2027a61152fe7c7415)
@@ -230,7 +230,7 @@ print(color[beforeEndIndex]) // => "n"
```
`color.startIndex` 是第一个字符的索引,所以 `color[startIndex]` 表示为 `g`。
-`color.endIndex` 表示**结束**位置,或者简单的说是比最后一个有效小标参数大的位置。要访问最后一个字符,你必须计算它的前一个索引:`color.index(before: color.endIndex)`
+`color.endIndex` 表示**结束**位置,或者简单的说是比最后一个有效下标参数大的位置。要访问最后一个字符,你必须计算它的前一个索引:`color.index(before: color.endIndex)`
要通过偏移访问字符的位置, 在 `index(theIndex, offsetBy: theOffset)` 方法中使用 `offsetBy` 参数:
@@ -459,9 +459,9 @@ if let index = weather.characters.index(of: " ") {
上面描述的许多字符串操作都是直接应用于字符串中的字符视图。
-而更方便的直接使用一个字符序列可能是更好的选择。
+如果你觉得直接对字符序列进行操作更加方便的话,那也是个不错的选择。
-比如你可以删除特定索引出的字符,或者直接删除第一个或者最好一个字符:
+比如你可以删除特定索引出的字符,或者直接删除第一个或者最后一个字符:
[Try in Swift sandbox](http://swiftlang.ng.bluemix.net/#/repl/57f4bea927a61152fe7c7425)
@@ -531,7 +531,7 @@ print(numberOfStars) // => 2
首先要说,大家对于字符串内容持有的不同观点看起来似乎过于复杂。
-而在我看来这是一个很好的实现。字符串可以从不同的角度来例假:昨晚字形集合、UTF-8 或 UTF-16 码位和简单是 Unicode 标量。
+而在我看来这是一个很好的实现。字符串可以从不同的角度来看待:作为字形集合、UTF-8 / UTF-16 码位或者简单的 Unicode 标量。
根据你的任务来选择合适的视图。在大多数情况下,`CharacterView` 都很合适。
From fccb3e734190d7323c5c224ec48fab9e89d0e646 Mon Sep 17 00:00:00 2001
From: zhuzi
Date: Tue, 11 Apr 2017 23:49:37 +0800
Subject: [PATCH 187/945] translation half finished
---
TODO/secure-web-app-http-headers.md | 119 ++++++++++++++++++----------
1 file changed, 79 insertions(+), 40 deletions(-)
diff --git a/TODO/secure-web-app-http-headers.md b/TODO/secure-web-app-http-headers.md
index e2f1066fd33..17aad2f24b1 100644
--- a/TODO/secure-web-app-http-headers.md
+++ b/TODO/secure-web-app-http-headers.md
@@ -1,29 +1,30 @@
> * 原文地址:[How To Secure Your Web App With HTTP Headers](https://www.smashingmagazine.com/2017/04/secure-web-app-http-headers/)
> * 原文作者:[Hagay Lupesko](https://www.smashingmagazine.com/author/hagaylupesko/)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[bambooom](https://github.com/bambooom)
> * 校对者:
-## [How To Secure Your Web App With HTTP Headers](https://www.smashingmagazine.com/2017/04/secure-web-app-http-headers/) ##
+## 如何使用 HTTP Headers 来保护你的 Web 应用 ##
-Web applications, be they thin websites or thick single-page apps, are notorious targets for cyber-attacks. In 2016, approximately [40% of data breaches](http://www.verizonenterprise.com/verizon-insights-lab/dbir/2016/) originated from attacks on web apps — the leading attack pattern. Indeed, these days, understanding cyber-security is not a luxury but rather **a necessity for web developers**, especially for developers who build consumer-facing applications.
+> Web applications, be they thin websites or thick single-page apps, are notorious targets for cyber-attacks. In 2016, approximately [40% of data breaches](http://www.verizonenterprise.com/verizon-insights-lab/dbir/2016/) originated from attacks on web apps — the leading attack pattern. Indeed, these days, understanding cyber-security is not a luxury but rather **a necessity for web developers**, especially for developers who build consumer-facing applications.
-HTTP response headers can be leveraged to tighten up the security of web apps, typically just by adding a few lines of code. In this article, we’ll show how web developers can use HTTP headers to build secure apps. While the code examples are for Node.js, setting HTTP response headers is supported across all major server-side-rendering platforms and is typically simple to set up.
+Web 应用,无论是简单的小网页还是复杂的单页应用,众所周知都是网络攻击的目标。2016年,大约 40% 的数据泄露源自对 Web 应用的攻击,这是主要的攻击模式。事实上,现在来说,了解网络安全并不是锦上添花, 而是 Web 开发者的必需任务,特别对于构建面向消费者的产品的开发人员。
-#### Further Reading on SmashingMag:
+> HTTP response headers can be leveraged to tighten up the security of web apps, typically just by adding a few lines of code. In this article, we’ll show how web developers can use HTTP headers to build secure apps. While the code examples are for Node.js, setting HTTP response headers is supported across all major server-side-rendering platforms and is typically simple to set up.
-- [Facing The Challenge: Building A Responsive Web Application](https://www.smashingmagazine.com/2013/06/building-a-responsive-web-application/)
-- [Getting Ready For HTTP2: A Guide For Web Designers And Developers](https://www.smashingmagazine.com/2016/02/getting-ready-for-http2/)
-- [Common Security Mistakes in Web Applications](https://www.smashingmagazine.com/2010/10/common-security-mistakes-in-web-applications/)
-- [Web Security: Are You Part Of The Problem?](https://www.smashingmagazine.com/2010/01/web-security-primer-are-you-part-of-the-problem/)
+开发者可以利用 HTTP 响应头来加强 Web 应用程序的安全性,通常只需要添加几行代码即可。本文将结束 web 开发者如何利用 HTTP Headers 来构建安全的应用。虽然本文的示例代码是 Node.js,但是设置 HTTP 响应头基本在所有主要的服务端语言都是简单易设置的。
-### About HTTP Headers ###
+### 关于 HTTP Headers ###
-Technically, HTTP headers are simply fields, encoded in clear text, that are part of the HTTP request and response message header. They are designed to enable both the HTTP client and server to send and receive meta data about the connection to be established, the resource being requested, as well as the returned resource itself.
+> Technically, HTTP headers are simply fields, encoded in clear text, that are part of the HTTP request and response message header. They are designed to enable both the HTTP client and server to send and receive meta data about the connection to be established, the resource being requested, as well as the returned resource itself.
-Plain-text HTTP response headers can be examined easily using cURL, with the `--head` option, like so:
+技术上,HTTP 头只是简单的字段,咦明文形式编码,这是 HTTP 请求和响应消息头的一部分。它们旨在使客户端和服务端都能够发送和接受有关要建立的连接元数据、所请求的资源,以及返回的资源本身的元数据。
-```
+> Plain-text HTTP response headers can be examined easily using cURL, with the `--head` option, like so:
+
+可以使用 cURL `--head` 选项轻松检查纯文本 HTTP 响应头,例如:
+
+```sh
$ curl --head https://www.google.com
HTTP/1.1 200 OK
Date: Thu, 05 Jan 2017 08:20:29 GMT
@@ -36,33 +37,51 @@ Vary: Accept-Encoding
…
```
-Today, hundreds of headers are used by web apps, some standardized by the [Internet Engineering Task Force](https://www.ietf.org/) (IETF), the open organization that is behind many of the standards that power the web as we know it today, and some proprietary. HTTP headers provide a flexible and extensible mechanism that enables the rich and varying use cases found on the web today.
+> Today, hundreds of headers are used by web apps, some standardized by the [Internet Engineering Task Force](https://www.ietf.org/) (IETF), the open organization that is behind many of the standards that power the web as we know it today, and some proprietary. HTTP headers provide a flexible and extensible mechanism that enables the rich and varying use cases found on the web today.
+
+现在,数百种响应头正在被 web 应用所使用,其中一部分由[互联网工程任务组, IETF](https://www.ietf.org/)标准化。IETF 是一个开发性组织,今天我们所熟知的许多 web 标准或专利都是由他们进行推进的。HTTP 头提供了一种灵活可扩展的机制,造就了现今的网络各种丰富多变的用例。
+
+### 机密资源禁用缓存 ###
+
+> Caching is a valuable and effective technique for optimizing performance in client-server architectures, and HTTP, which leverages caching extensively, is no exception. However, in cases where the cached resource is confidential, caching can lead to vulnerabilities — and must be avoided. As an example, consider a web app that renders and caches a page with sensitive information and is being used on a shared PC. Anyone can view confidential information rendered by that web app simply by visiting the browser’s cache, or sometimes even as easily as clicking the browser’s “back” button!
+
+缓存是优化客户端-服务端架构性能中有效的技术,广泛利用缓存的 HTTP 也不例外。但是,在缓存的资源是保密的情况下,缓存可能导致漏斗,所以必须避免。假设一个 web 应用对含有敏感信息的网页进行缓存,并且是在一台公用的 PC 上使用。任何人可以通过访问浏览器的缓存看到这个 web 应用上的敏感信息,甚至有时仅仅通过点击浏览器的返回按钮就可以看到。
-### Disabling Caching Of Confidential Resources ###
+> The IETF’s [RFC 7234](https://tools.ietf.org/html/rfc7234), which defines HTTP caching, specifies the default behavior of HTTP clients, both browsers and intermediary Internet proxies, to *always* cache responses to HTTP `GET` requests — unless specified otherwise. While this enables HTTP to boost performance and reduce network congestion, it could also expose end users to theft of personal information, as mentioned above. The good news is that the HTTP specification also defines a pretty simple way to instruct clients not to cache a given response, through the use of — you guessed it! — HTTP response headers.
-Caching is a valuable and effective technique for optimizing performance in client-server architectures, and HTTP, which leverages caching extensively, is no exception. However, in cases where the cached resource is confidential, caching can lead to vulnerabilities — and must be avoided. As an example, consider a web app that renders and caches a page with sensitive information and is being used on a shared PC. Anyone can view confidential information rendered by that web app simply by visiting the browser’s cache, or sometimes even as easily as clicking the browser’s “back” button!
+IETF 的 [RFC 7234](https://tools.ietf.org/html/rfc7234) 上定义了 HTTP 缓存,指定 HTTP 客户端(浏览器以及网络代理)的默认行为,也就是始终缓存对 HTTP GET 请求的相应,除非另行指定。虽然这样可以使 HTTP 提升性能减少网络拥塞,但如上所述,它也有可能使终端用户个人信息被盗。好消息是,HTTP 规范还廷议了一种非常简单的方式来指示客户端对特定响应不进行缓存,通过使用 —— 对,你猜到了 —— HTTP 响应头。
-The IETF’s [RFC 7234](https://tools.ietf.org/html/rfc7234), which defines HTTP caching, specifies the default behavior of HTTP clients, both browsers and intermediary Internet proxies, to *always* cache responses to HTTP `GET` requests — unless specified otherwise. While this enables HTTP to boost performance and reduce network congestion, it could also expose end users to theft of personal information, as mentioned above. The good news is that the HTTP specification also defines a pretty simple way to instruct clients not to cache a given response, through the use of — you guessed it! — HTTP response headers.
+> There are three headers to return when you are returning sensitive information and would like to disable caching by HTTP clients:
-There are three headers to return when you are returning sensitive information and would like to disable caching by HTTP clients:
+当你返回敏感信息并希望禁用 HTTP 客户端的缓存时,有三个头可以返回:
- `Cache-Control`
-This response header, introduced in HTTP 1.1, may contain one or more directives, each carrying a specific caching semantic, and instructing HTTP clients and proxies on how to treat the response being annotated by the header. My recommendation is to format the header as follows: `cache-control: no-cache, no-store, must-revalidate`. These three directives pretty much instruct clients and intermediary proxies not to use a previously cached response, not to store the response, and that even if the response is somehow cached, the cache must be revalidated on the origin server.
+> This response header, introduced in HTTP 1.1, may contain one or more directives, each carrying a specific caching semantic, and instructing HTTP clients and proxies on how to treat the response being annotated by the header. My recommendation is to format the header as follows: `cache-control: no-cache, no-store, must-revalidate`. These three directives pretty much instruct clients and intermediary proxies not to use a previously cached response, not to store the response, and that even if the response is somehow cached, the cache must be revalidated on the origin server.
+
+从 HTTP 1.1 引入的此响应头可能包含一个或多个指令,每个指令带有特定的缓存语义,指示 HTTP 客户端和代理如何处理有此响应头注释的响应。我推荐如下指定响应头,`cache-control: no-cache, no-store, must-revalidate`。这三个指令基本上可以指示客户端和中间代理不可使用之前缓存的响应,不可存储响应,甚至就算响应被缓存,也必须从源服务器上重新验证。
- `Pragma: no-cache`
-
-For backwards-compatibility with HTTP 1.0, you will want to include this header as well. Some HTTP clients, especially intermediary proxies, still might not fully support HTTP 1.1 and so will not correctly handle the `Cache-Control` header mentioned above. Use `Pragma: no-cache` to ensure that these older clients do not cache your response.
+
+> For backwards-compatibility with HTTP 1.0, you will want to include this header as well. Some HTTP clients, especially intermediary proxies, still might not fully support HTTP 1.1 and so will not correctly handle the `Cache-Control` header mentioned above. Use `Pragma: no-cache` to ensure that these older clients do not cache your response.
+
+为了与 HTTP 1.0 的向后兼容性,你还需要包含此响应头。有部分客户端,特别是中间代理,可能仍然没有完全支持 HTTP 1.1,所以不能正确处理前面提到的 `Cache-Control` 响应头,所以使用 `Pragma: no-cache` 确保较旧的客户端不缓存你的响应。
- `Expires: -1`
-This header specifies a timestamp after which the response is considered stale. By specifying `-1`, instead of an actual future time, you ensure that clients immediately treat this response as stale and avoid caching.
+> This header specifies a timestamp after which the response is considered stale. By specifying `-1`, instead of an actual future time, you ensure that clients immediately treat this response as stale and avoid caching.
-Note that, while disabling caching enhances the security of your web app and helps to protect confidential information, is does come at the price of a performance hit. Make sure to disable caching only for resources that actually require confidentiality and not just for any response rendered by your server! For a deeper dive into best practices for caching web resources, I highly recommend reading [Jake Archibald’s post](https://jakearchibald.com/2016/caching-best-practices/) on the subject.
+此标头指定了该响应过时的时间戳。如果不指定为未来某个真实时间而指定为 `-1`,可以保证客户端立即将此响应视为过时并避免缓存。
-Here’s how you would program these headers in Node.js:
+> Note that, while disabling caching enhances the security of your web app and helps to protect confidential information, is does come at the price of a performance hit. Make sure to disable caching only for resources that actually require confidentiality and not just for any response rendered by your server! For a deeper dive into best practices for caching web resources, I highly recommend reading [Jake Archibald’s post](https://jakearchibald.com/2016/caching-best-practices/) on the subject.
-```
+需要注意的是,禁用缓存提高安全性及保护机密资源的同时,也的确会带来性能上的折损。所以确保仅对实际需要保密性的资源禁用缓存,而不是对任何服务器的响应禁用。想要更深入了解 web 资源缓存的最佳实践,我推荐阅读 [Jake Archibald 的文章](https://jakearchibald.com/2016/caching-best-practices/)。
+
+> Here’s how you would program these headers in Node.js:
+
+下面是 Node.js 中设置响应头的示例代码:
+
+```javascript
function requestHandler(req, res) {
res.setHeader('Cache-Control','no-cache,no-store,max-age=0,must-revalidate');
res.setHeader('Pragma','no-cache');
@@ -70,37 +89,58 @@ function requestHandler(req, res) {
}
```
-### Enforcing HTTPS ###
+### 强制 HTTPS ###
-Today, the importance of HTTPS is widely recognized by the tech community. More and more web apps configure secured endpoints and are redirecting unsecure traffic to secured endpoints (i.e. HTTP to HTTPS redirects). Unfortunately, end users have yet to fully comprehend the importance of HTTPS, and this lack of comprehension exposes them to various man-in-the-middle (MitM) attacks. The typical user navigates to a web app without paying much attention to the protocol being used, be it secure (HTTPS) or unsecure (HTTP). Moreover, many users will just click past browser warnings when their browser presents a certificate error or warning!
+> Today, the importance of HTTPS is widely recognized by the tech community. More and more web apps configure secured endpoints and are redirecting unsecure traffic to secured endpoints (i.e. HTTP to HTTPS redirects). Unfortunately, end users have yet to fully comprehend the importance of HTTPS, and this lack of comprehension exposes them to various man-in-the-middle (MitM) attacks. The typical user navigates to a web app without paying much attention to the protocol being used, be it secure (HTTPS) or unsecure (HTTP). Moreover, many users will just click past browser warnings when their browser presents a certificate error or warning!
-The importance of interacting with web apps over a valid HTTPS connection cannot be overstated: An unsecure connection exposes the user to various attacks, which could lead to cookie theft or worse. As an example, it is not very difficult for an attacker to spoof network frames within a public Wi-Fi network and to extract the session cookies of users who are not using HTTPS. To make things even worse, even users interacting with a web app over a secured connection may be exposed to downgrade attacks, which try to force the connection to be downgraded to an unsecure connection, thus exposing the user to MitM attacks.
+今天,HTTPS 的重要性已经得到了科技界的广泛认可。越来越多的 web 应用配置了安全端点,并将不安全网路重定向到安全端点(即 HTTP 重定向至 HTTPS)。不幸的是,终端用户还未完全理解 HTTPS 的重要性,这种缺乏理解使他们面临着各种中间人攻击(MitM)。典型的用户访问到一个 web 应用时,并不会注意到正在使用的网络协议是安全的(HTTPS)还是不安全的(HTTP)。
-How can we help users avoid these attacks and better enforce the usage of HTTPS? Enter the HTTP Strict Transport Security (HSTS) header. Put simply, HSTS makes sure all communications with the origin host are using HTTPS. Specified in [RFC 6797](https://tools.ietf.org/html/rfc6797), HSTS enables a web app to instruct browsers to allow *only* HTTPS connections to the origin host, to internally redirect all unsecure traffic to secured connections, and to automatically upgrade all unsecure resource requests to be secure.
+> The importance of interacting with web apps over a valid HTTPS connection cannot be overstated: An unsecure connection exposes the user to various attacks, which could lead to cookie theft or worse. As an example, it is not very difficult for an attacker to spoof network frames within a public Wi-Fi network and to extract the session cookies of users who are not using HTTPS. To make things even worse, even users interacting with a web app over a secured connection may be exposed to downgrade attacks, which try to force the connection to be downgraded to an unsecure connection, thus exposing the user to MitM attacks.
-HSTS directives include the following:
+通过有效的 HTTPS 连接与 web 应用进行交互的重要性怎么说都不算夸大:不安全的连接将用户暴露给各种攻击,这可能导致 cookie 被盗甚至更糟。举个例子,攻击者可以轻易在公共 Wi-Fi 网络下骗过网络帧并提起不使用 HTTPS 的用户的会话 cookie。更糟的情况是,即使用户通过安全连接与 web 永盈进行交互也可能遭受降级攻击,这种攻击试图强制将连接降级到不安全的连接,从而是用户收到中间人攻击。
+
+> How can we help users avoid these attacks and better enforce the usage of HTTPS? Enter the HTTP Strict Transport Security (HSTS) header. Put simply, HSTS makes sure all communications with the origin host are using HTTPS. Specified in [RFC 6797](https://tools.ietf.org/html/rfc6797), HSTS enables a web app to instruct browsers to allow *only* HTTPS connections to the origin host, to internally redirect all unsecure traffic to secured connections, and to automatically upgrade all unsecure resource requests to be secure.
+
+我们如何帮助用户避免这些攻击,并更好地实施 HTTPS 的使用呢?使用 HTTP 严格传输安全头(HSTS)。简单来说,HSTS 确保与源主机间的所有通信都使用 HTTPS。[RFC 6797](https://tools.ietf.org/html/rfc6797) 中说明了,HSTS可以使 web 应用程序指示浏览器仅允许与源主机之间的 HTTPS 连接,将所有不安全的连接内部重定向到安全连接,并自动将所有不安全的资源请求升级为安全请求。
+
+> HSTS directives include the following:
+
+HSTS 的指令如下:
- `max-age=`
-This instructs the browser to cache this header, for this domain, for the specified number of seconds. This can ensure tightened security for a long duration!
+>This instructs the browser to cache this header, for this domain, for the specified number of seconds. This can ensure tightened security for a long duration!
+
+此项指示浏览器对此域缓存此响应头指定的秒数。这样可以保证长时间的加固安全。
- `includeSubDomains`
-This instructs the browser to apply HSTS for all subdomains of the current domain. This can be useful to cover all current and future subdomains you may have.
+>This instructs the browser to apply HSTS for all subdomains of the current domain. This can be useful to cover all current and future subdomains you may have.
+
+此项指示浏览器对当前域的所有子域应用 HSTS,这可以用于覆盖你可能有的所有当前和未来的子域。
- `preload`
-This is a powerful directive that forces browsers to *always* load your web app securely, even on the first hit, before the response is even received! This works by hardcoding a list of HSTS preload-enabled domains into the browser’s code. To enable the preloading feature, you need to register your domain with [HSTS Preload List Submission](https://hstspreload.org), a website maintained by Google’s Chrome team. Once registered, the domain will be prebuilt into supporting browsers to always enforce HSTS. The preload directive within the HTTP response header is used to confirm registration, indicating that the web app and domain owner are indeed interested in being on the preload list.
+> This is a powerful directive that forces browsers to *always* load your web app securely, even on the first hit, before the response is even received! This works by hardcoding a list of HSTS preload-enabled domains into the browser’s code. To enable the preloading feature, you need to register your domain with [HSTS Preload List Submission](https://hstspreload.org), a website maintained by Google’s Chrome team. Once registered, the domain will be prebuilt into supporting browsers to always enforce HSTS. The preload directive within the HTTP response header is used to confirm registration, indicating that the web app and domain owner are indeed interested in being on the preload list.
-A word of caution: using the `preload` directive also means it cannot be easily undone, and carries an update lead time of months! While preload certainly improves your app’s security, it also means you need to be fully confident your app can support HTTPS-only!
+这是一个强大的指令,强制浏览器始终安全加载你的 web 应用程序,即使是第一次收到响应之前加载!这是通过将启用 HSTS 预加载域的列表硬编码到浏览器的代码中实现的。要启用预加载功能,你需要在 Google Chrome 团队维护的网站 [HSTS 预加载列表提交](https://hstspreload.org)注册你的域。
-My recommendation is to use `Strict-Transport-Security: max-age=31536000; includeSubDomains;` which instructs the browser to enforce a valid HTTPS connection to the origin host and to all subdomains for a year. If you are confident that your app can handle HTTPS-only, I would also recommend adding the `preload` directive, in which case don’t forget to register your website on the preload list as well, as noted above!
+> A word of caution: using the `preload` directive also means it cannot be easily undone, and carries an update lead time of months! While preload certainly improves your app’s security, it also means you need to be fully confident your app can support HTTPS-only!
-Here’s what implementing HSTS looks like in Node.js:
+注意谨慎使用 `preload`,因为这意味着它不能轻易撤销,并带有几个月前的更新。虽然预加载肯定会应用程序的安全性,但也意味着你需要充分确信你的应用程序可以支持仅 HTTPS!
-```
-functionrequestHandler(req, res){
- res.setHeader('Strict-Transport-Security','max-age=31536000; includeSubDomains; preload');}
+> My recommendation is to use `Strict-Transport-Security: max-age=31536000; includeSubDomains;` which instructs the browser to enforce a valid HTTPS connection to the origin host and to all subdomains for a year. If you are confident that your app can handle HTTPS-only, I would also recommend adding the `preload` directive, in which case don’t forget to register your website on the preload list as well, as noted above!
+
+我推荐的用法是 `Strict-Transport-Security: max-age=31536000; includeSubDomains;`,这样指示了浏览器强制通过 HTTPS 连接到源主机并且有效期为一年。如果你对你的 app 处理 HTTPS 限定很有信心,我也推荐加上 `preload` 指令,当然别忘记去前面提到的预加载列表注册你的网站。
+
+> Here’s what implementing HSTS looks like in Node.js:
+
+以下是在 Nodes.js 中实现 HSTS 的方法:
+
+```javascript
+function requestHandler(req, res){
+ res.setHeader('Strict-Transport-Security','max-age=31536000; includeSubDomains; preload');
+}
```
### Enabling XSS Filtering ###
@@ -248,7 +288,6 @@ Remember that for the web to be truly awesome and engaging, it has to be secure.
*Front page image credits: [Pexels.com](https://www.pexels.com/photo/coffee-writing-computer-blogging-34600/).*
-
---
> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 535f0275fb15aac1dd4b5def4cd27bfc3362d470 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Wed, 12 Apr 2017 10:15:25 +0800
Subject: [PATCH 188/945] Update css-is-fine-its-just-really-hard.md
---
TODO/css-is-fine-its-just-really-hard.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/css-is-fine-its-just-really-hard.md b/TODO/css-is-fine-its-just-really-hard.md
index 423250fcfb8..20cb8558bcd 100644
--- a/TODO/css-is-fine-its-just-really-hard.md
+++ b/TODO/css-is-fine-its-just-really-hard.md
@@ -1,5 +1,5 @@
> * 原文地址:[CSS is Fine, It’s Just Really Hard](https://medium.com/@jdan/css-is-fine-its-just-really-hard-638da7a3dce0)
-> * 原文作者:[Jordan Scales](https://medium.com/@jdan)
+> * 原文作者:该文章已获原作者 [Jordan Scales](https://medium.com/@jdan) 授权
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者: [ZhangFe](https://github.com/ZhangFe)
> * 校对者:[bambooom](https://github.com/bambooom),[gy134340](https://github.com/gy134340)
From 88f2c4e5960ea4f8351d8095b3f71418d5346a7e Mon Sep 17 00:00:00 2001
From: ivyxuan
Date: Wed, 12 Apr 2017 10:45:39 +0800
Subject: [PATCH 189/945] =?UTF-8?q?=E4=BF=AE=E6=94=B9=E9=94=99=E8=AF=AF?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/the-details-that-matter.md | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/TODO/the-details-that-matter.md b/TODO/the-details-that-matter.md
index b4ccfb5d9f8..69114a7588b 100644
--- a/TODO/the-details-that-matter.md
+++ b/TODO/the-details-that-matter.md
@@ -80,7 +80,7 @@ Airbnb 的提示听起来像人说的话而且语气还很随和。
-**像“出错啦”这种警告对所有的用户都会造成困扰,而且还会惹恼专家级用户。**但是,一个精心设计过的错误信息,会顿时化失望为欣喜。所以,把报错变得人性化、不用技术性的语言并且适合你的用户群体。
+**像“出错啦”这种警告对所有的用户都会造成困扰,而且还会惹恼专家级用户。** 但是,一个精心设计过的错误信息,会顿时化失望为欣喜。所以,把报错变得人性化、不用技术性的语言并且适合你的用户群体。
@@ -100,7 +100,7 @@ Airbnb 的提示听起来像人说的话而且语气还很随和。
#### 让用户界面更容易理解 ####
-**杂乱的堆砌是很糟糕的一件事情。**在界面上杂乱堆砌元素会给用户带来过多的信息:每一个被添加的按钮、图片和文字都会让界面显得更加复杂。如果你不想你的设计有任何刻意的留白的话,下面这个例子就能很明白的告诉你,有太多东西一起吸引你的注意力是多么可怕的事情。
+**杂乱的堆砌是很糟糕的一件事情。** 在界面上杂乱堆砌元素会给用户带来过多的信息:每一个被添加的按钮、图片和文字都会让界面显得更加复杂。如果你不想你的设计有任何刻意的留白的话,下面这个例子就能很明白的告诉你,有太多东西一起吸引你的注意力是多么可怕的事情。
From 6e460df0aceb6017d65043c6bdcf1042bbce31bc Mon Sep 17 00:00:00 2001
From: reid
Date: Wed, 12 Apr 2017 11:55:31 +0800
Subject: [PATCH 190/945] =?UTF-8?q?=E5=AE=8C=E6=88=90=E7=BF=BB=E8=AF=91?=
=?UTF-8?q?=E5=88=9D=E7=A8=BF?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...gher-order-functions-composing-software.md | 89 +++++++++----------
1 file changed, 44 insertions(+), 45 deletions(-)
diff --git a/TODO/higher-order-functions-composing-software.md b/TODO/higher-order-functions-composing-software.md
index aa432f1141e..67336f3fd47 100644
--- a/TODO/higher-order-functions-composing-software.md
+++ b/TODO/higher-order-functions-composing-software.md
@@ -1,22 +1,22 @@
-> * 原文地址:[Higher Order Functions (Composing Software)(part 4)](https://medium.com/javascript-scene/higher-order-functions-composing-software-5365cf2cbe99)
-> * 原文作者:[Eric Elliott](https://medium.com/@_ericelliott?source=post_header_lockup)
-> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
-> * 校对者:
+> * 原文地址:[Higher Order Functions (Composing Software)(part 4)](https://medium.com/javascript-scene/higher-order-functions-composing-software-5365cf2cbe99)
+> * 原文作者:[Eric Elliott](https://medium.com/@_ericelliott?source=post_header_lockup)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:[reid3290](https://github.com/reid3290)
+> * 校对者:
-# Higher Order Functions (Composing Software) #
+# 高阶函数(软件编写)(第四部分) #
-Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0)
-> Note: This is part of the “Composing Software” series on learning functional programming and compositional software techniques in JavaScript ES6+ from the ground up. Stay tuned. There’s a lot more of this to come!
-> [< Previous](https://medium.com/javascript-scene/a-functional-programmers-introduction-to-javascript-composing-software-d670d14ede30#.gof8dsqh9) | [<< Start over at Part 1](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [Next >](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d)
+Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0)(译注:该图是用 PS 将烟雾处理成方块状后得到的效果,参见 [flickr](https://www.flickr.com/photos/68397968@N07/11432696204)。)
+> 注意:这是“软件编写”系列文章的第四部分,该系列主要阐述如何在 JavaScript ES6+ 中从零开始学习函数式编程和组合化软件(compositional software)技术(译注:关于软件可组合性的概念,参见维基百科 [Composability](https://en.wikipedia.org/wiki/Composability))。后续还有更多精彩内容,敬请期待!
+> [< 上一篇](https://medium.com/javascript-scene/a-functional-programmers-introduction-to-javascript-composing-software-d670d14ede30#.gof8dsqh9) | [<< 第一篇](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [下一篇 >](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d)
-A **higher order function** is a function that takes a function as an argument, or returns a function. Higher order function is in contrast to first order functions, which don’t take a function as an argument or return a function as output.
+**高阶函数**是一种接收一个函数作为输入或输出一个函数的函数(译注:参见维基百科[高阶函数](https://zh.wikipedia.org/wiki/%E9%AB%98%E9%98%B6%E5%87%BD%E6%95%B0)),这是和一阶函数截然不同的。
-Earlier we saw examples of `.map()` and `.filter()`. Both of them take a function as an argument. They're both higher order functions.
+之前我们看到的 `.map()` 和 `.filter()` 都是高阶函数 —— 它们都接受一个函数作为参数,
-Let’s look at an example of a first-order function which filters all the 4-letter words from a list of words:
+先来看个一阶函数的例子,该函数会将单词数组中 4 个字母的单词过滤掉:
```
const censor = words => {
@@ -32,7 +32,7 @@ censor(['oops', 'gasp', 'shout', 'sun']);
// [ 'shout', 'sun' ]
```
-Now what if we want to select all the words that begin with ‘s’? We could create another function:
+如果又要选择出所有以 's' 开头的单词呢?可以再定义一个函数:
```
const startsWithS = words => {
@@ -48,29 +48,29 @@ startsWithS(['oops', 'gasp', 'shout', 'sun']);
// [ 'shout', 'sun' ]
```
-You may already be recognizing a lot of repeated code. There’s a pattern forming here that could be abstracted into a more generalized solution. These two functions have a whole lot in common. They both iterate over a list and filter it on a given condition.
+显然可以看出这里面有很多重复的代码,这两个函数的主体是相同的 —— 都是遍历一个数组并根据给定的条件进行过滤。这便形成了一种特定的模式,可以从中抽象出更为通用的解决方案。
-Both the iteration and the filtering seem like they’re begging to be abstracted so they can be shared and reused to build all sorts of similar functions. After all, selecting things from lists of things is a very common task.
+不难看出, “遍历”和“过滤”都是亟待抽象出来的,以便分享和复用到其他所有类似的函数中去。毕竟,从数组中选取某些特定元素是很常见的需求。
-Luckily for us, JavaScript has first class functions. What does that mean? Just like numbers, strings, or objects, functions can be:
+幸运的是,函数是 JavaScript 中的一等公民,就像数字、字符串和对象一样,函数可以:
-- Assigned as an identifier (variable) value
-- Assigned to object property values
-- Passed as arguments
-- Returned from functions
+- 像变量一样赋值给其他变量
+- 作为对象的属性值
+- 作为参数进行传递
+- 作为函数的返回值
-Basically, we can use functions just like any other bits of data in our programs, and that makes abstraction a lot easier. For instance, we can create a function that abstracts the process of iterating over a list an accumulating a return value by passing in a function that handles *the bits that are different.* We’ll call that function the *reducer:*
+函数基本上可以像其他任何数据类型一样被使用,这点使得“抽象”容易了许多。例如,可以定义一种函数,将遍历数组并累计出一个返回值的过程抽象出来,该函数接收一个函数作为参数来决定具体的**累计**过程,不妨将此函数称为 **reducer**:
```
const reduce = (reducer, initial, arr) => {
- // shared stuff
+ // 共享的
let acc = initial;
for (let i = 0, length = arr.length; i < length; i++) {
- // unique stuff in reducer() call
+ // 独特的
acc = reducer(acc, arr[i]);
- // more shared stuff
+ // 又是共享的
}
return acc;
};
@@ -78,14 +78,14 @@ const reduce = (reducer, initial, arr) => {
reduce((acc, curr) => acc + curr, 0, [1,2,3]); // 6
```
-This `reduce()` implementation takes a reducer function, an initial value for the accumulator, and an array of data to iterate over. For each item in the array, the reducer is called, passing it the accumulator and the current array element. The return value is assigned to the accumulator. When it's finished applying the reducer to all of the values in the list, the accumulated value is returned.
+该 `reduce()` 接受 3 个参数:一个 reducer 函数、一个累计的初始值和一个用于遍历的数组。对数组中的每个元素都会调用 reducer,传入累计器和当前数组元素,返回值又会赋给累计器。对数组中的所有元素都执行过 reducer 之后,返回最终的累计结果。
-In the usage example, we call reduce and pass it the function, `(acc, curr) => acc + curr`, which takes the accumulator and the current value in the list and returns a new accumulated value. Next we pass an initial value, `0`, and finally, the data to iterate over.
+在用例中,调用 `reduce` 并传给它 3 个参数:`reducer` 函数、初始值 0 以及需要遍历的数组。其中 `reducer` 函数以累计器和当前数组元素为参数,返回累计后的结果。
-With the iteration and value accumulation abstracted, now we can implement a more generalized `filter()` function:
+如此将遍历和累计的过程抽象出来之后,便可实现更为通用的 `filter()` 函数:
```
-const filter = (
+ const filter = (
fn, arr
) => reduce((acc, curr) => fn(curr) ?
acc.concat([curr]) :
@@ -93,11 +93,11 @@ const filter = (
);
```
-In the `filter()` function, everything is shared except the `fn()` function that gets passed in as an argument. That `fn()`argument is called a predicate. A **predicate** is a function that returns a boolean value.
+在此 `filter()` 函数中,除了以参数形式传进来的 `fn()` 函数以外,所有代码都是可复用的。其中 `fn()` 参数被称为**断言(predicate)** —— 返回一个布尔值的函数。
-We call `fn()` with the current value, and if the `fn(curr)` test returns `true`, we concat the `curr` value to the accumulator array. Otherwise, we just return the current accumulator value.
+将当前值传给 `fn()`,如果 `fn(curr)` 返回 `true`,则将 `curr` 添加到结果数组中并返回之;否则,直接返回当前数组。
-Now we can implement `censor()` with `filter()` to filter out 4-letter words:
+现在便可借助 `filter()` 函数来实现过滤 4 字母单词的 `censor()` 函数:
```
const censor = words => filter(
@@ -106,20 +106,20 @@ const censor = words => filter(
);
```
-Wow! With all the common stuff abstracted out, `censor()` is a tiny function.
+喔!将所有公共代码抽象出来之后,`censor()` 函数便十分简洁了。
-And so is `startsWithS()`:
+`startsWithS()` 也是如此:
```
-const startsWithS = words => filter(
+ const startsWithS = words => filter(
word => word.startsWith('s'),
words
);
```
-If you’re paying attention, you probably know that JavaScript has already done this abstraction work for us. We have the `Array.prototype` methods, `.reduce()` and `.filter()` and `.map()` and a few more for good measure.
+ 你若稍加留意便会发现 JavaScript 其实已经为我们做了这些抽象,即 `Array.prototype` 的相关方法,例如 `.reduce()`、`.filter()`、`.map()` 等等。
-Higher order functions are also commonly used to abstract how to operate on different data types. For instance, `.filter()` doesn't have to operate on arrays of strings. It could just as easily filter numbers, because you can pass in a function that knows how to deal with a different data type. Remember the `highpass()` example?
+ 高阶函数也常常被用于对不同数据类型的操作进行抽象。例如,`.filter()` 函数不一定非得作用于字符串数组。只需传入一个能够处理不同数据类型的函数,`.filter()` 便能过滤数字了。还记得 `highpass` 的例子吗?
```
const highpass = cutoff => n => n >= cutoff;
@@ -127,22 +127,21 @@ const gt3 = highpass(3);
[1, 2, 3, 4].filter(gt3); // [3, 4];
```
-In other words, you can use higher order functions to make a function polymorphic. As you can see, higher order functions can be a whole lot more reusable and versatile than their first order cousins. Generally speaking, you’ll use higher order functions in combination with very simple first order functions in your real application code.
+换言之,高阶函数可以用来实现函数的多态性。如你所见,相较于一阶函数而言,高阶函数的复用性和通用性非常好。一般来讲,在实际编码中会组合使用高阶函数和一些非常简单的一阶函数。
-[**Continue to “Reduce” >**](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d)
+[**再续 “Reduce” >**](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d)
-### Next Steps ###
+### 接下来 ###
-Want to learn more about functional programming in JavaScript?
+想学习更多 JavaScript 函数式编程吗?
-[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/). If you’re not a member, you’re missing out!
+[跟着 Eric Elliott 学 Javacript](http://ericelliottjs.com/product/lifetime-access-pass/),机不可失时不再来!
-[
-](https://ericelliottjs.com/product/lifetime-access-pass/)
+[](https://ericelliottjs.com/product/lifetime-access-pass/)
-***Eric Elliott*** is the author of [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly), and *[*“Learn JavaScript with Eric Elliott”*](http://ericelliottjs.com/product/lifetime-access-pass/). He has contributed to software experiences for **Adobe Systems, Zumba Fitness, The Wall Street Journal, ESPN, BBC, and top recording artists including Usher, Frank Ocean, Metallica**, and many more.
+**Eric Elliott** 是 [**“编写 JavaScript 应用”**](http://pjabook.com) (O’Reilly) 以及 [**“跟着 Eric Elliott 学 Javascript”**](http://ericelliottjs.com/product/lifetime-access-pass/) 两书的作者。他为许多公司和组织作过贡献,例如 **Adobe Systems**、**Zumba Fitness**、**The Wall Street Journal**、**ESPN** 和 **BBC**等 , 也是很多机构的顶级艺术家,包括但不限于 **Usher**、**Frank Ocean** 以及 **Metallica**。
-*He spends most of his time in the San Francisco Bay Area with the most beautiful woman in the world.*
+大多数时间,他都在 San Francisco Bay Area,同这世上最美丽的女子在一起。
---
From c38df1b848dcd5e272debf8e17d3d0d99d7c3452 Mon Sep 17 00:00:00 2001
From: lsvih
Date: Wed, 12 Apr 2017 12:57:17 +0800
Subject: [PATCH 191/945] =?UTF-8?q?=E4=BA=8C=E6=A0=A1=E6=84=8F=E8=A7=81?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/webpack-and-rollup-the-same-but-different.md | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/TODO/webpack-and-rollup-the-same-but-different.md b/TODO/webpack-and-rollup-the-same-but-different.md
index b80a7886547..dd15131dc7b 100644
--- a/TODO/webpack-and-rollup-the-same-but-different.md
+++ b/TODO/webpack-and-rollup-the-same-but-different.md
@@ -2,7 +2,7 @@
> * 原文作者:[Rich Harris](https://medium.com/@Rich_Harris?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[lsvih](https://github.com/lsvih)
-> * 校对者:
+> * 校对者:[avocadowang](https://github.com/avocadowang),[Aladdin-ADD](https://github.com/Aladdin-ADD)
# 同中有异的 Webpack 与 Rollup #
@@ -18,14 +18,14 @@
Webpack 由 [Tobias Koppers](https://medium.com/@sokra) 在 2012 年创建,用于解决当时的工具不能处理的问题:构建复杂的单页应用(SPA)。尤其是它的两个特点改变了一切:
-1. **代码分割**可以将你的 app 分割成许多个容易管理的分块,这些分块能够在用户使用你的 app 时按需加载。这意味着你的用户可以有更快的交互体验。因为访问那些网站必须要等待整个应用都被下载并解析完成。当然,你**也可以**自己手动去进行代码分割,但是……总之,祝你好运。
+1. **代码分割**可以将你的 app 分割成许多个容易管理的分块,这些分块能够在用户使用你的 app 时按需加载。这意味着你的用户可以有更快的交互体验。因为访问那些没有使用代码分割的应用时,必须要等待整个应用都被下载并解析完成。当然,你**也可以**自己手动去进行代码分割,但是……总之,祝你好运。
2. **静态资源**的导入:图片、CSS 等静态资源可以直接导入到你的 app 中,就和其它的模块、节点一样能够进行依赖管理。因此,我们再也不用小心翼翼地将各个静态文件放在特定的文件夹中,然后再去用脚本给文件 URL 加上哈希串了。Webpack 已经帮你完成了这一切。
而 Rollup 的开发理念则不同:它利用 ES2015 模块的巧妙设计,尽可能高效地构建精简且易分发的 JavaScript 库。而其它的模块打包器(包括 Webpack在内)都是通过将模块分别封装进函数中,然将这些函数通过能在浏览器中实现的 `require` 方法打包,最后依次处理这些函数。在你需要实现按需加载的时候,这种做法非常的方便,但是这样做引入了很多无关代码,比较浪费资源。当[你有很多模块要打包的时候,这种情况会变得更糟糕](https://nolanlawson.com/2016/08/15/the-cost-of-small-modules/)。
ES2015 模块则启用了一种不同的实现方法,Rollup 用的也就是这种方法。所有代码都将被放置在同一个地方,并且会在一起进行处理。因此得到的最终代码相较而言会更加的精简,运行起来自然也就更快。你可以[点击这儿亲自试试 Rollup 交互式解释器(REPL)](https://rollupjs.org/repl)。
-但这儿也存在一些需要权衡的点:代码分割是一个很棘手的问题,而 Rollup 并不能做到这一点。同样的,Rollup 也不支持模块热替换(HMR)。而且对于打算使用 Rollup 的人来说,还有一个最大的痛点:它通过[插件](https://github.com/rollup/rollup-plugin-commonjs)处理大多数 CommonJS 文件的时候,一些代码将无法被翻译回 ES2015。而与之相反,你可以把这一切的事全部放心交给 Webpack 去处理。
+但这儿也存在一些需要权衡的点:代码分割是一个很棘手的问题,而 Rollup 并不能做到这一点。同样的,Rollup 也不支持模块热替换(HMR)。而且对于打算使用 Rollup 的人来说,还有一个最大的痛点:它通过[插件](https://github.com/rollup/rollup-plugin-commonjs)处理大多数 CommonJS 文件的时候,一些代码将无法被翻译为 ES2015。而与之相反,你可以把这一切的事全部放心交给 Webpack 去处理。
### 那么我到底应该选用哪一个呢? ###
From 60d29b02c6319a3063e1f9630a741a3f84261cd2 Mon Sep 17 00:00:00 2001
From: lsvih
Date: Wed, 12 Apr 2017 12:57:44 +0800
Subject: [PATCH 192/945] =?UTF-8?q?=E4=BA=8C=E6=A0=A1=E6=84=8F=E8=A7=81?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
From 6cc14ad5fa43bbcfee01296502a37a1ec1fe999b Mon Sep 17 00:00:00 2001
From: reid
Date: Wed, 12 Apr 2017 15:23:34 +0800
Subject: [PATCH 193/945] fix a typo
---
...bpack-bits-getting-the-most-out-of-the-commonschunkplugin.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md b/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
index 20692bd6b00..228984136a1 100644
--- a/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
+++ b/TODO/webpack-bits-getting-the-most-out-of-the-commonschunkplugin.md
@@ -10,7 +10,7 @@ webpack 核心团队隔三差五地就会在 Twitter 上作一些寓教于乐的
![Markdown](http://i4.buimg.com/1949/614a949156a09f9e.png)
-这次的“游戏规则”很简单:安装 `webpacl-bundle-analyzer`,生成一张包含所有 bundles 信息的酷炫图片分享给我,然后 webpack 团队会帮忙指出任何潜在的问题。
+这次的“游戏规则”很简单:安装 `webpack-bundle-analyzer`,生成一张包含所有 bundles 信息的酷炫图片分享给我,然后 webpack 团队会帮忙指出任何潜在的问题。
### 我们发现了什么? ###
From ed2b2bf92d3d20a03c69f824b063742d52ea65c8 Mon Sep 17 00:00:00 2001
From: DeepMissea <398752853@qq.com>
Date: Wed, 12 Apr 2017 19:15:35 +0800
Subject: [PATCH 194/945] =?UTF-8?q?=E6=96=9C=20=E2=99=82=20=E4=BD=93?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/mvvmc-with-swift.md | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/TODO/mvvmc-with-swift.md b/TODO/mvvmc-with-swift.md
index eb77db7fd87..a4119cbe373 100644
--- a/TODO/mvvmc-with-swift.md
+++ b/TODO/mvvmc-with-swift.md
@@ -454,15 +454,15 @@ MVVM-C 有很多优点,可以提高应用程序的质量。你应该注意使
# FAQ
-***MVVM-C 有什么限制吗?***
+**MVVM-C 有什么限制吗?**
是的,当然有。如果你正做一个复杂的项目,你可能会遇到一些边缘案例,MVVM-C 可能无法使用,或者在一些小功能上使用过度。如果你开始使用 MVVM-C,并不意味着你必须在每个地方都强制的使用它,你应该始终选择更适合你需求的架构。
-***我能用 RxSwift 同时使用函数式和命令式编程吗?***
+**我能用 RxSwift 同时使用函数式和命令式编程吗?**
是的,你可以。但是我建议你在遗留的代码中保持命令式的方式,而在新的实现里使用函数式编程,这样你可以利用 RxSwift 强大的优势。如果你使用 RxSwift 仅仅为了 UI 绑定,你可以轻松使用命令式编写程序,而只用函数响应式编程来设置绑定。
-***我可以在企业项目中使用 RxSwift 吗?***
+**我可以在企业项目中使用 RxSwift 吗?**
这取决于你要开新项目,还是要维护旧代码。在有遗留代码的项目中,你可能无法使用 RxSwift,因为你需要重构很多的类。如果你有时间和资源来做,我建议你新开一项目一点一点的做,否则还是尝试其他的方法来解决 UI 绑定的问题。
From 38d15c20727c6208b97ab4ac9b5ae6b2c3447667 Mon Sep 17 00:00:00 2001
From: zhuzi
Date: Wed, 12 Apr 2017 22:24:57 +0800
Subject: [PATCH 195/945] fix Punctuation
---
TODO/beyond-browser-web-desktop-apps.md | 22 +++++++++++-----------
1 file changed, 11 insertions(+), 11 deletions(-)
diff --git a/TODO/beyond-browser-web-desktop-apps.md b/TODO/beyond-browser-web-desktop-apps.md
index 1899ee8f1d9..f40c59cd68e 100644
--- a/TODO/beyond-browser-web-desktop-apps.md
+++ b/TODO/beyond-browser-web-desktop-apps.md
@@ -16,7 +16,7 @@
### 为什么?
-首先,为什么会有人开发桌面应用?任何现有的 web 应用(不同于网站,如果你认为它们是不同的)都可能适合变成一个桌面应用。你可以围绕任何可以从与用户系统集成中获益的 web 应用构建桌面应用;例如本地通知,开机启动,与文件的交互等。有些用户单纯更喜欢在自己的电脑中永久保存一些 app,无论是否联网都可以访问。
+首先,为什么会有人开发桌面应用?任何现有的 web 应用(不同于网站,如果你认为它们是不同的)都可能适合变成一个桌面应用。你可以围绕任何可以从与用户系统集成中获益的 web 应用构建桌面应用;例如本地通知、开机启动、与文件的交互等。有些用户单纯更喜欢在自己的电脑中永久保存一些 app,无论是否联网都可以访问。
也许你有个想法,但只能用作桌面应用,有些事情只是在 web 应用中不可能实现(至少还有一点,但更多的是这一点)。你可能想要为公司内部创建一个独立的功能性应用程序,而不需要任何人安装除了你的 app 之外的任何内容(因为内置 Node.js )。也许你有个有关 Mac 应用商店的想法,也许只是你的一个个人兴趣的小项目。
@@ -26,23 +26,23 @@
### NW.js
-桌面应用已经有很长一段时间了,我知道你没有很多时间,所以我们跳过一些历史,从2011年的上海开始。来自 Intel 开源技术中心的 Roger Wang 开发了 node-webkit,一个概念验证的 Node.js 模块,这个模块可以让用户创建一个 WebKit 内核的浏览器窗口并直接在 `
```
-As you may realize, this URL will make the browser run the injected script and send the user’s cookies, potentially including confidential session cookies, to evil.com!
+> As you may realize, this URL will make the browser run the injected script and send the user’s cookies, potentially including confidential session cookies, to evil.com!
+
+你可能意识到了,这个 URL 会让浏览器执行注入的脚本,并发送用户的 cookies,极有可能包含机密的会话 cookie,至 evil.com。
-To help protect users against reflective XSS attacks, some browsers have implemented protection mechanisms. These mechanisms try to identify these attacks by looking for matching code patterns in the HTTP request and response. Internet Explorer was the first browser to introduce such a mechanism with its XSS filter, introduced in Internet Explorer 8 back in 2008, and WebKit later introduced XSS Auditor, available today in Chrome and Safari. (Firefox has no similar mechanism built in, but users can use add-ons to gain this functionality.) These various protection mechanisms are not perfect: They may fail to detect a real XSS attack (a false negative), and in other cases may block legitimate code (a false positive). Due to the latter, browsers allow users to disable the XSS filter via the settings. Unfortunately, this is typically a global setting, which turns off this security feature completely for all web apps loaded by the browser.
+> To help protect users against reflective XSS attacks, some browsers have implemented protection mechanisms. These mechanisms try to identify these attacks by looking for matching code patterns in the HTTP request and response. Internet Explorer was the first browser to introduce such a mechanism with its XSS filter, introduced in Internet Explorer 8 back in 2008, and WebKit later introduced XSS Auditor, available today in Chrome and Safari. (Firefox has no similar mechanism built in, but users can use add-ons to gain this functionality.) These various protection mechanisms are not perfect: They may fail to detect a real XSS attack (a false negative), and in other cases may block legitimate code (a false positive). Due to the latter, browsers allow users to disable the XSS filter via the settings. Unfortunately, this is typically a global setting, which turns off this security feature completely for all web apps loaded by the browser.
-Luckily, there is a way for a web app to override this configuration and ensure that the XSS filter is turned on for the web app being loaded by the browser. This is done via the `X-XSS-Protection` header. This header, supported by Internet Explorer (from version 8), Edge, Chrome and Safari, instructs the browser to turn on or off the browser’s built-in protection mechanism and to override the browser’s local configuration.
+为了帮助保护用户抵抗反射型 XSS 攻击,有些浏览器实施了保护机制。这些保护机制尝试通过在 HTTP 请求和响应中寻找匹配的代码模式来辨识这些攻击。Internet Explorer 是第一个推出这种机制的,在 2008 年的 IE 8 中引入了 XSS 过滤器的机制,而 WebKit 后来推出了 XSS 审计,现今在 Chrome 和 Safari 上可用。(Firefox 没有内置类似的机制,但是用户可以使用插件来获得此功能)。这些保护机制并不完美,它们可能无法检测到真正的 XSS 攻击(漏报),在其他情况可能会阻止合法代码(误判)。由于后一种情况的出现,浏览器允许用户可设置禁用 XSS 过滤功能。不幸的是,这通常是一个全局设置,这会完全关闭所有浏览器加载的 web 应用程序的安全功能。
-`X-XSS-Protection` directives include these:
+> Luckily, there is a way for a web app to override this configuration and ensure that the XSS filter is turned on for the web app being loaded by the browser. This is done via the `X-XSS-Protection` header. This header, supported by Internet Explorer (from version 8), Edge, Chrome and Safari, instructs the browser to turn on or off the browser’s built-in protection mechanism and to override the browser’s local configuration.
-- `1` or `0`
+幸运的是,有方法可以让 web 应用程序覆盖此配置,并确保浏览器加载的 web 应用已打开 XSS 过滤器。这是通过设定 `X-XSS-Protection` 响应头来达到的。此响应头支持 Internet Explorer (8以上)、Edge、Chrome 和 Safar,指示浏览器打开或关闭浏览器内置的保护机制,及覆盖浏览器的本地配置。
-This enables or disables the filter.
+`X-XSS-Protection` 指令包括:
+
+- `1` 或者 `0`
+
+使用或禁用 CSS 过滤器。
- `mode=block`
-This instructs the browser to prevent the entire page from rendering when an XSS attack is detected.
+> This instructs the browser to prevent the entire page from rendering when an XSS attack is detected.
+
+当检测到 XSS 攻击时,这会指示浏览器不渲染整个页面。
-I recommend always turning on the XSS filter, as well as block mode, to maximize user protection. Such a response header looks like this:
+> I recommend always turning on the XSS filter, as well as block mode, to maximize user protection. Such a response header looks like this:
+
+我推荐永远打开 XSS 过滤器以及 block 模式,以求最大化保护用户。这样的响应头应该是这样的:
```
X-XSS-Protection: 1; mode=block
```
-Here’s how you would configure this response header in Node.js:
+以下是在 Node.js 中配置此响应头的方法:
-```
+```javascript
functionrequestHandler(req, res){
res.setHeader('X-XSS-Protection','1;mode=block');}
```
From d9d798c63b6ebe5e5b001cdc6323f7d10a4f6440 Mon Sep 17 00:00:00 2001
From: lsvih
Date: Wed, 12 Apr 2017 23:53:12 +0800
Subject: [PATCH 207/945] Update swift-lazy-initialization-with-closures.md
---
...swift-lazy-initialization-with-closures.md | 230 ++++++++++--------
1 file changed, 130 insertions(+), 100 deletions(-)
diff --git a/TODO/swift-lazy-initialization-with-closures.md b/TODO/swift-lazy-initialization-with-closures.md
index 051ab41f287..41885f1449f 100644
--- a/TODO/swift-lazy-initialization-with-closures.md
+++ b/TODO/swift-lazy-initialization-with-closures.md
@@ -1,23 +1,23 @@
-> * 原文地址:[Swift Lazy Initialization with Closures](https://blog.bobthedeveloper.io/swift-lazy-initialization-with-closures-a9ef6f6312c)
-> * 原文作者:[Bob Lee](https://blog.bobthedeveloper.io/@bobthedev)
-> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 原文地址:[Swift Lazy Initialization with Closures][1]
+> * 原文作者:[Bob Lee][2]
+> * 译文出自:[掘金翻译计划][3]
+> * 译者:[lsvih][4]
> * 校对者:
-# Swift Lazy Initialization with Closures #
+# 在 Swift 中使用闭包实现懒加载
-## Learn how to create objects with modularity and readability ##
+## 学习如何兼顾模块化与可读性来创建对象
![](https://cdn-images-1.medium.com/max/2000/1*KNmIy5QAOeokXPW86TtVyA.png)
-Magic Keyboard 2 and Magic Mouse 2
+(图为苹果的 Magic Keyboard 2 与 Magic Mouse 2)
-*Welcome my lovely readers. Good to see you here today. For those who are new, I’m Bob. Just for real quick, if you wish to be on my mailing list, you can sign up* [*here*](https://boblee.typeform.com/to/oR9Nt2) *and get more value for your learning with iOS development :)*
+*亲爱的读者你们好!我是 Bob,很高兴能在这篇文章中与你们相遇!如你想加入我的邮件列表,获取更多学习 iOS 开发的文章,请点击*[*这儿*][5]*注册,很快就能完成的哦 :)*
-### Motivation ###
+### 动机
-In the beginning of my iOS journey, I followed tutorials on YouTube. I saw a few using something like below to create UI objects.
+在我刚开始学习 iOS 开发的时候,我在 YouTube 上找了一个教程。我发现这个教程有时候会用下面这种方式来创建 UI 对象:
```
let makeBox: UIView = {
@@ -26,42 +26,42 @@ let makeBox: UIView = {
}()
```
-As a learner, I copied the practice and used it. One day, however, one of my readers asked me, “why do you add `{}` and why does `()` exist at the end? Is it a computed property?” I could not answer. I was a zombie.
+作为一个初学者,我自然而然地复制并使用了这个例子。直到有一天,我的一个读者问我:“为什么你要加上`{}`呢?最后为什么要加上一对`()`呢?这是一个计算属性吗?”我哑口无言,因为我自己也不知道答案。
-I wrote this tutorial for my younger self. Yet, some may find it useful.
+因此,我为过去年轻的自己写下了这份教程。说不定还能帮上其他人的忙。
-### Objectives ###
+### 目标
-There are three objectives. First, understand how to initialize an object using the unconventional way as shown above. Second, learn when to use `lazy var` in Swift. Last, join my mailing list.
+这篇教程有一下三个目标:第一,了解如何像前面的代码一样,非常规地创建对象;第二,知道编在写 Swfit 代码时,什么时候该使用 `lazy var`;第三,快加入我的邮件列表呀。
-#### Prerequisites ####
+#### 预备知识
-To fully enjoy the ride with me, I highly recommend you to be familiar with the topics below.
+为了让你能轻松愉快地和我一起完成这篇教程,我强烈推荐你先了解下面这几个概念。
-1. [*Closures*](https://blog.bobthedeveloper.io/no-fear-closure-in-swift-3-with-bob-72a10577c564)
-2. [*Capture List and retention cycle [weak self]*](https://blog.bobthedeveloper.io/swift-retention-cycle-in-closures-and-delegate-836c469ef128)
-3. *Descent Object Oriented Programming*
+1. [*闭包*][6]
+2. [*捕获列表与保留周期 \[weak self]*][7]
+3. *面向对象程序设计*
-### Create UI Components ###
+### 创建 UI 组件
-Before I explain the unconventional method above, let’s look into your past. In order to create a button in Swift, you probably have done something like this,
+在我介绍“非常规”方法之前,让我们先复习一下“常规”方法。在 Swift 中,如果你要创建一个按钮,你应该会这么做:
```
-// Determine Size
+// 设定尺寸
let buttonSize = CGRect(x: 0, y: 0, width: 100, height: 100)
-// Create Instance
+// 创建控件
let bobButton = UIButton(frame: buttonSize)
bobButton.backgroundColor = .black
bobButton.titleLabel?.text = "Bob"
bobButton.titleLabel?.textColor = .white
```
-This is *Okay.*
+这样做**没问题**。
-Assume, you have to create three other buttons, you probably have to copy the code above and then change the name from `bobButton` to `bobbyButton`.
+假设现在你要创建另外三个按钮,你很可能会把上面的代码复制,然后把变量名从 `bobButton` 改成 `bobbyButton`。
-It’s quite tedious.
+这未免也太单调了吧。
```
// New Button
@@ -71,13 +71,13 @@ bobbyButton.titleLabel?.text = "Bob"
bobbyButton.titleLabel?.textColor = .white
```
-To make things just a bit easier, you may
+为了方便,你可以:
![](https://cdn-images-1.medium.com/max/800/1*oDIPy0i4YzUnKVR4XYI4kg.gif)
-This works too with the keyboard shortcut: ctrl-cmd-e
+使用快捷键:ctrl-cmd-e 来完成这个工作。
-If you don’t wish to repeat yourself, you may create a function instead.
+如果你不想做重复的工作,你也可以创建一个函数。
```
func createButton(enterTitle: String) -> UIButton {
@@ -86,27 +86,26 @@ func createButton(enterTitle: String) -> UIButton {
button.titleLabel?.text = enterTitle
return button
}
-
createButton(enterTitle: "Yoyo") // 👍
```
-However, in iOS development, it is rare that custom buttons look similar. Therefore, a function may require a lot more parameters including background color, title, border radius, shadow, and so on. You function may end up looking like,
+然而,在 iOS 开发中,很少会看到一堆一模一样的按钮。因此,这个函数需要接受更多的参数,如背景颜色、文字、圆角尺寸、阴影等等。你的函数最后可能会变成这样:
```
func createButton(title: String, borderWidth: Double, backgrounColor, ...) -> Button
```
-The code above is not ideal even if you add default parameters to the function. It decreases readability. Therefore, it’s better to stay with the tedious method above.
+但是,上面的代码并不理想。即使你为这个函数加上了默认参数,它的可读性很差。因此,比起这个方法,我们还是采用上面那个”单调“的方法为妙。
-But, is there any way we can make it less tedious and more organized? Of course. We’ve looked into your past — It’s time to step up and look into your future.
+到底有没有办法让我们既不那么无聊,还能让代码更有条理呢?当然咯。我们现在只是复习你过去的做法——是时候更上一层楼,展望你未来的做法了。
-### Introducing the Unconventional Way ###
+### 介绍”非常规“方法
-Before we create UI components with the unconventional way, let’s first answer the initial questions my reader asked. What does `{}` mean, and is it a `computed property`?
+在我们使用”非常规“方法创建 UI 组件之前,让我们先回答一下最开始那个读者的问题。`{}`是什么意思,它是一个`计算属性`吗?
-*Nope, it’s just a* ***closure block***.
+*当然不是,它只是一个* ***闭包***。
-First, let’s demonstrate how to create an object using a closure. We will design a struct called `Human`.
+首先,让我来示范一下如何用闭包来创建一个对象。我们设计一个名为`Human`的结构:
```
struct Human {
@@ -116,7 +115,7 @@ struct Human {
}
```
-Now, this is how you create an object with a closure
+现在,让你看看怎么用闭包创建对象:
```
let createBob = { () -> Human in
@@ -127,11 +126,11 @@ let createBob = { () -> Human in
let babyBob = createBob() // "Born 1996"
```
-*If the syntax above doesn’t look familiar to you, you may stop reading now, and go to* [*Fear No Closure with Bob*](https://blog.bobthedeveloper.io/no-fear-closure-in-swift-3-with-bob-72a10577c564) *, and bring some bullets.*
+*如果你不熟悉这段语法,请先停止阅读这篇文章,去看看* [*Fear No Closure with Bob*][8] *充充电吧。*
-Just to explain, `createBob` is a closure whose type is `() -> Human`. You’ve created an instance called, `babyBob` by calling `createBob()` .
+解释一下,`createBob` 是一个类型为 `()-> Human` 的闭包。你已经通过调用 `createBob()` 创建好了一个 `babyBob` 实例。
-However, you had to create two constants: `createBob` and `babyBob`. What if you want to do everything in a single statement? Here you go.
+然而,这样做你创建了两个常量:`createBob` 与 `babyBob`。如何把所有的东西都放在一个声明中呢?请看:
```
let bobby = { () -> Human in
@@ -140,11 +139,11 @@ let bobby = { () -> Human in
}()
```
-Now, the closure block executes itself through adding `()` at the end and `bobby` now has a `Human` object attached. Pretty good stuff.
+现在,这个闭包通过在最后加上 `()` 执行了自己,`bobby` 现在附上了一个 `Human` 对象。干的漂亮!
-**You’ve learned how to initialize an object with a closure block.**
+**现在你已经学会了使用闭包来创建一个对象**
-Now, let’s apply to creating an UI object which should be similar to the example right above.
+让我们应用这个方法,模仿上面的例子来创建一个 UI 对象吧。
```
let bobView = { () -> UIView in
@@ -153,8 +152,7 @@ let bobView = { () -> UIView in
return view
}()
```
-
-Great, we can make it shorter. In fact, we don’t need to specify the type of the closure block. Instead, all we have to do is specify the type of the instance, `bobView`, for example.
+很好,我们还能让它更简洁。实际上,我们不需要为闭包指定类型,我们只需要指定 `bobView` 实例的类型就够了。例如:
```
let bobbyView: **UIView** = {
@@ -164,17 +162,17 @@ let bobbyView: **UIView** = {
}()
```
-Swift is able to infer that the closure block is `() -> UIView` based on the keyword, `return`.
+Swift 能够通过关键字 `return` 推导出这个闭包的类型是 `() -> UIView`。
-Now, take a look. The example right above should look identical to the “unconventional way” I feared.
+现在看看,上面的例子已经和我之前害怕的“非常规方式”一样了。
-### Benefits of Init with Closures ###
+### 使用闭包创建的好处
-We discussed the tediousness of creating objects and the problem that arises from using a function. In your head, you must be thinking, “why should I use a closure block instead?”
+我们已经讨论了:直接创建对象是很单调无聊的,使用函数构造是会出现问题的。现在你可能会想“为什么我非得用闭包来创建?”
-#### Easy to Duplicate ####
+#### 重复起来更容易
-I don’t like to use Storyboard, I love copy and pasting UI objects. In fact, I’ve a “library” of code in my computer. Let us assume that there is a button as shown below in the library.
+我不喜欢用 Storyboard,我比较喜欢复制粘贴 UI 对象。实际上,在我电脑里有一个“库”的代码。假设库里有个按钮,代码如下:
```
let myButton: UIButton = {
@@ -188,14 +186,14 @@ return button
}()
```
-All I have to do is copy the entire lines, and then just change the name of `myButton` to `newButton` for the usage. Had I not used the closure method, I probably had to change the name of `button` to `newButton` 7–8 times. We could use the Xcode shortcut above, but why not make it just simpler.
+我只需要把它整个复制,然后把名字从 `myButton` 改成 `newButtom` 就行了。在我用闭包之前,我得重复地把 `myButton` 改成 `newButtom` ,甚至要改上七八遍。我们虽然可以用 Xcode 的快捷键,但为啥不使用闭包,让这件事更简单呢?
-#### Look Cleaner ####
+#### 看起来更干净
-Since objects are grouped together, it feels cleaner based on my eyes. Let’s compare
+由于对象对象会自己编好组,在我看来它更加的干净。让我们对比一下:
```
-// Init with Closure
+// 使用闭包创建
let leftCornerButton: UIButton = {
let button = UIButton(frame: buttonSize)
button.backgroundColor = .black
@@ -220,7 +218,7 @@ return button
vs
```
-// Init With Fingers
+// 使用我的手指创建..
let leftCornerButton = UIButton(frame: buttonSize)
leftCornerButton.backgroundColor = .black
leftCornerButton.titleLabel?.text = "Button"
@@ -236,81 +234,81 @@ rightCornerButton.layer.cornerRadius =
rightCornerButton.layer.masksToBounds = true
```
-Although creating an object with the closure add a couple lines more, I feel less overwhelmed since I only have to add attributes to `button` rather than `rightCornerButton` or `leftCornerButton`.
+虽然使用闭包创建对象甚至还多出了一行,但是比起要在 `rightCornerButton` 或者 `leftCornerButton` 后面狂加属性,我还是更喜欢在 `button` 后面加属性。
-*In fact, if the name of a button gets more descriptive, often times it requires fewer lines to create an object with a closure block.*
+*实际上如果按钮的命名特别详细时,用闭包创建对象还可以少几行。*
-**You’ve accomplished the first objective. Congratulations**
+**恭喜你,你已经完成了我们的第一个目标**
-### Lazy Init Application ###
+### 懒加载的应用
-You’ve come a long way. It’s time to meet the second objective of this tutorial.
+辛苦了!现在让我们来看看这个教程的第二个目标把。
-You might have seen something like this below
+你可能看过与下面类似的代码:
```
class IntenseMathProblem {
- lazyvar complexNumber: Int = {
- // imagine it requires a lot of CPU
+ lazy var complexNumber: Int = {
+ // 请想象这儿要耗费很多CPU资源
1 * 1
}()
}
```
-What `lazy` allows you to do is, the `complexNumber` property will be only calculated when you try to access it. For example,
+`lazy` 的作用是,让 `complexNumber` 属性只有在你试图访问它的时候才会被计算。例如:
```
let problem = IntenseMathProblem
-problem() // No value for complexNumber
+problem() // 此时complexNumber没有值
```
-Currently, there is no value for `complexNumber`. However, once you access the property,
+没错,现在 `complexNumber` 没有值。然而,一旦你访问这个属性:
```
-problem().complexNumber // Now returns 1
+problem().complexNumber // 现在回返回1
```
-The `lazy var` is often used to sort database and fetch data from any backend services because you definitely don’t want to calculate and sort everything when you create an object.
+`lazy var` 经常用于数据库排序或者从后端取数据,因为你并不想在创建对象的时候就把所有东西都计算、排序。
-*In fact, your phone will crash since the object is super bloated and the RAM can’t handle.*
+*实际上,由于对象太大了导致 RAM 撑不住,你的手机就会崩溃。*
-### Application ###
+### 应用
-Below is just an application of `lazy var`.
+以下是 `lazy var` 的应用:
-#### Sorting ####
+#### 排序
```
class SortManager {
lazy var sortNumberFromDatabase: [Int] = {
- // Sorting logic
+ // 排序逻辑
return [1, 2, 3, 4]
}()
}
```
-#### Image Compression ####
+#### 图片压缩
```
class CompressionManager {
lazy var compressedImage: UIImage = {
let image = UIImage()
- // Compress the image
- // Logic
+ // 压缩图片的
+ // 逻辑
return image
}()
}
```
-### Rules with `Lazy` ###
+### `Lazy`的一些规定
-1. You can’t use `lazy` with `let` since there is no initial value, and it is attained later when it is accessed.
-2. You can’t use it with a `computed property` since computed property is always recalculated (requires CPU) when you modify any of the variables that has a relationship with the `lazy` property.
-3. `Lazy` is only valid for members of a struct or class
+1. 你不能把 `lazy` 和 `let` 一起用,因为用 `lazy` 时没有初值,只有当被访问时才会获得值。
+2. 你不能把它和 `计算属性` 一起用,因为在你修改任何与 `lazy` 的计算属性有关的变量时,计算属性都会被重新计算(耗费 CPU 资源)。
+3. `Lazy` 只能是结构或类的成员。
-### Does Lazy Capture? ###
+### Lazy 能被捕获吗?
-So, if you’ve read the previous article on [Retention Cycle in Closures and Delegate](https://blog.bobthedeveloper.io/swift-retention-cycle-in-closures-and-delegate-836c469ef128) , you might wonder. Let’s test it out. Create a class called `BobGreet`. It has two properties: `name` whose type is `String` and `greeting` whose type is also `String` but initialized with a closure block.
+如果你读过我的前一篇文章[《Swift 闭包和代理中的保留周期》][9],你就会明白这个问题。让我们试一试吧。创建一个名叫 `BobGreet` 的类,它有两个属性:一个是类型为 `String` 的 `name`,一个是类型为 `String` 但是使用闭包创建的 `greeting`。
```
class BobGreet {
@@ -320,43 +318,75 @@ class BobGreet {
}()
deinit {
- print("I'm gone, bruh 🙆")}
+ print("I'm gone, bruh 🙆")}
}
}
```
-The closure block *might* have a strong reference to `BobGuest` but let’s attempt to deallocate.
+闭包**可能**对 `BobGuest` 有强引用,让我们尝试着 deallocate 它。
```
var bobGreet: BobGreet? = BobClass()
bobGreet?.greeting
-bobClass = nil // I'm gone, bruh 🙆
+bobClass = nil // I'm gone, bruh 🙆
```
-No need to worry about `[unowned self]` The closure block does not have a reference to the object. Instead, it just copies `self` within the closure block. If you are confused by the previous statement, feel free to read [Swift Capture Lists](https://blog.bobthedeveloper.io/swift-capture-list-in-closures-e28282c71b95) to learn more. 👍
+不用担心 `[unowned self]`,闭包并没有对对象存在引用。相反,它仅仅是在闭包内复制了 `self`。如果你对前面的代码声明有疑问,放松一会儿,读读 [Swift Capture Lists][10] 来了解更多这方面的知识。👍
-### Last Remarks ###
+### 最后的唠叨
-I learned a quite a bit while preparing for this tutorial. I hope you did as well. I’d appreciate your genuine fat ❤️. But, there is just one more. As the last objective, if you wish to on my mailing list and receive greater value from me, you can sign up right [**here**](https://boblee.typeform.com/to/oR9Nt2) .
+我在准备这篇教程的过程中也学到了很多,希望你也一样。感谢你们的热情❤️!不过这篇文章还剩一点:我的最后一个目标。如果你希望加入我的邮件列表以获得更多有价值的信息的话,你可以点 [**这里**][11]注册。
-As you can see by the cover photo, I recently bought Magic Keyboard and Mouse. They are pretty good and increase my productivity a lot. You can get the mouse [here](http://amzn.to/2noHxgl) or the keyboard [here](http://amzn.to/2noHxgl). I never regret despite the price. 😓
+正如封面照片所示,我最近买了 Magic Keyboard 和 Magic Mouse。它们超级棒,帮我提升了很多的效率。你可以在 [这儿][12]买鼠标,在 [这儿][13]买键盘。我才不会因为它们的价格心疼呢。😓
-> [Source Code](https://github.com/bobthedev/Blog_Lazy_Init_with_Closures)
+> [本文的源码][14]
-### Swift Conference I Will Join ###
+### 我将要参加 Swift 讨论会
-I will be joining my first conference @[SwiftAveir](https://twitter.com/SwiftAveiro) o from June 1–2. A friend of mine, [Joao](https://twitter.com/NSMyself) , is helping organize the conference, so I’m super excited. You can learn more about the event [here](http://swiftaveiro.xyz) !
+我将在 6 月 1 日至 6 月 2 日 参加我有生以来的第一次讨论会 @[SwiftAveir][15], 我的朋友 [Joao][16]协助组织了这次会议,所以我非常 excited。你可以点[这儿][17]了解这件事 的详情!
-#### Recommended Articles ####
+#### 文章推荐
-> Intro to Functional Programming ([Blog](https://blog.bobthedeveloper.io/intro-to-swift-functional-programming-with-bob-9c503ca14f13))
+> 函数式编程简介 ([Blog][18])
-> My Favorite Xcode Shortcuts ([Blog](https://blog.bobthedeveloper.io/intro-to-swift-functional-programming-with-bob-9c503ca14f13) )
+> 我最爱的 XCode 快捷键 ([Blog][19] )
-### Bob the Developer ###
+### 关于我
-I’m an iOS instructor from Seoul, 🇰🇷. Feel free to get to know me on [Instagram](https://instagram.com/bobthedev) . I post regular updates on [Facebook Page](https://facebook.com/bobthedeveloper) and 🖨 on Sat 8pm EST.
+我是一名来自首尔的 iOS 课程教师,你可以在 [Instagram][20] 上了解我。我会经常在 [Facebook Page][21] 投稿,投稿时间一般在北京时间上午9点(Sat 8pm EST)。
---
-> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
+> [掘金翻译计划][22] 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金][23] 上的英文分享文章。内容覆盖 [Android][24]、[iOS][25]、[React][26]、[前端][27]、[后端][28]、[产品][29]、[设计][30] 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划][31]。
+
+[1]: https://blog.bobthedeveloper.io/swift-lazy-initialization-with-closures-a9ef6f6312c
+[2]: https://blog.bobthedeveloper.io/@bobthedev
+[3]: https://github.com/xitu/gold-miner
+[4]: https://github.com/lsvih
+[5]: https://boblee.typeform.com/to/oR9Nt2
+[6]: https://blog.bobthedeveloper.io/no-fear-closure-in-swift-3-with-bob-72a10577c564
+[7]: https://juejin.im/post/58e4ac5d44d904006d2a9a19
+[8]: https://blog.bobthedeveloper.io/no-fear-closure-in-swift-3-with-bob-72a10577c564
+[9]: https://juejin.im/post/58e4ac5d44d904006d2a9a19
+[10]: https://blog.bobthedeveloper.io/swift-capture-list-in-closures-e28282c71b95
+[11]: https://boblee.typeform.com/to/oR9Nt2
+[12]: http://amzn.to/2noHxgl
+[13]: http://amzn.to/2noHxgl
+[14]: https://github.com/bobthedev/Blog_Lazy_Init_with_Closures
+[15]: https://twitter.com/SwiftAveiro
+[16]: https://twitter.com/NSMyself
+[17]: http://swiftaveiro.xyz
+[18]: https://blog.bobthedeveloper.io/intro-to-swift-functional-programming-with-bob-9c503ca14f13
+[19]: https://blog.bobthedeveloper.io/intro-to-swift-functional-programming-with-bob-9c503ca14f13
+[20]: https://instagram.com/bobthedev
+[21]: https://facebook.com/bobthedeveloper
+[22]: https://github.com/xitu/gold-miner
+[23]: https://juejin.im
+[24]: https://github.com/xitu/gold-miner#android
+[25]: https://github.com/xitu/gold-miner#ios
+[26]: https://github.com/xitu/gold-miner#react
+[27]: https://github.com/xitu/gold-miner#%E5%89%8D%E7%AB%AF
+[28]: https://github.com/xitu/gold-miner#%E5%90%8E%E7%AB%AF
+[29]: https://github.com/xitu/gold-miner#%E4%BA%A7%E5%93%81
+[30]: https://github.com/xitu/gold-miner#%E8%AE%BE%E8%AE%A1
+[31]: https://github.com/xitu/gold-miner
From 69bca85e75f4b99b33c193c21f577db93622ee8b Mon Sep 17 00:00:00 2001
From: gy134340
Date: Thu, 13 Apr 2017 00:20:55 +0800
Subject: [PATCH 208/945] =?UTF-8?q?=E6=A0=A1=E5=AF=B9=E4=BF=AE=E6=94=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...amming-in-javascript-composing-software.md | 74 +++++++++----------
1 file changed, 37 insertions(+), 37 deletions(-)
diff --git a/TODO/why-learn-functional-programming-in-javascript-composing-software.md b/TODO/why-learn-functional-programming-in-javascript-composing-software.md
index 3203cee2ce5..c260bf55297 100644
--- a/TODO/why-learn-functional-programming-in-javascript-composing-software.md
+++ b/TODO/why-learn-functional-programming-in-javascript-composing-software.md
@@ -4,29 +4,29 @@
> * 译者:
> * 校对者:
-# 为什么用 JavaScript 学习函数式编程?(组成化软件)(第二部分)
+# 为什么用 JavaScript 学习函数式编程?(软件构建)(第二部分)
烟雾的方块艺术 —MattysFlicks —(CC BY 2.0)
-> 注意:这是从基础学习函数式编程和使用 JavaScript ES6+ 组成软件的第一部分。保持关注,接下来还有很多!
+> 注意:这是从基础学习函数式编程和使用 JavaScript ES6+ 撰写软件的第二部分。保持关注,接下来还有很多!
> [从第一部分开始](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [接下来的 >](https://medium.com/javascript-scene/a-functional-programmers-introduction-to-javascript-composing-software-d670d14ede30#.2e4youss2)
-忘了你认为知道的关于 JavaScript 的一切,用初学者的眼光去看待它。为了帮助你做到这一点,我们将会从头复习一下 JavaScript 的基础,就像你与其尚未谋面一样。如果你是初学者,那你就很幸运了。最终从零开始探索 ES6 和函数式编程!希望所有的概念都被解释清楚 — 但不要太依赖于此。
+忘掉你认为知道的关于 JavaScript 的一切,用初学者的眼光去看待它。为了帮助你做到这一点,我们将会从头复习一下 JavaScript 的基础,就像你与其尚未谋面一样。如果你是初学者,那你就很幸运了。最终从零开始探索 ES6 和函数式编程!希望所有的概念都被解释清楚 — 但不要太依赖于此。
如果你是已经熟悉 JavaScript 或者纯函数式语言的老开发者了,也许你会认为 JavaScript 是探索函数式编程有趣的选择。把这些想法放在一边,用更开放的思想接触它,你会发现 JavaScript 编程更高层次的东西。一些你从来不知道的东西。
-由于这个被称为“组成式软件”,同时函数式编程是明显的组成软件的方法(使用函数组合,高阶函数等等),你也许想知道为什么我不用 Haskell, ClojureScript,或者 Elm,而是 JavaScript。
+由于这个被称为“组合式软件”,同时函数式编程是明显的构建软件的方法(使用函数组合,高阶函数等等),你也许想知道为什么我不用 Haskell、ClojureScript,或者 Elm,而是 JavaScript。
JavaScript 有函数式编程所需要的最重要的特性:
-1. **一级函数:**使用函数作为数据值的能力:用函数传参,返回函数,用函数做变量和对象属性。这个属性允许更高级别的函数,使局部应用、柯里化和组合成为可能。
-2. **匿名函数和简介的 lambda 语法:**`x => x * 2` 是 JavaScript 中有效的函数表达式。简介的 lambda 语法使得它更好的跟高阶函数合作。
-3. **闭包:**闭包是一个有着自己独立作用域的捆绑函数。闭包在函数被创建时被创建。当一个函数在另一个函数内部被创建,它可以访问外部函数的变量,即使在外部函数退出后。闭包时使局部应用或者固定的参数。固定的参数时绑定在返回函数的作用域范围内的参数。在 `add2(1)(2)` 中,`1` 是 `add2(1)` 返回的函数中的固定参数。
+1. **一级公民函数:**使用函数作为数据值的能力:用函数传参,返回函数,用函数做变量和对象属性。这个属性允许更高级别的函数,使偏函数应用、柯里化和组合成为可能。
+2. **匿名函数和简洁的 lambda 语法:**`x => x * 2` 是 JavaScript 中有效的函数表达式。简洁的 lambda 语法使得高阶函数变的简单。
+3. **闭包:**闭包是一个有着自己独立作用域的捆绑函数。闭包在函数被创建时被创建。当一个函数在另一个函数内部被创建,它可以访问外部函数的变量,即使在外部函数退出后。通过闭包偏函数应用可以获取内部固定参数。固定的参数时绑定在返回函数的作用域范围内的参数。在 `add2(1)(2)` 中,`1` 是 `add2(1)` 返回的函数中的固定参数。
### JavaScript 缺少了什么
-JavaScript 是多范式语言,意味着它支持多种风格的编程。其他被 JavaScript 支持的风格包括过程式的(命令式)编程(比如 C),把函数看作可以被重复调用和组织的子程序指令;面向对象编程,对象 — 而不是函数 — 作为初始构造块;当然,还有函数式编程。多范式编程语言的劣性在于命令式和面向对象往往意味着所有东西都是可变的。
+JavaScript 是多范式语言,意味着它支持多种风格的编程。其他被 JavaScript 支持的风格包括过程式(命令式)编程(比如 C),把函数看作可以被重复调用和组织的子程序指令;面向对象编程,对象— 而不是函数— 作为初始构造块;当然,还有函数式编程。多范式编程语言的劣性在于命令式和面向对象往往意味着所有东西都是可变的。
可变性指的是数据结构上的变化。比如:
@@ -34,68 +34,68 @@ JavaScript 是多范式语言,意味着它支持多种风格的编程。其他
bar: 'baz'
};
- foo.bar = 'qux'; // mutation
+ foo.bar = 'qux'; // 改变
对象通常需要可变性以便于被方法更新值,在命令式的语言中,大部分的数据结构可变以便于数组和对象的高效操作。
下面是一些函数式语言拥有但是 JavaScript 没有的特性:
1. **纯粹性:**在一些函数式语言中,纯粹性是强制的,有副作用的表达式是不被允许的。
-2. **不可变性:**一些函数式语言不允许转变,采用表达式来产生新的数据结构来代替更改一个已存的数据结构,比如说数组或者对象。这样看起来可能不够搞笑,但是大多数函数式语言在引擎下使用 trie 数据结构,具有结构共享的特点:意味着旧的对象和新的对象是对相同数据的引用。
+2. **不可变性:**一些函数式语言不允许转变,采用表达式来产生新的数据结构来代替更改一个已存的数据结构,比如说数组或者对象。这样看起来可能不够高效,但是大多数函数式语言在引擎下使用 trie 数据结构,具有结构共享的特点:意味着旧的对象和新的对象是对相同数据的引用。
3. **递归:**递归是函数引用自身来进行迭代的能力。在大多数函数式语言中,递归是迭代的唯一方式,它们没有像 `for` 、`while`、`do` 这类循环语句。
**纯粹性:**在 JavaScript 中,纯粹性由约定来达成,如果你不是使用纯函数来构成你的大多数应用,那么你就不是在进行函数式风格的编程。很不幸,在 JavaScript 中,你很容易就会不小心创建和使用一些不纯的函数。
-**不可变性:**在纯函数式语言中,可变性通常是强制的,JavaScript 缺少函数式语言中高效的、基于 trie 树的数据结构,但是又一些你可以使用的库,包括 [Immutable.js](https://facebook.github.io/immutable-js/) 和 [Mori](https://github.com/swannodette/mori),真是期望未来的 ECMAScript 规范版本可以拥抱不可变数据结构。
+**不可变性:**在纯函数式语言中,不可变性通常是强制的,JavaScript 缺少函数式语言中高效的、基于 trie 树的数据结构,但是你可以使用一些库,包括 [Immutable.js](https://facebook.github.io/immutable-js/) 和 [Mori](https://github.com/swannodette/mori),由衷期望未来的 ECMAScript 规范版本可以拥抱不可变数据结构。
-有一些未来的迹象,比如说在 ES6 中添加了 `const` 关键字,`const` 声明的变量不能被重新赋值,知道 `const` 并不实际代表不可改变的值也很重要。
+有一些迹象带来了希望,比如说在 ES6 中添加了 `const` 关键字,`const` 声明的变量不能被重新赋值,重要的是要理解 `const` 所声明的值并不是不可改变的。
-`const` 声明的对象不能被重新声明为新的对象,但是对象的属性却是可变的,JavaScript 有 `freeze()` 对象的能力,但是这些对象只能在根实例上被冻结,意味着嵌套着的对象还是可以改变它的属性。换句话说,在 JavaScript 规范中看到真正的不可变还有很多路要走。
+`const` 声明的对象不能被重新声明为新的对象,但是对象的属性却是可变的,JavaScript 有 `freeze()` 对象的能力,但是这些对象只能在根实例上被冻结,意味着嵌套着的对象还是可以改变它的属性。换句话说,在 JavaScript 规范中看到真正的不可变还有很长的路要走。
-**递归:**JavaScript 支持递归,但是大多数函数式语言都有尾部调用优化的特性,尾部调用优化是一个允许递归的函数重用堆栈帧来递归调用的特性。
+**递归:**JavaScript 技术上支持递归,但是大多数函数式语言都有尾部调用优化的特性,尾部调用优化是一个允许递归的函数重用堆栈帧来递归调用的特性。
-没有尾部调用优化,一个调用的栈很可能没有边界导致堆栈溢出。JavaScript 在 ES6 规范中有一个有限的尾部调用优化。不幸的是,只有一个主要的浏览器引擎支持它,这个优化被部分应用随后从 Babel(最流行的 JavaScript 编译器,在旧的浏览器中被用来把 ES6 编译到 ES5)。
+没有尾部调用优化,一个调用的栈很可能没有边界导致堆栈溢出。JavaScript 在 ES6 规范中有一个有限的尾调用优化。不幸的是,只有一个主要的浏览器引擎支持它,这个优化被部分应用随后从 Babel(最流行的 JavaScript 编译器,在旧的浏览器中被用来把 ES6 编译到 ES5) 中移除。
-最后一行:现在使用递归来作为大的迭代还不是很安全 — 即使你很小心的调用尾部的函数。
+最重要的事实:现在使用递归来作为大的迭代还不是很安全 — 即使你很小心的调用尾部的函数。
-### 什么又是 JavaScript 所有但是纯函数式语言没有的
+### 什么又是 JavaScript 拥有但是纯函数式语言缺乏的
-一个纯粹主义者会告诉你 JavaScript 的可变性是它的重要缺点,这是事实。但是,引起的副作用和改变有时候很有用。事实上,不可能在规避所有副作用的情况下开发有用的现代应用。纯函数式语言比如说 Haskell 使用副作用,但是使用 monads 包来伪装纯函数,从而使程序保持纯净,尽管用 monads 所带来的副作用是不纯净的。
+一个纯粹主义者会告诉你 JavaScript 的可变性是它的重大缺点,这是事实。但是,引起的副作用和改变有时候很有用。事实上,不可能在规避所有副作用的情况下开发有用的现代应用。纯函数式语言比如说 Haskell 使用副作用,使用 monads 包将有副作用的函数伪装成纯函数,从而使程序保持纯净,尽管用 Monads 所带来的副作用是不纯净的。
-monads 的问题是,尽管它的使用很简单,但是对一个不是很熟悉它的人解释清楚它有点像对牛谈琴。
+Monads 的问题是,尽管它的使用很简单,但是对一个不是很熟悉它的人解释清楚它有点像“对牛谈琴”。
-> “monad 是 endofunctor 范畴的幺半群,有什么问题?” ~James Iry 所引用 Philip Wadler的话,解释一个 Saunders Mac Lane 说过的名言。[*“A Brief, Incomplete, and Mostly Wrong History of Programming Languages”*](http://james-iry.blogspot.com/2009/05/brief-incomplete-and-mostly-wrong.html)
+> “Monad说白了不过就是自函子范畴上的一个幺半群而已,这有什么难以理解的?” ~James Iry 所引用 Philip Wadler 的话,解释一个 Saunders Mac Lane 说过的名言。[*“编程语言简要、不完整之黑历史”*](http://james-iry.blogspot.com/2009/05/brief-incomplete-and-mostly-wrong.html)
-典型的,这是在调侃这有趣的一点。在上面的引用中,关于 monads 的解释相比最初的有了很大的简化,原来是下面这样:
+典型的,这是在调侃这有趣的一点。在上面的引用中,关于 Monads 的解释相比最初的有了很大的简化,原来是下面这样:
> “`X` 中的 monad 是其 endofunctor 范畴的幺半群,生成 endofunctor 和被 endofunctor 单位 set 组合所代替的 `X` ” ~ Saunders Mac Lane。 [*"Categories for the Working Mathematician"*](https://www.amazon.com/Categories-Working-Mathematician-Graduate-Mathematics/dp/0387984038//ref=as_li_ss_tl?ie=UTF8&linkCode=ll1&tag=eejs-20&linkId=de6f23899da4b5892f562413173be4f0)
-尽管这样,在我的观点看来,害怕 monads 是没有必要的,学习 monads 最好的方法不是去读关于它的一堆书和博客,而是立刻去使用它。对于大部分的函数式编程语言来说,晦涩的学术词汇比它实际概念难的多,相信我,你不必通过了解 Saunders Mac Lane 来了解函数式编程。
+尽管这样,在我的观点看来,害怕 Monads 是没有必要的,学习 Monads 最好的方法不是去读关于它的一堆书和博客,而是立刻去使用它。对于大部分的函数式编程语言来说,晦涩的学术词汇比它实际概念难的多,相信我,你不必通过了解 Saunders Mac Lane 来了解函数式编程。
-尽管它不是对所有的编程风格都完全的理想,JavaScript 无疑是作为适应各种编程风格和背景的人的通用编程语言被设计出来的。
+尽管它不是对所有的编程风格都绝对完美,JavaScript 无疑是作为适应各种编程风格和背景的人的通用编程语言被设计出来的。
根据 [Brendan Eric](https://brendaneich.com/2008/04/popularity/) 所言,在一开始的时候,网景公司就有意适应两类开发者:
> “...写组件的,比如说 C++ 或者 Java;写脚本的、业余的和爱好者,比如直接写嵌在 HTML 里的代码的。”
-最初,网景公司的意向是支持两种不同的语言,同时脚本语言大致要像 Scheme (一个 Lisp 的方言),而且,Brendan Eich:
+本来,网景公司的意向是支持两种不同的语言,同时脚本语言大致要像 Scheme (一个 Lisp 的方言),而且,Brendan Eich:
> “我被招聘到网景公司,目的是在浏览器中 **做一些 Scheme**”。
JavaScript 应当是一门新的语言:
-> “上面工程管理的命令是这门语言**应当像 Java**,这就排除了 Perl,Python,和 Tcl,以及 Scheme。”
+> “上级工程管理的命令是这门语言**应当像 Java**,这就排除了 Perl,Python,和 Tcl,以及 Scheme。”
所以,Brendan Eich 最初脑子里的想法是:
1. 浏览器中的 Scheme。
2. 看起来像 Java。
-最终的结果稍微有点复杂:
+它最终更像是个大杂烩:
->“我不骄傲,但我很高兴我选择了 Scheme 的一类函数和 Self(尽管奇怪)的原型作为主要的元素。”由于 Java 的影响,特别是 y2k 的 Date 问题以及对象的区别(比如string 和 String),就不幸了。”
+>“我不骄傲,但我很高兴我选择了 Scheme 的一类函数和 Self(尽管奇怪)的原型作为主要的元素。”由于 Java 的影响,特别是 y2k 的 Date 问题以及对象的区别(比如 string 和 String),就不幸了。”
-我列出了这些 “不好的” 的类 Java 特性,最后转换成 JavaScript:
+我列出了这些 “不好的” 的类 Java 特性,最后整理成 JavaScript:
* 构造函数和 `new` 关键子,跟工厂函数有着不同的调用和使用语义。
* `class` 的关键字和单一父类 `extends` 作为最初的继承机制。
@@ -107,31 +107,31 @@ JavaScript 应当是一门新的语言:
我们最终创作了一个直接被浏览器支持的语言:JavaScript。
-那意味着浏览器可以减少臃肿和问题,因为它们现在只需要支持一种语言:JavaScript。你也许认为 WebAssembly 是异常,但是 WebAssembly 设计之初的目的是使用兼容的抽象语法树来共享JavaScript的语言绑定(AST)。事实上,最早的把 WebAssembly 编译成 JavaScript 的子集的示范是 ASM.js。
+那意味着浏览器可以减少臃肿和问题,因为它们现在只需要支持一种语言:JavaScript。你也许认为 WebAssembly 是例外,但是 WebAssembly 设计之初的目的是使用兼容的抽象语法树来共享 JavaScript 的语言绑定(AST)。事实上,最早的把 WebAssembly 编译成 JavaScript 的子集的示范是 ASM.js。
作为 web 平台唯一的通用标准编程语言,JavaScript 在软件历史潮流中乘风直上:
App 吞食世界, web 吞食 app, 同时 JavaScript 吞食 web。
-根据[各方](http://redmonk.com/sogrady/2016/07/20/language-rankings-6-16/)[调查](http://stackoverflow.com/research/developer-survey-2016),[JavaScript](https://octoverse.github.com/)是目前世界上最流行的语言。
+根据[多个平台](http://redmonk.com/sogrady/2016/07/20/language-rankings-6-16/)[调查](http://stackoverflow.com/research/developer-survey-2016),[JavaScript](https://octoverse.github.com/) 是目前世界上最流行的语言。
-JavaScript 并不是函数式编程的理想化工具,但是它却是为大型的分布式的团队开发大型应用的好工具,因为不同的团队也许有不同建立应用的想法。
+JavaScript 并不是函数式编程的理想化工具,但是它却是为大型的分布式的团队开发大型应用的好工具,因为不同的团队对于如何构建一个应用或许有不同的看法。
一些团队致力于脚本化,那么命令式的编程就特别有用,另外一些更精于抽象架构,那么一点保留的面向对象方法也许不失为坏。还有一些拥抱函数式编程,使用纯函数来确保稳定性、可测试性和项目状态管理以便减少用户的反馈。团队里的这些人可以使用相同的语言,意味着他们可以更好的交换想法,互相学习和在其他人的基础上更进一步的开发。
在 JavaScript 中,所有这些想法可以共存,这样就让更多的人开始拥抱 JavaScript,然后就产生了[世界上最大的开源包管理器](http://www.modulecounts.com/) (2017 年 2 月),[npm](https://www.npmjs.com/)。
-JavaScript 的真正优势在于其生态系统中的思想和用户的多样性。它也许不是纯函数式编程的理想语言,但它是你可以想象的工作在不同平台的人共同合作的理想语言,比如说 Java、Lisp 或者 C。JavaScript 也许对这些有这些背景的用户完全友好,但是这些人很乐意学习这门语言并迅速投入生产。
+JavaScript 的真正优势在于其生态系统中的思想和用户的多样性。它也许不是纯函数式编程最理想的语言,但它是你可以想象的工作在不同平台的人共同合作的理想语言,比如说 Java、Lisp 或者 C。JavaScript 也许并不对有这些背景的用户完全友好,但是这些人很乐意学习这门语言并迅速投入生产。
-我同意 JavaScript 并不是函数式编程着最好的语言。但是,没有任何其他语言可以声称他们可以被所有人使用,同时正如 ES6 所述:JavaScript 可以满足到更与喜欢函数式编程的人的需要,同时也越来越好。相比于抛弃 JavaScript 和它不可思议的被世界上所有公司使用的生态系统,为什么不拥抱它,把它变成一个更适合软件组成化的语言?
+我同意 JavaScript 并不是对函数式编程者最好的语言。但是,没有任何其他语言可以声称他们可以被所有人使用,同时正如 ES6 所述:JavaScript 可以满足到更与喜欢函数式编程的人的需要,同时也越来越好。相比于抛弃 JavaScript 和世界上几乎每家公司都使用的令人难以置信的生态系统,为什么不拥抱它,把它变成一个更适合软件组合化的语言?
-现在,JavaScript 已经是一门足够优秀的函数式编程语言,意味着人们可以使用 JavaScript 的函数式编程方法来构造所有有趣的和有用的东西。Netflix(和其他使用 Angular 2+ 的应用)使用基于 RxJS 的函数式功能。[Facebook](https://github.com/facebook/react/wiki/sites-using-react)在 React 中使用纯函数、高阶函数和高级组件来开发 Facebook 和 Instagram,[PayPal, KhanAcademy, and Flipkart](https://github.com/reactjs/redux/issues/310)使用 Redux 来进行状态管理。
+现在,JavaScript 已经是一门**足够优秀**的函数式编程语言,意味着人们可以使用 JavaScript 的函数式编程方法来构造很多有趣的和有用的东西。Netflix(和其他使用 Angular 2+ 的应用)使用基于 RxJS 的函数式功能。[Facebook](https://github.com/facebook/react/wiki/sites-using-react)在 React 中使用纯函数、高阶函数和高级组件来开发 Facebook 和 Instagram,[PayPal、KhanAcademy、和Flipkart](https://github.com/reactjs/redux/issues/310)使用 Redux 来进行状态管理。
-它们并不孤单:Angular、React、Redux 和 Lodash 是 JavaScript 生态系统中主要的框架和库,同时它们都被函数式编程很深的影响到 — 或者在 Lodash 和 Redux中,明确的表达是为了在实际的 JavaScript 应用中使用函数式编程模式。
+它们并不孤单:Angular、React、Redux 和 Lodash 是 JavaScript 生态系统中主要的框架和库,同时它们都被函数式编程很深的影响到— 在 Lodash 和 Redux 中,明确地表达是为了在实际的 JavaScript 应用中使用函数式编程模式。
-“为什么是 JavaScript?”因为实际上所有的软件呢公司用来开发软件的语言。无论如何,JavaScript 从 Lisp 这个数十年来的标志头上偷取了 “最受欢迎的函数式编程语言” 的头衔。事实上,Haskell 更适合当今函数式编程概念的标准,但是人们并不使用它来开发实际应用。
+“为什么是 JavaScript?”因为 JavaScript 是实际上大多数公司开发真实的软件所使用的语言。无论你对它是爱是恨,JavaScript 已经取代了 Lisp 这个数十年来 “最受欢迎的函数式编程语言”。事实上,Haskell 更适合当今函数式编程概念的标准,但是人们并不使用它来开发实际应用。
-在任何时候,在美国都有一百万的 JavaScript 工作需求,世界其他地方也有数百万的量。学习 Haskell 可以帮助你很好的学习函数式编程,但学习 JavaScript 将会教会你在实际工作中开发应用。
+在任何时候,在美国都有近十万的 JavaScript 工作需求,世界其他地方也有数十万的量。学习 Haskell 可以帮助你很好的学习函数式编程,但学习 JavaScript 将会教会你在实际工作中开发应用。
App 正在吞食世界, web 正在吞食 app, 同时 JavaScript 正在吞食 web。
From c019bf6346020a777f1149e030de6607bf819294 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E5=90=B4=E6=99=93=E5=86=9B?=
Date: Thu, 13 Apr 2017 00:42:01 +0800
Subject: [PATCH 209/945] functors & category
---
TODO/functors-categories.md | 173 ++++++++++++++++++------------------
1 file changed, 88 insertions(+), 85 deletions(-)
diff --git a/TODO/functors-categories.md b/TODO/functors-categories.md
index a5b81f432c4..31ee4219b6f 100644
--- a/TODO/functors-categories.md
+++ b/TODO/functors-categories.md
@@ -1,120 +1,121 @@
> * 原文地址:[Functors & Categories](https://medium.com/javascript-scene/functors-categories-61e031bac53f)
> * 原文作者:[Eric Elliott](https://medium.com/@_ericelliott?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[yoyoyohamapi](https://github.com/yoyoyohamapi) [reid3290](https://github.com/reid3290)
> * 校对者:
-# Functors & Categories #
-
-## Composable Software ##
+# Functor 与 Category (软件编写)(第五部分)#
-Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0)
-> Note: This is part of the “Composing Software” series on learning functional programming and compositional software techniques in JavaScript ES6+ from the ground up. Stay tuned. There’s a lot more of this to come!
-> [< Previous](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d#.w4y0mlpcs) | [<< Start over at Part 1](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe)
+Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0) (译注:该图是用 PS 将烟雾处理成方块状后得到的效果,参见 [flickr](https://www.flickr.com/photos/68397968@N07/11432696204)。))
+
+> 注意:这是 “软件编写” 系列文章的第四部分,该系列主要阐述如何在 JavaScript ES6+ 中从零开始学习函数式编程和组合化软件(compositional software)技术(译注:关于软件可组合性的概念,参见维基百科 [Composability](https://en.wikipedia.org/wiki/Composability))。后续还有更多精彩内容,敬请期待!
+> [<上一篇](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d#.w4y0mlpcs) | [<< 返回第一章](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe)
-A **functor** is something that can be mapped over. In other words, it’s a container which has an interface which can be used to apply a function to the values inside it. When you see the word functor, you should think *“mappable”.*
+所谓 **functor(函子)**,是能够对其进行 map 操作的对象。换言之,**functor** 可以被认为是一个容器,该容器容纳了一个值,并且暴露了一个接口(译注:即 map 接口),该接口使得外界的函数能够操纵容器中的值。所以当你见到 **functor**,别被其来自范畴学的名字唬住,简单把他当做个 *“mappable”* 对象就行。
-The term “functor” comes from category theory. In category theory, a functor is a mapping between categories. Loosely, a **category** is a group of things, where each “thing” can be any value. In code, a functor is sometimes represented as an object with a `.map()` method that maps from one set of values to another.
+**“functor”** 一词源于范畴学。在范畴学中,一个 functor 代表了两个范畴(category)间的映射。简单说来,一个 **范畴** 是一系列事物的分组,这里的 “事物” 可以指代一切的值。对于编码来说,一个 functor 通常代表了一个具有 `.map()` 方法的对象,该方法能够将某一集合映射到另一集合。
-A functor supplies a box with zero or more things inside, and a mapping interface. An array is a good example of a functor, but many other kinds of objects can be mapped over as well, including single valued-objects, streams, trees, objects, etc…
+上文说到,一个 functor 可以被看做是一个容器,比如我们将其看做是一个盒子,盒子里面容纳了一些事物,或者空空如也,最重要的是,盒子暴露了一个 mapping(映射)接口。在 JavaScript 中,数组对象就是 functor 的绝佳例子(译注:`[1,2,3].map(x => x + 1)`),但是,其他类型的对象,只要能够被 map 操作,也可以算作是 functor,这些对象包括了单值对象(single valued-objects),流(streams),树(trees),对象(objects)等等。
-For collections (arrays, streams, etc…), `.map()` typically iterates over the collection and applies the given function to each value in the collection, but not all functors iterate.
+对于如数组和流这样的集合(collections)来说,`.map()` 方法指的是,在集合上进行迭代操作,在此过程中,应用一个预先指定的函数对每次迭代到的值进行处理。但是,不是所有的 functor 都可以被迭代。
-In JavaScript, Arrays and Promises are functors (`.then()` obeys the functor laws), but lots of libraries exist that will turn a variety of other things into functors, too.
+在 JavaScript 中,数组和 Promise 对象都是 **functor**(Promise 对象虽然没有 `.map()` 方法,但其 `.then()` 方法也遵从 functor 的定律),除此之外,非常多的第三方库也能够将各种各样的一般事物给转换成 functor(译注:大名鼎鼎的 [Bluebird](https://github.com/petkaantonov/bluebird/) 就能将异步过程封装为 Promise functor)。
-In Haskell, the functor type is defined as:
+在 Haskell 中,functor 类型被定义为如下形式:
```
fmap :: (a -> b) -> f a -> f b
```
-Given a function that takes an `a` and returns a `b` and a functor with zero or more `a`s inside it: `fmap` returns a box with zero or more `b`s inside it. The `f a` and `f b` bits can be read as “a functor of `a`” and “a functor of `b`”, meaning `f a` has `a`s inside the box, and `f b` has `b`s inside the box.
+fmap 接受一个函数参数,该函数接受一个参数 `a`,并返回一个 `b`,最终,fmap 完成了从 `f a` 到 `f b` 的映射。`f a` 及 `f b` 可以被读作 “一个 `a` 的 functor” 和“一个 `b` 的 functor”,亦即 `f a` 这个盒子容纳了 `a`,`f b` 这个盒子容纳了 `b`。
-Using a functor is easy — just call `map()`:
+使用一个 functor 是非常简单的,仅需要调用 `map()` 方法即可:
```
const f = [1, 2, 3];
f.map(double); // [2, 4, 6]
```
-### Functor Laws ###
+### Functor 定律 ###
-Categories have two important properties:
+一个范畴含有两个基本的定律:
-1. Identity
-2. Composition
+1. 同一性(Identity)
+2. 组合性(Composition)
-Since a functor is a mapping between categories, functors must respect identity and composition. Together, they’re known as the functor laws.
+由于 functor 是两个范畴间的映射,其就必须遵守同一性和组合性,二者也构成了 functor 的基本定律。
-### Identity ###
+### 同一性 ###
-If you pass the identity function (`x => x`) into `f.map()`, where `f` is any functor, the result should be equivalent to (have the same meaning as) `f`:
+如果你将函数(`x => x`)传入 `f.map()`,并且,`f` 是任意的一个 functor,那么 `f.map()` 执行结果就等于 `f`。
```
const f = [1, 2, 3];
f.map(x => x); // [1, 2, 3]
```
-### Composition ###
+### 组合性 ###
-Functors must obey the composition law: `F.map(x => f(g(x)))` is equivalent to `F.map(g).map(f)`.
+functor 还必须具有组合性:`F.map(x => f(g(x))) == F.map(g).map(f)`
-Function Composition is the application of one function to the result of another, e.g., given an `x` and the functions, `f` and `g`, the composition `(f ∘ g)(x)` (usually shortened to `f ∘ g` - the `(x)` is implied) means `f(g(x))`.
+函数组合是将一个函数的输出作为另一个函数输入的过程。例如,给定一个值 `x`及函数 `f` 和函数 `g`,函数的组合就是 `(f ∘ g)(x)`(通常简写为 `f ∘ g`,简写形式已经暗示了 `(x)`),其意味着 `f(g(x))`。
-A lot of functional programming terms come from category theory, and the essence of category theory is composition. Category theory is scary at first, but easy. Like jumping off a diving board or riding a roller coaster. Here’s the foundation of category theory in a few bullet points:
+很多函数式编程的术语都源于范畴学,而范畴学的实质即是组合。初看范畴学,就像初次进行高台跳水或者乘坐过山车,慌张,恐惧,但是并不难完成。你只需明确下面几个范畴学基础要点:
-- A category is a collection of objects and arrows between objects (where “object” can mean literally anything).
-- Arrows are known as morphisms. Morphisms can be thought of and represented in code as functions.
-- For any group of connected objects, `a -> b -> c`, there must be a composition which goes directly from `a -> c`.
-- All arrows can be represented as compositions (even if it’s just a composition with the object’s identity arrow). All objects in a category have identity arrows.
+- 一个范畴(category)是一个容纳了一系列对象及对象间箭头(`->`)的集合。
+- 箭头只是形式上的描述,实际上,箭头代表了态射(morphismms)。在编程中,态射可以被认为是函数。
+- 对于任何被箭头相连接的对象,如 `a -> b -> c`,必须存在一个 `a -> c ` 的组合。
+- 所有的箭头表示都代表了组合(即便这个对象间的组合只是一个同一(identity)箭头:`a->c`)。所有的对象都存在一个同一箭头,即存在同一态射(`a -> a`)。
-Say you have a function `g` that takes an `a` and returns a `b`, and another function `f` that takes a `b` and returns a `c`; there must also be a function `h` that represents the composition of `f` and `g`. So, the composition from `a -> c`, is the composition `f ∘ g` (`f`*after*`g`). So, `h(x) = f(g(x))`. Function composition works right to left, not left to right, which is why `f ∘ g` is frequently called `f`*after*`g`.
+如果你有一个函数 `g`,该函数接受一个参数 `a` 并且返回一个 `b`,另一个函数 `f` 接受一个 `b` 并返回一个 `c`。那么,必然存在一个函数 `h`,其代表了 `f` 及 `g` 的组合。而 `a -> c` 的组合,就是 `f ∘ g`(读作`f` **紧接着** `g`),进而,也就知 `h(x)=f(g(x))`。函数组合的方向是由右向左的,这也就是就是 `f ∘ g` 常被叫做 `f` **紧接着** `g` 的原因。
-Composition is associative. Basically that means that when you’re composing multiple functions (morphisms if you’re feeling fancy), you don’t need parenthesis:
+函数组合是满足结合律的,这就意味着你在组合多个函数时,免去了添加括号的烦恼:
```
h∘(g∘f) = (h∘g)∘f = h∘g∘f
```
-Let’s take another look at the composition law in JavaScript:
+让我们再看一眼 JavaScript 中组合律:
-Given a functor, `F`:
+给定一个 functor,`F`:
```
const F = [1, 2, 3];
```
-The following are equivalent:
+下面的两段是等效的:
```
F.map(x => f(g(x)));
-// is equivalent to...
+// 等效于。。。
F.map(g).map(f);
```
-### Endofunctors ###
+> 译注:functor 中函数组合的结合率可以被理解为:对 functor 中保存的值使用组合后的函数进行 map,等效于先后对该值用不同的函数进行 map。
+
+### Endofunctors(自函子) ###
-An endofunctor is a functor that maps from a category back to the same category.
+一个 endofunctor(自函子)是一个能将一个范畴映射回相同范畴的 functor。
-A functor can map from category to category: `F a -> F b`
+一个 functor 能够完成任意范畴间映射: `F a -> F b`
-An endofunctor maps from a category to the same category: `F a -> F a`
+一个 endofunctor 能够完成相同范畴间的映射:`F a -> F a`
-`F` here represents a *functor type* and `a` represents a category variable (meaning it can represent any category, including a set or a category of all possible values in a data type).
+在这里,`F` 代表了一个 **functor 类型**,而 `a` 代表了一个范畴变量(意味着其能够代表任意的范畴,无论是一个集合,还是一个包含了某一数据类型所有可能取值的范畴)。
-A monad is an endofunctor. Remember:
+而一个 monad 则是一个 endofunctor,先记住下面这句话:
-> *“A monad is just a monoid in the category of endofunctors. What’s the problem?”*
+> “monad 是 endofunctor 范畴的 monoids(幺半群),有什么问题?”(译注:这句话的出处在该系列第一篇已有提及)
-Hopefully that quote is starting to make a little more sense. We’ll get to monoids and monads later.
+现在,我们希望第一篇提及的这句话能在之后多一点意义,monoids(幺半群)及 monad 将在之后作介绍。
-### Build Your Own Functor ###
+### 自定义一个 Functor ###
-Here’s a simple example of a functor:
+下面将展示一个简单的 functor 例子:
```
const Identity = value => ({
@@ -122,11 +123,11 @@ const Identity = value => ({
});
```
-As you can see, it satisfies the functor laws:
+显然,其满足了 functor 定律:
```
-// trace() is a utility to let you easily inspect
-// the contents.
+// trace() 是一个简单的工具函数来帮助审查内容
+// 内容
const trace = x => {
console.log(x);
return x;
@@ -134,14 +135,14 @@ const trace = x => {
const u = Identity(2);
-// Identity law
+// 同一性
u.map(trace); // 2
u.map(x => x).map(trace); // 2
const f = n => n + 1;
const g = n => n * 2;
-// Composition law
+// 组合性
const r1 = u.map(x => f(g(x)));
const r2 = u.map(g).map(f);
@@ -149,11 +150,11 @@ r1.map(trace); // 5
r2.map(trace); // 5
```
-Now you can map over any data type, just like you can map over an array. Nice!
+现在,你可以对存在该 functor 中的任何数据类型进行 map 操作,就像你对一个数组进行 map 时那样。这简直太美妙了。
-That’s about as simple as a functor can get in JavaScript, but it’s missing some features we expect from data types in JavaScript. Let’s add them. Wouldn’t it be cool if the `+` operator could work for number and string values?
+上面的代码片展示了 JavaScript 中 functor 的简单实现,但是其缺失了 JavaScript 中常见数据类型的一些特性。现在我们逐个添加它们。首先,我们会想到,假如能够直接通过 `+` 操作符操作我们的 functor 是不是太好了,就像我们在数值或者字符串对象间使用 `+` 号那样。
-To make that work, all we need to do is implement `.valueOf()` -- which also seems like a convenient way to unwrap the value from the functor:
+为了使该想法变现,我们首先要为该 functor 对象添加 `.valueOf()` 方法 —— 这可被看作是提供了一个便捷的渠道来将值从 functor 盒子中取出。
```
const Identity = value => ({
@@ -169,13 +170,13 @@ const hi = (Identity('h') + Identity('i'));
trace(hi); // "hi"
```
-Nice. But what if we want to inspect an `Identity` instance in the console? It would be cool if it would say `"Identity(value)"`, right. Let's add a `.toString()` method:
+现在代码更漂亮了。但是如果我们还想要在控制台审查 `Identity` 实例呢?如果控制台能够输出 `"Identity(value)"` 就太好了,为此,我们只需要添加一个 `.toString()` 方法即可(译注:亦即重载原型链上原有的 `.toString()` 方法):
```
toString: () => `Identity(${value})`,
```
-Cool. We should probably also enable the standard JS iteration protocol. We can do that by adding a custom iterator:
+代码又有所进步。现在,我们可能也想 functor 能够满足标准的 JavaScript 迭代协议(译注:[MDN - 迭代协议](https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Iteration_protocols))。为此,我们可以为 `Identity` 添加一个自定义的迭代器:
```
[Symbol.iterator]: () => {
@@ -197,7 +198,7 @@ Cool. We should probably also enable the standard JS iteration protocol. We can
},
```
-Now this will work:
+现在,我们的 functor 还能这样工作:
```
// [Symbol.iterator] enables standard JS iterations:
@@ -205,28 +206,30 @@ const arr = [6, 7, ...Identity(8)];
trace(arr); // [6, 7, 8]
```
-What if you want to take an `Identity(n)` and return an array of Identities containing `n + 1`, `n + 2`, and so on? Easy, right?
+假如你想借助 `Identity(n)` 来返回包含了 `n+1`,`n+2` 等等的 Identity 数组,这非常容易:
```
const fRange = (
start,
end
) => Array.from(
- { length: end - start + 1 },
+ {length: end - start + 1},
(x, i) => Identity(i + start)
);
```
-Ah, but what if you want this to work with any functor? What if we had a spec that said that each instance of a data type must have a reference to its constructor? Then you could do this:
+> 译注:[MDN -- Array.from()](https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Array/from)
+
+但是,如果你想上面的操作方式能够应用于任何 functor,该怎么办?假如我们规定了每种数据类型对应的实例必须有一个关于其构造函数的引用,那么你可以这样改造之前的逻辑:
```
const fRange = (
start,
end
) => Array.from(
- { length: end - start + 1 },
+ {length: end - start + 1},
- // change `Identity` to `start.constructor`
+ // 将 `Identity` 变更为 `start.constructor`
(x, i) => start.constructor(i + start)
);
@@ -234,7 +237,7 @@ const range = fRange(Identity(2), 4);
range.map(x => x.map(trace)); // 2, 3, 4
```
-What if you want to test to see if a value is a functor? We could add a static method on `Identity` to check. We should throw in a static `.toString()` while we're at it:
+假如你还想知道一个值是否在一个 functor 中,又怎么办?我们可以为 `Identity` 添加一个静态方法 `.is()` 来进行检测,另外,我们也顺便添加了一个静态的 `.toString()` 方法来告知这个 functor 的种类:
```
Object.assign(Identity, {
@@ -243,7 +246,8 @@ Object.assign(Identity, {
});
```
-Let’s put all this together:
+
+现在,我们整合一下上面的代码片:
```
const Identity = value => ({
@@ -280,14 +284,14 @@ Object.assign(Identity, {
});
```
-Note you don’t need all this extra stuff for something to qualify as a functor or an endofunctor. It’s strictly for convenience. All you *need* for a functor is a `.map()` interface that satisfies the functor laws.
+注意,无论是 functor,还是 endofunctor,不一定需要上述那么多的条条框框。以上工作只是为了我们在使用 functor 时更加便捷,而非必须。一个 functor 的所有需求只是一个满足了 functor 定律 `.map()` 接口。
-### Why Functors? ###
+### 为什么要使用 functor? ###
-Functors are great for lots of reasons. Most importantly, they’re an abstraction that you can use to implement lots of useful things in a way that works with any data type. For instance, what if you want to kick off a chain of operations, but only if the value inside the functor is not `undefined` or `null`?
+说 functor 多么多么好是不无理由的。最重要的一点是,functor 作为一种抽象,能让开发者以同一种方式实现大量有用的,能够操纵任何数据类型的事物。例如,如果你想要在 functor 中值不为 `null` 或者不为 `undefined` 前提下,构建一串地链式操作:
```
-// Create the predicate
+// 创建一个 predicte
const exists = x => (x.valueOf() !== undefined && x.valueOf() !== null);
const ifExists = x => ({
@@ -297,9 +301,9 @@ const ifExists = x => ({
const add1 = n => n + 1;
const double = n => n * 2;
-// Nothing happens...
+// undefined
ifExists(Identity(undefined)).map(trace);
-// Still nothing...
+// null
ifExists(Identity(null)).map(trace);
// 42
@@ -310,9 +314,9 @@ ifExists(Identity(20))
;
```
-Of course, functional programming is all about composing tiny functions to create higher level abstractions. What if you want a generic map that works with any functor? That way you can partially apply arguments to create new functions.
+函数式编程一直探讨的是将各个小的函数进行组合,以创建出更高层次的抽象。假如你想要一个更通用的,能够工作在任何 functor 上的 `map()` 方法,那么你可以通过参数的部分应用(译注:即 [偏函数](https://en.wikipedia.org/wiki/Partial_application))来完成。
-Easy. Pick your favorite auto-curry, or use this magic spell from before:
+你可以使用自己喜欢的 curry 化方法(译注:Underscore,Lodash,Ramda 等第三方库都提供了 curry 化一个函数的方法),或者使用下面这个之前篇章提到的,基于 ES6 的,充满魅力的 curry 化方法来实现参数的部分应用:
```
const curry = (
@@ -324,7 +328,7 @@ const curry = (
)([...arr, ...args]);
```
-Now we can customize map:
+现在,我们可以自定义 `map()` 方法:
```
const map = curry((fn, F) => F.map(fn));
@@ -335,28 +339,27 @@ const mdouble = map(double);
mdouble(Identity(4)).map(trace); // 8
```
-### Conclusion ###
+### 总结 ###
-Functors are things we can map over. More specifically, a functor is a mapping from category to category. A functor can even map from a category back to the same category (i.e., an *endofunctor*).
+functor 是能够对其进行 map 操作的对象。更进一步地,一个 functor 能够将一个范畴映射到另一个范畴。一个 functor 甚至可以将某一范畴映射回相同范畴(例如 endofunctor)。
-A category is a collection of objects, with arrows between objects. Arrows represent morphisms (aka functions, aka compositions). Each object in a category has an identity morphism (`x => x`). For any chain of objects `A -> B -> C` there must exist a composition `A -> C`.
+一个范畴是一个容纳了对象和对象间箭头的集合。箭头代表了态射(也可理解为函数或者组合)。一个范畴中的每个对象都具有一个同一态射(`x -> x`)。对于任何链接起来的对象 `A -> B -> C`,必存在一个 `A -> C` 的组合。
-Functors are great higher-order abstractions that allow you to create a variety of generic functions that will work for any data type.
+总之,functor 是一个极佳的高阶抽象,能然你创建各种各样的通用函数来操作任何的数据类型。
-**To be continued…**
+**未完待续。。。**
-### Next Steps ###
+### 接下来 ###
-Want to learn more about functional programming in JavaScript?
+想学习更多 JavaScript 函数式编程吗?
-[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/). If you’re not a member, you’re missing out!
+[跟着 Eric Elliott 学 Javacript](http://ericelliottjs.com/product/lifetime-access-pass/),机不可失时不再来!
-[
-](https://ericelliottjs.com/product/lifetime-access-pass/)
+[](https://ericelliottjs.com/product/lifetime-access-pass/)
-***Eric Elliott*** is the author of [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly), and [*“Learn JavaScript with Eric Elliott”*](http://ericelliottjs.com/product/lifetime-access-pass/) . He has contributed to software experiences for **Adobe Systems, Zumba Fitness, The Wall Street Journal, ESPN, BBC**, and top recording artists including **Usher, Frank Ocean, Metallica**, and many more.
+**Eric Elliott** 是 [**“编写 JavaScript 应用”**](http://pjabook.com) (O’Reilly) 以及 [**“跟着 Eric Elliott 学 Javascript”**](http://ericelliottjs.com/product/lifetime-access-pass/) 两书的作者。他为许多公司和组织作过贡献,例如 **Adobe Systems**、**Zumba Fitness**、**The Wall Street Journal**、**ESPN** 和 **BBC** 等 , 也是很多机构的顶级艺术家,包括但不限于 **Usher**、**Frank Ocean** 以及 **Metallica**。
-*He spends most of his time in the San Francisco Bay Area with the most beautiful woman in the world.*
+大多数时间,他都在 San Francisco Bay Area,同这世上最美丽的女子在一起。
---
From 2c3a7a1b733624be4e533cd2b2fda859caa04ddd Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E5=90=B4=E6=99=93=E5=86=9B?=
Date: Thu, 13 Apr 2017 00:50:54 +0800
Subject: [PATCH 210/945] title
---
TODO/functors-categories.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/functors-categories.md b/TODO/functors-categories.md
index 31ee4219b6f..491a023bc29 100644
--- a/TODO/functors-categories.md
+++ b/TODO/functors-categories.md
@@ -4,7 +4,7 @@
> * 译者:[yoyoyohamapi](https://github.com/yoyoyohamapi) [reid3290](https://github.com/reid3290)
> * 校对者:
-# Functor 与 Category (软件编写)(第五部分)#
+# Functor 与 Category (软件编写)(第六部分)#
From 8e7e318c4a1391a3088eeb6fbab18e5b61807797 Mon Sep 17 00:00:00 2001
From: sunxinlei
Date: Thu, 13 Apr 2017 02:07:57 +0800
Subject: [PATCH 211/945] =?UTF-8?q?=E7=BF=BB=E8=AF=91=E4=B8=80=E8=BF=87?=
=?UTF-8?q?=E5=8D=8A?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...uction-to-javascript-composing-software.md | 124 +++++++++---------
1 file changed, 63 insertions(+), 61 deletions(-)
diff --git a/TODO/a-functional-programmers-introduction-to-javascript-composing-software.md b/TODO/a-functional-programmers-introduction-to-javascript-composing-software.md
index c831ac252e4..df1ca8810a3 100644
--- a/TODO/a-functional-programmers-introduction-to-javascript-composing-software.md
+++ b/TODO/a-functional-programmers-introduction-to-javascript-composing-software.md
@@ -1,29 +1,30 @@
> * 原文地址:[A Functional Programmer’s Introduction to JavaScript (Composing Software)(part 3)](https://medium.com/javascript-scene/a-functional-programmers-introduction-to-javascript-composing-software-d670d14ede30)
> * 原文作者:[Eric Elliott](https://medium.com/@_ericelliott?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[sun](http://suncafe.cc)
> * 校对者:
-# A Functional Programmer’s Introduction to JavaScript (Composing Software) #
+# 函数式程序员的 JavaScript 简介 (软件构建系列) #
-Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0)
+烟雾艺术模仿 — MattysFlicks — (CC BY 2.0)
> Note: This is part of the “Composing Software” series on learning functional programming and compositional software techniques in JavaScript ES6+ from the ground up. Stay tuned. There’s a lot more of this to come!
-> [< Previous](https://medium.com/javascript-scene/why-learn-functional-programming-in-javascript-composing-software-ea13afc7a257#.ia0aq3fmb) | [<< Start over at Part 1](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [Next >](https://medium.com/javascript-scene/higher-order-functions-composing-software-5365cf2cbe99#.egoxjg6x7)
+> [< Previous](https://medium.com/javascript-scene/why-learn-functional-programming-in-javascript-composing-software-ea13afc7a257#.ia0aq3fmb) | [<< Start over at Part 1](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [Next >](https://medium.com/javascript-scene/higher-order-functions-composing-software-5365cf2cbe99#.egoxjg6x7)
-For those unfamiliar with JavaScript or ES6+, this is intended as a brief introduction. Whether you’re a beginner or experienced JavaScript developer, you may learn something new. The following is only meant to scratch the surface and get you excited. If you want to know more, you’ll just have to explore deeper. There’s a lot more ahead.
+对于不熟悉 JavaScript 或 ES6+ 的同学,这里做一个简短的介绍。无论你是 JavaScript 开发新手还是有经验的老兵,你都可能学到一些新东西。以下内容仅是毛羽鳞鬣,吊吊大家的兴致。如果想知道更多,还需深入学习。敬请期待吧。
-The best way to learn to code is to code. I recommend that you follow along using an interactive JavaScript programming environment such as [CodePen](https://codepen.io/) or the [Babel REPL](https://babeljs.io/repl/) .
-Alternatively, you can get away with using the Node or browser console REPLs.
+学习编程最好的方法就是动手编程。我建议您使用交互式 JavaScript 编程环境(如 [CodePen](https://codepen.io/) 或 [Babel REPL](https://babeljs.io/repl/))。
-### Expressions and Values ###
+或者,您也可以使用 Node 或浏览器控制台 REPL。
-An expression is a chunk of code that evaluates to a value.
+### 表达式和值 ###
-The following are all valid expressions in JavaScript:
+表达式是可以求得数据值的代码块。
+
+下面这些都是 JavaScript 中合法的表达式:
```
7;
@@ -35,42 +36,42 @@ The following are all valid expressions in JavaScript:
'Hello'; // Hello
```
-The value of an expression can be given a name. When you do so, the expression is evaluated first, and the resulting value is assigned to the name. For this, we’ll use the `const` keyword. It's not the only way, but it's the one you'll use most, so we'll stick with `const` for now:
+表达式的值可以被赋予一个名称。执行此操作时,表达式首先被计算,取得的结果值被分配给该名称。对于这一点我们将使用 `const` 关键字。这不是唯一的方式,但这将是你使用最多的,所以目前我们就可以坚持使用 `const`。
```
const hello = 'Hello';
hello; // Hello
```
-### var, let, and const ###
+### var、let 和 const ###
-JavaScript supports two more variable declaration keywords: `var`, and `let`. I like to think of them in terms of order of selection. By default, I select the strictest declaration: `const`. A variable declared with the `const` keyword can't be reassigned. The final value must be assigned at declaration time. This may sound rigid, but the restriction is a good thing. It's a signal that tells you, "the value assigned to this name is not going to change". It helps you fully understand what the name means right away, without needing to read the whole function or block scope.
+JavaScript 支持另外两种变量声明关键字:`var`,还有 `let`。我喜欢根据选择的顺序来考虑它们。默认情况下,我选择最严格的声明方式:`const`。用 `const` 关键字声明的变量不能被重新赋值。最终值必须在声明时分配。这可能听起来很严格,但限制是一件好事。这是个标识在提醒你“赋给这个名称的值将不会改变”。它可以帮你全面了解这个名称的意义,而无需阅读整个函数或块级作用域。
-Sometimes it’s useful to reassign variables. For example, if you’re using manual, imperative iteration rather than a more functional approach, you can iterate a counter assigned with `let`.
+有时,给变量重新赋值很有用。比如,如果你正在写一个手动的强制性迭代,而不是一个更具功能性的方法,你可以迭代一个用 `let` 赋值的计数器。
-Because `var` tells you the least about the variable, it is the weakest signal. Since I started using ES6, I have never intentionally declared a `var` in a real software project.
+因为 `var` 能告诉你很少关于这个变量的信息,所以它是最无力的声明标识。自从开始用 ES6,我就再也没在实际软件项目中有意使用 `var` 作声明了。
-Be aware that once a variable is declared with `let` or `const`, any attempt to declare it again will result in an error. If you prefer more experimental flexibility in the REPL (Read, Eval, Print Loop) environment, you may use `var` instead of `const` to declare variables. Redeclaring `var` is allowed.
+注意一下,一个变量一旦用 `let` 或 `const` 声明,任何再次声明的尝试都将导致报错。如果你在 REPL(读取-求值-输出循环)环境中更喜欢多一些实验性和灵活性,那么建议你使用 `var` 声明变量,而不是 `const`。
-This text will use `const` in order to get you in the habit of defaulting to `const` for actual programs, but feel free to substitute `var` for the purpose of interactive experimentation.
+本文将使用 const 来让您习惯于为实际程序中用 `const`,而出于试验的目的自由切换回 `var`。
-### Types ###
+### 数据类型 ###
-So far we’ve seen two types: numbers and strings. JavaScript also has booleans (`true` or `false`), arrays, objects, and more. We'll get to other types later.
+目前为止我们见到了两种数据类型:数字和字符串。JavaScript 也有布尔值(`true` 或 `false`)、数组、对象等。稍后我们再看其他类型。
-An array is an ordered list of values. Think of it as a box that can hold many items. Here’s the array literal notation:
+数组是一系列值的有序列表。可以把它比作一个能够装很多元素的盒子。这是一个数组字面量:
```
[1, 2, 3];
```
-Of course, that’s an expression which can be given a name:
+当然,它也是一个可被赋予名称的表达式:
```
const arr = [1, 2, 3];
```
-An object in JavaScript is a collection of key: value pairs. It also has a literal notation:
+在 JavaScript 中,对象是一系列键值对的集合。它也有字面量:
```
{
@@ -78,7 +79,7 @@ An object in JavaScript is a collection of key: value pairs. It also has a liter
}
```
-And of course, you can assign an object to a name:
+当然,你也可以给对象赋予名称:
```
const foo = {
@@ -86,40 +87,40 @@ const foo = {
}
```
-If you want to assign existing variables to object property keys of the same name, there’s a shortcut for that. You can just type the variable name instead of providing both a key and a value:
+如果你想将现有变量赋值给同名的对象属性,这有个捷径。你可以仅输入变量名,而不用同时提供一个键和一个值:
```
const a = 'a';
-const oldA = { a: a }; // long, redundant way
-const oA = { a }; // short an sweet!
+const oldA = { a: a }; // 长而冗余的写法
+const oA = { a }; // 短小精悍!
```
-Just for fun, let’s do that again:
+为了好玩而已,让我们再来一次:
```
const b = 'b';
const oB = { b };
```
-Objects can be easily composed together into new objects:
+对象可以轻松合并到新的对象中:
```
const c = {...oA, ...oB}; // { a: 'a', b: 'b' }
```
-Those dots are the object spread operator. It iterates over the properties in `oA` and assigns them to the new object, then does the same for `oB`, overriding any keys that already exist on the new object. As of this writing, object spread is a new, experimental feature that may not be available in all the popular browsers yet, but if it's not working for you, there is a substitute: `Object.assign()`:
+这些点是对象扩展运算符。它迭代 `oA` 的属性并分配到新的对象中,`oB` 也是一样,在新对象中已经存在的键都会被重写。在撰写本文时,对象扩展是一个新的试验特性,可能还没有被所有主流浏览器支持,但如果你那不能用,还可以用 `Object.assign()` 替代:
```
const d = Object.assign({}, oA, oB); // { a: 'a', b: 'b' }
```
-Only a little more typing in the `Object.assign()` example, and if you're composing lots of objects, it may even save you some typing. Note that when you use `Object.assign()`, you must pass a destination object as the first parameter. It is the object that properties will be copied to. If you forget, and omit the destination object, the object you pass in the first argument will be mutated.
+这个 `Object.assign()` 的例子代码很少,如果你想合并很多对象,它甚至可以节省一些打字。注意当你使用 `Object.assign()` 时,你必须传一个目标对象作为第一个参数。它就是那个源对象的属性将被复制过去的对象。如果你忘了传,第一个参数传递的对象将被改变。
-In my experience, mutating an existing object rather than creating a new object is usually a bug. At the very least, it is error-prone. Be careful with `Object.assign()`.
+以我的经验,改变一个已经存在的对象而不创建一个新的对象常常引发 bug。至少至少,它很容易出错。要小心使用 `Object.assign()`。
-### Destructuring ###
+### 解构 ###
-Both objects and arrays support destructuring, meaning that you can extract values from them and assign them to named variables:
+对象和数组都支持解构,这意味着你可以从中提取值分配给命过名的变量:
```
const [t, u] = ['a', 'b'];
@@ -130,19 +131,19 @@ const blep = {
blop: 'blop'
};
-// The following is equivalent to:
+// 下面等同于:
// const blop = blep.blop;
const { blop } = blep;
blop; // 'blop'
```
-As with the array example above, you can destructure to multiple assignments at once. Here’s a line you’ll see in lots of Redux projects:
+和上面数组的例子类似,你可以一次解构多次分配。下面这行你在大量的 Redux 项目中都能见到。
```
const { type, payload } = action;
```
-Here’s how it’s used in the context of a reducer (much more on that topic coming later):
+下面是它在一个 reducer(后面的话题再详细说) 的上下文中的使用方法。
```
const myReducer = (state = {}, action = {}) => {
@@ -154,75 +155,76 @@ const myReducer = (state = {}, action = {}) => {
};
```
-If you don’t want to use a different name for the new binding, you can assign a new name:
+如果不想为新绑定使用不同的名称,你可以分配一个新名称:
```
const { blop: bloop } = blep;
bloop; // 'blop'
```
-Read: Assign `blep.blop` as `bloop`.
+读作:把 `blep.blop` 分配给 `bloop`。
-### Comparisons and Ternaries ###
+### 比较运算符和三元表达式 ###
-You can compare values with the strict equality operator (sometimes called “triple equals”):
+你可以用严格的相等操作符(有时称为“三等于”)来比较数据值:
```
3 + 1 === 4; // true
```
-There’s also a sloppy equality operator. It’s formally known as the “Equal” operator. Informally, “double equals”. Double equals has a valid use-case or two, but it’s almost always better to default to the `===` operator, instead.
+还有另外一种宽松的相等操作符。它正式地被称为“等于”运算符。非正式地可以叫“双等于”。双等于有一两个有效的用例,但大多数时候默认使用 `===` 操作符是更好的选择。
+
-Other comparison operators include:
+其它比较操作符有:
-- `>` Greater than
-- `<` Less than
-- `>=` Greater than or equal to
-- `<=` Less than or equal to
-- `!=` Not equal
-- `!==` Not strict equal
-- `&&` Logical and
-- `||` Logical or
+- `>` 大于
+- `<` 小于
+- `>=` 大于或等于
+- `<=` 小于或等于
+- `!=` 不等于
+- `!==` 严格不等于
+- `&&` 逻辑与
+- `||` 逻辑或
-A ternary expression is an expression that lets you ask a question using a comparator, and evaluates to a different answer depending on whether or not the expression is truthy:
+三元表达式是一个可以让你使用一个比较器来问问题的表达式,运算出的不同答案取决于表达式是否为真:
```
14 - 7 === 7 ? 'Yep!' : 'Nope.'; // Yep!
```
-### Functions ###
+### 函数 ###
-JavaScript has function expressions, which can be assigned to names:
+JavaScript 支持函数表达式,函数可以这样分配名称:
```
const double = x => x * 2;
```
-This means the same thing as the mathematical function `f(x) = 2x`. Spoken out loud, that function reads `f` of `x` equals `2x`. This function is only interesting when you apply it to a specific value of `x`. To use the function in other equations, you'd write `f(2)`, which has the same meaning as `4`.
+这和数学表达式 `f(x) = 2x` 是一个意思。大声说出来,这个函数读作 `x` 的 `f` 等于 `2x`。这个函数只有当你用一个具体的 `x` 的值应用它的时候才有意思。在其它方程式里面你写 `f(2)`,就等同于 `4`。
-In other words, `f(2) = 4`. You can think of a math function as a mapping from inputs to outputs. `f(x)` in this case is a mapping of input values for `x` to corresponding output values equal to the product of the input value and `2`.
+换种说话就是 `f(2) = 4`。您可以将数学函数视为从输入到输出的映射。这个例子里 `f(x)` 是输入数值 `x` 到相应的输出数值的映射,等于输入数值和 `2` 的乘积。
-In JavaScript, the value of a function expression is the function itself:
+在 JavaScript 中,函数表达式的值是函数本身:
```
double; // [Function: double]
```
-You can see the function definition using the `.toString()` method:
+你可以使用 `.toString()` 方法看到这个函数的定义。
```
double.toString(); // 'x => x * 2'
```
-If you want to apply a function to some arguments, you must invoke it with a function call. A function call applies a function to its arguments and evaluates to a return value.
+如果要将函数应用于某些参数,则必须使用函数调用来调用它。函数调用会接收参数并且计算一个返回值。
-You can invoke a function using `(argument1, argument2, ...rest)`. For example, to invoke our double function, just add the parentheses and pass in a value to double:
+你可以使用 `(argument1, argument2, ...rest)` 调用一个函数。比如调用我们的 double 函数,就加一对括号并传进去一个值:
```
double(2); // 4
```
-Unlike some functional languages, those parentheses are meaningful. Without them, the function won’t be called:
+和一些函数式语言不同,这对括号是有意义的。没有它们,函数将不会被调用。
```
double 4; // SyntaxError: Unexpected number
@@ -481,7 +483,7 @@ Want to learn more about functional programming in JavaScript?
[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/). If you’re not a member, you’re missing out!
[
-](https://ericelliottjs.com/product/lifetime-access-pass/)
+](https://ericelliottjs.com/product/lifetime-access-pass/)
***Eric Elliott*** is the author of [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly), and [*“Learn JavaScript with Eric Elliott”*](http://ericelliottjs.com/product/lifetime-access-pass/) . He has contributed to software experiences for **Adobe Systems, Zumba Fitness, The Wall Street Journal, ESPN, BBC and top recording artists including Usher, Frank Ocean, Metallica**, and many more.
From 1656973000dc535edb97f9447f47325a89444f7d Mon Sep 17 00:00:00 2001
From: Wenlin Ou
Date: Wed, 12 Apr 2017 19:19:12 -0400
Subject: [PATCH 212/945] errors fixes
---
TODO/nothing-will-change-until-you-start-building.md | 9 ++++-----
1 file changed, 4 insertions(+), 5 deletions(-)
diff --git a/TODO/nothing-will-change-until-you-start-building.md b/TODO/nothing-will-change-until-you-start-building.md
index 034e8ae1484..f063274822b 100644
--- a/TODO/nothing-will-change-until-you-start-building.md
+++ b/TODO/nothing-will-change-until-you-start-building.md
@@ -10,15 +10,15 @@
# 真正行动之前 你将一无所成
-就在上周我打了一辆 Lyft 出租车,司机跟我聊了很多关于她的天花乱坠的想法。她的想法里,她有写一本儿童书籍的想法,有开发一个帮助人们找到停车位置的软件的想法,有寻找到一种更高效打包礼物的想法,但问题是她总是犹豫不决:她有很多想法但却不知道从何开始。
+就在上周我打了一辆 Lyft 出租车,司机跟我聊了很多关于她的天花乱坠的想法。她既想写一本儿童书籍,又想开发一个帮助人们找到停车位置的软件,还想找到一种更高效的打包礼物的方式,但问题是她总是犹豫不决:她有很多想法但却不知道从何开始。
-接下来这段话是我对她说的:**开始一些实际行动吧。选一个项目然后尽可能的去完成它。** 如果你想写一本书,那么从每天写一页开始;如果你想做一个APP,那就从构建草图开始。任何人都可以做到这些。
+接下来这段话是我对她说的:**开始一些实际行动吧。选一个项目然后尽可能的去完成它。** 如果你想写一本书,那么从每天写一页开始;如果你想做一个 APP ,那就从构建草图开始。任何人都可以做到这些。
这个建议适用于所有的创作者。一旦你开始着手在你的项目上,你将会不由自主的继续做下去。创新将会成为你人格的一部分 —— 即使你的项目失败了。
你今天迈出的一小步将会是你人生的一大步。看看那些伟大的产品缔造者吧,比如 [Drew Wilson](https://twitter.com/drewwilson), [Pieter Levels](https://twitter.com/levelsio?), 和 [Sebastian Dobrincu](https://twitter.com/Sebyddd)。**他们不仅仅是等待合适的时机。** 他们每周都在更新他们的产品。
-在一个 [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics) 的节目中, [Josh Pigford](https://twitter.com/Shpigford) 透露,在做 Baremetrics 之前,他已经尝试过 *一百个* 失败的点子。 然而现在, Baremetrics 每个月都会有 60,000 美金进账。
+在一个 [IndieHackers](https://www.indiehackers.com/podcast/006-josh-pigford-of-baremetrics) 的采访中, [Josh Pigford](https://twitter.com/Shpigford) 透露,在做 Baremetrics 之前,他已经尝试过 *一百个* 失败的点子。 然而现在, Baremetrics 每个月都会有 60,000 美金进账。
![](https://cdn-images-1.medium.com/max/800/1*BzmVaqAKEzNRybUmMYOxdA.png)
@@ -39,7 +39,7 @@
![](https://cdn-images-1.medium.com/max/800/1*ESildSVTSSOnFXGDxD-l9w.png)
-另一个让你保持动力的建议就是让你的项目更加公开。在网上和你的同行们分享你的进展。人们很喜欢看到线框(wireframe 翻译成 ”框架“ 会不会好一点?)类的东西,并常常带来有价值的投入。
+另一个让你保持动力的建议就是让你的项目更加公开。在网上和你的同行们分享你的进展。人们很喜欢看到流程图一类的东西,并常常带来有价值的投入。
我自己找到的一个非常有用的保持动力的小秘诀是和朋友打赌。我对他们说,如果我在截止日期前没有完成某个任务,我就给他们一百刀。比如,我会在一个项目开始的两周内找到二十个人打赌。
@@ -83,7 +83,6 @@
我在 Medium 上每周都会发文。也可以关注我的 [Twitter](https://twitter.com/JonathanZWhite),我会发一些关于设计、前端开发和虚拟现实的杂想。
-*P.S. If you enjoyed this article, it would mean a lot if you click the 💚 and share with friends.*(可删?)
---
From 9adaad4fe374f2caf5ea1914f62f9a81d30c60be Mon Sep 17 00:00:00 2001
From: sqrtthree
Date: Thu, 13 Apr 2017 10:05:57 +0800
Subject: [PATCH 213/945] :memo: Adjust the translator's information format
(#1435)
---
TODO/beyond-browser-web-desktop-apps.md | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/TODO/beyond-browser-web-desktop-apps.md b/TODO/beyond-browser-web-desktop-apps.md
index f40c59cd68e..56eefc24934 100644
--- a/TODO/beyond-browser-web-desktop-apps.md
+++ b/TODO/beyond-browser-web-desktop-apps.md
@@ -1,8 +1,8 @@
> * 原文地址:[Beyond The Browser: From Web Apps To Desktop Apps](https://www.smashingmagazine.com/2017/03/beyond-browser-web-desktop-apps/)
> * 原文作者:本文已获原作者 [Adam Lynch](https://www.smashingmagazine.com/author/adamlynch/) 授权
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者: [bambooom](https://github.com/bambooom)/[imink](https://github.com/imink)
-> * 校对者:[bambooom](https://github.com/bambooom)/[imink](https://github.com/imink)/[sunui](https://github.com/sunui)
+> * 译者: [bambooom](https://github.com/bambooom)、[imink](https://github.com/imink)
+> * 校对者:[bambooom](https://github.com/bambooom)、[imink](https://github.com/imink)、[sunui](https://github.com/sunui)
## 超越浏览器:从 web 应用到桌面应用
From 38ccbc8b663fc62fb2dce80af2bc6d760b3b4dbf Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Thu, 13 Apr 2017 10:32:35 +0800
Subject: [PATCH 214/945] Add image.
---
...-and-rise-of-functional-programming-composable-software.md | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
index 608366c3f41..dce44040646 100644
--- a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
+++ b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
@@ -90,8 +90,8 @@ lambda 表达式对软件设计产生了很大的影响,在 1980 年之前,
[![](https://i.embed.ly/1/display/resize?url=https%3A%2F%2Fpbs.twimg.com%2Fprofile_images%2F3722564505%2Fb8030b8f3990875e8f38ed877fdf8d25_bigger.png&key=4fce0568f2ce49e8b54624ef71a8a5bd&width=40)](https://video.twimg.com/tweet_video/CmectVVVUAAsvpo.mp4)
-
-我一直在告诉大家 [JavaScript] 并不是一门玩具语言。现在我需要展示它。[#StarWars](https://twitter.com/hashtag/StarWars?src=hash) [@BrendanEich](https://twitter.com/BrendanEich)
+![](http://ww1.sinaimg.cn/large/006tNbRwgy1fekui0p6i3j30j50hcmyn.jpg)
+我一直在告诉大家 #JavaScript 并不是一门玩具语言。现在我需要展示它。
在 2015 年,使用函数的组合来编写软件又开始流行起来。为了更简单化,JavaScript 规范获得的数十年来第一次主要的更新并且添加了箭头函数,为了更简单的编写和读取函数、柯里化,和 lambda 语句。
From d7bb3d2a8c355b62b64ac5c987746005bde080fd Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Thu, 13 Apr 2017 10:34:39 +0800
Subject: [PATCH 215/945] Delete unnecessary image.
---
...ll-and-rise-of-functional-programming-composable-software.md | 2 --
1 file changed, 2 deletions(-)
diff --git a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
index dce44040646..577d3d97d34 100644
--- a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
+++ b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
@@ -88,8 +88,6 @@ lambda 表达式对软件设计产生了很大的影响,在 1980 年之前,
在 2010 年左右,一些有趣的事情发生了:JavaScript 的崛起。在大概 2006 年以前,JavaScript 被广泛的看作玩具语言和被用制作浏览器中好玩的动画,但是它里面隐藏着一些极其强大的特性。即 lambda 表达式中最重要的特性。人们开始暗中讨论一个叫做 “函数式编程的” 酷东西。
-[![](https://i.embed.ly/1/display/resize?url=https%3A%2F%2Fpbs.twimg.com%2Fprofile_images%2F3722564505%2Fb8030b8f3990875e8f38ed877fdf8d25_bigger.png&key=4fce0568f2ce49e8b54624ef71a8a5bd&width=40)](https://video.twimg.com/tweet_video/CmectVVVUAAsvpo.mp4)
-
![](http://ww1.sinaimg.cn/large/006tNbRwgy1fekui0p6i3j30j50hcmyn.jpg)
我一直在告诉大家 #JavaScript 并不是一门玩具语言。现在我需要展示它。
From 09e5fb0658df415a4da89ce8b0fc8d68d6f5cee6 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Thu, 13 Apr 2017 10:35:19 +0800
Subject: [PATCH 216/945] Update
the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
---
...all-and-rise-of-functional-programming-composable-software.md | 1 +
1 file changed, 1 insertion(+)
diff --git a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
index 577d3d97d34..1b0a3b7c98c 100644
--- a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
+++ b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
@@ -89,6 +89,7 @@ lambda 表达式对软件设计产生了很大的影响,在 1980 年之前,
在 2010 年左右,一些有趣的事情发生了:JavaScript 的崛起。在大概 2006 年以前,JavaScript 被广泛的看作玩具语言和被用制作浏览器中好玩的动画,但是它里面隐藏着一些极其强大的特性。即 lambda 表达式中最重要的特性。人们开始暗中讨论一个叫做 “函数式编程的” 酷东西。
![](http://ww1.sinaimg.cn/large/006tNbRwgy1fekui0p6i3j30j50hcmyn.jpg)
+
我一直在告诉大家 #JavaScript 并不是一门玩具语言。现在我需要展示它。
在 2015 年,使用函数的组合来编写软件又开始流行起来。为了更简单化,JavaScript 规范获得的数十年来第一次主要的更新并且添加了箭头函数,为了更简单的编写和读取函数、柯里化,和 lambda 语句。
From 61ba93b7f49d64895f6a01c150acf9c57f1ff8b1 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Thu, 13 Apr 2017 10:38:50 +0800
Subject: [PATCH 217/945] Update article link.
---
...ll-and-rise-of-functional-programming-composable-software.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
index 1b0a3b7c98c..a529896a3c1 100644
--- a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
+++ b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
@@ -100,7 +100,7 @@ lambda 表达式对软件设计产生了很大的影响,在 1980 年之前,
你在阅读下一部分时,可以使用实例实验,记住要把你自己当孩子一样把其他的思想扔在一边在学习中去探索和玩耍。重新发现孩童时发现新事物的欣喜。让我们来做一些魔术吧。
-[接下来的第二部分:“为什么要用 JavaScript 学习函数式编程?”]
+[接下来的第二部分:“为什么要用 JavaScript 学习函数式编程?”](https://github.com/xitu/gold-miner/blob/master/TODO/why-learn-functional-programming-in-javascript-composing-software.md)
### 下一步
From 4aa2181333f014def4d8f91742f5381eb0c788fc Mon Sep 17 00:00:00 2001
From: PhxNirvana
Date: Thu, 13 Apr 2017 11:16:51 +0800
Subject: [PATCH 218/945] Update multithreading-with-rxjava-dadddc.md
---
TODO/multithreading-with-rxjava-dadddc.md | 73 ++++++++++++-----------
1 file changed, 37 insertions(+), 36 deletions(-)
diff --git a/TODO/multithreading-with-rxjava-dadddc.md b/TODO/multithreading-with-rxjava-dadddc.md
index f2c11d0df96..d81eb058383 100644
--- a/TODO/multithreading-with-rxjava-dadddc.md
+++ b/TODO/multithreading-with-rxjava-dadddc.md
@@ -1,24 +1,25 @@
+
> * 原文地址:[Multithreading with RxJava](https://android.jlelse.eu/multithreading-with-rxjava-dadddc4f7a63#.yghtx4u43)
> * 原文作者:[Pierce Zaifman](https://android.jlelse.eu/@PierceZaifman?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[PhxNirvana](https://github.com/phxnirvana)
> * 校对者:
-# Multithreading with RxJava #
+# RxJava 中的多线程 #
-Most of the code I’ve ever written for Android is fast enough that speed has never been a concern. Until a few weeks ago when I was working on an app that reads and analyzes large files.
+大多数情况下,我写的 Android 代码都是可以流畅运行的,因此,速度从不是我考虑的问题。但几周前我处理一个读取和分析大文件的 app 时,浮现出了这一问题。
-While I expect users to understand that a larger file will take longer to process, at some point they will give up on my app. Either because they think the app isn’t working, or they simply aren’t willing to wait that long. So if I can cut that time in half or less, it’s certainly worth doing.
+尽管我期望用户明白文件越大,耗时越长的道理,有时候他们仍会放弃我的应用。他们可能认为应用卡住了,也可能是因为他们就不想等那么久。所以如果我能把时间砍一半或更多的话,一定会大有裨意的。
-#### First attempt ####
+#### 初体验 ####
-Since I’ve been using RxJava for all of my background jobs, it made sense to try to do the same for this. Essentially I have some code that looks like this:
+因为我所有后台任务都用 RxJava 重写了,所以继续用 RxJava 来解决这个问题也是自然而然的。尤其是我还有一些如下所示的代码:
```
List dataList;
-//dataList gets populated here
+//这里是数据列表
List result = new ArrayList<>();
for (String data : dataList) {
@@ -26,80 +27,80 @@ for (String data : dataList) {
}
```
-So, I just tried to stick each iteration of the loop into a background thread. You can see what I did below:
+所以我只是想把循环的每个操作放到一个后台线程中。如下所示:
```
List dataList;
-//dataList gets populated here
+//这里是数据列表
List> tasks = new ArrayList<>();
for (String data : dataList) {
tasks.add(Observable.just(data).subscribeOn(Schedulers.io()).map(s -> {
- // returns a DataModel object
+ // 返回一个 DataModel 对象
return DataParser.createData(s);
}));
}
List result = new ArrayList<>();
-// Wait for everything to finish and collect the results
+// 等待运行结束并收集结果
for (DataModel dataModel : Observable.merge(tasks).toBlocking().toIterable()) {
result.add(dataModel);
}
```
-It did speed it up, it was running in about half the time. But, it was causing a lot of garbage collection (GC), which made the UI very stuttery and janky while it was loading. To try to understand what was happening, I added a log to print out `Thread.currentThread().getName()`. This made me realize that I was actually spawning a new thread for each piece of data. As it turns out, spawning thousands of threads is not a good idea.
+的确起作用了,时间减少了近一半。但也导致大量垃圾回收(GC),这使得加载时的 UI 又卡又慢。为了搞清楚问题的原因,我加了一句 log 打印如下信息 `Thread.currentThread().getName()`。 这样我就搞清楚了,我在处理每一段数据时都新建了线程。正如结果所示,创建上千个线程并不是什么好主意。
-#### Second attempt ####
+#### 二进宫 ####
-I’ve already accomplished my goal of speeding up the data processing, but it’s not running very smoothly. I was also wondering if it could run even faster if it didn’t cause so much GC. So instead of spawning a new thread, I created my own limited thread pool for RxJava to use:
+我已经完成了加速数据处理的目标,但运行起来并不那么流畅。我想知道如果不触发这么多 GC 的话还能不能跑得再快点。所以作为直接创建新线程的代替,我建立了我自己的限定线程数量的线程池提供给 RxJava 使用:
```
List dataList;
-//dataList gets populated here
+//这里是数据列表
List> tasks = new ArrayList<>();
-// Get the max number of threads we could have
+// 取得能够使用的最大线程数
int threadCount = Runtime.getRuntime().availableProcessors();
ExecutorService threadPoolExecutor = Executors.newFixedThreadPool(threadCount);
Scheduler scheduler = Schedulers.from(threadPoolExecutor);
for (String data : dataList) {
tasks.add(Observable.just(data).subscribeOn(scheduler).map(s -> {
- // returns a DataModel object
+ // 返回一个 DataModel 对象
return DataParser.createData(s);
}));
}
List result = new ArrayList<>();
-// Wait for everything to finish and collect the results
+// 等待运行结束并收集结果
for (DataModel dataModel : Observable.merge(tasks).toBlocking().toIterable()) {
result.add(dataModel);
}
```
-For a dataset where each piece of data was quite large, this resulted in a further time savings of roughly 10%. However, for a dataset where each piece of data was small, it reduced the processing time by roughly 30%. It also reduced the number of GC calls, but there were still too many.
+对于单个数据都很大的数据集来说,这样减少了约 10% 的数据处理时间。然而,对于单个数据都很小的数据集就减少了约 30% 的时间。这同时也减少了 GC 的数量,但 GC 还是太频繁。
#### Third attempt ####
-I had one more idea — what if the overhead of managing and switching threads was slowing it down? To get around this, I could group my data into one sublist per thread. That way it would still run in parallel, but the management of each thread would be minimal. I adapted the solution [here](https://github.com/ReactiveX/RxJava/issues/3532#issuecomment-157509946) to implement this idea:
+我有一个新想法——如果性能的瓶颈是线程的管理和切换会怎么样?为了克服这个问题,我可以将数据按照线程来拆分。这样虽然仍是并发运行,但对每个线程的管理将大大减小。我尝试使用 [这里](https://github.com/ReactiveX/RxJava/issues/3532#issuecomment-157509946) 的解决方法来实现我的想法:
```
List dataList;
-//dataList gets populated here
+//这里是数据列表
-// Get the max number of threads we could have
+// 取得能够使用的最大线程数
int threadCount = Runtime.getRuntime().availableProcessors();
ExecutorService threadPoolExecutor = Executors.newFixedThreadPool(threadCount);
Scheduler scheduler = Schedulers.from(threadPoolExecutor);
AtomicInteger groupIndex = new AtomicInteger();
-// Split the data into groups by thread number, then process each group on their own thread
+// 以线程数量为依据分组数据,将每组数据放到它们自己的线程中
Iterable> resultGroups =
Observable.from(dataList).groupBy(k -> groupIndex.getAndIncrement() % threadCount)
.flatMap(group -> group.observeOn(scheduler).toList().map(sublist -> {
@@ -112,24 +113,24 @@ Iterable> resultGroups =
List result = new ArrayList<>();
-// Wait for everything to finish and collect the results
+// 等待运行结束并收集结果
for (List dataModels : resultGroups) {
result.addAll(dataModels);
}
```
-Previously I mentioned testing with two datasets. The first had large individual items but fewer total items. The second had many more total items, but each item was smaller. When I ran this solution against the first dataset, the difference was negligible. But, when I ran it on the second, longer dataset, the change was dramatic. With my previous approach it took about 20 seconds to process this dataset. Now, it takes about 5 seconds.
+上文中我提到用两组数据测试,一组单个数据大但总数少,另一组单个数据小但数量多。当我再次测试时,第一组数据几乎没差别,而第二组改变相当大。之前几乎要 20秒,现在只需 5秒。
-The reason the second dataset was sped up so much, is because of the number of data items. For each item, it has to schedule a thread to do the work. Now that I’ve reduced this scheduling work, there’s very little overhead.
+第二组数据运行时间改进如此大的原因是数量的改变。每个数据都需要指定工作线程,而现在减少了指定线程时的工作量,性能也从而提升了。
#### Clean up ####
-There’s a few places in my code where I need to do this parallel work. So I refactored this solution into a util method that can be called anywhere:
+上面的代码要执行并发还有一些地方需要修改,所以我重新用 util 整理了代码,增加了这段代码的通用性:
```
/**
* Will process the data with the callable by splitting the data into the specified number of threads.
- * T is ths type of data being parsed, and U is the type of data being returned.
+ * T 是要被处理的数据类型,U 是返回的数据类型
*/
public static Iterable parseDataInParallel(List data, Func1, U> worker) {
int threadCount = Runtime.getRuntime().availableProcessors();
@@ -161,20 +162,20 @@ for (List dataModels : resultGroups) {
}
```
-Here `T` is the type being processed, in the example that would be `DataModel`. You pass in a `List` to be processed, and expect a result `U`. In my example `U` is `List`, but it could be anything, it doesn’t have to be a list either. The worker function being passed in is what does the processing of each sublist, and returns a result.
+这里 `T` 是被处理的数据类型,样例中是`DataModel`。传入待处理的 `List` 并期望结果是 `U`。在我的样例中 `U` 是 `List`,但它可以是任何东西,并不一定是一个 list。被传入的工作方法处理每个子列表并返回结果。
-#### Can we make it faster? ####
+#### 可以再快点么? ####
-There are many factors at play here that affect how fast this can run. Like how the threads are managed, how many threads are available, what kind of device the app is running on, etc. Most of these I can’t control, but there was something else I didn’t consider with my solution.
+事实上影响运行速度的因素有许多。比如线程管理方式,线程数,设备等。大多数因素我无法控制,但总有一些是我没有考虑到的。
-What if the data isn’t split evenly? For example, if I have 4 threads, what if every data item assigned to the 4th thread is 10 times larger than any other item? Then the 4th thread will end up running roughly 10 times longer than any other thread. In that case I won’t save much time with multithreading. The second approach I had actually solves this problem, since a thread is only assigned as it becomes available. But that solution was too slow.
+如果数据不是平均分配的会怎么样?举个例子,如果有 4 个线程,第四个线程数据量是其他的 10 倍会怎么样?这时第四个线程的耗时就是其他线程的大约 10 倍。这种情况下使用多线程就不会减少多少时间。我的第二次尝试基本解决了这个问题,因为线程只在需要时才初始化。但这个方法太慢了。
-I also tried to change how the data is grouped into sublists. Instead of doing it arbitrarily, I could keep track of the total size of the data in each group. Then, assign each data item to whichever list is the smallest. This way, the amount of work done by each thread is close to being equal. Unfortunately, in testing this solution I found the time it takes to do this extra setup usually costs more time than it saves.
+我也试过改变数据分组方式。作为随意分配的取代,我可以跟踪每一组数据的总量,然后将数据分配给最少的那组。这样每个线程的工作量就接近平均了。倒霉的是,测试之后发现这样做增加的时间远大于它节省的时间。
-Depending on how unevenly you expect your data to be split up, running it this way may actually be faster. But for the most part, it seems like just randomly splitting up the data is faster. Ideally if each thread could be assigned when available, with minimal overhead, it would be the most efficient solution. But, I couldn’t find a way to reduce the overhead enough to make it worth it.
+这么做可能会更快,(结果)取决于原始数据有多不平均。但大多数情况下,随机分配看起来更快些。理想情况下是每个线程一有空就分配任务,同时执行分配所消耗的资源也少,这是最高效的。但我找不到一个足够高效的可以减少分配瓶颈的方法。
#### Final thoughts ####
-So if you need to run some code in parallel, here’s one approach. Let me know what you think, there are many variables at play here. It’s difficult to determine the optimal solution, if there even is one. Also remember, just because you *can* run something in parallel doesn’t mean you *should*.
+所以如果你想要并发运行一些代码,这里提供一种方法。让我知道你的想法,这里还有许多考验控制的变量。得到一个最优解(如果有的话)总是很难的。以及,**能**并发执行并不意味着**必须**用并发。
-### If You Enjoyed Reading, Please Click That Little Heart. If You Want To Read More Like This, Follow Me On [Medium](https://medium.com/@piercezaifman) . Thanks! ###
+### 如果有收获的话,轻轻扎一下小红心吧老铁。想阅读更多,在 [Medium](https://medium.com/@piercezaifman) 关注我。谢谢!(顺便关注一下 [译者](https://juejin.im/user/57a16f4e6be3ff00650682d8) 233) ###
From eb33e6ae1b32c8e99838d3a9195cdeba9759e8ac Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E9=94=85=E7=82=89=E5=B7=A5?=
Date: Thu, 13 Apr 2017 12:50:46 +0800
Subject: [PATCH 219/945] modified according to @GitFuture 's revison
---
...ava-8-stream-android-rxjava2-hell-part4.md | 92 +++++--------------
1 file changed, 21 insertions(+), 71 deletions(-)
diff --git a/TODO/war-learning-curve-rx-java-2-java-8-stream-android-rxjava2-hell-part4.md b/TODO/war-learning-curve-rx-java-2-java-8-stream-android-rxjava2-hell-part4.md
index b0b63eeb7c5..c496724c8e2 100644
--- a/TODO/war-learning-curve-rx-java-2-java-8-stream-android-rxjava2-hell-part4.md
+++ b/TODO/war-learning-curve-rx-java-2-java-8-stream-android-rxjava2-hell-part4.md
@@ -1,8 +1,8 @@
> * 原文地址:[War against Learning Curve of RxJava2 + Java8 Stream [ Android RxJava2 ] ( What the hell is this ) Part4](http://www.uwanttolearn.com/android/war-learning-curve-rx-java-2-java-8-stream-android-rxjava2-hell-part4/)
> * 原文作者:[Hafiz Waleed Hussain](http://www.uwanttolearn.com/author/admin/)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
-> * 校对者:
+> * 译者: [Boiler Yao](https://github.com/boileryao)
+> * 校对者: [Vivienmm](https://github.com/Vivienmm)、[GitFuture](https://github.com/GitFuture)
## 大战 RxJava2 和 Java8 Stream [ Android RxJava2 ] (这到底是什么) 第四部分 ##
@@ -11,17 +11,16 @@
又是新的一天,如果学点新东西,这一天一定会很酷炫。
-小伙伴们一切顺利啊,这是我们的 RxJava2 Android 系列的第四部分 [ [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md), [第二部分](http://www.uwanttolearn.com/android/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2/), [第三部分](https://github.com/xitu/gold-miner/blob/master/TODO/functional-interfaces-functional-programming-and-lambda-expressions-reactive-programming-android-rxjava2-what-the-hell-is-this-part3.md) ]。 好消息是我们已经做好准备,可以开始使用 Rx 了。在使用 RxJava2 Android Observable 之前,我会先用 Java8 的 Stream 来做响应式编程。我认为我们应该了解 Java8,我还感觉通过使用 Java8 的流式 API 会让 RxJava2 Android 的学习曲线更简单。
-
+小伙伴们一切顺利啊,这是我们的 RxJava2 Android 系列的第四部分 [ [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md), [第二部分](http://www.uwanttolearn.com/android/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2/), [第三部分](https://github.com/xitu/gold-miner/blob/master/TODO/functional-interfaces-functional-programming-and-lambda-expressions-reactive-programming-android-rxjava2-what-the-hell-is-this-part3.md) ]。 好消息是我们已经做好准备,可以开始使用 Rx 了。在使用 RxJava2 Android Observable 之前,我会先用 Java8 的 Stream 来做响应式编程。我认为我们应该了解 Java8,而且通过使用 Java8 的 Stream API 让我感觉学习 RxJava2 Android 的过程更简单。
**动机:**
-动机跟我在 [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md) 和大家分享过的一样。在我开始学习 RxJava2 Android 的时候,我并不知道我该怎样使用、在什么地方使用它。
+动机跟我在 [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md) 和大家分享过的一样。在我开始学习 RxJava2 Android 的时候,我并不知道自己会在什么地方,以何种方式使用到它。
-现在我们已经学会了一些预备知识,但当时我什么都不知道。我那时从学习如何从数据或对象创建 Observable 开始。然后学会当 Observable 数据上发生变化时,去调用哪些接口(或者可以叫做“回调”)。这在理论上很棒,但是当我付诸实践的时候,GG了。我发现很多理论上应该成立的模式在我去用的时候完全不起作用。对我来说最大的问题,是不能用响应或者函数式响应的思维思考问题。我有命令式编程和面向对象编程的背景,由于先入为主,所以对我来说理解会有些难。我一直在问这些问题:我该在哪里实现?我应该怎么实现?如果你跟着这篇文章看下来,我可以 100% 保证你会知道怎样把命令式代码转换到 Rx 代码,虽然写出来的 Rx 代码可能不是很好,但是你起码知道怎么开始。
+现在我们已经学会了一些预备知识,但当时我什么都不懂。因此我开始学习如何根据数据或对象创建 Observable 。然后知道了当 Observable 的数据发生变化时,应该调用哪些接口(或者可以叫做“回调”)。这在理论上很好,但是当我付诸实践的时候,却 GG 了。我发现很多理论上应该成立的模式在我去用的时候完全不起作用。对我来说最大的问题,是不能用响应或者函数式响应的思维思考问题。我熟悉命令式编程和面向对象编程,由于先入为主,所以对我来说理解响应式会有些难。我一直在问这些问题:我该在哪里实现?我应该怎么实现?如果你能坚持看完这篇文章,我可以 100% 保证你会知道怎样把命令式代码转换成 Rx 代码,虽然写出来的 Rx 代码不是最好的,但至少你知道该从哪里入手了。
**回顾:**
-我想回顾之前三篇文章中我们提到过的所有概念 [ [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md)、[第二部分](http://www.uwanttolearn.com/android/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2/)、 [第三部分](https://github.com/xitu/gold-miner/blob/master/TODO/functional-interfaces-functional-programming-and-lambda-expressions-reactive-programming-android-rxjava2-what-the-hell-is-this-part3.md) ]。因为现在我们要用到这些概念了。在 [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md) 我们学习了观察者模式; 在 [第二部分](http://www.uwanttolearn.com/android/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2/) 学习了拉模式和推模式、命令式和响应式;在 [第三部分](https://github.com/xitu/gold-miner/blob/master/TODO/functional-interfaces-functional-programming-and-lambda-expressions-reactive-programming-android-rxjava2-what-the-hell-is-this-part3.md) 我们学习了函数式接口(Functional Interfaces)、 接口默认方法(Default Methods)、高阶函数(Higher Order Functions)、函数的副作用(Side Effects in Functions)、纯函数(Pure Functions)、Lambda 表达式和函数式编程。我在下面写了一些定义(很无聊的东西)。如果你早就记得的话,可以跳到下一部分。
+我想回顾之前三篇文章中我们提到过的所有概念 [ [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md)、[第二部分](http://www.uwanttolearn.com/android/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2/)、 [第三部分](https://github.com/xitu/gold-miner/blob/master/TODO/functional-interfaces-functional-programming-and-lambda-expressions-reactive-programming-android-rxjava2-what-the-hell-is-this-part3.md) ]。因为现在我们要用到这些概念了。在 [第一部分](https://github.com/xitu/gold-miner/blob/master/TODO/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2.md) 我们学习了观察者模式; 在 [第二部分](http://www.uwanttolearn.com/android/pull-vs-push-imperative-vs-reactive-reactive-programming-android-rxjava2-hell-part2/) 学习了拉模式和推模式、命令式和响应式;在 [第三部分](https://github.com/xitu/gold-miner/blob/master/TODO/functional-interfaces-functional-programming-and-lambda-expressions-reactive-programming-android-rxjava2-what-the-hell-is-this-part3.md) 我们学习了函数式接口(Functional Interfaces)、 接口默认方法(Default Methods)、高阶函数(Higher Order Functions)、函数的副作用(Side Effects in Functions)、纯函数(Pure Functions)、Lambda 表达式和函数式编程。我在下面写了一些定义(很无聊的东西)。如果你清楚这些定义,可以跳到下一部分。
**函数式接口是只有一个抽象方法的接口。**
**在 Java8 我们可以在接口中定义方法,这种方法叫做“默认方法”。**
**至少有一个参数是函数的函数和返回类型为函数的函数称为高阶函数。**
@@ -30,7 +29,7 @@
**简介:**
-今天我们将向 RxJava 的学习曲线宣战。我确定在最后我们会取得胜利。
+今天我们将向 RxJava 的学习宣战。我确定在最后我们会取得胜利。
作战策略:
@@ -54,7 +53,7 @@ Stream:
第一个问题:在英语中 Stream 是什么意思?
-答案:一条很窄的小河,或者源源不断流动的液体、空气、气体。在编程的时候把数据转化成“流”的形式,比如我有一个字符串但是我想把它变成“流”来使用我需要干些什么,我需要创建一个机制,使这个字符串满足“源源不断流动的液体、空气、气体 {**或者数据**}”的定义。问题是,我们为什么想要自己的数据变成“流”呢,下面是个简单的例子。
+答案:一条很窄的小河,或者源源不断流动的液体、空气、气体。在编程的时候把数据转化成“流”的形式,比如我有一个字符串,但是我想把它变成“流”来使用的话我需要干些什么,我需要创建一个机制,使这个字符串满足“源源不断流动的液体、空气、气体 {**或者数据**}”的定义。问题是,我们为什么想要自己的数据变成“流”呢,下面是个简单的例子。
就像下面这幅图中画的那样,我有一杯混合着大大小小石子的蓝色的水。
@@ -90,9 +89,9 @@ Stream:
把水转换成水流后,我们做了很多事情。我先用一个过滤器去掉了大石子,然后用另一个过滤器去掉了小石子, 最后用一个转换器(map)把水的颜色从蓝色变成黑色。
-当我将数据转换成流时,我将在编程中得到同样的好处。现在,我将把这个例子转换成代码。我要显示的代码是真正的代码。可能示例代码不能工作,但我将要使用的操作符和API是真实的,我们将在后面的实例中使用。所以,同志们不要把关注点放在编译上。通过这个例子,我有一种感觉,我们将很容易地把握这些概念。在这个例子中,重要的一点是,我使用Java8 的 Stream API 而不是 Rx API。我不想让事情变困难,但稍后我也会使用 Rx。
+当我将数据转换成流时,我将在编程中得到同样的好处。现在,我将把这个例子转换成代码。我要显示的代码是真正的代码。可能示例代码不能工作,但我将要使用的操作符和 API 是真实的,我们将在后面的实例中使用。所以,同志们不要把关注点放在编译上。通过这个例子,我有一种感觉,我们将很容易地把握这些概念。在这个例子中,重要的一点是,我使用 Java8 的 Stream API 而不是 Rx API。我不想让事情变困难,但稍后我也会使用 Rx。
-图像中的水 和 代码中的水:
+图像中的水 & 代码中的水:
[![](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_1-300x253.jpg)](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_1.jpg)
@@ -107,36 +106,22 @@ public static void main(String [] args){
```
输出:
-
water
-
water
-
big stone
-
water
-
water
-
small stone
-
water
-
small stone
-
small stone
-
water
-
water
-
water
-
water
-
water
-图像中的水流 和 代码中的水流:
+图像中的水流 & 代码中的水流:
[![](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_2-237x300.jpg)](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_2.jpg)
@@ -150,7 +135,7 @@ public static void main(String[] args) {
//输出和上面那个一样
```
-图像中的“大石子过滤器” 和 代码中的“大石子过滤器”:
+图像中的“大石子过滤器” & 代码中的“大石子过滤器”:
[![](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_3-300x252.jpg)](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_3.jpg)
@@ -167,13 +152,13 @@ private static Predicate BigStoneFilter = new Predicate() {
};
```
-正如我们在 [第三部分](https://github.com/xitu/gold-miner/blob/master/TODO/functional-interfaces-functional-programming-and-lambda-expressions-reactive-programming-android-rxjava2-what-the-hell-is-this-part3.md) 已经知道的,任何函数式接口都可以转换成 Lambda 表达式。把上面的代码转换成 Lambda 表达式:
+正如我们在 [第三部分](https://github.com/xitu/gold-miner/blob/master/TODO/functional-interfaces-functional-programming-and-lambda-expressions-reactive-programming-android-rxjava2-what-the-hell-is-this-part3.md) 所学到的,任何函数式接口都可以转换成 Lambda 表达式。把上面的代码转换成 Lambda 表达式:
```
private static Predicate BigStoneFilter = s -> !s.equals("big stone");
```
-图像中和代码中作用在水流上的”大石子过滤器“:
+图像和代码中的作用在水流上的“大石子过滤器”:
[![](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_4-204x300.jpeg)](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_4.jpeg)
@@ -189,36 +174,23 @@ private static Predicate BigStoneFilter = s -> !s.equals("big stone");
```
这里我使用了 forEach 方法,暂时把这当作流上的 for 循环。用在这里仅仅是为了输出。除去没有这个方法,我们也已经实现了我们在图像中表示的内容。是时候看看输出了:
-
water
-
water
-
water
-
water
-
small stone
-
water
-
small stone
-
small stone
-
water
-
water
-
water
-
water
-
water
没有大石子了,这意味着我们成功过滤了水。
-图像中的“小石子过滤器” 和 代码中的“小石子过滤器”:
+图像中的“小石子过滤器” & 代码中的“小石子过滤器”:
[![](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_5-300x229.jpg)](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_5.jpg)
@@ -226,7 +198,7 @@ water
private static Predicate SmallStoneFilter = s -> !s.equals("small stone");
```
-在图像和代码中使用”小石子过滤器“:
+在图像和代码中使用“小石子过滤器”:
[![](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_6-228x300.png)](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_6.png)
@@ -246,32 +218,23 @@ private static Predicate SmallStoneFilter = s -> !s.equals("small stone
我不打算解释 **SmallStoneFilter**,它的实现和 **BigStoneFilter** 是一样一样的。这里我只展示输出。
water
-
water
-
water
-
water
-
water
-
water
-
water
-
water
-
water
-
water
-图像中的”水颜色转换器“ 和 代码中的”水颜色转换器“
+图像中的“水颜色转换器” 和 代码中的“水颜色转换器”
[![](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_7-300x171.jpg)](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_7.jpg)
同志们这里需要注意!
-在 Java8 Stream 中有个叫做 Function 的函数式接口。所以,当我想进行转换的时候,需要把这个函数式接口送到流的转换(map)函数里面。现在,我给大家展示在我们的代码中如何创建“水颜色过滤器”。
+在 Java8 Stream 中有个叫做 Function 的函数式接口。所以,当我想进行转换的时候,需要把这个函数式接口送到流的转换(map)函数里面。现在,我给大家展示在我们的代码中如何创建“水颜色转换器”。
```
private static Function convertWaterColour = new Function() {
@@ -316,25 +279,15 @@ private static Function convertWaterColour = s -> s + " black";
```
输出:
-
water black
-
water black
-
water black
-
water black
-
water black
-
water black
-
water black
-
water black
-
water black
-
water black
完活!现在我们再次回顾一些内容。
@@ -567,13 +520,9 @@ protected void onCreate(Bundle savedInstanceState) {
```
输出:
-
03-12 16:13:32.432 14918-14918/async.waleed.rx I/Android: 1
-
03-12 16:13:32.432 14918-14918/async.waleed.rx I/Android: 4
-
03-12 16:13:32.432 14918-14918/async.waleed.rx I/Android: 9
-
03-12 16:13:32.432 14918-14918/async.waleed.rx I/Android: 16
```
@@ -605,7 +554,8 @@ Observable.fromArray(data)
我在一个项目中使用了 [OnBoarding](https://www.google.com/search?q=onboarding+ui&) 界面,根据 UI 设计需要在每个 OnBoarding 界面上显示点点,如下图所示:
-[![](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_15-300x287.jpg)](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_15.jpg)如果你观察得很仔细的话,可以看到我需要将选定的界面对应的点设置成黑色。
+[![](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_15-300x287.jpg)](http://www.uwanttolearn.com/wp-content/uploads/2017/03/war_against_learning_curve_of_rx_java_2_java_8_stream_15.jpg)
+如果你观察得很仔细的话,可以看到我需要将选定的界面对应的点设置成黑色。
命令式编程的代码:
@@ -767,7 +717,7 @@ Observable.range(0, 10)
**总结:**
-同志们干得好!今天我们学 Rx Android 学得很开心。我们从图画开始,然后使用了 Java8 的流(Stream)。之后将 Java8 的流转换到 RxJava 2 Android 的 Observable。再之后,我们看到了真实项目中的示例并且展示了在现有的项目中如何开始使用 Rx。最后,我展示了一些转换到 Rx 的技巧:把循环用 forEach 替换,把 if 换成 filter,用 map 进行数据转化,用 flatmap 代替嵌套的循环。下篇文章: [Dialogue between Rx Observable and a Developer (Me) [ Android RxJava2 ] ( What the hell is this ) Part5](http://www.uwanttolearn.com/android/dialogue-rx-observable-developer-android-rxjava2-hell-part5/).
+同志们干得好!今天我们学 Rx Android 学得很开心。我们从图画开始,然后使用了 Java8 的流(Stream)。之后将 Java8 的流转换到 RxJava 2 Android 的 Observable。再之后,我们看到了实际项目中的示例并且展示了在现有的项目中如何开始使用 Rx。最后,我展示了一些转换到 Rx 的技巧:把循环用 forEach 替换,把 if 换成 filter,用 map 进行数据转化,用 flatmap 代替嵌套的循环。下篇文章: [Dialogue between Rx Observable and a Developer (Me) [ Android RxJava2 ] ( What the hell is this ) Part5](http://www.uwanttolearn.com/android/dialogue-rx-observable-developer-android-rxjava2-hell-part5/).
希望你们开心,同志们再见!
From f4f068d917286b48fc9ceb648b99c3a76304fb77 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E5=90=B4=E6=99=93=E5=86=9B?=
Date: Thu, 13 Apr 2017 14:18:42 +0800
Subject: [PATCH 220/945] reduce
---
TODO/reduce-composing-software.md | 96 ++++++++++++++++---------------
1 file changed, 49 insertions(+), 47 deletions(-)
diff --git a/TODO/reduce-composing-software.md b/TODO/reduce-composing-software.md
index 8baf2c381ad..312cc27ea78 100644
--- a/TODO/reduce-composing-software.md
+++ b/TODO/reduce-composing-software.md
@@ -4,18 +4,18 @@
> * 译者:
> * 校对者:
-# Reduce (Composing Software) #
+# Reduce (软件编写) (第五部分)#
-Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0)
+Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0) (译注:该图是用 PS 将烟雾处理成方块状后得到的效果,参见 [flickr](https://www.flickr.com/photos/68397968@N07/11432696204)。))
-> Note: This is part of the “Composing Software” series on learning functional programming and compositional software techniques in JavaScript ES6+ from the ground up. Stay tuned. There’s a lot more of this to come!
-> [< Previous](https://medium.com/javascript-scene/higher-order-functions-composing-software-5365cf2cbe99#.su6cmn4f7) | [<< Start over at Part 1](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [Next >](https://medium.com/javascript-scene/functors-categories-61e031bac53f#.4hqndcx22)
+> 注意:这是 “软件编写” 系列文章的第四部分,该系列主要阐述如何在 JavaScript ES6+ 中从零开始学习函数式编程和组合化软件(compositional software)技术(译注:关于软件可组合性的概念,参见维基百科 [Composability](https://en.wikipedia.org/wiki/Composability))。后续还有更多精彩内容,敬请期待!
+> [<上一篇](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d#.w4y0mlpcs) | [<< 返回第一章](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe)
-**Reduce** (aka: fold, accumulate) utility commonly used in functional programming that lets you iterate over a list, applying a function to an accumulated value and the next item in the list, until the iteration is complete and the accumulated value gets returned. Many useful things can be implemented with reduce. Frequently, it’s the most elegant way to do any non-trivial processing on a collection of items.
+在函数式编程中,**reduce**(也称为:fold,ccumulate)允许你在一个序列上迭代,并应用一个函数来处理预先声明的累积值和当前迭代到的元素。当迭代完成时,将返回这个累积值。许多其他有用的事物都可以通过 reduce 实现。多数时候,reduce 可以说是处理集合中元素最优雅的方式。
-Reduce takes a reducer function and an initial value, and returns the accumulated value. For `Array.prototype.reduce()`, the initial list is provided by `this`, so it's not one of the arguments:
+reduce 接受一个 reducer 函数以及一个初始值,最终返回一个累积值。对于 `Array.prototype.reduce()` 来说, 初始列表将由 `this` 指明, 所以列表本身不会作为该函数的参数:
```
array.reduce(
@@ -24,28 +24,28 @@ array.reduce(
) => accumulator: Any
```
-Let’s sum an array:
+我们利用如下方式对一个数组进行求和:
```
[2, 4, 6].reduce((acc, n) => acc + n, 0); // 12
```
-For each element in the array, the reducer is called and passed the accumulator and the current value. The reducer’s job is to “fold” the current value into the accumulated value somehow. How is not specified, and specifying how is the purpose of the reducer function. The reducer returns the new accumulated value, and `reduce()` moves on to the next value in the array. The reducer may need an initial value to start with, so most implementations take an initial value as a parameter.
+对于数组的每步迭代,reducer 函数都会被调用,并且向其传入了累积值和当前迭代到的数组元素。reducer 的职责在于以某种方式将当前迭代的元素 “合拢(fold)” 到累加值中。reducer 规定了 “合拢” 的手段和方式,完成了对当前元素的 “合拢” 后,reducer 将返回新的累加值,然后, `.reduce()` 将开始处理数组中的下一个元素。reducer 需要一个初始值才能开始工作,所以绝大多数的 `.reduce()` 实现都需要接收一个初始值作为参数。
-In the case of this summing reducer, the first time the reducer is called, `acc` starts at `0` (the value we passed to `.reduce()` as the second parameter). The reducer returns `0` + `2` (`2` was the first element in the array), which is `2`. For the next call, `acc = 2, n = 4` and the reducer returns the result of `2 + 4` (`6`). In the last iteration, `acc = 6, n = 6`, and the reducer returns `12`. Since the iteration is finished, `.reduce()` returns the final accumulated value, `12`.
+在求数组元素和一例中,reducer 函数第一次调用时,`acc` 将会以 `0` 值(该值是传入 `.reduce()` 方法的第二个参数)开始。然后,reducer 返回了 `0` + `2`(`2` 是数组的第一个元素), 也就是返回了 `2` 作为新的累积值。下一步,`acc = 2, n = 4` 传入了 reducer,reducer返回了 `2 + 4`(`6`)。在最后一步迭代中,`acc = 6, n = 6`, reducer 返回了 `12`。迭代完成,`.reduce()` 返回了最终的累积值 `12`。
-In this case, we passed in an anonymous reducing function, but we can abstract it and give it a name:
+在这一例子中,我们传入了一个匿名函数作为 reducer,但是我们也可以抽象出每次求和的过程为一个具名函数,这使得我们代码的复用成都更高:
```
const summingReducer = (acc, n) => acc + n;
[2, 4, 6].reduce(summingReducer, 0); // 12
```
-Normally, `reduce()` works left to right. In JavaScript, we also have `[].reduceRight()`, which works right to left. In other words, if you applied `.reduceRight()` to `[2, 4, 6]`, the first iteration would use `6` as the first value for `n`, and it would work its way back and finish with `2`.
+通常,`reduce` 的工作过程为由左向右。在 JavaScript 中,我们也有一个 `[].reduceRight()` (译注:[MDN -- Array.prototype.reduceRight()](https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Array/ReduceRight))方法来让 reduce 由右向左地工作。 具体说来,如果你对数组 `[2, 4, 6]` 应用 `.reduceRight()` ,第一个被迭代到的元素就将是 `6`,最后一个迭代到的元素就是 `2`。
-### Reduce is Versatile ###
+### 无所不能的 reduce ###
-Reduce is versatile. It’s easy to define `map()`, `filter()`, `forEach()` and lots of other interesting things using reduce:
+别吃惊,reduce 确实无所不能,你所熟悉的 `map()`,`filter()`,`forEach()` 以及其他函数都可借助于 reduce 来创建。
**Map:**
@@ -55,7 +55,7 @@ const map = (fn, arr) => arr.reduce((acc, item, index, arr) => {
}, []);
```
-For map, our accumulated value is a new array with a new element for each value in the original array. The new values are generated by applying the passed in mapping function (`fn`) to each element in the `arr` argument. We accumulate the new array by calling `fn` with the current element, and concatenating the result to the accumulator array, `acc`.
+对于 map 来说,我们的累积值就是一个新的数组对象,该数组对象中的每个元素都由原数组对应元素映射得到。累积数组中新的元素由传入 map 的映射函数(`fn`)所确定:对于当前迭代到的元素 `item`,我们通过 `fn` 计算出新的元素,并将其拼接入累加数组 `arr` 中。
**Filter:**
@@ -65,37 +65,37 @@ const filter = (fn, arr) => arr.reduce((newArr, item) => {
}, []);
```
-Filter works in much the same way as map, except that we take a predicate function and *conditionally* append the current value to the new array if the element passes the predicate check (`fn(item)` returns `true`).
+filter 的工作方式与 map 类似,只不过原数组的元素只有通过一个真值检测函数(predicate function)才能被送入新的累积数组中。亦即,相较于 map,filter 是**有条件**地选择元素到累积数组中,并且不会改变元素的值。
-For each of the above examples, you have a list of data, iterate over that data applying some function and folding the results into an accumulated value. Lots of applications spring to mind. But what if *your data is a list of functions?*
+上面几个例子,你处理的数据都是一些数值序列,你在数值序列上应用指定的函数迭代数据,并将结果合拢到累积值中。大多数应用都因此开始雏形初备,但是你想过这个问题没有:**假如你的序列是函数序列呢?**
**Compose:**
-Reduce is also a convenient way to compose functions. Remember function composition: If you want to apply the function `f` to the result of `g` of `x` i.e., the composition, `f . g`, you could use the following JavaScript:
+reduce 也是实现函数组合的便捷渠道。假如你想用将函数 `g` 的输出作为函数 `f` 的输入,即组合这两个函数: `f . g`,那么你可以使用下面的 JavaScript 代码片,它没有任何的抽象:
```
f(g(x))
```
-Reduce lets us abstract that process to work on any number of functions, so you could easily define a function that would represent:
+reduce 让我们能抽象出函数组合过程,从而让你也能轻易地实现更多层次的函数组合:
```
f(g(h(x)))
```
-To make that happen, we’ll need to run reduce in reverse. That is, right-to-left, rather than left-to-right. Thankfully, JavaScript provides a `.reduceRight()` method:
+由于函数组合是由右向左的,我们就要使用上面提到的 `.reduceRight()` 方法来抽象函数组合过程:
```
const compose = (...fns) => x => fns.reduceRight((v, f) => f(v), x);
```
-> Note: If JavaScript had not provided `[].reduceRight()`, you could still implement `reduceRight()` -- using `reduce()`. I'll leave it to adventurous readers to figure out how.
+> 注意:如果 JavaScript 的版本没有提供 `[].reduceRight()`,你可以借助于 `reduce` 实现该方法。该实现留给读者自己思考。
**Pipe:**
-`compose()` is great if you want to represent the composition from the inside-out -- that is, in the math notation sense. But what if you want to think of it as a sequence of events?
+`compose()` 很好地描述了由内至外的组合过程,某种程度上,这是数学上的关于输入输出的组合。如果你想从事件发生顺序上来思考函数组合呢?
-Imagine we want to add `1` to a number and then double it. With `compose(), that would be:
+假设我们想要对一个数值加 `1`,然后对新得到的数值进行翻倍。如果是利用 `compose()`,就需要这么做:
```
const add1 = n => n + 1;
@@ -110,15 +110,15 @@ add1ThenDouble(2); // 6
// ((2 + 1 = 3) * 2 = 6)
```
-See the problem? The first step is listed last, so in order to understand the sequence, you’ll need to start at the bottom of the list and work your way backwards to the top.
+发现问题没有?第一步(加1操作)是 compose 序列上的最后一个元素,所以,`compose` 需要你自底向上地分析流程的执行。
-Or we can reduce left-to-right as you normally would, instead of right-to-left:
+如果我们以由左向右的方式进行 reduce 来创建函数组合,以示区别,我们用 `pipe` 来描述新的组合方式:
```
const pipe = (...fns) => x => fns.reduce((v, f) => f(v), x);
```
-Now you can write `add1ThenDouble()` like this:
+现在,新的流程就可以这么撰写:
```
const add1ThenDouble = pipe(
@@ -130,7 +130,7 @@ add1ThenDouble(2); // 6
// ((2 + 1 = 3) * 2 = 6)
```
-This is important because sometimes if you compose backwards, you get a different result:
+你也看到了,在组合中,顺序是非常重要的,如果你调换了 `double` 和 `add1` 的顺序,你将得到截然不同的结果:
```
const doubleThenAdd1 = pipe(
@@ -141,25 +141,25 @@ const doubleThenAdd1 = pipe(
doubleThenAdd1(2); // 5
```
-We’ll go into more details on `compose()` and `pipe()` later. What you should understand right now is that `reduce()` is a very powerful tool, and you really need to learn it. Just be aware that if you get very tricky with reduce, some people may have a hard time following along.
+之后,我们还会讨论跟多的关于 `compose()` 和 `pipe()` 的细节。现在,你所要知道的只是,`reduce()` 是一个极为强大的工具,因此一定要掌握它。 如果在学习过程中遇到了挫折,也大可不必灰心,很多开发者都花了大量时间在 reduce 的学习上。
-### A Word on Redux ###
+### Redux 中的 reduce ###
-You may have heard the term “reducer” used to describe the important state update bits of Redux. As of this writing, Redux is the most popular state management library/architecture for web applications built using React and Angular (the latter via `ngrx/store`).
+你可能听说过 “reducer” 这个术语被用在了 [Redux](https://github.com/reactjs/redux) 的状态更新中。这篇文章撰写之时,对于使用了 React 或者 Angular 进行构建的 web 应用来说,Redux 是最流行的状态管理库/架构(Angualar 中的类 Redux 管理是 ngrx/store )。
-Redux uses reducer functions to manage application state. A Redux-style reducer takes the current state and an action object and returns the new state:
+Redux 使用了 reducer 函数来管理应用状态。一个 Redux 风格的 reducer 接收一个当前应用状态 `state` 和 和交互对象 `action` 作为参数(译注:当前状态就相当于累积值,而 action 就相当于目前处理的元素),处理完成后,返回一个新的应用状态:
```
reducer(state: Any, action: { type: String, payload: Any}) => newState: Any
```
-Redux has some reducer rules you need to keep in mind:
+Redux 的一些 reducer 规则需要你牢记在心:
-1. A reducer called with no parameters should return its valid initial state.
-2. If the reducer isn’t going to handle the action type, it still needs to return the state.
-3. Redux reducers **must be pure functions.**
+1. 一个 reducer 如果进行了无参调用,它要返回它的初始状态。
+2. 如果 reducer 操纵的 action 没有声明类型,他要返回当前状态。
+3. 最最重要的是,Redux reducer 必须是纯函数。
-Let’s rewrite our summing reducer as a Redux-style reducer that reduces over action objects:
+现在,我们以 Redux 风格重写上面的求和 reducer,该 reducer 的行为将由 action 类型决定:
```
const ADD_VALUE = 'ADD_VALUE';
@@ -175,7 +175,9 @@ const summingReducer = (state = 0, action = {}) => {
};
```
-The cool thing about Redux is that the reducers are just standard reducers that you can plug into any `reduce()` implementation which respects the reducer function signature, including `[].reduce()`. That means you can create an array of action objects and reduce over them to get a snapshot of state representing the same state you'd have if those same actions were dispatched to your store:
+关于 Redux 的一个非常美妙的事儿就是,其 reducer 都是标准的 reducer (译注:即接收 `accumulator` 和 `current` 两个参数的 reducer ),这意味着你将 Redux 中的 reducer 插入到任何现有的 `reduce()` 实现中去,比如最常用的 `[].reduce()`。以此为例,我们可以创建一个 action 对象的数组,并对其进行 reduce 操作,传入 `reduce()` 的将是我们定义好的 `summingReducer`,据此,我们获得一个状态快照。之后,一旦对 Redux 中的状态树(store)分派了同样的 action 序列,那么一定能俘获到相同的状态快照:
+
+The cool thing about Redux is that the reducers are just standard reducers that you can plug into any `reduce()` implementation which respects the reducer function signature, including `[].reduce()`.
```
const actions = [
@@ -187,26 +189,26 @@ const actions = [
actions.reduce(summingReducer, 0); // 3
```
-That makes unit testing Redux-style reducers a breeze.
+这使得对 Redux 风格的 reducer 的单元测试变得极为容易。
+
+### 总结 ###
-### Conclusion ###
+现在,你应该可以瞥见 recude 的强大甚至是无所不能了。它不只能帮助我们定义出 map 或者 filter,还是函数式编程中重要的工具,这个工具强大在它是一个基础工具,能够通过它构建出更多更强大的工具。
-You should be starting to see that reduce is an incredibly useful and versatile abstraction. It’s definitely a little trickier to understand than map or filter, but it is an essential tool in your functional programming utility belt — one you can use to make a lot of other great tools.
+[**下一篇: Functors(函子) & Categories(范畴) >**](https://medium.com/javascript-scene/functors-categories-61e031bac53f#.4hqndcx22)
-[**Next: Functors & Categories >**](https://medium.com/javascript-scene/functors-categories-61e031bac53f#.4hqndcx22)
-### Next Steps ###
+### 接下来 ###
-Want to learn more about functional programming in JavaScript?
+想学习更多 JavaScript 函数式编程吗?
-[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/). If you’re not a member, you’re missing out!
+[跟着 Eric Elliott 学 Javacript](http://ericelliottjs.com/product/lifetime-access-pass/),机不可失时不再来!
-[
-](https://ericelliottjs.com/product/lifetime-access-pass/)
+[](https://ericelliottjs.com/product/lifetime-access-pass/)
-***Eric Elliott*** is the author of [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly), and [*“Learn JavaScript with Eric Elliott”*](http://ericelliottjs.com/product/lifetime-access-pass/) . He has contributed to software experiences for **Adobe Systems, Zumba Fitness, The Wall Street Journal, ESPN, BBC, and top recording artists including Usher, Frank Ocean, Metallica**, and many more.
+**Eric Elliott** 是 [**“编写 JavaScript 应用”**](http://pjabook.com) (O’Reilly) 以及 [**“跟着 Eric Elliott 学 Javascript”**](http://ericelliottjs.com/product/lifetime-access-pass/) 两书的作者。他为许多公司和组织作过贡献,例如 **Adobe Systems**、**Zumba Fitness**、**The Wall Street Journal**、**ESPN** 和 **BBC** 等 , 也是很多机构的顶级艺术家,包括但不限于 **Usher**、**Frank Ocean** 以及 **Metallica**。
-*He spends most of his time in the San Francisco Bay Area with the most beautiful woman in the world.*
+大多数时间,他都在 San Francisco Bay Area,同这世上最美丽的女子在一起。
---
From 76137a1e95bcb09c9a9d061ce0a416e99a756ea9 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E5=90=B4=E6=99=93=E5=86=9B?=
Date: Thu, 13 Apr 2017 14:23:43 +0800
Subject: [PATCH 221/945] reset
---
TODO/reduce-composing-software.md | 96 +++++++++++++++----------------
1 file changed, 47 insertions(+), 49 deletions(-)
diff --git a/TODO/reduce-composing-software.md b/TODO/reduce-composing-software.md
index 312cc27ea78..85591dbfa4e 100644
--- a/TODO/reduce-composing-software.md
+++ b/TODO/reduce-composing-software.md
@@ -4,18 +4,18 @@
> * 译者:
> * 校对者:
-# Reduce (软件编写) (第五部分)#
+# Reduce (Composing Software) #
-Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0) (译注:该图是用 PS 将烟雾处理成方块状后得到的效果,参见 [flickr](https://www.flickr.com/photos/68397968@N07/11432696204)。))
+Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0)
-> 注意:这是 “软件编写” 系列文章的第四部分,该系列主要阐述如何在 JavaScript ES6+ 中从零开始学习函数式编程和组合化软件(compositional software)技术(译注:关于软件可组合性的概念,参见维基百科 [Composability](https://en.wikipedia.org/wiki/Composability))。后续还有更多精彩内容,敬请期待!
-> [<上一篇](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d#.w4y0mlpcs) | [<< 返回第一章](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe)
+> Note: This is part of the “Composing Software” series on learning functional programming and compositional software techniques in JavaScript ES6+ from the ground up. Stay tuned. There’s a lot more of this to come!
+> [< Previous](https://medium.com/javascript-scene/higher-order-functions-composing-software-5365cf2cbe99#.su6cmn4f7) | [<< Start over at Part 1](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [Next >](https://medium.com/javascript-scene/functors-categories-61e031bac53f#.4hqndcx22)
-在函数式编程中,**reduce**(也称为:fold,ccumulate)允许你在一个序列上迭代,并应用一个函数来处理预先声明的累积值和当前迭代到的元素。当迭代完成时,将返回这个累积值。许多其他有用的事物都可以通过 reduce 实现。多数时候,reduce 可以说是处理集合中元素最优雅的方式。
+**Reduce** (aka: fold, accumulate) utility commonly used in functional programming that lets you iterate over a list, applying a function to an accumulated value and the next item in the list, until the iteration is complete and the accumulated value gets returned. Many useful things can be implemented with reduce. Frequently, it’s the most elegant way to do any non-trivial processing on a collection of items.
-reduce 接受一个 reducer 函数以及一个初始值,最终返回一个累积值。对于 `Array.prototype.reduce()` 来说, 初始列表将由 `this` 指明, 所以列表本身不会作为该函数的参数:
+Reduce takes a reducer function and an initial value, and returns the accumulated value. For `Array.prototype.reduce()`, the initial list is provided by `this`, so it's not one of the arguments:
```
array.reduce(
@@ -24,28 +24,28 @@ array.reduce(
) => accumulator: Any
```
-我们利用如下方式对一个数组进行求和:
+Let’s sum an array:
```
[2, 4, 6].reduce((acc, n) => acc + n, 0); // 12
```
-对于数组的每步迭代,reducer 函数都会被调用,并且向其传入了累积值和当前迭代到的数组元素。reducer 的职责在于以某种方式将当前迭代的元素 “合拢(fold)” 到累加值中。reducer 规定了 “合拢” 的手段和方式,完成了对当前元素的 “合拢” 后,reducer 将返回新的累加值,然后, `.reduce()` 将开始处理数组中的下一个元素。reducer 需要一个初始值才能开始工作,所以绝大多数的 `.reduce()` 实现都需要接收一个初始值作为参数。
+For each element in the array, the reducer is called and passed the accumulator and the current value. The reducer’s job is to “fold” the current value into the accumulated value somehow. How is not specified, and specifying how is the purpose of the reducer function. The reducer returns the new accumulated value, and `reduce()` moves on to the next value in the array. The reducer may need an initial value to start with, so most implementations take an initial value as a parameter.
-在求数组元素和一例中,reducer 函数第一次调用时,`acc` 将会以 `0` 值(该值是传入 `.reduce()` 方法的第二个参数)开始。然后,reducer 返回了 `0` + `2`(`2` 是数组的第一个元素), 也就是返回了 `2` 作为新的累积值。下一步,`acc = 2, n = 4` 传入了 reducer,reducer返回了 `2 + 4`(`6`)。在最后一步迭代中,`acc = 6, n = 6`, reducer 返回了 `12`。迭代完成,`.reduce()` 返回了最终的累积值 `12`。
+In the case of this summing reducer, the first time the reducer is called, `acc` starts at `0` (the value we passed to `.reduce()` as the second parameter). The reducer returns `0` + `2` (`2` was the first element in the array), which is `2`. For the next call, `acc = 2, n = 4` and the reducer returns the result of `2 + 4` (`6`). In the last iteration, `acc = 6, n = 6`, and the reducer returns `12`. Since the iteration is finished, `.reduce()` returns the final accumulated value, `12`.
-在这一例子中,我们传入了一个匿名函数作为 reducer,但是我们也可以抽象出每次求和的过程为一个具名函数,这使得我们代码的复用成都更高:
+In this case, we passed in an anonymous reducing function, but we can abstract it and give it a name:
```
const summingReducer = (acc, n) => acc + n;
[2, 4, 6].reduce(summingReducer, 0); // 12
```
-通常,`reduce` 的工作过程为由左向右。在 JavaScript 中,我们也有一个 `[].reduceRight()` (译注:[MDN -- Array.prototype.reduceRight()](https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Array/ReduceRight))方法来让 reduce 由右向左地工作。 具体说来,如果你对数组 `[2, 4, 6]` 应用 `.reduceRight()` ,第一个被迭代到的元素就将是 `6`,最后一个迭代到的元素就是 `2`。
+Normally, `reduce()` works left to right. In JavaScript, we also have `[].reduceRight()`, which works right to left. In other words, if you applied `.reduceRight()` to `[2, 4, 6]`, the first iteration would use `6` as the first value for `n`, and it would work its way back and finish with `2`.
-### 无所不能的 reduce ###
+### Reduce is Versatile ###
-别吃惊,reduce 确实无所不能,你所熟悉的 `map()`,`filter()`,`forEach()` 以及其他函数都可借助于 reduce 来创建。
+Reduce is versatile. It’s easy to define `map()`, `filter()`, `forEach()` and lots of other interesting things using reduce:
**Map:**
@@ -55,7 +55,7 @@ const map = (fn, arr) => arr.reduce((acc, item, index, arr) => {
}, []);
```
-对于 map 来说,我们的累积值就是一个新的数组对象,该数组对象中的每个元素都由原数组对应元素映射得到。累积数组中新的元素由传入 map 的映射函数(`fn`)所确定:对于当前迭代到的元素 `item`,我们通过 `fn` 计算出新的元素,并将其拼接入累加数组 `arr` 中。
+For map, our accumulated value is a new array with a new element for each value in the original array. The new values are generated by applying the passed in mapping function (`fn`) to each element in the `arr` argument. We accumulate the new array by calling `fn` with the current element, and concatenating the result to the accumulator array, `acc`.
**Filter:**
@@ -65,37 +65,37 @@ const filter = (fn, arr) => arr.reduce((newArr, item) => {
}, []);
```
-filter 的工作方式与 map 类似,只不过原数组的元素只有通过一个真值检测函数(predicate function)才能被送入新的累积数组中。亦即,相较于 map,filter 是**有条件**地选择元素到累积数组中,并且不会改变元素的值。
+Filter works in much the same way as map, except that we take a predicate function and *conditionally* append the current value to the new array if the element passes the predicate check (`fn(item)` returns `true`).
-上面几个例子,你处理的数据都是一些数值序列,你在数值序列上应用指定的函数迭代数据,并将结果合拢到累积值中。大多数应用都因此开始雏形初备,但是你想过这个问题没有:**假如你的序列是函数序列呢?**
+For each of the above examples, you have a list of data, iterate over that data applying some function and folding the results into an accumulated value. Lots of applications spring to mind. But what if *your data is a list of functions?*
**Compose:**
-reduce 也是实现函数组合的便捷渠道。假如你想用将函数 `g` 的输出作为函数 `f` 的输入,即组合这两个函数: `f . g`,那么你可以使用下面的 JavaScript 代码片,它没有任何的抽象:
+Reduce is also a convenient way to compose functions. Remember function composition: If you want to apply the function `f` to the result of `g` of `x` i.e., the composition, `f . g`, you could use the following JavaScript:
```
f(g(x))
```
-reduce 让我们能抽象出函数组合过程,从而让你也能轻易地实现更多层次的函数组合:
+Reduce lets us abstract that process to work on any number of functions, so you could easily define a function that would represent:
```
f(g(h(x)))
```
-由于函数组合是由右向左的,我们就要使用上面提到的 `.reduceRight()` 方法来抽象函数组合过程:
+To make that happen, we’ll need to run reduce in reverse. That is, right-to-left, rather than left-to-right. Thankfully, JavaScript provides a `.reduceRight()` method:
```
const compose = (...fns) => x => fns.reduceRight((v, f) => f(v), x);
```
-> 注意:如果 JavaScript 的版本没有提供 `[].reduceRight()`,你可以借助于 `reduce` 实现该方法。该实现留给读者自己思考。
+> Note: If JavaScript had not provided `[].reduceRight()`, you could still implement `reduceRight()` -- using `reduce()`. I'll leave it to adventurous readers to figure out how.
**Pipe:**
-`compose()` 很好地描述了由内至外的组合过程,某种程度上,这是数学上的关于输入输出的组合。如果你想从事件发生顺序上来思考函数组合呢?
+`compose()` is great if you want to represent the composition from the inside-out -- that is, in the math notation sense. But what if you want to think of it as a sequence of events?
-假设我们想要对一个数值加 `1`,然后对新得到的数值进行翻倍。如果是利用 `compose()`,就需要这么做:
+Imagine we want to add `1` to a number and then double it. With `compose(), that would be:
```
const add1 = n => n + 1;
@@ -110,15 +110,15 @@ add1ThenDouble(2); // 6
// ((2 + 1 = 3) * 2 = 6)
```
-发现问题没有?第一步(加1操作)是 compose 序列上的最后一个元素,所以,`compose` 需要你自底向上地分析流程的执行。
+See the problem? The first step is listed last, so in order to understand the sequence, you’ll need to start at the bottom of the list and work your way backwards to the top.
-如果我们以由左向右的方式进行 reduce 来创建函数组合,以示区别,我们用 `pipe` 来描述新的组合方式:
+Or we can reduce left-to-right as you normally would, instead of right-to-left:
```
const pipe = (...fns) => x => fns.reduce((v, f) => f(v), x);
```
-现在,新的流程就可以这么撰写:
+Now you can write `add1ThenDouble()` like this:
```
const add1ThenDouble = pipe(
@@ -130,7 +130,7 @@ add1ThenDouble(2); // 6
// ((2 + 1 = 3) * 2 = 6)
```
-你也看到了,在组合中,顺序是非常重要的,如果你调换了 `double` 和 `add1` 的顺序,你将得到截然不同的结果:
+This is important because sometimes if you compose backwards, you get a different result:
```
const doubleThenAdd1 = pipe(
@@ -141,25 +141,25 @@ const doubleThenAdd1 = pipe(
doubleThenAdd1(2); // 5
```
-之后,我们还会讨论跟多的关于 `compose()` 和 `pipe()` 的细节。现在,你所要知道的只是,`reduce()` 是一个极为强大的工具,因此一定要掌握它。 如果在学习过程中遇到了挫折,也大可不必灰心,很多开发者都花了大量时间在 reduce 的学习上。
+We’ll go into more details on `compose()` and `pipe()` later. What you should understand right now is that `reduce()` is a very powerful tool, and you really need to learn it. Just be aware that if you get very tricky with reduce, some people may have a hard time following along.
-### Redux 中的 reduce ###
+### A Word on Redux ###
-你可能听说过 “reducer” 这个术语被用在了 [Redux](https://github.com/reactjs/redux) 的状态更新中。这篇文章撰写之时,对于使用了 React 或者 Angular 进行构建的 web 应用来说,Redux 是最流行的状态管理库/架构(Angualar 中的类 Redux 管理是 ngrx/store )。
+You may have heard the term “reducer” used to describe the important state update bits of Redux. As of this writing, Redux is the most popular state management library/architecture for web applications built using React and Angular (the latter via `ngrx/store`).
-Redux 使用了 reducer 函数来管理应用状态。一个 Redux 风格的 reducer 接收一个当前应用状态 `state` 和 和交互对象 `action` 作为参数(译注:当前状态就相当于累积值,而 action 就相当于目前处理的元素),处理完成后,返回一个新的应用状态:
+Redux uses reducer functions to manage application state. A Redux-style reducer takes the current state and an action object and returns the new state:
```
reducer(state: Any, action: { type: String, payload: Any}) => newState: Any
```
-Redux 的一些 reducer 规则需要你牢记在心:
+Redux has some reducer rules you need to keep in mind:
-1. 一个 reducer 如果进行了无参调用,它要返回它的初始状态。
-2. 如果 reducer 操纵的 action 没有声明类型,他要返回当前状态。
-3. 最最重要的是,Redux reducer 必须是纯函数。
+1. A reducer called with no parameters should return its valid initial state.
+2. If the reducer isn’t going to handle the action type, it still needs to return the state.
+3. Redux reducers **must be pure functions.**
-现在,我们以 Redux 风格重写上面的求和 reducer,该 reducer 的行为将由 action 类型决定:
+Let’s rewrite our summing reducer as a Redux-style reducer that reduces over action objects:
```
const ADD_VALUE = 'ADD_VALUE';
@@ -175,9 +175,7 @@ const summingReducer = (state = 0, action = {}) => {
};
```
-关于 Redux 的一个非常美妙的事儿就是,其 reducer 都是标准的 reducer (译注:即接收 `accumulator` 和 `current` 两个参数的 reducer ),这意味着你将 Redux 中的 reducer 插入到任何现有的 `reduce()` 实现中去,比如最常用的 `[].reduce()`。以此为例,我们可以创建一个 action 对象的数组,并对其进行 reduce 操作,传入 `reduce()` 的将是我们定义好的 `summingReducer`,据此,我们获得一个状态快照。之后,一旦对 Redux 中的状态树(store)分派了同样的 action 序列,那么一定能俘获到相同的状态快照:
-
-The cool thing about Redux is that the reducers are just standard reducers that you can plug into any `reduce()` implementation which respects the reducer function signature, including `[].reduce()`.
+The cool thing about Redux is that the reducers are just standard reducers that you can plug into any `reduce()` implementation which respects the reducer function signature, including `[].reduce()`. That means you can create an array of action objects and reduce over them to get a snapshot of state representing the same state you'd have if those same actions were dispatched to your store:
```
const actions = [
@@ -189,26 +187,26 @@ const actions = [
actions.reduce(summingReducer, 0); // 3
```
-这使得对 Redux 风格的 reducer 的单元测试变得极为容易。
-
-### 总结 ###
+That makes unit testing Redux-style reducers a breeze.
-现在,你应该可以瞥见 recude 的强大甚至是无所不能了。它不只能帮助我们定义出 map 或者 filter,还是函数式编程中重要的工具,这个工具强大在它是一个基础工具,能够通过它构建出更多更强大的工具。
+### Conclusion ###
-[**下一篇: Functors(函子) & Categories(范畴) >**](https://medium.com/javascript-scene/functors-categories-61e031bac53f#.4hqndcx22)
+You should be starting to see that reduce is an incredibly useful and versatile abstraction. It’s definitely a little trickier to understand than map or filter, but it is an essential tool in your functional programming utility belt — one you can use to make a lot of other great tools.
+[**Next: Functors & Categories >**](https://medium.com/javascript-scene/functors-categories-61e031bac53f#.4hqndcx22)
-### 接下来 ###
+### Next Steps ###
-想学习更多 JavaScript 函数式编程吗?
+Want to learn more about functional programming in JavaScript?
-[跟着 Eric Elliott 学 Javacript](http://ericelliottjs.com/product/lifetime-access-pass/),机不可失时不再来!
+[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/). If you’re not a member, you’re missing out!
-[](https://ericelliottjs.com/product/lifetime-access-pass/)
+[
+](https://ericelliottjs.com/product/lifetime-access-pass/)
-**Eric Elliott** 是 [**“编写 JavaScript 应用”**](http://pjabook.com) (O’Reilly) 以及 [**“跟着 Eric Elliott 学 Javascript”**](http://ericelliottjs.com/product/lifetime-access-pass/) 两书的作者。他为许多公司和组织作过贡献,例如 **Adobe Systems**、**Zumba Fitness**、**The Wall Street Journal**、**ESPN** 和 **BBC** 等 , 也是很多机构的顶级艺术家,包括但不限于 **Usher**、**Frank Ocean** 以及 **Metallica**。
+***Eric Elliott*** is the author of [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly), and [*“Learn JavaScript with Eric Elliott”*](http://ericelliottjs.com/product/lifetime-access-pass/) . He has contributed to software experiences for **Adobe Systems, Zumba Fitness, The Wall Street Journal, ESPN, BBC, and top recording artists including Usher, Frank Ocean, Metallica**, and many more.
-大多数时间,他都在 San Francisco Bay Area,同这世上最美丽的女子在一起。
+*He spends most of his time in the San Francisco Bay Area with the most beautiful woman in the world.*
---
From ce75287cbd1757f21d6ef2a84961df549a35d6b6 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E5=90=B4=E6=99=93=E5=86=9B?=
Date: Thu, 13 Apr 2017 14:32:12 +0800
Subject: [PATCH 222/945] reduce
---
TODO/reduce-composing-software.md | 98 ++++++++++++++++---------------
1 file changed, 50 insertions(+), 48 deletions(-)
diff --git a/TODO/reduce-composing-software.md b/TODO/reduce-composing-software.md
index 8baf2c381ad..2c42b1116aa 100644
--- a/TODO/reduce-composing-software.md
+++ b/TODO/reduce-composing-software.md
@@ -1,21 +1,21 @@
> * 原文地址:[Reduce (Composing Software)(part 5)](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d)
> * 原文作者:[Eric Elliott](https://medium.com/@_ericelliott?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
+> * 译者:[yoyoyohamapi](https://github.com/yoyoyohamapi) [reid3290](https://github.com/reid3290)
> * 校对者:
-# Reduce (Composing Software) #
+# Reduce (软件编写) (第五部分) #
-Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0)
+Smoke Art Cubes to Smoke — MattysFlicks — (CC BY 2.0) (译注:该图是用 PS 将烟雾处理成方块状后得到的效果,参见 [flickr](https://www.flickr.com/photos/68397968@N07/11432696204)。))
-> Note: This is part of the “Composing Software” series on learning functional programming and compositional software techniques in JavaScript ES6+ from the ground up. Stay tuned. There’s a lot more of this to come!
-> [< Previous](https://medium.com/javascript-scene/higher-order-functions-composing-software-5365cf2cbe99#.su6cmn4f7) | [<< Start over at Part 1](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [Next >](https://medium.com/javascript-scene/functors-categories-61e031bac53f#.4hqndcx22)
+> 注意:这是 “软件编写” 系列文章的第四部分,该系列主要阐述如何在 JavaScript ES6+ 中从零开始学习函数式编程和组合化软件(compositional software)技术(译注:关于软件可组合性的概念,参见维基百科 [Composability](https://en.wikipedia.org/wiki/Composability))。后续还有更多精彩内容,敬请期待!
+> [<上一篇](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d#.w4y0mlpcs) | [<< 返回第一章](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe)
-**Reduce** (aka: fold, accumulate) utility commonly used in functional programming that lets you iterate over a list, applying a function to an accumulated value and the next item in the list, until the iteration is complete and the accumulated value gets returned. Many useful things can be implemented with reduce. Frequently, it’s the most elegant way to do any non-trivial processing on a collection of items.
+在函数式编程中,**reduce**(也称为:fold,ccumulate)允许你在一个序列上迭代,并应用一个函数来处理预先声明的累积值和当前迭代到的元素。当迭代完成时,将返回这个累积值。许多其他有用的事物都可以通过 reduce 实现。多数时候,reduce 可以说是处理集合中元素最优雅的方式。
-Reduce takes a reducer function and an initial value, and returns the accumulated value. For `Array.prototype.reduce()`, the initial list is provided by `this`, so it's not one of the arguments:
+reduce 接受一个 reducer 函数以及一个初始值,最终返回一个累积值。对于 `Array.prototype.reduce()` 来说, 初始列表将由 `this` 指明, 所以列表本身不会作为该函数的参数:
```
array.reduce(
@@ -24,28 +24,28 @@ array.reduce(
) => accumulator: Any
```
-Let’s sum an array:
+我们利用如下方式对一个数组进行求和:
```
[2, 4, 6].reduce((acc, n) => acc + n, 0); // 12
```
-For each element in the array, the reducer is called and passed the accumulator and the current value. The reducer’s job is to “fold” the current value into the accumulated value somehow. How is not specified, and specifying how is the purpose of the reducer function. The reducer returns the new accumulated value, and `reduce()` moves on to the next value in the array. The reducer may need an initial value to start with, so most implementations take an initial value as a parameter.
+对于数组的每步迭代,reducer 函数都会被调用,并且向其传入了累积值和当前迭代到的数组元素。reducer 的职责在于以某种方式将当前迭代的元素 “合拢(fold)” 到累加值中。reducer 规定了 “合拢” 的手段和方式,完成了对当前元素的 “合拢” 后,reducer 将返回新的累加值,然后, `.reduce()` 将开始处理数组中的下一个元素。reducer 需要一个初始值才能开始工作,所以绝大多数的 `.reduce()` 实现都需要接收一个初始值作为参数。
-In the case of this summing reducer, the first time the reducer is called, `acc` starts at `0` (the value we passed to `.reduce()` as the second parameter). The reducer returns `0` + `2` (`2` was the first element in the array), which is `2`. For the next call, `acc = 2, n = 4` and the reducer returns the result of `2 + 4` (`6`). In the last iteration, `acc = 6, n = 6`, and the reducer returns `12`. Since the iteration is finished, `.reduce()` returns the final accumulated value, `12`.
+在求数组元素和一例中,reducer 函数第一次调用时,`acc` 将会以 `0` 值(该值是传入 `.reduce()` 方法的第二个参数)开始。然后,reducer 返回了 `0` + `2`(`2` 是数组的第一个元素), 也就是返回了 `2` 作为新的累积值。下一步,`acc = 2, n = 4` 传入了 reducer,reducer返回了 `2 + 4`(`6`)。在最后一步迭代中,`acc = 6, n = 6`, reducer 返回了 `12`。迭代完成,`.reduce()` 返回了最终的累积值 `12`。
-In this case, we passed in an anonymous reducing function, but we can abstract it and give it a name:
+在这一例子中,我们传入了一个匿名函数作为 reducer,但是我们也可以抽象出每次求和的过程为一个具名函数,这使得我们代码的复用成都更高:
```
const summingReducer = (acc, n) => acc + n;
[2, 4, 6].reduce(summingReducer, 0); // 12
```
-Normally, `reduce()` works left to right. In JavaScript, we also have `[].reduceRight()`, which works right to left. In other words, if you applied `.reduceRight()` to `[2, 4, 6]`, the first iteration would use `6` as the first value for `n`, and it would work its way back and finish with `2`.
+通常,`reduce` 的工作过程为由左向右。在 JavaScript 中,我们也有一个 `[].reduceRight()` (译注:[MDN -- Array.prototype.reduceRight()](https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Array/ReduceRight))方法来让 reduce 由右向左地工作。 具体说来,如果你对数组 `[2, 4, 6]` 应用 `.reduceRight()` ,第一个被迭代到的元素就将是 `6`,最后一个迭代到的元素就是 `2`。
-### Reduce is Versatile ###
+### 无所不能的 reduce ###
-Reduce is versatile. It’s easy to define `map()`, `filter()`, `forEach()` and lots of other interesting things using reduce:
+别吃惊,reduce 确实无所不能,你所熟悉的 `map()`,`filter()`,`forEach()` 以及其他函数都可借助于 reduce 来创建。
**Map:**
@@ -55,7 +55,7 @@ const map = (fn, arr) => arr.reduce((acc, item, index, arr) => {
}, []);
```
-For map, our accumulated value is a new array with a new element for each value in the original array. The new values are generated by applying the passed in mapping function (`fn`) to each element in the `arr` argument. We accumulate the new array by calling `fn` with the current element, and concatenating the result to the accumulator array, `acc`.
+对于 map 来说,我们的累积值就是一个新的数组对象,该数组对象中的每个元素都由原数组对应元素映射得到。累积数组中新的元素由传入 map 的映射函数(`fn`)所确定:对于当前迭代到的元素 `item`,我们通过 `fn` 计算出新的元素,并将其拼接入累加数组 `arr` 中。
**Filter:**
@@ -65,37 +65,37 @@ const filter = (fn, arr) => arr.reduce((newArr, item) => {
}, []);
```
-Filter works in much the same way as map, except that we take a predicate function and *conditionally* append the current value to the new array if the element passes the predicate check (`fn(item)` returns `true`).
+filter 的工作方式与 map 类似,只不过原数组的元素只有通过一个真值检测函数(predicate function)才能被送入新的累积数组中。亦即,相较于 map,filter 是**有条件**地选择元素到累积数组中,并且不会改变元素的值。
-For each of the above examples, you have a list of data, iterate over that data applying some function and folding the results into an accumulated value. Lots of applications spring to mind. But what if *your data is a list of functions?*
+上面几个例子,你处理的数据都是一些数值序列,你在数值序列上应用指定的函数迭代数据,并将结果合拢到累积值中。大多数应用都因此开始雏形初备,但是你想过这个问题没有:**假如你的序列是函数序列呢?**
**Compose:**
-Reduce is also a convenient way to compose functions. Remember function composition: If you want to apply the function `f` to the result of `g` of `x` i.e., the composition, `f . g`, you could use the following JavaScript:
+reduce 也是实现函数组合的便捷渠道。假如你想用将函数 `g` 的输出作为函数 `f` 的输入,即组合这两个函数: `f . g`,那么你可以使用下面的 JavaScript 代码片,它没有任何的抽象:
```
f(g(x))
```
-Reduce lets us abstract that process to work on any number of functions, so you could easily define a function that would represent:
+reduce 让我们能抽象出函数组合过程,从而让你也能轻易地实现更多层次的函数组合:
```
f(g(h(x)))
```
-To make that happen, we’ll need to run reduce in reverse. That is, right-to-left, rather than left-to-right. Thankfully, JavaScript provides a `.reduceRight()` method:
+由于函数组合是由右向左的,我们就要使用上面提到的 `.reduceRight()` 方法来抽象函数组合过程:
```
const compose = (...fns) => x => fns.reduceRight((v, f) => f(v), x);
```
-> Note: If JavaScript had not provided `[].reduceRight()`, you could still implement `reduceRight()` -- using `reduce()`. I'll leave it to adventurous readers to figure out how.
+> 注意:如果 JavaScript 的版本没有提供 `[].reduceRight()`,你可以借助于 `reduce` 实现该方法。该实现留给读者自己思考。
**Pipe:**
-`compose()` is great if you want to represent the composition from the inside-out -- that is, in the math notation sense. But what if you want to think of it as a sequence of events?
+`compose()` 很好地描述了由内至外的组合过程,某种程度上,这是数学上的关于输入输出的组合。如果你想从事件发生顺序上来思考函数组合呢?
-Imagine we want to add `1` to a number and then double it. With `compose(), that would be:
+假设我们想要对一个数值加 `1`,然后对新得到的数值进行翻倍。如果是利用 `compose()`,就需要这么做:
```
const add1 = n => n + 1;
@@ -110,15 +110,15 @@ add1ThenDouble(2); // 6
// ((2 + 1 = 3) * 2 = 6)
```
-See the problem? The first step is listed last, so in order to understand the sequence, you’ll need to start at the bottom of the list and work your way backwards to the top.
+发现问题没有?第一步(加1操作)是 compose 序列上的最后一个元素,所以,`compose` 需要你自底向上地分析流程的执行。
-Or we can reduce left-to-right as you normally would, instead of right-to-left:
+如果我们以由左向右的方式进行 reduce 来创建函数组合,以示区别,我们用 `pipe` 来描述新的组合方式:
```
const pipe = (...fns) => x => fns.reduce((v, f) => f(v), x);
```
-Now you can write `add1ThenDouble()` like this:
+现在,新的流程就可以这么撰写:
```
const add1ThenDouble = pipe(
@@ -130,7 +130,7 @@ add1ThenDouble(2); // 6
// ((2 + 1 = 3) * 2 = 6)
```
-This is important because sometimes if you compose backwards, you get a different result:
+你也看到了,在组合中,顺序是非常重要的,如果你调换了 `double` 和 `add1` 的顺序,你将得到截然不同的结果:
```
const doubleThenAdd1 = pipe(
@@ -141,25 +141,25 @@ const doubleThenAdd1 = pipe(
doubleThenAdd1(2); // 5
```
-We’ll go into more details on `compose()` and `pipe()` later. What you should understand right now is that `reduce()` is a very powerful tool, and you really need to learn it. Just be aware that if you get very tricky with reduce, some people may have a hard time following along.
+之后,我们还会讨论跟多的关于 `compose()` 和 `pipe()` 的细节。现在,你所要知道的只是,`reduce()` 是一个极为强大的工具,因此一定要掌握它。 如果在学习过程中遇到了挫折,也大可不必灰心,很多开发者都花了大量时间在 reduce 的学习上。
-### A Word on Redux ###
+### Redux 中的 reduce ###
-You may have heard the term “reducer” used to describe the important state update bits of Redux. As of this writing, Redux is the most popular state management library/architecture for web applications built using React and Angular (the latter via `ngrx/store`).
+你可能听说过 “reducer” 这个术语被用在了 [Redux](https://github.com/reactjs/redux) 的状态更新中。这篇文章撰写之时,对于使用了 React 或者 Angular 进行构建的 web 应用来说,Redux 是最流行的状态管理库/架构(Angualar 中的类 Redux 管理是 ngrx/store )。
-Redux uses reducer functions to manage application state. A Redux-style reducer takes the current state and an action object and returns the new state:
+Redux 使用了 reducer 函数来管理应用状态。一个 Redux 风格的 reducer 接收一个当前应用状态 `state` 和 和交互对象 `action` 作为参数(译注:当前状态就相当于累积值,而 action 就相当于目前处理的元素),处理完成后,返回一个新的应用状态:
```
reducer(state: Any, action: { type: String, payload: Any}) => newState: Any
```
-Redux has some reducer rules you need to keep in mind:
+Redux 的一些 reducer 规则需要你牢记在心:
-1. A reducer called with no parameters should return its valid initial state.
-2. If the reducer isn’t going to handle the action type, it still needs to return the state.
-3. Redux reducers **must be pure functions.**
+1. 一个 reducer 如果进行了无参调用,它要返回它的初始状态。
+2. 如果 reducer 操纵的 action 没有声明类型,他要返回当前状态。
+3. 最最重要的是,Redux reducer 必须是纯函数。
-Let’s rewrite our summing reducer as a Redux-style reducer that reduces over action objects:
+现在,我们以 Redux 风格重写上面的求和 reducer,该 reducer 的行为将由 action 类型决定:
```
const ADD_VALUE = 'ADD_VALUE';
@@ -175,7 +175,9 @@ const summingReducer = (state = 0, action = {}) => {
};
```
-The cool thing about Redux is that the reducers are just standard reducers that you can plug into any `reduce()` implementation which respects the reducer function signature, including `[].reduce()`. That means you can create an array of action objects and reduce over them to get a snapshot of state representing the same state you'd have if those same actions were dispatched to your store:
+关于 Redux 的一个非常美妙的事儿就是,其 reducer 都是标准的 reducer (译注:即接收 `accumulator` 和 `current` 两个参数的 reducer ),这意味着你将 Redux 中的 reducer 插入到任何现有的 `reduce()` 实现中去,比如最常用的 `[].reduce()`。以此为例,我们可以创建一个 action 对象的数组,并对其进行 reduce 操作,传入 `reduce()` 的将是我们定义好的 `summingReducer`,据此,我们获得一个状态快照。之后,一旦对 Redux 中的状态树(store)分派了同样的 action 序列,那么一定能俘获到相同的状态快照:
+
+The cool thing about Redux is that the reducers are just standard reducers that you can plug into any `reduce()` implementation which respects the reducer function signature, including `[].reduce()`.
```
const actions = [
@@ -187,26 +189,26 @@ const actions = [
actions.reduce(summingReducer, 0); // 3
```
-That makes unit testing Redux-style reducers a breeze.
+这使得对 Redux 风格的 reducer 的单元测试变得极为容易。
+
+### 总结 ###
-### Conclusion ###
+现在,你应该可以瞥见 recude 的强大甚至是无所不能了。它不只能帮助我们定义出 map 或者 filter,还是函数式编程中重要的工具,这个工具强大在它是一个基础工具,能够通过它构建出更多更强大的工具。
-You should be starting to see that reduce is an incredibly useful and versatile abstraction. It’s definitely a little trickier to understand than map or filter, but it is an essential tool in your functional programming utility belt — one you can use to make a lot of other great tools.
+[**下一篇: Functors(函子) & Categories(范畴) >**](https://medium.com/javascript-scene/functors-categories-61e031bac53f#.4hqndcx22)
-[**Next: Functors & Categories >**](https://medium.com/javascript-scene/functors-categories-61e031bac53f#.4hqndcx22)
-### Next Steps ###
+### 接下来 ###
-Want to learn more about functional programming in JavaScript?
+想学习更多 JavaScript 函数式编程吗?
-[Learn JavaScript with Eric Elliott](http://ericelliottjs.com/product/lifetime-access-pass/). If you’re not a member, you’re missing out!
+[跟着 Eric Elliott 学 Javacript](http://ericelliottjs.com/product/lifetime-access-pass/),机不可失时不再来!
-[
-](https://ericelliottjs.com/product/lifetime-access-pass/)
+[](https://ericelliottjs.com/product/lifetime-access-pass/)
-***Eric Elliott*** is the author of [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly), and [*“Learn JavaScript with Eric Elliott”*](http://ericelliottjs.com/product/lifetime-access-pass/) . He has contributed to software experiences for **Adobe Systems, Zumba Fitness, The Wall Street Journal, ESPN, BBC, and top recording artists including Usher, Frank Ocean, Metallica**, and many more.
+**Eric Elliott** 是 [**“编写 JavaScript 应用”**](http://pjabook.com) (O’Reilly) 以及 [**“跟着 Eric Elliott 学 Javascript”**](http://ericelliottjs.com/product/lifetime-access-pass/) 两书的作者。他为许多公司和组织作过贡献,例如 **Adobe Systems**、**Zumba Fitness**、**The Wall Street Journal**、**ESPN** 和 **BBC** 等 , 也是很多机构的顶级艺术家,包括但不限于 **Usher**、**Frank Ocean** 以及 **Metallica**。
-*He spends most of his time in the San Francisco Bay Area with the most beautiful woman in the world.*
+大多数时间,他都在 San Francisco Bay Area,同这世上最美丽的女子在一起。
---
From c273b62d7998d0e0fd49332aa5d841010ded0763 Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Thu, 13 Apr 2017 22:52:38 +0800
Subject: [PATCH 223/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20=E5=86=99=E7=BB=99?=
=?UTF-8?q?=E2=80=9C=E8=80=81=E6=B4=BE=E2=80=9D=20Web=20=E5=BC=80=E5=8F=91?=
=?UTF-8?q?=E8=80=85=E7=9A=84=E2=80=9C=E7=8E=B0=E4=BB=A3=E2=80=9D=20JavaSc?=
=?UTF-8?q?ript=20=E6=8C=87=E5=8D=97=20=E7=9A=84=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 5 ++---
front-end.md | 1 +
2 files changed, 3 insertions(+), 3 deletions(-)
diff --git a/README.md b/README.md
index 54648b75c5d..255e14d40e7 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [453](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [454](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -37,11 +37,10 @@
## 前端
+* [写给“老派” Web 开发者的“现代” JavaScript 指南](https://juejin.im/post/58ebab0c8d6d81006191376f?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([SunCafe](http://suncafe.cc/) 翻译)
* [CSS很棒,只是真的太难了](https://juejin.im/entry/58eae24a61ff4b0061a6a102?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([ZhangFe](https://github.com/ZhangFe) 翻译)
* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
-* [如何使用 JavaScript 构建响应式引擎 —— Part 2:计算属性和依赖追踪](https://juejin.im/post/58ddeb1a570c3500579016ef?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([IridescentMia](https://github.com/IridescentMia) 翻译)
-* [如何使用 JavaScript 构建响应式引擎 —— Part 1:可观察的对象](https://juejin.im/post/58dc9da661ff4b0061547ca0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github)([IridescentMia](https://github.com/IridescentMia) 翻译)
* [所有前端译文>>](https://github.com/xitu/gold-miner/blob/master/front-end.md)
diff --git a/front-end.md b/front-end.md
index 52e31749ac5..bcbc316bce0 100644
--- a/front-end.md
+++ b/front-end.md
@@ -1,3 +1,4 @@
+* [写给“老派” Web 开发者的“现代” JavaScript 指南](https://juejin.im/post/58ebab0c8d6d81006191376f?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([SunCafe](http://suncafe.cc/) 翻译)
* [CSS很棒,只是真的太难了](https://juejin.im/entry/58eae24a61ff4b0061a6a102?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([ZhangFe](https://github.com/ZhangFe) 翻译)
* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
From 30f822f51a7b1b0d7c03045e84f24c87d700d648 Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Thu, 13 Apr 2017 23:01:01 +0800
Subject: [PATCH 224/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20webpack=20?=
=?UTF-8?q?=E6=8B=BE=E7=BF=A0=EF=BC=9A=E5=85=85=E5=88=86=E5=88=A9=E7=94=A8?=
=?UTF-8?q?=20CommonsChunkPlugin()=20=E7=9A=84=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 4 ++--
front-end.md | 1 +
2 files changed, 3 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index 255e14d40e7..0179de489e5 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [454](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [455](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -37,10 +37,10 @@
## 前端
+* [webpack 拾翠:充分利用 CommonsChunkPlugin()](https://juejin.im/post/58ec4e3f5c497d0062c470bf?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [写给“老派” Web 开发者的“现代” JavaScript 指南](https://juejin.im/post/58ebab0c8d6d81006191376f?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([SunCafe](http://suncafe.cc/) 翻译)
* [CSS很棒,只是真的太难了](https://juejin.im/entry/58eae24a61ff4b0061a6a102?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([ZhangFe](https://github.com/ZhangFe) 翻译)
* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
-* [setState() 门事件](https://juejin.im/post/58e5aeccb123db15eb80ecb0?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [所有前端译文>>](https://github.com/xitu/gold-miner/blob/master/front-end.md)
diff --git a/front-end.md b/front-end.md
index bcbc316bce0..1d501a043c1 100644
--- a/front-end.md
+++ b/front-end.md
@@ -1,3 +1,4 @@
+* [webpack 拾翠:充分利用 CommonsChunkPlugin()](https://juejin.im/post/58ec4e3f5c497d0062c470bf?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [写给“老派” Web 开发者的“现代” JavaScript 指南](https://juejin.im/post/58ebab0c8d6d81006191376f?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([SunCafe](http://suncafe.cc/) 翻译)
* [CSS很棒,只是真的太难了](https://juejin.im/entry/58eae24a61ff4b0061a6a102?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([ZhangFe](https://github.com/ZhangFe) 翻译)
* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
From 06127d45b22e9bdd043a5b082064c8354000ef57 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Thu, 13 Apr 2017 23:05:54 +0800
Subject: [PATCH 225/945] Create debugging-tips-tricks.md
---
TODO/debugging-tips-tricks.md | 229 ++++++++++++++++++++++++++++++++++
1 file changed, 229 insertions(+)
create mode 100644 TODO/debugging-tips-tricks.md
diff --git a/TODO/debugging-tips-tricks.md b/TODO/debugging-tips-tricks.md
new file mode 100644
index 00000000000..dca2d0b0200
--- /dev/null
+++ b/TODO/debugging-tips-tricks.md
@@ -0,0 +1,229 @@
+> * 原文地址:[Debugging Tips and Tricks](https://css-tricks.com/debugging-tips-tricks/)
+> * 原文作者:[SARAH DRASNER](https://css-tricks.com/author/sdrasner/)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+# Debugging Tips and Tricks #
+
+Writing code is only one small piece of being a developer. In order to be efficient and capable at our jobs, we must also excel at debugging. When I dedicate some time to learning new debugging skills, I often find I can move much quicker, and add more value to the teams I work on. I have a few tips and tricks I rely on pretty heavily and found that I give the same advice again and again during workshops, so here’s a compilation of some of them, as well as some from the community. We'll start with some core tenants and then drill down to more specific examples.
+
+[![](https://s3.buysellads.com/1279518/7386844-1491489478.jpg)](https://srv.buysellads.com/ads/click/x/GTND4237CVYDL23JCV74YKQWF67DTKJNCT7DKZ3JCEAI45QUCT7D6K7KC6BDLKJEF67DTK3EHJNCLSIZ)
+
+
+### # Main Concepts ###
+
+#### # Isolate the Problem ####
+
+Isolation is possibly the strongest core tenant in all of debugging. Our codebases can be sprawling, with different libraries, frameworks, and they can include many contributors, even people who aren't working on the project anymore. Isolating the problem helps us slowly whittle away non-essential parts of the problem so that we can singularly focus on a solution.
+
+Some of the benefits of isolation include, but are not limited to:
+
+- Figuring out if it’s actually the root cause we think it is or some sort of conflict
+- For time-based tasks, understanding whether or not there is a race condition
+- Taking a hard look at whether or not our code can be simplified, which can help with both writing it and maintaining it
+- Untangling it and seeing if it’s one issue or perhaps more
+
+It's very important to make the issue reproducible. Without being able to discern exactly what the issue is in a way where you can reproduce it, it’s very difficult to solve for it. This also allows you to compare it to working model that is similar so that you can see what changed or what is different between the two.
+
+I have a lot of different methods of isolation in practice. One is to create a reduced test case on a local instance, or a private CodePen, or a JSBin. Another is to create breakpoints in the code so that I can see it execute bit by bit. There are a few ways to define breakpoints:
+
+You can literally write `debugger;` inline in your code. You can see how this will fire small pieces at a time.
+
+You can take this one step further in Chrome DevTools and even walk through the next events that are fired or choose specific event listeners:
+
+![Step into the next function call](https://cdn.css-tricks.com/wp-content/uploads/2017/04/stepintonextfunctioncall.gif)
+
+Good 'ol `console.log` is a form of isolation. (Or `echo` in PHP, or `print` in python, etc…). You are taking one tiny piece of execution and testing your assumptions, or checking to see if something is altering. This is probably the most time-tested form of debugging, that no matter how advanced you become, still has it's uses. Arrow functions in ES6 have allowed us to step up our console debugging game as well, as it’s now a lot easier to write useful one-liners in the console.
+
+The `console.table` function is also a favorite tool of mine, especially great for when you have a lot of data you need to represent- large arrays, large objects and the like. The `console.dir` function is also a nice alternative. It will log an interactive listing of an object's properties.
+
+![](https://cdn.css-tricks.com/wp-content/uploads/2017/04/dir.png)
+
+console.dir gives an interactive listing
+
+#### #Be Methodical ####
+
+When I teach workshops and help students in my class, the number one thing that I find holds them back as they try to debug a problem is not being methodical enough. This is truly a tortoise-and-the-hare kind of situation. They understandably want to move quickly, so they change a ton of things at once, and then when something stops working, they don’t know which thing they changed is causing the error. Then, to debug, they change many things at once and get a little lost trying to figure out what is working and what isn't.
+
+We all do this to some extent. As we become more proficient with a tool, we can write more and more code without testing an assumption. But if you’re new to a syntax or technology, being slow and careful behooves you. You have a much better shot at backing out of an issue you might have inadvertently created for yourself. And indeed, once you have created an issue, debugging one thing at a time might seem slower, but exposes exactly what changes have happened and where the error lies in a way that a *seemingly* faster pace doesn’t allow. I say seemingly because the time isn’t actually recovered working this way.
+
+**Do you remember when you were a kid and your parents said, "if you get lost, stay where you are?"** My parents did, at least. It's because if they were moving around to find me and I was also moving around to find them, we'd have fewer chances of bumping into one another. Code works the same way. The less moving pieces you have, the better- the more you are returning consistent results, the easier it will be to track things down. So while you’re debugging, try not to also install anything, or put in new dependencies. If you see a different error every time you should be returning a static result, that’s a big red flag you should be headed right for with your sleuth hat on.
+
+### # Choose Good Tools ###
+
+There are a million different tools for solving a variety of problems. I’m going to work through some of the tools I find the most useful and then we’ll link off to a bevy of resources.
+
+#### # Syntax Highlighting ####
+
+Sure, it’s damn fun to pick out the new hotness in colors and flavors for your syntax highlighting theme, but spending some time thinking about clarity here matters. I often pick dark themes where a skip in syntax will turn all of my code a lighter color, I find errors are really easy to see right away. I tend to like Oceanic Next or Panda, but really, to each their own on this one. It’s important to keep in mind that when looking for a good syntax highlighter, awesome-looking is great, but functional for calling out your mistakes is most important, and it's totally possible to do both.
+
+#### # Linting ####
+
+Linting helps flag suspicious code and calls out errors we might have overlooked. Linting is incredibly important, but which linter you choose has so much to do with what language/framework you’re writing in, and then on top of that, what your agreed-upon code style is.
+
+Different companies have different code styles and rules. Personally, I like [AirBnB's](https://github.com/airbnb/javascript), but take care and don't just use any old linter. Your linter enforces patterns that, if you yourself don’t want to enforce, can stall your build process. I had a CSS linter that complained whenever I wrote a browser hack, and ended up having to circumvent it so often that it stopped being useful. But a good linter can shine light on small errors you might have missed that are lurking.
+
+Here are some resources:
+
+- I recently found [this responsive images linter](https://github.com/ausi/respimagelint), that tells you what opportunities you might have to use picture, srcset, or sizes.
+- Here’s a [pretty good breakdown](https://www.sitepoint.com/comparison-javascript-linting-tools/) of some JS linters
+
+#### # Browser Extensions ####
+
+Extensions can be really awesome because they can be enabled and disabled so readily, and they can work with really specific requirements. If you’re working with a particular library or framework, chances are, having their extension for DevTools enabled is going to give you all sorts of clarity that you can’t find otherwise. Take care though- not only can extensions bog a browser down, but they have permissions to execute scripts, so do a little homework into the extension author, ratings, and background. All that said, here are some of my favorites:
+
+- [Accessibility extension](https://chrome.google.com/webstore/detail/axe/lhdoppojpmngadmnindnejefpokejbdd) from Dequeue Systems
+- [React DevTools](https://chrome.google.com/webstore/detail/react-developer-tools/fmkadmapgofadopljbjfkapdkoienihi) really vital, in my opinion, if you're working with React, to see their virtual DOM
+- [Vue DevTools](https://chrome.google.com/webstore/detail/vuejs-devtools/nhdogjmejiglipccpnnnanhbledajbpd) same endorsement as above.
+- [Codopen](https://chrome.google.com/webstore/detail/codopen/agnkphdgffianchpipdbkeaclfbobaak): pops you out of the editor mode into a debug window for CodePen. Full disclosure: my husband made this for me as a present because he was sick of watching me manually opening the debug window (best gift ever!)
+- [Pageruler](https://chrome.google.com/webstore/detail/page-ruler/jlpkojjdgbllmedoapgfodplfhcbnbpn): get pixel dimensions and measure anything on a page. I like this one because I’m super duper anal about my layout. This helps me feed the beast.
+
+### # DevTools ###
+
+This is probably the most obvious of debugging tools, and there are so many things you can do with them. They can have so many packed-in features that can be easy to miss, so in the next section of specific tips, we'll go into a deep dive of some favorites.
+
+Umar Hansa has great materials for learning what the DevTools can do. He has [a weekly newsletter and GIFs](https://umaar.com/dev-tips/), a new course linked in the last section, and [an article on our site](https://css-tricks.com/six-tips-for-chrome-devtools/).
+
+One of my favorite recent ones is this [CSS Tracker Enhancement](https://umaar.com/dev-tips/126-css-tracker/), shown here with permission from Umar. This shows all of the unused CSS so that you can understand the performance impact.
+
+![](https://cdn.css-tricks.com/wp-content/uploads/2017/04/Screen-Shot-2017-04-10-at-10.20.11-AM.png)
+
+The CSS tracker shows color-coded rules for used and unused sets.
+
+#### # Misc Tools ####
+
+- [What input](https://ten1seven.github.io/what-input/) is a global utility for tracking the current input method (mouse, keyboard or touch), as well as the current intent- this can be really good for tracking down accessiblity leaks (hat tip to Marcy Sutton, accessibility expert for this tipoff)
+- [Ghostlabapp](https://www.vanamco.com/ghostlab/) is a pretty snazzy tool if you’re doing responsive development or checking anything deployed across a ton of devices. It offers synchronized web development, testing, and inspection.
+- [Eruda is an awesome tool](http://eruda.liriliri.io/) that helps debug on mobile devices. I really like it because it takes a simulator a step further, gives a console and real devtools to help you gain understanding.
+
+![eruda gives you a mobile console](https://cdn.css-tricks.com/wp-content/uploads/2017/04/Screen-Shot-2017-04-10-at-10.38.57-AM.png)
+
+### # Specific Tips ###
+
+I am always interested in what other people do to debug, so I asked the community through the CSS-Tricks account and my own what they were really into. This list is a mixture of tips I like as well as a roundup of tips from the community.
+
+#### # Accessibility ####
+
+```
+$('body').on('focusin',function(){
+ console.log(document.activeElement);});
+```
+
+> This logs the currently focused element, useful because opening the Devtools blurs the activeElement
+
+-[Marcy Sutton](https://twitter.com/marcysutton)
+
+#### # Debugging CSS ####
+
+We got quite a lot of responses saying that people put red borders on elements to see what they’re doing
+
+> [@sarah_edo](https://twitter.com/sarah_edo) for CSS, I'll usually have a .debug class with a red border that I slap on troublesome elements.
+>
+> — Jeremy Wagner (@malchata) [March 15, 2017](https://twitter.com/malchata/status/842029469246324736)
+
+I do this too, I even have a little CSS file that drops in some classes I can access for different colors easily.
+
+#### # Checking State in React ####
+
+> [@sarah_edo](https://twitter.com/sarah_edo)
{JSON.stringify(this.state, null, 2)}
+>
+> — MICHAEL JACKSON (@mjackson) [March 15, 2017](https://twitter.com/mjackson/status/842041642760646657)
+
+Props to Michael, this is one of the most useful debugging tools I know of. That snippet "pretty prints" the state of the component you're working with onto the component so that you can see what’s going on. You can validate that the state is working the way that you think it should be, and it helps track down any errors between the state and how you're using it.
+
+#### # Animation ####
+
+We got a lot of responses that said they slow the animation way down:
+
+> [@sarah_edo](https://twitter.com/sarah_edo)[@Real_CSS_Tricks](https://twitter.com/Real_CSS_Tricks) * { animation-duration: 10s !important; }
+>
+> — Thomas Fuchs (@thomasfuchs) [March 15, 2017](https://twitter.com/thomasfuchs/status/842029720820695040)
+
+I mentioned this on a post I wrote right here on CSS Tricks about [debugging CSS Keyframe animations](https://css-tricks.com/debugging-css-keyframe-animations/), there are more tips too, like how to hardware accelerate, or work with multiple transforms in different percentages.
+
+I also slow down my animations in JavaScript- in GreenSock that would look like: `timeline.timeScale(0.5)` (you can slow down the whole timeline, not just one thing at a time, which is super useful), in mo.js that would look like `{speed: 0.5}`.
+
+[Val Head has a great screencast](https://www.youtube.com/watch?v=MjRipmP7ffM&feature=youtu.be) going through both chrome and firefox devtools offering on animation.
+
+If you want to use the Chrome Devtools timeline to do performance audits, it's worth mentioning that painting is the most expense of the tasks, so all things being equal, pay a little more attention to a high percentage of that green.
+
+#### # Checking different connection speeds and loads ####
+
+I tend to work on fast connections, so I will throttle my connection to check and see what the performance would look like for people who don’t have my internet speed.
+
+![throttle connection in devtools](https://cdn.css-tricks.com/wp-content/uploads/2017/04/Screen-Shot-2017-04-10-at-9.29.00-AM.png)
+
+This is also useful in conjunction with a hard reload, or with the cache empty
+
+> [@sarah_edo](https://twitter.com/sarah_edo) Not so secret trick. But still many people are unaware. You need DevTools open, and then right click over the refresh button. [pic.twitter.com/FdAfF9Xtxm](https://t.co/FdAfF9Xtxm)
+>
+> — David Corbacho (@dcorbacho) [March 15, 2017](https://twitter.com/dcorbacho/status/842033259664035840)
+
+#### # Set a Timed Debugger ####
+
+This one came from Chris. We have a whole writeup on it [right here](https://css-tricks.com/set-timed-debugger-web-inspect-hard-grab-elements/):
+
+```
+setTimeout(function() {
+ debugger;
+}, 3000);
+```
+
+It’s similar to the debugger; tool I mentioned earlier, except you can put it in a setTimeout function and get even more fine-tuned information
+
+#### # Simulators ####
+
+> [@Real_CSS_Tricks](https://twitter.com/Real_CSS_Tricks) And just in case any Mac users didn't know this, iOS simulator + Safari is sweet. [pic.twitter.com/Uz4XO3e6uD](https://t.co/Uz4XO3e6uD)
+>
+> — Chris Coyier (@chriscoyier) [March 15, 2017](https://twitter.com/chriscoyier/status/842034009060302848)
+
+I mentioned simulators with Eruda before. iOS users also get a pretty sweet simulator. I was going to tell you you have to install XCode first, but this tweet showed another way:
+
+> [@chriscoyier](https://twitter.com/chriscoyier)[@Real_CSS_Tricks](https://twitter.com/Real_CSS_Tricks) Or, you can use this approach if you didn't want to bother with installing xCode: [https://t.co/WtAnZNo718](https://t.co/WtAnZNo718)
+>
+> — Chris Harrison (@cdharrison) [March 15, 2017](https://twitter.com/cdharrison/status/842038887904088065)
+
+Chrome also has a device toggle which is helpful.
+
+#### # Remote Debuggers ####
+
+> [@chriscoyier](https://twitter.com/chriscoyier)[@Real_CSS_Tricks](https://twitter.com/Real_CSS_Tricks)[https://t.co/q3OfWKNlUo](https://t.co/q3OfWKNlUo) is a good tool.
+>
+> — Gilles 💾⚽ (@gfra54) [March 15, 2017](https://twitter.com/gfra54/status/842035375304523777)
+
+I actually didn't know about this tool until seeing this tweet. Pretty useful!
+
+#### # CSS Grid Debugging #### (#article-header-id-18 .has-header-link)
+
+Rachel Andrew gave a presentation at Smashing and mentioned a little waffle thing you can click on in Firefox that will illuminate the gutters in the grid. [Her video](http://gridbyexample.com/learn/2016/12/17/learning-grid-day17/) explains it really eloquently.
+
+![](https://cdn.css-tricks.com/wp-content/uploads/2017/04/Screen-Shot-2017-04-10-at-9.58.14-AM.png)
+
+Rachel Andrew shows how to highlight gutters in Firefox DevTools.
+
+#### # Array Debugging ####
+
+Wes Bos with a really useful tip for searching for a single item in an array:
+
+> If you are just looking for a single item array.find() is 🔥 [https://t.co/AuRtyFwnq7](https://t.co/AuRtyFwnq7)
+>
+> — Wes Bos (@wesbos) [March 15, 2017](https://twitter.com/wesbos/status/842069915158884354)
+
+### # Further Debugging Resources ###
+
+Jon Kuperman has a [Frontend Masters course](https://frontendmasters.com/courses/chrome-dev-tools/) that can help you master devtools it goes along [with this app](https://github.com/jkup/mastering-chrome-devtools).
+
+There’s a small course on code school called [discover devtools](https://www.codeschool.com/courses/discover-devtools).
+
+Umar Hansa has a new online course called [Modern DevTools](https://moderndevtools.com/).
+
+Julia Evans has a great article [about debugging here](http://jvns.ca/blog/2015/11/22/how-i-got-better-at-debugging/), hat tip to Jamison Dance for showing it to me.
+
+Paul Irish does some [advanced performance audits with devtools](https://docs.google.com/document/d/1K-mKOqiUiSjgZTEscBLjtjd6E67oiK8H2ztOiq5tigk/pub) if you're super nerdy like me and want to dig into the timeline.
+
+Finally, I'll put in a bittersweet resource. My friend James Golick who was an excellent programmer and even more excellent human gave this great conference talk about debugging anything many years ago. Sadly James has passed, but we can still honor his memory and learn from him:
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 863a5bc2e3f1f08165f87df9039e310393325ee6 Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Thu, 13 Apr 2017 23:07:26 +0800
Subject: [PATCH 226/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20=E7=BB=86=E8=8A=82?=
=?UTF-8?q?=E6=98=AF=E4=BA=A7=E5=93=81=E8=AE=BE=E8=AE=A1=E7=9A=84=E9=87=8D?=
=?UTF-8?q?=E4=B8=AD=E4=B9=8B=E9=87=8D=20=E7=9A=84=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 4 ++--
product.md | 1 +
2 files changed, 3 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index 0179de489e5..982f0ed40da 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [455](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [456](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -81,8 +81,8 @@
## 产品
+* [细节是产品设计的重中之重](https://juejin.im/post/58ed96aaa22b9d00634732e9/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([iloveivyxuan](https://github.com/iloveivyxuan) 翻译)
* [单元测试,精益创业,以及两者之间的关系](https://juejin.im/post/58d90a3b44d90400694505c4/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [你正在阅读的用户体验文章是不是在向你进行推销?](https://juejin.im/post/58d4c501a22b9d00645544d9/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([ylq167](https://github.com/ylq167) 翻译)
* [直观设计 vs. 共享式设计](https://gold.xitu.io/entry/5862650a128fe1006d04d398/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([Funtrip](https://www.behance.net/Funtrip) 翻译)
-* [为何而设计?](https://gold.xitu.io/entry/5857969761ff4b00686ad66b/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([王子建](https://github.com/Romeo0906) 翻译)
* [所有产品译文>>](https://github.com/xitu/gold-miner/blob/master/product.md)
diff --git a/product.md b/product.md
index 308a8319f6f..be018fddc3b 100644
--- a/product.md
+++ b/product.md
@@ -1,3 +1,4 @@
+* [细节是产品设计的重中之重](https://juejin.im/post/58ed96aaa22b9d00634732e9/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([iloveivyxuan](https://github.com/iloveivyxuan) 翻译)
* [单元测试,精益创业,以及两者之间的关系](https://juejin.im/post/58d90a3b44d90400694505c4/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [你正在阅读的用户体验文章是不是在向你进行推销?](https://juejin.im/post/58d4c501a22b9d00645544d9/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([ylq167](https://github.com/ylq167) 翻译)
* [直观设计 vs. 共享式设计](https://gold.xitu.io/entry/5862650a128fe1006d04d398/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([Funtrip](https://www.behance.net/Funtrip) 翻译)
From 2cfde921be0d1ede43141d9b345f32aed73f45ed Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Thu, 13 Apr 2017 23:13:13 +0800
Subject: [PATCH 227/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20=E5=90=8C=E4=B8=AD?=
=?UTF-8?q?=E6=9C=89=E5=BC=82=E7=9A=84=20Webpack=20=E4=B8=8E=20Rollup=20?=
=?UTF-8?q?=E7=9A=84=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 4 ++--
front-end.md | 1 +
2 files changed, 3 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index 982f0ed40da..8db95c9f37f 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [456](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [457](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -37,10 +37,10 @@
## 前端
+* [同中有异的 Webpack 与 Rollup](https://juejin.im/post/58edb865570c350057f199a7?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([lsvih](https://github.com/lsvih) 翻译)
* [webpack 拾翠:充分利用 CommonsChunkPlugin()](https://juejin.im/post/58ec4e3f5c497d0062c470bf?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [写给“老派” Web 开发者的“现代” JavaScript 指南](https://juejin.im/post/58ebab0c8d6d81006191376f?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([SunCafe](http://suncafe.cc/) 翻译)
* [CSS很棒,只是真的太难了](https://juejin.im/entry/58eae24a61ff4b0061a6a102?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([ZhangFe](https://github.com/ZhangFe) 翻译)
-* [Preload,Prefetch 和它们在 Chrome 之中的优先级](https://juejin.im/post/58e8acf10ce46300585a7a42?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([gy134340](http://gy134340.com/) 翻译)
* [所有前端译文>>](https://github.com/xitu/gold-miner/blob/master/front-end.md)
diff --git a/front-end.md b/front-end.md
index 1d501a043c1..e646fb5aeff 100644
--- a/front-end.md
+++ b/front-end.md
@@ -1,3 +1,4 @@
+* [同中有异的 Webpack 与 Rollup](https://juejin.im/post/58edb865570c350057f199a7?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([lsvih](https://github.com/lsvih) 翻译)
* [webpack 拾翠:充分利用 CommonsChunkPlugin()](https://juejin.im/post/58ec4e3f5c497d0062c470bf?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([reid3290](https://github.com/reid3290) 翻译)
* [写给“老派” Web 开发者的“现代” JavaScript 指南](https://juejin.im/post/58ebab0c8d6d81006191376f?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([SunCafe](http://suncafe.cc/) 翻译)
* [CSS很棒,只是真的太难了](https://juejin.im/entry/58eae24a61ff4b0061a6a102?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([ZhangFe](https://github.com/ZhangFe) 翻译)
From 95989982386812d5e814660828cd3a7c0f224f25 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Thu, 13 Apr 2017 23:13:54 +0800
Subject: [PATCH 228/945] Update function-as-child-components.md
---
TODO/function-as-child-components.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/function-as-child-components.md b/TODO/function-as-child-components.md
index 761cef8ecbd..8a7ee9897c7 100644
--- a/TODO/function-as-child-components.md
+++ b/TODO/function-as-child-components.md
@@ -187,7 +187,7 @@ Ratio.defaultProps = {
Alright, so I did a lot there. We added some event listeners to listen for resize events as well as actually computing the width and height using the provided ratio. Neat, so we’ve got a width and height in our internal state, how can we share it with other components?
-This is one of those things that is hard to understand because it is so simple that when you see it you think, “That can’t be all there is to it.” but this ***is***all there is to it.
+This is one of those things that is hard to understand because it is so simple that when you see it you think, “That can’t be all there is to it.” but this is all there is to it.
#### Children is literally just a JavaScript function. ####
From f1201ddf9ef897e77cd6537a7deab9e62d135a42 Mon Sep 17 00:00:00 2001
From: sqrthree
Date: Thu, 13 Apr 2017 23:20:52 +0800
Subject: [PATCH 229/945] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=20MVVM-C=20=E4=B8=8E?=
=?UTF-8?q?=20Swift=20=E7=9A=84=E7=A7=AF=E5=88=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 4 ++--
ios.md | 1 +
2 files changed, 3 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index 8db95c9f37f..b8e4a9f3caa 100644
--- a/README.md
+++ b/README.md
@@ -5,7 +5,7 @@
[掘金翻译计划](https://juejin.im/tag/%E6%8E%98%E9%87%91%E7%BF%BB%E8%AF%91%E8%AE%A1%E5%88%92) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](#android)、[iOS](#ios)、[React](#react)、[前端](#前端)、[后端](#后端)、[产品](#产品)、[设计](#设计) 等领域,读者为热爱新技术的新锐开发者。
-掘金翻译计划目前翻译完成 [457](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
+掘金翻译计划目前翻译完成 [458](#近期文章列表) 篇文章,共有 [300](https://github.com/xitu/gold-miner/wiki/%E8%AF%91%E8%80%85%E7%A7%AF%E5%88%86%E8%A1%A8) 余名译者贡献翻译。
# 官方指南:
@@ -29,10 +29,10 @@
## iOS
+* [MVVM-C 与 Swift](https://juejin.im/post/58ef16b8da2f60005d180666/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([DeepMissea](https://github.com/DeepMissea)) 翻译)
* [Swift 闭包和代理中的保留周期](https://juejin.im/post/58e4ac5d44d904006d2a9a19/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([oOatuo](http://atuo.xyz/)) 翻译)
* [一名 iOS 开发者的 React Native 开发经历](https://juejin.im/post/58df4c3fa0bb9f0069e2f2bd/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([lsvih](https://github.com/lsvih)) 翻译)
* [优化 Swift 的编译时间](https://juejin.im/post/58df1d75570c35005796966b/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([DeepMissea](https://github.com/DeepMissea)) 翻译)
-* [看!Swift 里竟然有红绿灯 🚦!](https://juejin.im/post/58ddbc612f301e0062fed742/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([DeepMissea](https://github.com/DeepMissea)) 翻译)
* [所有 iOS 译文>>](https://github.com/xitu/gold-miner/blob/master/ios.md)
## 前端
diff --git a/ios.md b/ios.md
index 53fce8c3fce..89d29887e73 100644
--- a/ios.md
+++ b/ios.md
@@ -1,3 +1,4 @@
+* [MVVM-C 与 Swift](https://juejin.im/post/58ef16b8da2f60005d180666/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([DeepMissea](https://github.com/DeepMissea)) 翻译)
* [Swift 闭包和代理中的保留周期](https://juejin.im/post/58e4ac5d44d904006d2a9a19/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([oOatuo](http://atuo.xyz/)) 翻译)
* [一名 iOS 开发者的 React Native 开发经历](https://juejin.im/post/58df4c3fa0bb9f0069e2f2bd/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([lsvih](https://github.com/lsvih)) 翻译)
* [优化 Swift 的编译时间](https://juejin.im/post/58df1d75570c35005796966b/?utm_source=gold-miner&utm_medium=readme&utm_campaign=github) ([DeepMissea](https://github.com/DeepMissea)) 翻译)
From 548991b587ae880f4e24df0ce4c8720472e39781 Mon Sep 17 00:00:00 2001
From: gy134340
Date: Thu, 13 Apr 2017 23:25:29 +0800
Subject: [PATCH 230/945] =?UTF-8?q?=E5=AE=A1=E6=A0=B8=E4=BF=AE=E6=94=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...amming-in-javascript-composing-software.md | 30 +++++++++----------
1 file changed, 15 insertions(+), 15 deletions(-)
diff --git a/TODO/why-learn-functional-programming-in-javascript-composing-software.md b/TODO/why-learn-functional-programming-in-javascript-composing-software.md
index c260bf55297..76ee8309ca0 100644
--- a/TODO/why-learn-functional-programming-in-javascript-composing-software.md
+++ b/TODO/why-learn-functional-programming-in-javascript-composing-software.md
@@ -1,8 +1,8 @@
> * 原文地址:[Why Learn Functional Programming in JavaScript? (Composing Software)(part 2)](https://medium.com/javascript-scene/why-learn-functional-programming-in-javascript-composing-software-ea13afc7a257)
> * 原文作者:[Eric Elliott](https://medium.com/@_ericelliott?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
-> * 译者:
-> * 校对者:
+> * 译者:[gy134340](https://github.com/gy134340)
+> * 校对者:[sunui](https://github.com/sunui),[avocadowang](https://github.com/avocadowang)
# 为什么用 JavaScript 学习函数式编程?(软件构建)(第二部分)
@@ -10,7 +10,7 @@
烟雾的方块艺术 —MattysFlicks —(CC BY 2.0)
> 注意:这是从基础学习函数式编程和使用 JavaScript ES6+ 撰写软件的第二部分。保持关注,接下来还有很多!
-> [从第一部分开始](https://medium.com/javascript-scene/the-rise-and-fall-and-rise-of-functional-programming-composable-software-c2d91b424c8c#.2dfd6n6qe) | [接下来的 >](https://medium.com/javascript-scene/a-functional-programmers-introduction-to-javascript-composing-software-d670d14ede30#.2e4youss2)
+> [从第一部分开始](https://github.com/xitu/gold-miner/blob/master/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md) | [接下来的 >](https://github.com/xitu/gold-miner/blob/master/TODO/a-functional-programmers-introduction-to-javascript-composing-software.md)
忘掉你认为知道的关于 JavaScript 的一切,用初学者的眼光去看待它。为了帮助你做到这一点,我们将会从头复习一下 JavaScript 的基础,就像你与其尚未谋面一样。如果你是初学者,那你就很幸运了。最终从零开始探索 ES6 和函数式编程!希望所有的概念都被解释清楚 — 但不要太依赖于此。
@@ -20,9 +20,9 @@
JavaScript 有函数式编程所需要的最重要的特性:
-1. **一级公民函数:**使用函数作为数据值的能力:用函数传参,返回函数,用函数做变量和对象属性。这个属性允许更高级别的函数,使偏函数应用、柯里化和组合成为可能。
-2. **匿名函数和简洁的 lambda 语法:**`x => x * 2` 是 JavaScript 中有效的函数表达式。简洁的 lambda 语法使得高阶函数变的简单。
-3. **闭包:**闭包是一个有着自己独立作用域的捆绑函数。闭包在函数被创建时被创建。当一个函数在另一个函数内部被创建,它可以访问外部函数的变量,即使在外部函数退出后。通过闭包偏函数应用可以获取内部固定参数。固定的参数时绑定在返回函数的作用域范围内的参数。在 `add2(1)(2)` 中,`1` 是 `add2(1)` 返回的函数中的固定参数。
+1. **一级公民函数:** 使用函数作为数据值的能力:用函数传参,返回函数,用函数做变量和对象属性。这个属性允许更高级别的函数,使偏函数应用、柯里化和组合成为可能。
+2. **匿名函数和简洁的 lambda 语法:** `x => x * 2` 是 JavaScript 中有效的函数表达式。简洁的 lambda 语法使得高阶函数变的简单。
+3. **闭包:** 闭包是一个有着自己独立作用域的捆绑函数。闭包在函数被创建时被创建。当一个函数在另一个函数内部被创建,它可以访问外部函数的变量,即使在外部函数退出后。通过闭包偏函数应用可以获取内部固定参数。固定的参数时绑定在返回函数的作用域范围内的参数。在 `add2(1)(2)` 中,`1` 是 `add2(1)` 返回的函数中的固定参数。
### JavaScript 缺少了什么
@@ -40,19 +40,19 @@ JavaScript 是多范式语言,意味着它支持多种风格的编程。其他
下面是一些函数式语言拥有但是 JavaScript 没有的特性:
-1. **纯粹性:**在一些函数式语言中,纯粹性是强制的,有副作用的表达式是不被允许的。
-2. **不可变性:**一些函数式语言不允许转变,采用表达式来产生新的数据结构来代替更改一个已存的数据结构,比如说数组或者对象。这样看起来可能不够高效,但是大多数函数式语言在引擎下使用 trie 数据结构,具有结构共享的特点:意味着旧的对象和新的对象是对相同数据的引用。
-3. **递归:**递归是函数引用自身来进行迭代的能力。在大多数函数式语言中,递归是迭代的唯一方式,它们没有像 `for` 、`while`、`do` 这类循环语句。
+1. **纯粹性:** 在一些函数式语言中,纯粹性是强制的,有副作用的表达式是不被允许的。
+2. **不可变性:** 一些函数式语言不允许转变,采用表达式来产生新的数据结构来代替更改一个已存的数据结构,比如说数组或者对象。这样看起来可能不够高效,但是大多数函数式语言在引擎下使用 trie 数据结构,具有结构共享的特点:意味着旧的对象和新的对象是对相同数据的引用。
+3. **递归:** 递归是函数引用自身来进行迭代的能力。在大多数函数式语言中,递归是迭代的唯一方式,它们没有像 `for` 、`while`、`do` 这类循环语句。
-**纯粹性:**在 JavaScript 中,纯粹性由约定来达成,如果你不是使用纯函数来构成你的大多数应用,那么你就不是在进行函数式风格的编程。很不幸,在 JavaScript 中,你很容易就会不小心创建和使用一些不纯的函数。
+**纯粹性:** 在 JavaScript 中,纯粹性由约定来达成,如果你不是使用纯函数来构成你的大多数应用,那么你就不是在进行函数式风格的编程。很不幸,在 JavaScript 中,你很容易就会不小心创建和使用一些不纯的函数。
-**不可变性:**在纯函数式语言中,不可变性通常是强制的,JavaScript 缺少函数式语言中高效的、基于 trie 树的数据结构,但是你可以使用一些库,包括 [Immutable.js](https://facebook.github.io/immutable-js/) 和 [Mori](https://github.com/swannodette/mori),由衷期望未来的 ECMAScript 规范版本可以拥抱不可变数据结构。
+**不可变性:** 在纯函数式语言中,不可变性通常是强制的,JavaScript 缺少函数式语言中高效的、基于 trie 树的数据结构,但是你可以使用一些库,包括 [Immutable.js](https://facebook.github.io/immutable-js/) 和 [Mori](https://github.com/swannodette/mori),由衷期望未来的 ECMAScript 规范版本可以拥抱不可变数据结构。
有一些迹象带来了希望,比如说在 ES6 中添加了 `const` 关键字,`const` 声明的变量不能被重新赋值,重要的是要理解 `const` 所声明的值并不是不可改变的。
`const` 声明的对象不能被重新声明为新的对象,但是对象的属性却是可变的,JavaScript 有 `freeze()` 对象的能力,但是这些对象只能在根实例上被冻结,意味着嵌套着的对象还是可以改变它的属性。换句话说,在 JavaScript 规范中看到真正的不可变还有很长的路要走。
-**递归:**JavaScript 技术上支持递归,但是大多数函数式语言都有尾部调用优化的特性,尾部调用优化是一个允许递归的函数重用堆栈帧来递归调用的特性。
+**递归:** JavaScript 技术上支持递归,但是大多数函数式语言都有尾部调用优化的特性,尾部调用优化是一个允许递归的函数重用堆栈帧来递归调用的特性。
没有尾部调用优化,一个调用的栈很可能没有边界导致堆栈溢出。JavaScript 在 ES6 规范中有一个有限的尾调用优化。不幸的是,只有一个主要的浏览器引擎支持它,这个优化被部分应用随后从 Babel(最流行的 JavaScript 编译器,在旧的浏览器中被用来把 ES6 编译到 ES5) 中移除。
@@ -64,7 +64,7 @@ JavaScript 是多范式语言,意味着它支持多种风格的编程。其他
Monads 的问题是,尽管它的使用很简单,但是对一个不是很熟悉它的人解释清楚它有点像“对牛谈琴”。
-> “Monad说白了不过就是自函子范畴上的一个幺半群而已,这有什么难以理解的?” ~James Iry 所引用 Philip Wadler 的话,解释一个 Saunders Mac Lane 说过的名言。[*“编程语言简要、不完整之黑历史”*](http://james-iry.blogspot.com/2009/05/brief-incomplete-and-mostly-wrong.html)
+> “Monad说白了不过就是自函子范畴上的一个幺半群而已,这有什么难以理解的?” ~James Iry 所引用 Philip Wadler 的话,解释一个 Saunders Mac Lane 说过的名言。[**“编程语言简要、不完整之黑历史”**](http://james-iry.blogspot.com/2009/05/brief-incomplete-and-mostly-wrong.html)
典型的,这是在调侃这有趣的一点。在上面的引用中,关于 Monads 的解释相比最初的有了很大的简化,原来是下面这样:
@@ -135,7 +135,7 @@ JavaScript 的真正优势在于其生态系统中的思想和用户的多样性
App 正在吞食世界, web 正在吞食 app, 同时 JavaScript 正在吞食 web。
-[**接下来的第三部分: 函数式开发者的 JavScript 介绍…**](https://medium.com/javascript-scene/a-functional-programmers-introduction-to-javascript-composing-software-d670d14ede30#.zdpw16p65)
+[**接下来的第三部分: 函数式开发者的 JavScript 介绍…**](https://github.com/xitu/gold-miner/blob/master/TODO/a-functional-programmers-introduction-to-javascript-composing-software.md)
### 下一步
@@ -148,7 +148,7 @@ App 正在吞食世界, web 正在吞食 app, 同时 JavaScript 正在吞食
*Eric Elliott* 是 [*“Programming JavaScript Applications”*](http://pjabook.com) (O’Reilly) 和 “Learn JavaScript with Eric Elliott” 的作者。他曾效力于 *Adobe Systems, Zumba Fitness, he Wall Street Journal, ESPN, BBC, and top recording artists including Usher, Frank Ocean, Metallica* 和其他一些公司。
-*他和她的老婆(很漂亮)大部分时间都在旧金山湾区里。*
+**他和她的老婆(很漂亮)大部分时间都在旧金山湾区里。**
---
From faead526a8bbdb96a5e142a35150f99f78493622 Mon Sep 17 00:00:00 2001
From: gy134340
Date: Thu, 13 Apr 2017 23:29:35 +0800
Subject: [PATCH 231/945] Change name
---
...n-functional-programming-in-javascript-composing-software.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/why-learn-functional-programming-in-javascript-composing-software.md b/TODO/why-learn-functional-programming-in-javascript-composing-software.md
index 76ee8309ca0..1a9327f182e 100644
--- a/TODO/why-learn-functional-programming-in-javascript-composing-software.md
+++ b/TODO/why-learn-functional-programming-in-javascript-composing-software.md
@@ -4,7 +4,7 @@
> * 译者:[gy134340](https://github.com/gy134340)
> * 校对者:[sunui](https://github.com/sunui),[avocadowang](https://github.com/avocadowang)
-# 为什么用 JavaScript 学习函数式编程?(软件构建)(第二部分)
+# 为什么用 JavaScript 学习函数式编程?(软件编写)(第二部分)
From 5b1b51ebdafc4dff22796cad47f941d7edb82f82 Mon Sep 17 00:00:00 2001
From: gy134340
Date: Thu, 13 Apr 2017 23:31:51 +0800
Subject: [PATCH 232/945] Change title
---
...-and-rise-of-functional-programming-composable-software.md | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
index a529896a3c1..c02cffb517d 100644
--- a/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
+++ b/TODO/the-rise-and-fall-and-rise-of-functional-programming-composable-software.md
@@ -4,13 +4,13 @@
> * 译者:[gy134340](https://github.com/gy134340)
> * 校对者:[avocadowang](https://github.com/avocadowang),[Aladdin-ADD](https://github.com/Aladdin-ADD)
-# 跌宕起伏的函数式编程(软件构建)
+# 跌宕起伏的函数式编程(软件编写)
烟雾的方块艺术 —MattysFlicks —(CC BY 2.0)
-> 注意:这是从基础学习函数式编程和使用 JavaScript ES6+ 构建软件的第一部分。保持关注,接下来还有很多!
+> 注意:这是从基础学习函数式编程和使用 JavaScript ES6+ 编写软件的第一部分。保持关注,接下来还有很多!
当我 6 岁时,我花了很多时间跟我的小伙伴玩电脑游戏,他家有一个装满电脑的房间。对于我说,它们有不可抗拒的魔力。我花了很多时间探索所有的游戏。一天我问他,“我们怎样做一个游戏?”
From e4b0de65a66405a4f0afb4b5c5f897f8ec8efedd Mon Sep 17 00:00:00 2001
From: zhuzi
Date: Fri, 14 Apr 2017 01:05:56 +0800
Subject: [PATCH 233/945] update translation
---
TODO/secure-web-app-http-headers.md | 56 ++++++++++++++++++++---------
1 file changed, 39 insertions(+), 17 deletions(-)
diff --git a/TODO/secure-web-app-http-headers.md b/TODO/secure-web-app-http-headers.md
index af59d9b41f1..d15fe288045 100644
--- a/TODO/secure-web-app-http-headers.md
+++ b/TODO/secure-web-app-http-headers.md
@@ -219,13 +219,17 @@ functionrequestHandler(req, res){
res.setHeader('X-XSS-Protection','1;mode=block');}
```
-### Controlling Framing ###
+### Controlling Framing 控制 iframe ###
-An iframe (or HTML inline frame element, if you want to be more formal) is a DOM element that allows a web app to be nested within a parent web app. This powerful element enables some important web use cases, such as embedding third-party content into web apps, but it also has significant drawbacks, such as not being SEO-friendly and not playing nice with browser navigation — the list goes on.
+> An iframe (or HTML inline frame element, if you want to be more formal) is a DOM element that allows a web app to be nested within a parent web app. This powerful element enables some important web use cases, such as embedding third-party content into web apps, but it also has significant drawbacks, such as not being SEO-friendly and not playing nice with browser navigation — the list goes on.
-One of the caveats of iframes is that it makes clickjacking easier. Clickjacking is an attack that tricks the user into clicking something different than what they think they’re clicking. To understand a simple implementation of clickjacking, consider the HTML markup below, which tries to trick the user into buying a toaster when they think they are clicking to win a prize!
+iframe (正式来说,是一个 HTML 的行内框架元素)是一个 DOM 元素,它允许一个 web 应用嵌套在另一个 web 应用中。这个强大的元素有部分重要的使用场景,比如在 web 应用中嵌入第三方内容, 但它也有重大的缺点,例如对 SEO 不友好,对浏览器导航跳转也不友好,还有很多。
-```
+> One of the caveats of iframes is that it makes clickjacking easier. Clickjacking is an attack that tricks the user into clicking something different than what they think they’re clicking. To understand a simple implementation of clickjacking, consider the HTML markup below, which tries to trick the user into buying a toaster when they think they are clicking to win a prize!
+
+其中一个需要注意的事是它是的点击劫持变得更加容易。点击劫持是一种让用户点击与他们想要点击的不同的攻击。要理解一个简单的劫持实现,参考以下 HTML,当用户认为他们点击可以获得奖品时,实际上是试图欺骗用户购买面包机。
+
+```html
@@ -234,38 +238,56 @@ One of the caveats of iframes is that it makes clickjacking easier. Clickjacking
```
-Clickjacking has many malicious applications, such as tricking the user into confirming a Facebook like, purchasing an item online and even submitting confidential information. Malicious web apps can leverage iframes for clickjacking by embedding a legitimate web app inside their malicious web app, rendering the iframe invisible with the `opacity: 0` CSS rule, and placing the iframe’s click target directly on top of an innocent-looking button rendered by the malicious web app. A user who clicks the innocent-looking button will trigger a click on the embedded web app — without at all knowing the effect of their click.
+> Clickjacking has many malicious applications, such as tricking the user into confirming a Facebook like, purchasing an item online and even submitting confidential information. Malicious web apps can leverage iframes for clickjacking by embedding a legitimate web app inside their malicious web app, rendering the iframe invisible with the `opacity: 0` CSS rule, and placing the iframe’s click target directly on top of an innocent-looking button rendered by the malicious web app. A user who clicks the innocent-looking button will trigger a click on the embedded web app — without at all knowing the effect of their click.
+
+点击劫持有许多恶意应用程序,例如欺骗用户确认 Facebook 点赞,在线购买商品,甚至提交机密信息。恶意 web 应用程序可以通过在其恶意应用中嵌入合法的 web 应用来利用 iframe 进行点击劫持,这可以通过设置 `opacity: 0` 的 CSS 规则将其隐藏,并将 iframe 的点击目标直接放置在看起来无辜的按钮之上。点击了这个无辜按钮的用户会直接点击在嵌入的 web 应用上,并不知道点击后的作用。
-An effective way to block this attack is by restricting your web app from being framed. `X-Frame-Options`, specified in [RFC 7034](https://www.ietf.org/rfc/rfc7034.txt), is designed to do exactly that! This header instructs the browser to apply limitations on whether your web app can be embedded within another web page, thus blocking a malicious web page from tricking users into invoking various transactions on your web app. You can either block framing completely using the `DENY` directive, whitelist specific domains using the `ALLOW-FROM` directive, or whitelist only the web app’s origin using the `SAMEORIGIN` directive.
+> An effective way to block this attack is by restricting your web app from being framed. `X-Frame-Options`, specified in [RFC 7034](https://www.ietf.org/rfc/rfc7034.txt), is designed to do exactly that! This header instructs the browser to apply limitations on whether your web app can be embedded within another web page, thus blocking a malicious web page from tricking users into invoking various transactions on your web app. You can either block framing completely using the `DENY` directive, whitelist specific domains using the `ALLOW-FROM` directive, or whitelist only the web app’s origin using the `SAMEORIGIN` directive.
-My recommendation is to use the `SAMEORIGIN` directive, which enables iframes to be leveraged for apps on the same domain — which may be useful at times — and which maintains security. This recommended header looks like this:
+阻止这种攻击的一种有效的方法是限制你的 web 应用被框架化。`X-Frame-Options`,在 [RFC 7034](https://www.ietf.org/rfc/rfc7034.txt) 中引入,就是设计用来做这件事的。此响应头指示浏览器对你的 web 应用是否可以被嵌入另一个网页进行限制,从而阻止恶意网页欺骗用户调用你的应用程序上的各种事务。你可以使用 `DENY` 完全屏蔽,或者使用 `ALLOW-FROM` 指定将特定域列入白名单,也可以使用 `SAMEORIGIN` 指令将应用的源地址列入白名单。
+
+> My recommendation is to use the `SAMEORIGIN` directive, which enables iframes to be leveraged for apps on the same domain — which may be useful at times — and which maintains security. This recommended header looks like this:
+
+我的建议是使用 `SAMEORIGIN` 指令,因为它允许 iframe 被用于同一域上的可以保证安全性的应用程序,这有时是有用的。
```
X-Frame-Options: SAMEORIGIN
```
-Here’s an example of a configuration of this header to enable framing on the same origin in Node.js:
+以下是在 Node.js 中设置此响应头的示例代码:
-```
+```javascript
functionrequestHandler(req, res){
res.setHeader('X-Frame-Options','SAMEORIGIN');}
```
-### Explicitly Whitelisting Sources ###
+### 指定白名单资源 ###
-As we’ve noted earlier, you can add in-depth security to your web app by enabling the browser’s XSS filter. However, note that this mechanism is limited, is not supported by all browsers (Firefox, for instance, does not have an XSS filter) and relies on pattern-matching techniques that can be tricked.
+> As we’ve noted earlier, you can add in-depth security to your web app by enabling the browser’s XSS filter. However, note that this mechanism is limited, is not supported by all browsers (Firefox, for instance, does not have an XSS filter) and relies on pattern-matching techniques that can be tricked.
-Another layer of in-depth protection against XSS and other attacks can be achieved by explicitly whitelisting trusted sources and operations — which is what Content Security Policy (CSP) enables web app developers to do.
+如前所述,你可以通过启用浏览器的 XSS 过滤器,给你的 web 应用程序增强安全性。然而请注意,这种机制是有局限性的,不是所有浏览器都支持(例如 Firefox 就不支持 XSS 过滤),并且依赖的模式匹配技术可以被欺骗。
-CSP is a [W3C specification](https://www.w3.org/TR/2016/WD-CSP3-20160901/) that defines a powerful browser-based security mechanism, enabling granular control over resource-loading and script execution in a web app. With CSP, you can whitelist specific domains for operations such as script-loading, AJAX calls, image-loading and style sheet-loading. You can enable or disable inline scripts or dynamic scripts (the notorious `eval`) and control framing by whitelisting specific domains for framing. Another cool feature of CSP is that it allows you to configure a real-time reporting target, so that you can monitor your app in real time for CSP blocking operations.
+> Another layer of in-depth protection against XSS and other attacks can be achieved by explicitly whitelisting trusted sources and operations — which is what Content Security Policy (CSP) enables web app developers to do.
-This explicit whitelisting of resource loading and execution provides in-depth security that in many cases will fend off attacks. For example, by using CSP to disallow inline scripts, you can fend off many of the reflective XSS attack variants that rely on injecting inline scripts into the DOM.
+对抗 XSS 和其他攻击的更多一层的保护,可以通过明确列出可信来源和操作来实现 —— 这就是内容安全策略(CSP)。
-CSP is a relatively complex header, with a lot of directives, and I won’t go into the details of the various directives. HTML5 Rocks has a [great tutorial](https://www.html5rocks.com/en/tutorials/security/content-security-policy/) that provides an overview of CSP, and I highly recommend reading it and learning how to use CSP in your web app.
+> CSP is a [W3C specification](https://www.w3.org/TR/2016/WD-CSP3-20160901/) that defines a powerful browser-based security mechanism, enabling granular control over resource-loading and script execution in a web app. With CSP, you can whitelist specific domains for operations such as script-loading, AJAX calls, image-loading and style sheet-loading. You can enable or disable inline scripts or dynamic scripts (the notorious `eval`) and control framing by whitelisting specific domains for framing. Another cool feature of CSP is that it allows you to configure a real-time reporting target, so that you can monitor your app in real time for CSP blocking operations.
-Here’s a simple example of a CSP configuration to allow script-loading from the app’s origin only and to block dynamic script execution (`eval`) and inline scripts (as usual, on Node.js):
+CSP 是一种 W3C 规范,它定义了强大的基于浏览器的安全机制,可以对 web 应用中的资源加载以及脚本执行进行精细的控制。使用 CSP 可以将特定的域加入白名单进行例如脚本加载、AJAX 调用、图像加载和样式加载。你可以启用或禁用内嵌脚本或动态脚本(臭名昭著的 `eval`),并通过将特定域列入白名单来控制 iframe。CSP 的另一个很酷的功能是它允许配置实时报告目标,一遍实时监控应用程序进行 CSP 阻止操作。
-```
+> This explicit whitelisting of resource loading and execution provides in-depth security that in many cases will fend off attacks. For example, by using CSP to disallow inline scripts, you can fend off many of the reflective XSS attack variants that rely on injecting inline scripts into the DOM.
+
+这种对资源加载和脚本执行的明确的白名单提供了很强的安全性,在很多情况下都可以防范攻击。例如,使用 CSP 禁止内嵌脚本,你可以防范很多反射型 XSS 攻击,因为它们依赖于将内嵌脚本注入到 DOM。
+
+> CSP is a relatively complex header, with a lot of directives, and I won’t go into the details of the various directives. HTML5 Rocks has a [great tutorial](https://www.html5rocks.com/en/tutorials/security/content-security-policy/) that provides an overview of CSP, and I highly recommend reading it and learning how to use CSP in your web app.
+
+CSP 是一个相对复杂的响应头,它有很多种指令,在这里我不详细展开了,可以参考 HTML5 Rocks 里一篇很棒的[教程](https://www.html5rocks.com/en/tutorials/security/content-security-policy/),提供了 CSP 的概述,我非常推荐阅读它来学习如何在你的 web 应用中使用 CSP。
+
+> Here’s a simple example of a CSP configuration to allow script-loading from the app’s origin only and to block dynamic script execution (`eval`) and inline scripts (as usual, on Node.js):
+
+以下是一个设置 CSP 的示例代码,它仅允许从应用程序的源域加载脚本,并组织动态脚本的执行(eval)以及内嵌脚本(当然,还是 Node.js):
+
+```javascript
functionrequestHandler(req, res){
res.setHeader('Content-Security-Policy',"script-src 'self'");}
```
From 1d57ea5b8c23304f3943e32de02ca3710c1d0ec7 Mon Sep 17 00:00:00 2001
From: Wenlin Ou
Date: Thu, 13 Apr 2017 18:15:39 -0400
Subject: [PATCH 234/945] added proofreader info and error fixes
---
TODO/nothing-will-change-until-you-start-building.md | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/TODO/nothing-will-change-until-you-start-building.md b/TODO/nothing-will-change-until-you-start-building.md
index f063274822b..f5c5b91bc89 100644
--- a/TODO/nothing-will-change-until-you-start-building.md
+++ b/TODO/nothing-will-change-until-you-start-building.md
@@ -2,7 +2,7 @@
> * 原文作者:[Jonathan Z. White](https://medium.freecodecamp.com/@JonathanZWhite?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[owenlyn](https://github.com/owenlyn)
-> * 校对者:
+> * 校对者:[luoqiuyu](https://github.com/luoqiuyu),[Tina92](https://github.com/Tina92)
---
@@ -59,7 +59,7 @@
当你明确了你想要解决的问题并有了一个解决方案的时候,你可能会问自己这是不是一个好的点子。
-**相当一部分看起来很糟糕的点子最后被证明是超级棒的生意。** 比如,当年没人想投资 Airbnb。 Brian Chesky, Airbnb 的创始人之一,在他的文章 [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz) 里详细讲述了被投资人拒绝的故事。
+**相当一部分看起来很糟糕的点子最后却造就了一些伟大的公司。** 比如,当年没人想投资 Airbnb。 Brian Chesky, Airbnb 的创始人之一,在他的文章 [7 Rejections](https://medium.com/@bchesky/7-rejections-7d894cbaa084#.l8fdqlasz) 里详细讲述了被投资人拒绝的故事。
![](https://cdn-images-1.medium.com/max/800/1*WpxUxMCO-7NXr-o1yo023g.png)
From b30c68efa9815d4652b09acc573ffbbb3854c38a Mon Sep 17 00:00:00 2001
From: sqrtthree
Date: Fri, 14 Apr 2017 10:33:33 +0800
Subject: [PATCH 235/945] =?UTF-8?q?=E5=88=A0=E9=99=A4=E6=97=A0=E7=94=A8?=
=?UTF-8?q?=E7=9A=84=E5=9B=BE=E7=89=87?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/debugging-tips-tricks.md | 33 +++++++++++++++------------------
1 file changed, 15 insertions(+), 18 deletions(-)
diff --git a/TODO/debugging-tips-tricks.md b/TODO/debugging-tips-tricks.md
index dca2d0b0200..dee46a5eabc 100644
--- a/TODO/debugging-tips-tricks.md
+++ b/TODO/debugging-tips-tricks.md
@@ -4,18 +4,15 @@
> * 译者:
> * 校对者:
-# Debugging Tips and Tricks #
+# Debugging Tips and Tricks #
Writing code is only one small piece of being a developer. In order to be efficient and capable at our jobs, we must also excel at debugging. When I dedicate some time to learning new debugging skills, I often find I can move much quicker, and add more value to the teams I work on. I have a few tips and tricks I rely on pretty heavily and found that I give the same advice again and again during workshops, so here’s a compilation of some of them, as well as some from the community. We'll start with some core tenants and then drill down to more specific examples.
-[![](https://s3.buysellads.com/1279518/7386844-1491489478.jpg)](https://srv.buysellads.com/ads/click/x/GTND4237CVYDL23JCV74YKQWF67DTKJNCT7DKZ3JCEAI45QUCT7D6K7KC6BDLKJEF67DTK3EHJNCLSIZ)
-
-
### # Main Concepts ###
#### # Isolate the Problem ####
-Isolation is possibly the strongest core tenant in all of debugging. Our codebases can be sprawling, with different libraries, frameworks, and they can include many contributors, even people who aren't working on the project anymore. Isolating the problem helps us slowly whittle away non-essential parts of the problem so that we can singularly focus on a solution.
+Isolation is possibly the strongest core tenant in all of debugging. Our codebases can be sprawling, with different libraries, frameworks, and they can include many contributors, even people who aren't working on the project anymore. Isolating the problem helps us slowly whittle away non-essential parts of the problem so that we can singularly focus on a solution.
Some of the benefits of isolation include, but are not limited to:
@@ -36,7 +33,7 @@ You can take this one step further in Chrome DevTools and even walk through the
Good 'ol `console.log` is a form of isolation. (Or `echo` in PHP, or `print` in python, etc…). You are taking one tiny piece of execution and testing your assumptions, or checking to see if something is altering. This is probably the most time-tested form of debugging, that no matter how advanced you become, still has it's uses. Arrow functions in ES6 have allowed us to step up our console debugging game as well, as it’s now a lot easier to write useful one-liners in the console.
-The `console.table` function is also a favorite tool of mine, especially great for when you have a lot of data you need to represent- large arrays, large objects and the like. The `console.dir` function is also a nice alternative. It will log an interactive listing of an object's properties.
+The `console.table` function is also a favorite tool of mine, especially great for when you have a lot of data you need to represent- large arrays, large objects and the like. The `console.dir` function is also a nice alternative. It will log an interactive listing of an object's properties.
![](https://cdn.css-tricks.com/wp-content/uploads/2017/04/dir.png)
@@ -93,11 +90,11 @@ The CSS tracker shows color-coded rules for used and unused sets.
#### # Misc Tools ####
-- [What input](https://ten1seven.github.io/what-input/) is a global utility for tracking the current input method (mouse, keyboard or touch), as well as the current intent- this can be really good for tracking down accessiblity leaks (hat tip to Marcy Sutton, accessibility expert for this tipoff)
+- [What input](https://ten1seven.github.io/what-input/) is a global utility for tracking the current input method (mouse, keyboard or touch), as well as the current intent- this can be really good for tracking down accessiblity leaks (hat tip to Marcy Sutton, accessibility expert for this tipoff)
- [Ghostlabapp](https://www.vanamco.com/ghostlab/) is a pretty snazzy tool if you’re doing responsive development or checking anything deployed across a ton of devices. It offers synchronized web development, testing, and inspection.
- [Eruda is an awesome tool](http://eruda.liriliri.io/) that helps debug on mobile devices. I really like it because it takes a simulator a step further, gives a console and real devtools to help you gain understanding.
-![eruda gives you a mobile console](https://cdn.css-tricks.com/wp-content/uploads/2017/04/Screen-Shot-2017-04-10-at-10.38.57-AM.png)
+![eruda gives you a mobile console](https://cdn.css-tricks.com/wp-content/uploads/2017/04/Screen-Shot-2017-04-10-at-10.38.57-AM.png)
### # Specific Tips ###
@@ -119,7 +116,7 @@ $('body').on('focusin',function(){
We got quite a lot of responses saying that people put red borders on elements to see what they’re doing
> [@sarah_edo](https://twitter.com/sarah_edo) for CSS, I'll usually have a .debug class with a red border that I slap on troublesome elements.
->
+>
> — Jeremy Wagner (@malchata) [March 15, 2017](https://twitter.com/malchata/status/842029469246324736)
I do this too, I even have a little CSS file that drops in some classes I can access for different colors easily.
@@ -127,7 +124,7 @@ I do this too, I even have a little CSS file that drops in some classes I can ac
#### # Checking State in React ####
> [@sarah_edo](https://twitter.com/sarah_edo)
{JSON.stringify(this.state, null, 2)}
->
+>
> — MICHAEL JACKSON (@mjackson) [March 15, 2017](https://twitter.com/mjackson/status/842041642760646657)
Props to Michael, this is one of the most useful debugging tools I know of. That snippet "pretty prints" the state of the component you're working with onto the component so that you can see what’s going on. You can validate that the state is working the way that you think it should be, and it helps track down any errors between the state and how you're using it.
@@ -137,7 +134,7 @@ Props to Michael, this is one of the most useful debugging tools I know of. That
We got a lot of responses that said they slow the animation way down:
> [@sarah_edo](https://twitter.com/sarah_edo)[@Real_CSS_Tricks](https://twitter.com/Real_CSS_Tricks) * { animation-duration: 10s !important; }
->
+>
> — Thomas Fuchs (@thomasfuchs) [March 15, 2017](https://twitter.com/thomasfuchs/status/842029720820695040)
I mentioned this on a post I wrote right here on CSS Tricks about [debugging CSS Keyframe animations](https://css-tricks.com/debugging-css-keyframe-animations/), there are more tips too, like how to hardware accelerate, or work with multiple transforms in different percentages.
@@ -157,7 +154,7 @@ I tend to work on fast connections, so I will throttle my connection to check an
This is also useful in conjunction with a hard reload, or with the cache empty
> [@sarah_edo](https://twitter.com/sarah_edo) Not so secret trick. But still many people are unaware. You need DevTools open, and then right click over the refresh button. [pic.twitter.com/FdAfF9Xtxm](https://t.co/FdAfF9Xtxm)
->
+>
> — David Corbacho (@dcorbacho) [March 15, 2017](https://twitter.com/dcorbacho/status/842033259664035840)
#### # Set a Timed Debugger ####
@@ -175,13 +172,13 @@ It’s similar to the debugger; tool I mentioned earlier, except you can put it
#### # Simulators ####
> [@Real_CSS_Tricks](https://twitter.com/Real_CSS_Tricks) And just in case any Mac users didn't know this, iOS simulator + Safari is sweet. [pic.twitter.com/Uz4XO3e6uD](https://t.co/Uz4XO3e6uD)
->
+>
> — Chris Coyier (@chriscoyier) [March 15, 2017](https://twitter.com/chriscoyier/status/842034009060302848)
I mentioned simulators with Eruda before. iOS users also get a pretty sweet simulator. I was going to tell you you have to install XCode first, but this tweet showed another way:
> [@chriscoyier](https://twitter.com/chriscoyier)[@Real_CSS_Tricks](https://twitter.com/Real_CSS_Tricks) Or, you can use this approach if you didn't want to bother with installing xCode: [https://t.co/WtAnZNo718](https://t.co/WtAnZNo718)
->
+>
> — Chris Harrison (@cdharrison) [March 15, 2017](https://twitter.com/cdharrison/status/842038887904088065)
Chrome also has a device toggle which is helpful.
@@ -189,7 +186,7 @@ Chrome also has a device toggle which is helpful.
#### # Remote Debuggers ####
> [@chriscoyier](https://twitter.com/chriscoyier)[@Real_CSS_Tricks](https://twitter.com/Real_CSS_Tricks)[https://t.co/q3OfWKNlUo](https://t.co/q3OfWKNlUo) is a good tool.
->
+>
> — Gilles 💾⚽ (@gfra54) [March 15, 2017](https://twitter.com/gfra54/status/842035375304523777)
I actually didn't know about this tool until seeing this tweet. Pretty useful!
@@ -198,7 +195,7 @@ I actually didn't know about this tool until seeing this tweet. Pretty useful!
Rachel Andrew gave a presentation at Smashing and mentioned a little waffle thing you can click on in Firefox that will illuminate the gutters in the grid. [Her video](http://gridbyexample.com/learn/2016/12/17/learning-grid-day17/) explains it really eloquently.
-![](https://cdn.css-tricks.com/wp-content/uploads/2017/04/Screen-Shot-2017-04-10-at-9.58.14-AM.png)
+![](https://cdn.css-tricks.com/wp-content/uploads/2017/04/Screen-Shot-2017-04-10-at-9.58.14-AM.png)
Rachel Andrew shows how to highlight gutters in Firefox DevTools.
@@ -207,14 +204,14 @@ Rachel Andrew shows how to highlight gutters in Firefox DevTools.
Wes Bos with a really useful tip for searching for a single item in an array:
> If you are just looking for a single item array.find() is 🔥 [https://t.co/AuRtyFwnq7](https://t.co/AuRtyFwnq7)
->
+>
> — Wes Bos (@wesbos) [March 15, 2017](https://twitter.com/wesbos/status/842069915158884354)
### # Further Debugging Resources ###
Jon Kuperman has a [Frontend Masters course](https://frontendmasters.com/courses/chrome-dev-tools/) that can help you master devtools it goes along [with this app](https://github.com/jkup/mastering-chrome-devtools).
-There’s a small course on code school called [discover devtools](https://www.codeschool.com/courses/discover-devtools).
+There’s a small course on code school called [discover devtools](https://www.codeschool.com/courses/discover-devtools).
Umar Hansa has a new online course called [Modern DevTools](https://moderndevtools.com/).
From d7f45df5a6af8f59194ced27863aa89ceaa8c12b Mon Sep 17 00:00:00 2001
From: reid
Date: Fri, 14 Apr 2017 14:14:05 +0800
Subject: [PATCH 236/945] =?UTF-8?q?=E6=A0=B9=E6=8D=AE=E6=A0=A1=E5=AF=B9?=
=?UTF-8?q?=E6=84=8F=E8=A7=81=E4=BF=AE=E6=94=B9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
TODO/higher-order-functions-composing-software.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/TODO/higher-order-functions-composing-software.md b/TODO/higher-order-functions-composing-software.md
index 67336f3fd47..0f8e8def2bf 100644
--- a/TODO/higher-order-functions-composing-software.md
+++ b/TODO/higher-order-functions-composing-software.md
@@ -127,7 +127,7 @@ const gt3 = highpass(3);
[1, 2, 3, 4].filter(gt3); // [3, 4];
```
-换言之,高阶函数可以用来实现函数的多态性。如你所见,相较于一阶函数而言,高阶函数的复用性和通用性非常好。一般来讲,在实际编码中会组合使用高阶函数和一些非常简单的一阶函数。
+换言之,高阶函数可以用来实现函数的多态性。如你所见,相对于一阶函数而言,高阶函数的复用性和通用性更好。一般来讲,在实际编码中会组合使用高阶函数和一些非常简单的一阶函数。
[**再续 “Reduce” >**](https://medium.com/javascript-scene/reduce-composing-software-fe22f0c39a1d)
From 966426c87034b8bca35ff74ff6733d14a4149da2 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Fri, 14 Apr 2017 20:49:01 +0800
Subject: [PATCH 237/945] Update debugging-tips-tricks.md
---
TODO/debugging-tips-tricks.md | 42 +++++++++++++++++------------------
1 file changed, 21 insertions(+), 21 deletions(-)
diff --git a/TODO/debugging-tips-tricks.md b/TODO/debugging-tips-tricks.md
index dee46a5eabc..12b0aec52df 100644
--- a/TODO/debugging-tips-tricks.md
+++ b/TODO/debugging-tips-tricks.md
@@ -8,9 +8,9 @@
Writing code is only one small piece of being a developer. In order to be efficient and capable at our jobs, we must also excel at debugging. When I dedicate some time to learning new debugging skills, I often find I can move much quicker, and add more value to the teams I work on. I have a few tips and tricks I rely on pretty heavily and found that I give the same advice again and again during workshops, so here’s a compilation of some of them, as well as some from the community. We'll start with some core tenants and then drill down to more specific examples.
-### # Main Concepts ###
+### Main Concepts ###
-#### # Isolate the Problem ####
+#### Isolate the Problem ####
Isolation is possibly the strongest core tenant in all of debugging. Our codebases can be sprawling, with different libraries, frameworks, and they can include many contributors, even people who aren't working on the project anymore. Isolating the problem helps us slowly whittle away non-essential parts of the problem so that we can singularly focus on a solution.
@@ -39,7 +39,7 @@ The `console.table` function is also a favorite tool of mine, especially great f
console.dir gives an interactive listing
-#### #Be Methodical ####
+#### Be Methodical ####
When I teach workshops and help students in my class, the number one thing that I find holds them back as they try to debug a problem is not being methodical enough. This is truly a tortoise-and-the-hare kind of situation. They understandably want to move quickly, so they change a ton of things at once, and then when something stops working, they don’t know which thing they changed is causing the error. Then, to debug, they change many things at once and get a little lost trying to figure out what is working and what isn't.
@@ -47,15 +47,15 @@ We all do this to some extent. As we become more proficient with a tool, we can
**Do you remember when you were a kid and your parents said, "if you get lost, stay where you are?"** My parents did, at least. It's because if they were moving around to find me and I was also moving around to find them, we'd have fewer chances of bumping into one another. Code works the same way. The less moving pieces you have, the better- the more you are returning consistent results, the easier it will be to track things down. So while you’re debugging, try not to also install anything, or put in new dependencies. If you see a different error every time you should be returning a static result, that’s a big red flag you should be headed right for with your sleuth hat on.
-### # Choose Good Tools ###
+### Choose Good Tools ###
There are a million different tools for solving a variety of problems. I’m going to work through some of the tools I find the most useful and then we’ll link off to a bevy of resources.
-#### # Syntax Highlighting ####
+#### Syntax Highlighting ####
Sure, it’s damn fun to pick out the new hotness in colors and flavors for your syntax highlighting theme, but spending some time thinking about clarity here matters. I often pick dark themes where a skip in syntax will turn all of my code a lighter color, I find errors are really easy to see right away. I tend to like Oceanic Next or Panda, but really, to each their own on this one. It’s important to keep in mind that when looking for a good syntax highlighter, awesome-looking is great, but functional for calling out your mistakes is most important, and it's totally possible to do both.
-#### # Linting ####
+#### Linting ####
Linting helps flag suspicious code and calls out errors we might have overlooked. Linting is incredibly important, but which linter you choose has so much to do with what language/framework you’re writing in, and then on top of that, what your agreed-upon code style is.
@@ -66,7 +66,7 @@ Here are some resources:
- I recently found [this responsive images linter](https://github.com/ausi/respimagelint), that tells you what opportunities you might have to use picture, srcset, or sizes.
- Here’s a [pretty good breakdown](https://www.sitepoint.com/comparison-javascript-linting-tools/) of some JS linters
-#### # Browser Extensions ####
+#### Browser Extensions ####
Extensions can be really awesome because they can be enabled and disabled so readily, and they can work with really specific requirements. If you’re working with a particular library or framework, chances are, having their extension for DevTools enabled is going to give you all sorts of clarity that you can’t find otherwise. Take care though- not only can extensions bog a browser down, but they have permissions to execute scripts, so do a little homework into the extension author, ratings, and background. All that said, here are some of my favorites:
@@ -76,7 +76,7 @@ Extensions can be really awesome because they can be enabled and disabled so rea
- [Codopen](https://chrome.google.com/webstore/detail/codopen/agnkphdgffianchpipdbkeaclfbobaak): pops you out of the editor mode into a debug window for CodePen. Full disclosure: my husband made this for me as a present because he was sick of watching me manually opening the debug window (best gift ever!)
- [Pageruler](https://chrome.google.com/webstore/detail/page-ruler/jlpkojjdgbllmedoapgfodplfhcbnbpn): get pixel dimensions and measure anything on a page. I like this one because I’m super duper anal about my layout. This helps me feed the beast.
-### # DevTools ###
+### DevTools ###
This is probably the most obvious of debugging tools, and there are so many things you can do with them. They can have so many packed-in features that can be easy to miss, so in the next section of specific tips, we'll go into a deep dive of some favorites.
@@ -88,7 +88,7 @@ One of my favorite recent ones is this [CSS Tracker Enhancement](https://umaar.c
The CSS tracker shows color-coded rules for used and unused sets.
-#### # Misc Tools ####
+#### Misc Tools ####
- [What input](https://ten1seven.github.io/what-input/) is a global utility for tracking the current input method (mouse, keyboard or touch), as well as the current intent- this can be really good for tracking down accessiblity leaks (hat tip to Marcy Sutton, accessibility expert for this tipoff)
- [Ghostlabapp](https://www.vanamco.com/ghostlab/) is a pretty snazzy tool if you’re doing responsive development or checking anything deployed across a ton of devices. It offers synchronized web development, testing, and inspection.
@@ -96,11 +96,11 @@ The CSS tracker shows color-coded rules for used and unused sets.
![eruda gives you a mobile console](https://cdn.css-tricks.com/wp-content/uploads/2017/04/Screen-Shot-2017-04-10-at-10.38.57-AM.png)
-### # Specific Tips ###
+### Specific Tips ###
I am always interested in what other people do to debug, so I asked the community through the CSS-Tricks account and my own what they were really into. This list is a mixture of tips I like as well as a roundup of tips from the community.
-#### # Accessibility ####
+#### Accessibility ####
```
$('body').on('focusin',function(){
@@ -111,7 +111,7 @@ $('body').on('focusin',function(){
-[Marcy Sutton](https://twitter.com/marcysutton)
-#### # Debugging CSS ####
+#### Debugging CSS ####
We got quite a lot of responses saying that people put red borders on elements to see what they’re doing
@@ -121,7 +121,7 @@ We got quite a lot of responses saying that people put red borders on elements t
I do this too, I even have a little CSS file that drops in some classes I can access for different colors easily.
-#### # Checking State in React ####
+#### Checking State in React ####
> [@sarah_edo](https://twitter.com/sarah_edo)
{JSON.stringify(this.state, null, 2)}
>
@@ -129,7 +129,7 @@ I do this too, I even have a little CSS file that drops in some classes I can ac
Props to Michael, this is one of the most useful debugging tools I know of. That snippet "pretty prints" the state of the component you're working with onto the component so that you can see what’s going on. You can validate that the state is working the way that you think it should be, and it helps track down any errors between the state and how you're using it.
-#### # Animation ####
+#### Animation ####
We got a lot of responses that said they slow the animation way down:
@@ -145,7 +145,7 @@ I also slow down my animations in JavaScript- in GreenSock that would look like:
If you want to use the Chrome Devtools timeline to do performance audits, it's worth mentioning that painting is the most expense of the tasks, so all things being equal, pay a little more attention to a high percentage of that green.
-#### # Checking different connection speeds and loads ####
+#### Checking different connection speeds and loads ####
I tend to work on fast connections, so I will throttle my connection to check and see what the performance would look like for people who don’t have my internet speed.
@@ -157,7 +157,7 @@ This is also useful in conjunction with a hard reload, or with the cache empty
>
> — David Corbacho (@dcorbacho) [March 15, 2017](https://twitter.com/dcorbacho/status/842033259664035840)
-#### # Set a Timed Debugger ####
+#### Set a Timed Debugger ####
This one came from Chris. We have a whole writeup on it [right here](https://css-tricks.com/set-timed-debugger-web-inspect-hard-grab-elements/):
@@ -169,7 +169,7 @@ setTimeout(function() {
It’s similar to the debugger; tool I mentioned earlier, except you can put it in a setTimeout function and get even more fine-tuned information
-#### # Simulators ####
+#### Simulators ####
> [@Real_CSS_Tricks](https://twitter.com/Real_CSS_Tricks) And just in case any Mac users didn't know this, iOS simulator + Safari is sweet. [pic.twitter.com/Uz4XO3e6uD](https://t.co/Uz4XO3e6uD)
>
@@ -183,7 +183,7 @@ I mentioned simulators with Eruda before. iOS users also get a pretty sweet simu
Chrome also has a device toggle which is helpful.
-#### # Remote Debuggers ####
+#### Remote Debuggers ####
> [@chriscoyier](https://twitter.com/chriscoyier)[@Real_CSS_Tricks](https://twitter.com/Real_CSS_Tricks)[https://t.co/q3OfWKNlUo](https://t.co/q3OfWKNlUo) is a good tool.
>
@@ -191,7 +191,7 @@ Chrome also has a device toggle which is helpful.
I actually didn't know about this tool until seeing this tweet. Pretty useful!
-#### # CSS Grid Debugging #### (#article-header-id-18 .has-header-link)
+#### CSS Grid Debugging #### (#article-header-id-18 .has-header-link)
Rachel Andrew gave a presentation at Smashing and mentioned a little waffle thing you can click on in Firefox that will illuminate the gutters in the grid. [Her video](http://gridbyexample.com/learn/2016/12/17/learning-grid-day17/) explains it really eloquently.
@@ -199,7 +199,7 @@ Rachel Andrew gave a presentation at Smashing and mentioned a little waffle thin
Rachel Andrew shows how to highlight gutters in Firefox DevTools.
-#### # Array Debugging ####
+#### Array Debugging ####
Wes Bos with a really useful tip for searching for a single item in an array:
@@ -207,7 +207,7 @@ Wes Bos with a really useful tip for searching for a single item in an array:
>
> — Wes Bos (@wesbos) [March 15, 2017](https://twitter.com/wesbos/status/842069915158884354)
-### # Further Debugging Resources ###
+### Further Debugging Resources ###
Jon Kuperman has a [Frontend Masters course](https://frontendmasters.com/courses/chrome-dev-tools/) that can help you master devtools it goes along [with this app](https://github.com/jkup/mastering-chrome-devtools).
From 74a1cd716a7c51e6ed8e88571222cc79905207bc Mon Sep 17 00:00:00 2001
From: zhuzi
Date: Sat, 15 Apr 2017 15:04:50 +0800
Subject: [PATCH 238/945] translation completed
---
TODO/secure-web-app-http-headers.md | 40 +++++++++++++++++------------
1 file changed, 23 insertions(+), 17 deletions(-)
diff --git a/TODO/secure-web-app-http-headers.md b/TODO/secure-web-app-http-headers.md
index d15fe288045..9e1716b7e1f 100644
--- a/TODO/secure-web-app-http-headers.md
+++ b/TODO/secure-web-app-http-headers.md
@@ -292,39 +292,45 @@ functionrequestHandler(req, res){
res.setHeader('Content-Security-Policy',"script-src 'self'");}
```
-### Preventing Content-Type Sniffing ###
+### 防止 Content-Type 嗅探 ###
-In an effort to make the user experience as seamless as possible, many browsers have implemented a feature called content-type sniffing, or MIME sniffing. This feature enables the browser to detect the type of a resource provided as part of an HTTP response by “sniffing” the actual resource bits, regardless of the resource type declared through the `Content-Type` response header. While this feature is indeed useful in some cases, it introduces a vulnerability and an attack vector known as a MIME confusion attack. A MIME-sniffing vulnerability enables an attacker to inject a malicious resource, such as a malicious executable script, masquerading as an innocent resource, such as an image. With MIME sniffing, the browser will ignore the declared image content type, and instead of rendering an image will execute the malicious script.
+> In an effort to make the user experience as seamless as possible, many browsers have implemented a feature called content-type sniffing, or MIME sniffing. This feature enables the browser to detect the type of a resource provided as part of an HTTP response by “sniffing” the actual resource bits, regardless of the resource type declared through the `Content-Type` response header. While this feature is indeed useful in some cases, it introduces a vulnerability and an attack vector known as a MIME confusion attack. A MIME-sniffing vulnerability enables an attacker to inject a malicious resource, such as a malicious executable script, masquerading as an innocent resource, such as an image. With MIME sniffing, the browser will ignore the declared image content type, and instead of rendering an image will execute the malicious script.
-Luckily, the `X-Content-Type-Options` response header mitigates this vulnerability! This header, introduced in Internet Explorer 8 back in 2008 and currently supported by most major browsers (Safari is the only major browser not to support it), instructs the browser not to use sniffing when handling fetched resources. Because `X-Content-Type-Options` was only formally specified as part of the [“Fetch” specification](https://fetch.spec.whatwg.org/#x-content-type-options-header), the actual implementation varies across browsers; some (Internet Explorer and Edge) completely avoid MIME sniffing, whereas others (Firefox) still MIME sniff but rather block executable resources (JavaScript and CSS) when an inconsistency between declared and actual types is detected. The latter is in line with the latest Fetch specification.
+为了使用户体验尽可能无缝连接,许多浏览器实现了一个功能叫 内容类型嗅探,或者 MIME 嗅探。这个功能使得浏览器可以通过「嗅探」实际 HTTP 响应的资源的内容直接检测到资源的类型,无论在响应头中 `Content-Type` 是如何指定资源类型的。虽然这个功能在某些情况下确实是有用的,它引入了一个漏洞以及一种叫 MIME 混乱攻击。MIME 嗅探漏洞使攻击者可以注入恶意资源,例如恶意脚本,伪装成一个无辜的资源,例如一个图片。通过 MIME 嗅探,浏览器将忽略声明的图像内容类型,它不会渲染图片,而是执行恶意脚本。
-`X-Content-Type-Options` is a simple response header, with only one directive: `nosniff`. This header looks like this: `X-Content-Type-Options: nosniff`. Here’s an example of a configuration of the header:
+> Luckily, the `X-Content-Type-Options` response header mitigates this vulnerability! This header, introduced in Internet Explorer 8 back in 2008 and currently supported by most major browsers (Safari is the only major browser not to support it), instructs the browser not to use sniffing when handling fetched resources. Because `X-Content-Type-Options` was only formally specified as part of the [“Fetch” specification](https://fetch.spec.whatwg.org/#x-content-type-options-header), the actual implementation varies across browsers; some (Internet Explorer and Edge) completely avoid MIME sniffing, whereas others (Firefox) still MIME sniff but rather block executable resources (JavaScript and CSS) when an inconsistency between declared and actual types is detected. The latter is in line with the latest Fetch specification.
-```
+幸运的是,`X-Content-Type-Options` 响应头缓解了这个漏洞。此响应头在 2008 年引入 IE8,目前大多数主流浏览器都支持(Safari 是唯一不支持的主流浏览器),它指示浏览器在处理获取的资源时不使用嗅探。因为 `X-Content-Type-Options` 仅在 [「Fetch」规范](https://fetch.spec.whatwg.org/#x-content-type-options-header)中正式指定,实际的实现因浏览器而异。一部分浏览器(IE 和 Edge)完全阻止了 MIME 嗅探,而其他一些(Firefox)仍然会进行 MIME 嗅探,但会屏蔽掉可执行的资源(JavaScript 和 CSS)如果声明的内容类型与实际的类型不一致。后者符合最新的 Fetch 规范。
+
+> `X-Content-Type-Options` is a simple response header, with only one directive: `nosniff`. This header looks like this: `X-Content-Type-Options: nosniff`. Here’s an example of a configuration of the header:
+
+`X-Content-Type-Options` 是一个很简单的响应头,它只有一个指令,`nosniff`。它是这样指定的:`X-Content-Type-Options: nosniff`。以下是示例代码:
+
+```javascript
functionrequestHandler(req, res){
res.setHeader('X-Content-Type-Options','nosniff');}
```
-### Summary ###
+### 总结 ###
-In this article, we have seen how to leverage HTTP headers to reinforce the security of your web app, to fend off attacks and to mitigate vulnerabilities.
+> In this article, we have seen how to leverage HTTP headers to reinforce the security of your web app, to fend off attacks and to mitigate vulnerabilities.
-#### Takeaways ####
+本文中,我们了解了如何利用 HTTP 响应头来加强 web 应用的安全性,防止攻击和减轻漏洞。
-- Disable caching for confidential information using the `Cache-Control` header.
-- Enforce HTTPS using the `Strict-Transport-Security` header, and add your domain to Chrome’s preload list.
-- Make your web app more robust against XSS by leveraging the `X-XSS-Protection` header.
-- Block clickjacking using the `X-Frame-Options` header.
-- Leverage `Content-Security-Policy` to whitelist specific sources and endpoints.
-- Prevent MIME-sniffing attacks using the `X-Content-Type-Options` header.
+#### 要点 ####
-Remember that for the web to be truly awesome and engaging, it has to be secure. Leverage HTTP headers to build a more secure web!
+- 使用 `Cache-Control` 禁用对机密信息的缓存
+- 通过 `Strict-Transport-Security` 强制使用 HTTPS,并将你的域添加到 Chrome 预加载列表
+- 利用 `X-XSS-Protection` 使你的 web 应用更加能抵抗 XSS 攻击
+- 使用 `X-Frame-Options` 阻止点击劫持
+- 利用 `Content-Security-Policy` 将特定来源于端点列入白名单
+- 使用 `X-Content-Type-Options` 防止 MIME 嗅探攻击
+请记住,为了使 web 真正迷人,它必须是安全的。利用 HTTP 响应头构建更加安全的网页吧!
-(**Disclaimer:** The content of this post is my own and doesn’t represent my past or current employers in any way whatsoever.)
-*Front page image credits: [Pexels.com](https://www.pexels.com/photo/coffee-writing-computer-blogging-34600/).*
+(**声明:** 此文内容仅属本人,不代表本人过去或现在的雇主。)
---
From 08853d8784ccbda58465da8b99dac09a020019b6 Mon Sep 17 00:00:00 2001
From: zhuzi
Date: Sat, 15 Apr 2017 16:03:58 +0800
Subject: [PATCH 239/945] reviewed
---
TODO/secure-web-app-http-headers.md | 148 ++++++----------------------
1 file changed, 30 insertions(+), 118 deletions(-)
diff --git a/TODO/secure-web-app-http-headers.md b/TODO/secure-web-app-http-headers.md
index 9e1716b7e1f..7cf3e757680 100644
--- a/TODO/secure-web-app-http-headers.md
+++ b/TODO/secure-web-app-http-headers.md
@@ -6,21 +6,13 @@
## 如何使用 HTTP Headers 来保护你的 Web 应用 ##
-> Web applications, be they thin websites or thick single-page apps, are notorious targets for cyber-attacks. In 2016, approximately [40% of data breaches](http://www.verizonenterprise.com/verizon-insights-lab/dbir/2016/) originated from attacks on web apps — the leading attack pattern. Indeed, these days, understanding cyber-security is not a luxury but rather **a necessity for web developers**, especially for developers who build consumer-facing applications.
+Web 应用,无论是简单的小网页还是复杂的单页应用,众所周知都是网络攻击的目标。2016年,大约 40% 的数据泄露源自对 Web 应用的攻击,这是主要的攻击模式。事实上,现在来说,了解网络安全并不是锦上添花, 而是 **Web 开发者的必需任务**,特别对于构建面向消费者的产品的开发人员。
-Web 应用,无论是简单的小网页还是复杂的单页应用,众所周知都是网络攻击的目标。2016年,大约 40% 的数据泄露源自对 Web 应用的攻击,这是主要的攻击模式。事实上,现在来说,了解网络安全并不是锦上添花, 而是 Web 开发者的必需任务,特别对于构建面向消费者的产品的开发人员。
-
-> HTTP response headers can be leveraged to tighten up the security of web apps, typically just by adding a few lines of code. In this article, we’ll show how web developers can use HTTP headers to build secure apps. While the code examples are for Node.js, setting HTTP response headers is supported across all major server-side-rendering platforms and is typically simple to set up.
-
-开发者可以利用 HTTP 响应头来加强 Web 应用程序的安全性,通常只需要添加几行代码即可。本文将结束 web 开发者如何利用 HTTP Headers 来构建安全的应用。虽然本文的示例代码是 Node.js,但是设置 HTTP 响应头基本在所有主要的服务端语言都是简单易设置的。
+开发者可以利用 HTTP 响应头来加强 Web 应用程序的安全性,通常只需要添加几行代码即可。本文将介绍 web 开发者如何利用 HTTP Headers 来构建安全的应用。虽然本文的示例代码是 Node.js,但是设置 HTTP 响应头基本在所有主要的服务端语言中都是简单易设置的。
### 关于 HTTP Headers ###
-> Technically, HTTP headers are simply fields, encoded in clear text, that are part of the HTTP request and response message header. They are designed to enable both the HTTP client and server to send and receive meta data about the connection to be established, the resource being requested, as well as the returned resource itself.
-
-技术上,HTTP 头只是简单的字段,咦明文形式编码,这是 HTTP 请求和响应消息头的一部分。它们旨在使客户端和服务端都能够发送和接受有关要建立的连接元数据、所请求的资源,以及返回的资源本身的元数据。
-
-> Plain-text HTTP response headers can be examined easily using cURL, with the `--head` option, like so:
+技术上来说,HTTP 头只是简单的字段,以明文形式编码,它是 HTTP 请求和响应消息头的一部分。它们旨在使客户端和服务端都能够发送和接受有关要建立的连接、所请求的资源,以及返回的资源本身的元数据。
可以使用 cURL `--head` 选项轻松检查纯文本 HTTP 响应头,例如:
@@ -37,48 +29,30 @@ Vary: Accept-Encoding
…
```
-> Today, hundreds of headers are used by web apps, some standardized by the [Internet Engineering Task Force](https://www.ietf.org/) (IETF), the open organization that is behind many of the standards that power the web as we know it today, and some proprietary. HTTP headers provide a flexible and extensible mechanism that enables the rich and varying use cases found on the web today.
-
-现在,数百种响应头正在被 web 应用所使用,其中一部分由[互联网工程任务组, IETF](https://www.ietf.org/)标准化。IETF 是一个开发性组织,今天我们所熟知的许多 web 标准或专利都是由他们进行推进的。HTTP 头提供了一种灵活可扩展的机制,造就了现今的网络各种丰富多变的用例。
+现在,数百种响应头正在被 web 应用所使用,其中一部分由[互联网工程任务组(IETF)](https://www.ietf.org/)标准化。IETF 是一个开发性组织,今天我们所熟知的许多 web 标准或专利都是由他们推进的。HTTP 头提供了一种灵活可扩展的机制,造就了现今的网络各种丰富多变的用例。
### 机密资源禁用缓存 ###
-> Caching is a valuable and effective technique for optimizing performance in client-server architectures, and HTTP, which leverages caching extensively, is no exception. However, in cases where the cached resource is confidential, caching can lead to vulnerabilities — and must be avoided. As an example, consider a web app that renders and caches a page with sensitive information and is being used on a shared PC. Anyone can view confidential information rendered by that web app simply by visiting the browser’s cache, or sometimes even as easily as clicking the browser’s “back” button!
-
-缓存是优化客户端-服务端架构性能中有效的技术,广泛利用缓存的 HTTP 也不例外。但是,在缓存的资源是保密的情况下,缓存可能导致漏斗,所以必须避免。假设一个 web 应用对含有敏感信息的网页进行缓存,并且是在一台公用的 PC 上使用。任何人可以通过访问浏览器的缓存看到这个 web 应用上的敏感信息,甚至有时仅仅通过点击浏览器的返回按钮就可以看到。
+缓存是优化客户端-服务端架构性能中有效的技术,广泛利用缓存的 HTTP 并不少见。但是,在缓存的资源是保密的情况下,缓存可能导致漏洞,所以必须避免。假设一个 web 应用对含有敏感信息的网页进行缓存,并且是在一台公用的 PC 上使用,任何人可以通过访问浏览器的缓存看到这个 web 应用上的敏感信息,甚至有时仅仅通过点击浏览器的返回按钮就可以看到。
-> The IETF’s [RFC 7234](https://tools.ietf.org/html/rfc7234), which defines HTTP caching, specifies the default behavior of HTTP clients, both browsers and intermediary Internet proxies, to *always* cache responses to HTTP `GET` requests — unless specified otherwise. While this enables HTTP to boost performance and reduce network congestion, it could also expose end users to theft of personal information, as mentioned above. The good news is that the HTTP specification also defines a pretty simple way to instruct clients not to cache a given response, through the use of — you guessed it! — HTTP response headers.
+IETF [RFC 7234](https://tools.ietf.org/html/rfc7234) 上定义了 HTTP 缓存,指定 HTTP 客户端(浏览器以及网络代理)的默认行为,也就是始终缓存对 HTTP GET 请求的响应,除非另行指定。虽然这样可以使 HTTP 提升性能减少网络拥塞,但如上所述,它也有可能使终端用户个人信息被盗。好消息是,HTTP 规范还定义了一种非常简单的方式来指示客户端对特定响应不进行缓存,通过使用 —— 对,你猜到了 —— HTTP 响应头。
-IETF 的 [RFC 7234](https://tools.ietf.org/html/rfc7234) 上定义了 HTTP 缓存,指定 HTTP 客户端(浏览器以及网络代理)的默认行为,也就是始终缓存对 HTTP GET 请求的相应,除非另行指定。虽然这样可以使 HTTP 提升性能减少网络拥塞,但如上所述,它也有可能使终端用户个人信息被盗。好消息是,HTTP 规范还廷议了一种非常简单的方式来指示客户端对特定响应不进行缓存,通过使用 —— 对,你猜到了 —— HTTP 响应头。
-
-> There are three headers to return when you are returning sensitive information and would like to disable caching by HTTP clients:
-
-当你返回敏感信息并希望禁用 HTTP 客户端的缓存时,有三个头可以返回:
+当你返回敏感信息并希望禁用 HTTP 客户端的缓存时,有三个响应头可以返回:
- `Cache-Control`
-> This response header, introduced in HTTP 1.1, may contain one or more directives, each carrying a specific caching semantic, and instructing HTTP clients and proxies on how to treat the response being annotated by the header. My recommendation is to format the header as follows: `cache-control: no-cache, no-store, must-revalidate`. These three directives pretty much instruct clients and intermediary proxies not to use a previously cached response, not to store the response, and that even if the response is somehow cached, the cache must be revalidated on the origin server.
-
从 HTTP 1.1 引入的此响应头可能包含一个或多个指令,每个指令带有特定的缓存语义,指示 HTTP 客户端和代理如何处理有此响应头注释的响应。我推荐如下指定响应头,`cache-control: no-cache, no-store, must-revalidate`。这三个指令基本上可以指示客户端和中间代理不可使用之前缓存的响应,不可存储响应,甚至就算响应被缓存,也必须从源服务器上重新验证。
- `Pragma: no-cache`
-> For backwards-compatibility with HTTP 1.0, you will want to include this header as well. Some HTTP clients, especially intermediary proxies, still might not fully support HTTP 1.1 and so will not correctly handle the `Cache-Control` header mentioned above. Use `Pragma: no-cache` to ensure that these older clients do not cache your response.
-
-为了与 HTTP 1.0 的向后兼容性,你还需要包含此响应头。有部分客户端,特别是中间代理,可能仍然没有完全支持 HTTP 1.1,所以不能正确处理前面提到的 `Cache-Control` 响应头,所以使用 `Pragma: no-cache` 确保较旧的客户端不缓存你的响应。
+为了向后兼容 HTTP 1.0,你还需要包含此响应头。有部分客户端,特别是中间代理,可能仍然没有完全支持 HTTP 1.1,所以不能正确处理前面提到的 `Cache-Control` 响应头,所以使用 `Pragma: no-cache` 确保较旧的客户端不缓存你的响应。
- `Expires: -1`
-> This header specifies a timestamp after which the response is considered stale. By specifying `-1`, instead of an actual future time, you ensure that clients immediately treat this response as stale and avoid caching.
-
-此标头指定了该响应过时的时间戳。如果不指定为未来某个真实时间而指定为 `-1`,可以保证客户端立即将此响应视为过时并避免缓存。
-
-> Note that, while disabling caching enhances the security of your web app and helps to protect confidential information, is does come at the price of a performance hit. Make sure to disable caching only for resources that actually require confidentiality and not just for any response rendered by your server! For a deeper dive into best practices for caching web resources, I highly recommend reading [Jake Archibald’s post](https://jakearchibald.com/2016/caching-best-practices/) on the subject.
+此响应头指定了该响应过时的时间戳。如果不指定为未来某个真实时间而指定为 `-1`,可以保证客户端立即将此响应视为过期并避免缓存。
需要注意的是,禁用缓存提高安全性及保护机密资源的同时,也的确会带来性能上的折损。所以确保仅对实际需要保密性的资源禁用缓存,而不是对任何服务器的响应禁用。想要更深入了解 web 资源缓存的最佳实践,我推荐阅读 [Jake Archibald 的文章](https://jakearchibald.com/2016/caching-best-practices/)。
-> Here’s how you would program these headers in Node.js:
-
下面是 Node.js 中设置响应头的示例代码:
```javascript
@@ -91,49 +65,29 @@ function requestHandler(req, res) {
### 强制 HTTPS ###
-> Today, the importance of HTTPS is widely recognized by the tech community. More and more web apps configure secured endpoints and are redirecting unsecure traffic to secured endpoints (i.e. HTTP to HTTPS redirects). Unfortunately, end users have yet to fully comprehend the importance of HTTPS, and this lack of comprehension exposes them to various man-in-the-middle (MitM) attacks. The typical user navigates to a web app without paying much attention to the protocol being used, be it secure (HTTPS) or unsecure (HTTP). Moreover, many users will just click past browser warnings when their browser presents a certificate error or warning!
-
-今天,HTTPS 的重要性已经得到了科技界的广泛认可。越来越多的 web 应用配置了安全端点,并将不安全网路重定向到安全端点(即 HTTP 重定向至 HTTPS)。不幸的是,终端用户还未完全理解 HTTPS 的重要性,这种缺乏理解使他们面临着各种中间人攻击(MitM)。典型的用户访问到一个 web 应用时,并不会注意到正在使用的网络协议是安全的(HTTPS)还是不安全的(HTTP)。
-
-> The importance of interacting with web apps over a valid HTTPS connection cannot be overstated: An unsecure connection exposes the user to various attacks, which could lead to cookie theft or worse. As an example, it is not very difficult for an attacker to spoof network frames within a public Wi-Fi network and to extract the session cookies of users who are not using HTTPS. To make things even worse, even users interacting with a web app over a secured connection may be exposed to downgrade attacks, which try to force the connection to be downgraded to an unsecure connection, thus exposing the user to MitM attacks.
-
-通过有效的 HTTPS 连接与 web 应用进行交互的重要性怎么说都不算夸大:不安全的连接将用户暴露给各种攻击,这可能导致 cookie 被盗甚至更糟。举个例子,攻击者可以轻易在公共 Wi-Fi 网络下骗过网络帧并提起不使用 HTTPS 的用户的会话 cookie。更糟的情况是,即使用户通过安全连接与 web 永盈进行交互也可能遭受降级攻击,这种攻击试图强制将连接降级到不安全的连接,从而是用户收到中间人攻击。
-
-> How can we help users avoid these attacks and better enforce the usage of HTTPS? Enter the HTTP Strict Transport Security (HSTS) header. Put simply, HSTS makes sure all communications with the origin host are using HTTPS. Specified in [RFC 6797](https://tools.ietf.org/html/rfc6797), HSTS enables a web app to instruct browsers to allow *only* HTTPS connections to the origin host, to internally redirect all unsecure traffic to secured connections, and to automatically upgrade all unsecure resource requests to be secure.
+今天,HTTPS 的重要性已经得到了科技界的广泛认可。越来越多的 web 应用配置了安全端点,并将不安全网路重定向到安全端点(即 HTTP 重定向至 HTTPS)。不幸的是,终端用户还未完全理解 HTTPS 的重要性,这种缺乏理解使他们面临着各种中间人攻击(MitM)。普通用户访问到一个 web 应用时,并不会注意到正在使用的网络协议是安全的(HTTPS)还是不安全的(HTTP)。
-我们如何帮助用户避免这些攻击,并更好地实施 HTTPS 的使用呢?使用 HTTP 严格传输安全头(HSTS)。简单来说,HSTS 确保与源主机间的所有通信都使用 HTTPS。[RFC 6797](https://tools.ietf.org/html/rfc6797) 中说明了,HSTS可以使 web 应用程序指示浏览器仅允许与源主机之间的 HTTPS 连接,将所有不安全的连接内部重定向到安全连接,并自动将所有不安全的资源请求升级为安全请求。
+通过有效的 HTTPS 连接与 web 应用进行交互的重要性怎么说都不算夸大:不安全的连接将用户暴露给各种攻击,这可能导致 cookie 被盗甚至更糟。举个例子,攻击者可以轻易在公共 Wi-Fi 网络下骗过网络帧并提取不使用 HTTPS 的用户的会话 cookie。更糟的情况是,即使用户通过安全连接与 web 应用进行交互也可能遭受降级攻击,这种攻击试图强制将连接降级到不安全的连接,从而使用户收到中间人攻击。
-> HSTS directives include the following:
+我们如何帮助用户避免这些攻击,并更好地推行 HTTPS 的使用呢?使用 HTTP 严格传输安全头(HSTS)。简单来说,HSTS 确保与源主机间的所有通信都使用 HTTPS。[RFC 6797](https://tools.ietf.org/html/rfc6797) 中说明了,HSTS可以使 web 应用程序指示浏览器仅允许与源主机之间的 HTTPS 连接,将所有不安全的连接内部重定向到安全连接,并自动将所有不安全的资源请求升级为安全请求。
HSTS 的指令如下:
- `max-age=`
->This instructs the browser to cache this header, for this domain, for the specified number of seconds. This can ensure tightened security for a long duration!
-
此项指示浏览器对此域缓存此响应头指定的秒数。这样可以保证长时间的加固安全。
- `includeSubDomains`
->This instructs the browser to apply HSTS for all subdomains of the current domain. This can be useful to cover all current and future subdomains you may have.
-
-此项指示浏览器对当前域的所有子域应用 HSTS,这可以用于覆盖你可能有的所有当前和未来的子域。
+此项指示浏览器对当前域的所有子域应用 HSTS,这可以用于所有当前和未来可能的子域。
- `preload`
-> This is a powerful directive that forces browsers to *always* load your web app securely, even on the first hit, before the response is even received! This works by hardcoding a list of HSTS preload-enabled domains into the browser’s code. To enable the preloading feature, you need to register your domain with [HSTS Preload List Submission](https://hstspreload.org), a website maintained by Google’s Chrome team. Once registered, the domain will be prebuilt into supporting browsers to always enforce HSTS. The preload directive within the HTTP response header is used to confirm registration, indicating that the web app and domain owner are indeed interested in being on the preload list.
-
-这是一个强大的指令,强制浏览器始终安全加载你的 web 应用程序,即使是第一次收到响应之前加载!这是通过将启用 HSTS 预加载域的列表硬编码到浏览器的代码中实现的。要启用预加载功能,你需要在 Google Chrome 团队维护的网站 [HSTS 预加载列表提交](https://hstspreload.org)注册你的域。
-
-> A word of caution: using the `preload` directive also means it cannot be easily undone, and carries an update lead time of months! While preload certainly improves your app’s security, it also means you need to be fully confident your app can support HTTPS-only!
-
-注意谨慎使用 `preload`,因为这意味着它不能轻易撤销,并带有几个月前的更新。虽然预加载肯定会应用程序的安全性,但也意味着你需要充分确信你的应用程序可以支持仅 HTTPS!
+这是一个强大的指令,强制浏览器**始终**安全加载你的 web 应用程序,即使是第一次收到响应之前加载!这是通过将启用 HSTS 预加载域的列表硬编码到浏览器的代码中实现的。要启用预加载功能,你需要在 Google Chrome 团队维护的网站 [HSTS 预加载列表提交](https://hstspreload.org)注册你的域。
-> My recommendation is to use `Strict-Transport-Security: max-age=31536000; includeSubDomains;` which instructs the browser to enforce a valid HTTPS connection to the origin host and to all subdomains for a year. If you are confident that your app can handle HTTPS-only, I would also recommend adding the `preload` directive, in which case don’t forget to register your website on the preload list as well, as noted above!
+注意谨慎使用 `preload`,因为这意味着它不能轻易撤销,并带有几个月前的更新。虽然预加载肯定会加强应用程序的安全性,但也意味着你需要充分确信你的应用程序可以支持仅 HTTPS!
-我推荐的用法是 `Strict-Transport-Security: max-age=31536000; includeSubDomains;`,这样指示了浏览器强制通过 HTTPS 连接到源主机并且有效期为一年。如果你对你的 app 处理 HTTPS 限定很有信心,我也推荐加上 `preload` 指令,当然别忘记去前面提到的预加载列表注册你的网站。
-
-> Here’s what implementing HSTS looks like in Node.js:
+我建议的用法是 `Strict-Transport-Security: max-age=31536000; includeSubDomains;`,这样指示了浏览器强制通过 HTTPS 连接到源主机并且有效期为一年。如果你对你的 app 处理 HTTPS 限定很有信心,我也推荐加上 `preload` 指令,当然别忘记去前面提到的预加载列表注册你的网站。
以下是在 Nodes.js 中实现 HSTS 的方法:
@@ -145,13 +99,9 @@ function requestHandler(req, res){
### 启用 XSS 过滤 ###
-> In a reflected cross-site scripting attack (reflected XSS), an attacker injects malicious JavaScript code into an HTTP request, with the injected code “reflected” in the response and executed by the browser rendering the response, enabling the malicious code to operate within a trusted context, accessing potentially confidential information such as session cookies. Unfortunately, XSS is a pretty common web app attack, and a surprisingly effective one!
-
-在反射型跨站脚本攻击(reflected XSS)中,攻击者将恶意 JavaScript 代码注入到 HTTP 请求,注入的代码「映射」到响应中,并由浏览器执行,从而使恶意代码在可信任的上下文中执行,访问诸如会话 cookie 的潜在机密信息。不幸的是,XSS 是一个很常见的网络应用程序,且令人惊讶地有效!
-
->To understand a reflected XSS attack, consider the Node.js code below, rendering mywebapp.com, a mock and intentionally simple web app that renders search results alongside the search term requested by the user:
+在反射型跨站脚本攻击(reflected XSS)中,攻击者将恶意 JavaScript 代码注入到 HTTP 请求,注入的代码「映射」到响应中,并由浏览器执行,从而使恶意代码在可信任的上下文中执行,访问诸如会话 cookie 中的潜在机密信息。不幸的是,XSS 是一个很常见的网络应用攻击,且令人惊讶地有效!
-为了了解反射型 XSS 攻击,参考以下 Node.js 代码,渲染 `mywebapp.com` 一个模拟并有意简单的 web 应用程序,它将搜索结果以及用户请求的搜索关键词一起呈现:
+为了了解反射型 XSS 攻击,参考以下 Node.js 代码,渲染 `mywebapp.com`,模拟一个简单的 web 应用程序,它将搜索结果以及用户请求的搜索关键词一起呈现:
```javascript
function handleRequest(req, res) {
@@ -173,25 +123,19 @@ function handleRequest(req, res) {
};
```
->Now, consider how will the web app above handle a URL constructed with malicious executable code embedded within the URL, such as this:
-
现在,来考虑一下上面的 web 应用程序会如何处理在 URL 中嵌入的恶意可执行代码,例如:
```
https://mywebapp.com/search?
```
-> As you may realize, this URL will make the browser run the injected script and send the user’s cookies, potentially including confidential session cookies, to evil.com!
-
你可能意识到了,这个 URL 会让浏览器执行注入的脚本,并发送用户的 cookies,极有可能包含机密的会话 cookie,至 evil.com。
-> To help protect users against reflective XSS attacks, some browsers have implemented protection mechanisms. These mechanisms try to identify these attacks by looking for matching code patterns in the HTTP request and response. Internet Explorer was the first browser to introduce such a mechanism with its XSS filter, introduced in Internet Explorer 8 back in 2008, and WebKit later introduced XSS Auditor, available today in Chrome and Safari. (Firefox has no similar mechanism built in, but users can use add-ons to gain this functionality.) These various protection mechanisms are not perfect: They may fail to detect a real XSS attack (a false negative), and in other cases may block legitimate code (a false positive). Due to the latter, browsers allow users to disable the XSS filter via the settings. Unfortunately, this is typically a global setting, which turns off this security feature completely for all web apps loaded by the browser.
-
-为了帮助保护用户抵抗反射型 XSS 攻击,有些浏览器实施了保护机制。这些保护机制尝试通过在 HTTP 请求和响应中寻找匹配的代码模式来辨识这些攻击。Internet Explorer 是第一个推出这种机制的,在 2008 年的 IE 8 中引入了 XSS 过滤器的机制,而 WebKit 后来推出了 XSS 审计,现今在 Chrome 和 Safari 上可用。(Firefox 没有内置类似的机制,但是用户可以使用插件来获得此功能)。这些保护机制并不完美,它们可能无法检测到真正的 XSS 攻击(漏报),在其他情况可能会阻止合法代码(误判)。由于后一种情况的出现,浏览器允许用户可设置禁用 XSS 过滤功能。不幸的是,这通常是一个全局设置,这会完全关闭所有浏览器加载的 web 应用程序的安全功能。
+为了保护用户抵抗反射型 XSS 攻击,有些浏览器实施了保护机制。这些保护机制尝试通过在 HTTP 请求和响应中寻找匹配的代码模式来辨识这些攻击。Internet Explorer 是第一个推出这种机制的,在 2008 年的 IE 8 中引入了 XSS 过滤器的机制,而 WebKit 后来推出了 XSS 审计,现今在 Chrome 和 Safari 上可用(Firefox 没有内置类似的机制,但是用户可以使用插件来获得此功能)。这些保护机制并不完美,它们可能无法检测到真正的 XSS 攻击(漏报),在其他情况可能会阻止合法代码(误判)。由于后一种情况的出现,浏览器允许用户可设置禁用 XSS 过滤功能。不幸的是,这通常是一个全局设置,这会完全关闭所有浏览器加载的 web 应用程序的安全功能。
> Luckily, there is a way for a web app to override this configuration and ensure that the XSS filter is turned on for the web app being loaded by the browser. This is done via the `X-XSS-Protection` header. This header, supported by Internet Explorer (from version 8), Edge, Chrome and Safari, instructs the browser to turn on or off the browser’s built-in protection mechanism and to override the browser’s local configuration.
-幸运的是,有方法可以让 web 应用程序覆盖此配置,并确保浏览器加载的 web 应用已打开 XSS 过滤器。这是通过设定 `X-XSS-Protection` 响应头来达到的。此响应头支持 Internet Explorer (8以上)、Edge、Chrome 和 Safar,指示浏览器打开或关闭浏览器内置的保护机制,及覆盖浏览器的本地配置。
+幸运的是,有方法可以让 web 应用覆盖此配置,并确保浏览器加载的 web 应用已打开 XSS 过滤器。这是通过设定 `X-XSS-Protection` 响应头来达到的。此响应头支持 Internet Explorer (8以上)、Edge、Chrome 和 Safari,指示浏览器打开或关闭内置的保护机制,及覆盖浏览器的本地配置。
`X-XSS-Protection` 指令包括:
@@ -200,13 +144,11 @@ https://mywebapp.com/search?
```
-你可能意识到了,这个 URL 会让浏览器执行注入的脚本,并发送用户的 cookies,极有可能包含机密的会话 cookie,至 evil.com。
+你可能意识到了,这个 URL 会让浏览器执行注入的脚本,并发送极有可能包含机密会话的用户 cookies 到 evil.com。
为了保护用户抵抗反射型 XSS 攻击,有些浏览器实施了保护机制。这些保护机制尝试通过在 HTTP 请求和响应中寻找匹配的代码模式来辨识这些攻击。Internet Explorer 是第一个推出这种机制的,在 2008 年的 IE 8 中引入了 XSS 过滤器的机制,而 WebKit 后来推出了 XSS 审计,现今在 Chrome 和 Safari 上可用(Firefox 没有内置类似的机制,但是用户可以使用插件来获得此功能)。这些保护机制并不完美,它们可能无法检测到真正的 XSS 攻击(漏报),在其他情况可能会阻止合法代码(误判)。由于后一种情况的出现,浏览器允许用户可设置禁用 XSS 过滤功能。不幸的是,这通常是一个全局设置,这会完全关闭所有浏览器加载的 web 应用程序的安全功能。
-> Luckily, there is a way for a web app to override this configuration and ensure that the XSS filter is turned on for the web app being loaded by the browser. This is done via the `X-XSS-Protection` header. This header, supported by Internet Explorer (from version 8), Edge, Chrome and Safari, instructs the browser to turn on or off the browser’s built-in protection mechanism and to override the browser’s local configuration.
-
-幸运的是,有方法可以让 web 应用覆盖此配置,并确保浏览器加载的 web 应用已打开 XSS 过滤器。这是通过设定 `X-XSS-Protection` 响应头来达到的。此响应头支持 Internet Explorer (8以上)、Edge、Chrome 和 Safari,指示浏览器打开或关闭内置的保护机制,及覆盖浏览器的本地配置。
+幸运的是,有方法可以让 web 应用覆盖此配置,并确保浏览器加载的 web 应用已打开 XSS 过滤器。即通过设定 `X-XSS-Protection` 响应头实现。此响应头支持 Internet Explorer (8以上)、Edge、Chrome 和 Safari,指示浏览器打开或关闭内置的保护机制,及覆盖浏览器的本地配置。
`X-XSS-Protection` 指令包括:
- `1` 或者 `0`
-使用或禁用 CSS 过滤器。
+使用或禁用 XSS 过滤器。
- `mode=block`
当检测到 XSS 攻击时,这会指示浏览器不渲染整个页面。
-> I recommend always turning on the XSS filter, as well as block mode, to maximize user protection. Such a response header looks like this:
-
我建议永远打开 XSS 过滤器以及 block 模式,以求最大化保护用户。这样的响应头应该是这样的:
```
@@ -157,13 +153,13 @@ X-XSS-Protection: 1; mode=block
以下是在 Node.js 中配置此响应头的方法:
```javascript
-functionrequestHandler(req, res){
+function requestHandler(req, res){
res.setHeader('X-XSS-Protection','1;mode=block');}
```
### 控制 iframe ###
-iframe (正式来说,是 HTML 内联框架元素)是一个 DOM 元素,它允许一个 web 应用嵌套在另一个 web 应用中。这个强大的元素有部分重要的使用场景,比如在 web 应用中嵌入第三方内容,但它也有重大的缺点,例如对 SEO 不友好,对浏览器导航跳转也不友好,还有很多。
+iframe (正式来说,是 HTML 内联框架元素)是一个 DOM 元素,它允许一个 web 应用嵌套在另一个 web 应用中。这个强大的元素有部分重要的使用场景,比如在 web 应用中嵌入第三方内容,但它也有重大的缺点,例如对 SEO 不友好,对浏览器导航跳转也不友好等等。
其中一个需要注意的事是它使得点击劫持变得更加容易。点击劫持是一种诱使用户点击并非他们想要点击的目标的攻击。要理解一个简单的劫持实现,参考以下 HTML,当用户认为他们点击可以获得奖品时,实际上是试图欺骗用户购买面包机。
@@ -176,11 +172,11 @@ iframe (正式来说,是 HTML 内联框架元素)是一个 DOM 元素,
```
-点击劫持有许多恶意应用程序,例如欺骗用户确认 Facebook 点赞,在线购买商品,甚至提交机密信息。恶意 web 应用程序可以通过在其恶意应用中嵌入合法的 web 应用来利用 iframe 进行点击劫持,这可以通过设置 `opacity: 0` 的 CSS 规则将其隐藏,并将 iframe 的点击目标直接放置在看起来无辜的按钮之上。点击了这个无辜按钮的用户会直接点击在嵌入的 web 应用上,并不知道点击后的作用。
+有许多恶意应用程序都采用了点击劫持,例如诱导用户点赞,在线购买商品,甚至提交机密信息。恶意 web 应用程序可以通过在其恶意应用中嵌入合法的 web 应用来利用 iframe 进行点击劫持,这可以通过设置 `opacity: 0` 的 CSS 规则将其隐藏,并将 iframe 的点击目标直接放置在看起来无辜的按钮之上。点击了这个无害按钮的用户会直接点击在嵌入的 web 应用上,并不知道点击后的后果。
-阻止这种攻击的一种有效的方法是限制你的 web 应用被框架化。`X-Frame-Options`,在 [RFC 7034](https://www.ietf.org/rfc/rfc7034.txt) 中引入,就是设计用来做这件事的。此响应头指示浏览器对你的 web 应用是否可以被嵌入另一个网页进行限制,从而阻止恶意网页欺骗用户调用你的应用程序进行各项操作。你可以使用 `DENY` 完全屏蔽,或者使用 `ALLOW-FROM` 指定将特定域列入白名单,也可以使用 `SAMEORIGIN` 指令将应用的源地址列入白名单。
+阻止这种攻击的一种有效的方法是限制你的 web 应用被框架化。在 [RFC 7034](https://www.ietf.org/rfc/rfc7034.txt) 中引入的 `X-Frame-Options`,就是设计用来做这件事的。此响应头指示浏览器对你的 web 应用是否可以被嵌入另一个网页进行限制,从而阻止恶意网页欺骗用户调用你的应用程序进行各项操作。你可以使用 `DENY` 完全屏蔽,或者使用 `ALLOW-FROM` 指令将特定域列入白名单,也可以使用 `SAMEORIGIN` 指令将应用的源地址列入白名单。
-我的建议是使用 `SAMEORIGIN` 指令,因为它允许 iframe 被用于同一域上的可以保证安全性的应用程序,这有时是有用的。以下是响应头的示例:
+我的建议是使用 `SAMEORIGIN` 指令,因为它允许 iframe 被同域的应用程序所使用,这有时是有用的。以下是响应头的示例:
```
X-Frame-Options: SAMEORIGIN
@@ -189,7 +185,7 @@ X-Frame-Options: SAMEORIGIN
以下是在 Node.js 中设置此响应头的示例代码:
```javascript
-functionrequestHandler(req, res){
+function requestHandler(req, res){
res.setHeader('X-Frame-Options','SAMEORIGIN');}
```
@@ -197,31 +193,31 @@ functionrequestHandler(req, res){
如前所述,你可以通过启用浏览器的 XSS 过滤器,给你的 web 应用程序增强安全性。然而请注意,这种机制是有局限性的,不是所有浏览器都支持(例如 Firefox 就不支持 XSS 过滤),并且依赖的模式匹配技术可以被欺骗。
-对抗 XSS 和其他攻击的更多一层的保护,可以通过明确列出可信来源和操作来实现 —— 这就是内容安全策略(CSP)。
+对抗 XSS 和其他攻击的另一层的保护,可以通过明确列出可信来源和操作来实现 —— 这就是内容安全策略(CSP)。
-CSP 是一种 W3C 规范,它定义了强大的基于浏览器的安全机制,可以对 web 应用中的资源加载以及脚本执行进行精细的控制。使用 CSP 可以将特定的域加入白名单进行例如脚本加载、AJAX 调用、图像加载和样式加载。你可以启用或禁用内联脚本或动态脚本(臭名昭著的 `eval`),并通过将特定域列入白名单来控制框架化。CSP 的另一个很酷的功能是它允许配置实时报告目标,以便实时监控应用程序进行 CSP 阻止操作。
+CSP 是一种 W3C 规范,它定义了强大的基于浏览器的安全机制,可以对 web 应用中的资源加载以及脚本执行进行精细的控制。使用 CSP 可以将特定的域加入白名单进行脚本加载、AJAX 调用、图像加载和样式加载等操作。你可以启用或禁用内联脚本或动态脚本(臭名昭著的 `eval`),并通过将特定域列入白名单来控制框架化。CSP 的另一个很酷的功能是它允许配置实时报告目标,以便实时监控应用程序进行 CSP 阻止操作。
这种对资源加载和脚本执行的明确的白名单提供了很强的安全性,在很多情况下都可以防范攻击。例如,使用 CSP 禁止内联脚本,你可以防范很多反射型 XSS 攻击,因为它们依赖于将内联脚本注入到 DOM。
CSP 是一个相对复杂的响应头,它有很多种指令,在这里我不详细展开了,可以参考 HTML5 Rocks 里一篇很棒的[教程](https://www.html5rocks.com/en/tutorials/security/content-security-policy/),其中提供了 CSP 的概述,我非常推荐阅读它来学习如何在你的 web 应用中使用 CSP。
-以下是一个设置 CSP 的示例代码,它仅允许从应用程序的源域加载脚本,并组织动态脚本的执行(eval)以及内嵌脚本(当然,还是 Node.js):
+以下是一个设置 CSP 的示例代码,它仅允许从应用程序的源域加载脚本,并阻止动态脚本的执行(eval)以及内嵌脚本(当然,还是 Node.js):
```javascript
-functionrequestHandler(req, res){
+function requestHandler(req, res){
res.setHeader('Content-Security-Policy',"script-src 'self'");}
```
### 防止 Content-Type 嗅探 ###
-为了使用户体验尽可能无缝,许多浏览器实现了一个功能叫内容类型嗅探,或者 MIME 嗅探。这个功能使得浏览器可以通过「嗅探」实际 HTTP 响应的资源的内容直接检测到资源的类型,无论在响应头中 `Content-Type` 是如何指定资源类型的。虽然这个功能在某些情况下确实是有用的,它引入了一个漏洞以及一种叫 MIME 混乱攻击的攻击手法。MIME 嗅探漏洞使攻击者可以注入恶意资源,例如恶意脚本,伪装成一个无辜的资源,例如一个图片。通过 MIME 嗅探,浏览器将忽略声明的图像内容类型,它不会渲染图片,而是执行恶意脚本。
+为了使用户体验尽可能无缝,许多浏览器实现了一个功能叫内容类型嗅探,或者 MIME 嗅探。这个功能使得浏览器可以通过「嗅探」实际 HTTP 响应的资源的内容直接检测到资源的类型,无视响应头中 `Content-Type` 指定的资源类型。虽然这个功能在某些情况下确实是有用的,它引入了一个漏洞以及一种叫 MIME 类型混淆攻击的攻击手法。MIME 嗅探漏洞使攻击者可以注入恶意资源,例如恶意脚本,伪装成一个无害的资源,例如一张图片。通过 MIME 嗅探,浏览器将忽略声明的图像内容类型,它不会渲染图片,而是执行恶意脚本。
幸运的是,`X-Content-Type-Options` 响应头缓解了这个漏洞。此响应头在 2008 年引入 IE8,目前大多数主流浏览器都支持(Safari 是唯一不支持的主流浏览器),它指示浏览器在处理获取的资源时不使用嗅探。因为 `X-Content-Type-Options` 仅在 [「Fetch」规范](https://fetch.spec.whatwg.org/#x-content-type-options-header)中正式指定,实际的实现因浏览器而异。一部分浏览器(IE 和 Edge)完全阻止了 MIME 嗅探,而其他一些(Firefox)仍然会进行 MIME 嗅探,但会屏蔽掉可执行的资源(JavaScript 和 CSS)如果声明的内容类型与实际的类型不一致。后者符合最新的 Fetch 规范。
`X-Content-Type-Options` 是一个很简单的响应头,它只有一个指令,`nosniff`。它是这样指定的:`X-Content-Type-Options: nosniff`。以下是示例代码:
```javascript
-functionrequestHandler(req, res){
+function requestHandler(req, res){
res.setHeader('X-Content-Type-Options','nosniff');}
```
@@ -235,14 +231,16 @@ functionrequestHandler(req, res){
- 通过 `Strict-Transport-Security` 强制使用 HTTPS,并将你的域添加到 Chrome 预加载列表
- 利用 `X-XSS-Protection` 使你的 web 应用更加能抵抗 XSS 攻击
- 使用 `X-Frame-Options` 阻止点击劫持
-- 利用 `Content-Security-Policy` 将特定来源于端点列入白名单
+- 利用 `Content-Security-Policy` 将特定来源与端点列入白名单
- 使用 `X-Content-Type-Options` 防止 MIME 嗅探攻击
请记住,为了使 web 真正迷人,它必须是安全的。利用 HTTP 响应头构建更加安全的网页吧!
-(**声明:** 此文内容仅属本人,不代表本人过去或现在的雇主。)
+(**声明:** 此文内容仅属本人,不代表本人过去或现在的雇主。)
+
+(首页图片版权:[Pexels.com](https://www.pexels.com/photo/coffee-writing-computer-blogging-34600/))
---
From ae3e13d882eb36233da4bcedd7ff9eba72ec6c12 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Mon, 17 Apr 2017 22:45:12 +0800
Subject: [PATCH 257/945] Create testing-views-in-isolation-with-espresso.md
---
...esting-views-in-isolation-with-espresso.md | 170 ++++++++++++++++++
1 file changed, 170 insertions(+)
create mode 100644 TODO/testing-views-in-isolation-with-espresso.md
diff --git a/TODO/testing-views-in-isolation-with-espresso.md b/TODO/testing-views-in-isolation-with-espresso.md
new file mode 100644
index 00000000000..c746cbfcf68
--- /dev/null
+++ b/TODO/testing-views-in-isolation-with-espresso.md
@@ -0,0 +1,170 @@
+> * 原文地址:[Testing Views in Isolation with Espresso](https://www.novoda.com/blog/testing-views-in-isolation-with-espresso/)
+> * 原文作者:[Ataul Munim](https://www.novoda.com/blog/author/ataulm/)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+# Testing Views in Isolation with Espresso #
+
+
+In this post, we'll show you how and why you should use Espresso to test your custom Views on an Android device.
+
+You can use Espresso to test entire screens or flows at once. These tests launch an Activity and perform actions as a user would, including waiting for data to load or navigating to other screens.
+
+They're useful because you need end-to-end tests to validate common user flows. These automated tests should run at regular intervals, enabling you to spend manual QA time performing exploratory tests.
+
+That said, these aren’t tests that you can run frequently. Running the whole suite can take hours (imagine a suite that includes verifying offline sync for media content), so you might choose to run them at night.
+
+It's difficult because these types of tests incorporate multiple points of potential failure. Ideally when a single test fails, you want it to fail because of a single logical assertion.
+
+Most (or a lot) of the regressions you can introduce are in the UI. They might be subtle so we don't notice them when adding new features, but eagle-eyed QA teams often do.
+
+It's such a waste of time.
+
+## What can you do? ##
+
+Let's look at using Espresso to test that you have correctly bound data to Views.
+
+At Novoda, the Views we write are mostly extensions of existing View and ViewGroup classes on Android. They typically only expose one or two extra methods, which are used to bind callbacks and the data object/view model, like this:
+
+```
+public class MovieItemView extends RelativeLayout {
+ private TextView titleTextView;
+ private Callback callback;
+
+ public void attach(Callback callback) {
+ this.callback = callback;
+ }
+
+ public void bind(Movie movie) {
+ titleTextView.setText(movie.name());
+ setOnClickListener(new OnClickListener() {
+ @Override
+ public void onClick(View v) {
+ callback.onClick(movie);
+ }
+ });
+ }
+}
+```
+
+They group logical parts of the UI together and often encompass naming conventions from the business domain. You will rarely see ‘raw’ Android Views in Activity layouts in our work at Novoda.
+
+Let’s write these View tests in a BDD style, like "given MovieItemView is bound to Edward Scissorhands, then title is set as Edward Scissorhands" or "given MovieItemView is bound to Edward Scissorhands, when clicking on view, then onClick(Edward Scissorhands) is called", etc.
+
+## Couldn't you have caught these regressions with unit tests? ##
+
+Why do you need Espresso to run these tests if you’re using a presentation pattern like MVP or MVVM, which can be unit-tested?
+
+First, let’s go over the presentation flow and describe what you can test to see how the Espresso tests can augment it.
+
+- Presenters subscribe to data producers that send events
+- Events can be of type `loading`, `idle` or `error`, and may or may not contain data to display
+- Presenters will forward these events to "displayers" (“View” in MVP) using methods like `display(List)`, `displayCachedDataWhileLoading(List)` or `displayEmptyScreen()`, etc.
+- The implementation of displayers will show/hide Android Views and do things like `moviesView.bind(List)`
+
+You can unit-test the presenters completely, verifying that the correct methods on the displayers are called with the correct arguments.
+
+Can you test the displayers in the same way? Yes, you can mock the Android Views and verify you're calling the correct methods on them. It won't be at the correct granularity though:
+
+- your displayer could create or update the adapter of a RecyclerView or ViewPager, but this gives you no assurances of what the item/page is displaying
+- Android Views are setup in code with attributes inflated from XML (layouts and styles); verifying method calls isn't enough to assert what's displayed
+
+## Setting up for the tests ##
+
+Let’s use the [`espresso-support`](https://github.com/novoda/spikes/tree/master/espresso-support) library to get started.
+
+Add the dependencies to your build.gradle file (available from JCenter):
+
+```
+debugCompile 'com.novoda:espresso-support-extras:0.0.3'
+androidTestCompile 'com.novoda:espresso-support:0.0.3'
+```
+
+The `extras` artifact includes the `ViewActivity`, which needs to be part of your app under test. You can use this Activity to hold a single View, which you can test with Espresso.
+
+The core artifact (containing custom test rules) only needs to be included as part of your `androidTest` dependencies.
+
+The `ViewTestRule` is used in a similar way to the `ActivityTestRule`. Instead of passing the Activity class that you want to launch, you should pass the layout file containing the View you want to test:
+
+```
+@RunWith(AndroidJUnit4.class)publicclassMovieItemViewTest{
+ @Rule
+ public ViewTestRule viewTestRule=newViewTestRule<>(R.layout.test_movie_item_view);
+ ...
+```
+
+You can specify the View type for the root of the layout with `ViewTestRule`.
+
+The `ViewTestRule` extends `ActivityTestRule`, so it'll always open `ViewActivity`. `getActivityIntent()` is overridden so you can pass `R.layout.test_movie_item_view` to `ViewActivity` as an Intent extra.
+
+You can use Mockito in your tests to substitute for the callbacks:
+
+```
+@Rule
+public MockitoRule mockitoRule = MockitoJUnit.rule();
+
+@Mock
+MovieItemView.Listener movieItemListener;
+
+@Before
+publicvoidsetUp(){
+ MovieItemView view = viewTestRule.getView();
+ view.attachListener(movieItemListener);
+ ...
+ }
+```
+
+ViewTestRule has a method `bindViewUsing(Binder)`, which gives you a reference to the View so you can interact with it. While you could access the View directly with `viewTestRule.getView()`, you want to ensure any interaction with the View is performed on the main thread, not the test thread.
+
+```
+@Before
+public void setUp() {
+ MovieItemView view = viewTestRule.getView();
+ view.attachListener(movieItemListener);
+ viewTestRule.bindViewUsing(new ViewTestRule.Binder() {
+ @Override
+ public void bind(MovieItemView view) {
+ view.bind(EDWARD_SCISSORHANDS);
+ }
+ });
+}
+```
+
+## Ready to test ##
+
+Apps only really do two things, from the user's point of view:
+
+- they display information
+- they respond to user actions
+
+To write tests for these two cases, you can start by asserting that the correct information is displayed using standard Espresso ViewMatchers and ViewAssertions:
+
+```
+@Test
+public void titleSetToMovieName() {
+ onView(withId(R.id.movie_item_text_name))
+ .check(matches(withText(EDWARD_SCISSORHANDS.name)));
+}
+```
+
+Next, you should make sure that the user actions correspond to the correct event being fired, with the correct arguments:
+
+```
+@Test
+public void clickMovieItemView() {
+ onView(withClassName(is(MovieItemView.class.getName())))
+ .perform(click());
+
+ verify(movieItemListener)
+ .onClick(eq(EDWARD_SCISSORHANDS));
+}
+```
+
+That's it, I hope you find this useful.
+
+In a future post, I'll cover using Espresso to test your Views for TalkBack support.
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 0a04324fc5ff3a0e4ab4691953d980144358ff9e Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Mon, 17 Apr 2017 23:00:36 +0800
Subject: [PATCH 258/945] Create
es6-modules-support-lands-in-browsers-is-it-time-to-rethink-bundling.md
---
...browsers-is-it-time-to-rethink-bundling.md | 335 ++++++++++++++++++
1 file changed, 335 insertions(+)
create mode 100644 TODO/es6-modules-support-lands-in-browsers-is-it-time-to-rethink-bundling.md
diff --git a/TODO/es6-modules-support-lands-in-browsers-is-it-time-to-rethink-bundling.md b/TODO/es6-modules-support-lands-in-browsers-is-it-time-to-rethink-bundling.md
new file mode 100644
index 00000000000..2481f7b2bab
--- /dev/null
+++ b/TODO/es6-modules-support-lands-in-browsers-is-it-time-to-rethink-bundling.md
@@ -0,0 +1,335 @@
+> * 原文地址:[ES6 modules support lands in browsers: is it time to rethink bundling?](https://www.contentful.com/blog/2017/04/04/es6-modules-support-lands-in-browsers-is-it-time-to-rethink-bundling/)
+> * 原文作者:[Stefan Judis](https://www.contentful.com/about-us/)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+# ES6 modules support lands in browsers: is it time to rethink bundling? #
+
+![](http://images.contentful.com/256tjdsmm689/3xFvPzCb6wUek00gQAuU6q/0e8221e0e5c673f18d20448a9ba8924a/Contentful_ES6Modules_.png)
+
+Writing performant JavaScript applications is a complex matter these days. Years ago, everything started with script concatenation to save HTTP requests, and then it continued with minification and wrangling of variable names to squeeze out even the last tiny bit of the code we ship.
+
+Today we have [tree shaking](https://blog.engineyard.com/2016/tree-shaking) and module bundlers, and we go back to code splitting to not block the main thread on startup and speed up [the time to interactivity](https://developers.google.com/web/tools/lighthouse/audits/time-to-interactive). We're also transpiling everything: using future features today? No problem – thanks to Babel!
+
+ES6 modules have been defined in the ECMAScript specification [for a while already](http://2ality.com/2014/09/es6-modules-final.html). The community wrote tons of articles on how to use them with Babel and how `import` differs from `require` in Node.js, but it took a while until an actual implementation landed in browsers. I was surprised to see that Safari was the first one shipping ES6 modules in its technology preview channel, and now Edge and Firefox Nightly also ship this feature – even though it's behind a flag. After having used tools like [RequireJS](http://requirejs.org/) and [Browserify](http://browserify.org/) (remember [the AMD and CommonJS discussions](https://addyosmani.com/writing-modular-js/)?) it looks like modules are finally arriving in the browser landscape, so let's see a look what the bright future will bring. 🎉
+
+
+## The traditional setup ##
+
+The usual way to build web applications is to include one single bundle that is produced using Browserify, Rollup or Webpack (or any other tool out there). A classic website that's not a SPA (single page application) consists of server-side generated HTML, which then includes a single JavaScript bundle.
+
+```
+
+
+ ES6 modules tryout
+
+
+
+
+
+
+
+```
+
+The combined file includes three JavaScript files bundled with Webpack. These files make use of ES6 modules:
+
+```
+// app/index.js
+import dep1 from './dep-1';
+
+function getComponent () {
+ var element = document.createElement('div');
+ element.innerHTML = dep1();
+ return element;
+}
+
+document.body.appendChild(getComponent());
+
+// app/dep-1.js
+import dep2 from './dep-2';
+
+export default function() {
+ return dep2();
+}
+
+// app/dep-2.js
+export default function() {
+ return 'Hello World, dependencies loaded!';
+}
+```
+
+The result of this app will be a "Hello world" telling us that all files are loaded.
+
+### Shipping a bundle ###
+
+The Webpack configuration to create this bundle is relatively straightforward. There is not much happening right now except for the bundling and minification of the JavaScript files using UglifyJS.
+
+```
+// webpack.config.js
+
+const path = require('path');
+const UglifyJSPlugin = require('uglifyjs-webpack-plugin');
+
+module.exports = {
+ entry: './app/index.js',
+ output: {
+ filename: 'bundle.js',
+ path: path.resolve(__dirname, 'dist')
+ },
+ plugins: [
+ new UglifyJSPlugin()
+ ]
+};
+```
+
+The three base files are relatively small and have a total size of 347 bytes.
+
+```
+$ ll app
+total 24
+-rw-r--r-- 1 stefanjudis staff 75B Mar 16 19:33 dep-1.js
+-rw-r--r-- 1 stefanjudis staff 75B Mar 7 21:56 dep-2.js
+-rw-r--r-- 1 stefanjudis staff 197B Mar 16 19:33 index.js
+```
+
+When I ran this through Webpack, I got a bundle with the size of 856 bytes, which is roughly 500 bytes boilerplate. These additional bytes are acceptable, as it's nothing compared to the bundles most of us ship in production. Thanks to Webpack, we can already use ES6 modules.
+
+
+```
+$ webpack
+Hash: 4a237b1d69f142c78884
+Version: webpack 2.2.1
+Time: 114ms
+Asset Size Chunks Chunk Names
+bundle.js 856 bytes 0 [emitted] main
+ [0] ./app/dep-1.js 78 bytes {0}[built]
+ [1] ./app/dep-2.js 75 bytes {0}[built]
+ [2] ./app/index.js 202 bytes {0}[built]
+```
+
+## The new setup using native supported ES6 modules ##
+
+Now that we have the "traditional bundle" for all the browsers that don't support ES6 modules yet, we can start playing around with the cool stuff. To do so, let's add in the `index.html` file a new script element pointing to the ES6 module with `type="module"`.
+
+
+```
+ES6 modules tryout
+```
+
+When we take a look at Chrome, we'll see that there is not much more happening.
+
+![image01](http://images.contentful.com/256tjdsmm689/4JHwnbyrssomECAG2GI8se/e8e35adc37bc0627f0902bcc2fdb52df/image01.png)
+
+The bundle is loaded as before, "Hello world!" is shown, but that's it. And that's excellent, because this is how web works: browsers are forgiving, they won't throw errors when they don't understand markup we send down the wire. Chrome just ignores the script element with the type it doesn't know.
+
+Now, let's check the Safari technology preview:
+
+![Bildschirmfoto 2017-03-29 um 17.06.26](http://images.contentful.com/256tjdsmm689/1mefe0J3JKOiAoSguwMkka/0d76c5666300ed0b631a0fe548ac5b52/Bildschirmfoto_2017-03-29_um_17.06.26.png)
+
+Sadly, there is no additional "Hello world" showing up. The reason is the difference between build tools and native ES modules: whereas Webpack figures out which files to include during the build process, when running ES modules in the browser, we need to define concrete file paths.
+
+```
+// app/index.js
+
+// This needs to be changed
+// import dep1 from './dep-1';
+
+// This works
+import dep1 from './dep-1.js';
+```
+
+The adjusted file paths work great, except for the fact that Safari preview now loads the bundle and the three individual modules, meaning that our code will be executed twice.
+
+![image02](http://images.contentful.com/256tjdsmm689/6MeIDF7GuW6gy8om4Ceccc/a0dba00a4e0f301f2a7fd65449d044ab/image02.png)
+
+The solution is the `nomodule` attribute, which we can set on the script element requesting the bundle. This attribute [was added to the spec quite recently](https://github.com/whatwg/html/commit/a828019152213ae72b0ed2ba8e35b1c472091817) and Safari Preview supports it as of the [end of January](https://trac.webkit.org/changeset/211078/webkit). It tells Safari that this script is the "fallback" script for the lack of ES6 modules support, and in this case shouldn't be executed.
+
+```
+
+
+ ES6 modules tryout
+
+
+
+
+
+
+
+
+
+```
+
+![image03](http://images.contentful.com/256tjdsmm689/1YchZEromA2ueKUCoYqMsc/2c68c46ffd2a3ad73d99d17020d56093/image03.png)
+
+That's good. With the combination of `type="module"` and `nomodule`, we can load a classic bundle in not supporting browsers and load JavaScript modules in supporting browsers.
+
+You can check out this state in production at [es-module-on.stefans-playground.rocks](http://es-module-on.stefans-playground.rocks/).
+
+### Differences between modules and scripts ###
+
+There are a few gotchas here, though. First of all, JavaScript running in an ES6 module is not quite the same as in a regular script element. Axel Rauschmayer covers this quite nicely in [his book Exploring ES6](http://exploringjs.com/es6/ch_modules.html#sec_modules-vs-scripts). I highly recommend you check it out but let's just quickly mention the main differences:
+
+- ES6 modules are running in strict mode by default (no need for `'use strict'` anymore).
+- Top-level value of `this` is `undefined`.
+- Top-level variables are local to the module.
+- ES6 modules are loaded and executed asynchronously after the browser finished parsing the HTML.
+
+In my opinion, these are all huge advantages. Modules are local – there is no need for IIFEs around everything, and also we don't have to fear global variable leaking anymore. Also running in strict mode by default means that we can drop a lot of `'use strict'` statements.
+
+And from a performance point of view (probably the most important one) **modules load and execute deferred by default**. So we won't accidentally add blocking scripts to our website and there is no [SPOF](https://www.stevesouders.com/blog/2010/06/01/frontend-spof/) issue when dealing with script `type="module"` elements. We could place an `async` attribute on it, which overwrites the default deferred behavior, but `defer`[is a good choice these days](https://calendar.perfplanet.com/2016/prefer-defer-over-async/).
+
+```
+
+
+
+
+
+
+
+
+```
+
+In case you want to check the details around that, the [script element spec](https://html.spec.whatwg.org/multipage/scripting.html#the-script-element) is an understandable read and includes some examples.
+
+## Minifying of pure ES6 ##
+
+But we're not quite there yet! We serve a minified bundle for Chrome and individual not minified files for Safari Preview now. How can we make these smaller? UglifyJS should do the job just fine, right?
+
+It turns out that UglifyJS is not able to fully deal with ES6 code yet. There is a `harmony` development branch available, but unfortunately it didn't work with my three JavaScript files at the time of writing.
+
+```
+$ uglifyjs dep-1.js -o dep-1.min.js
+Parse error at dep-1.js:3,23
+export default function() {
+ ^
+SyntaxError: Unexpected token: punc (()
+// ..
+FAIL: 1
+```
+
+
+But UglifyJS is in every toolchain today, how does this work for all the projects written in ES6 out there?
+
+The usual flow is that tools like Babel transpile to ES5, and then Uglify comes into play to minify this ES5 code. I want to ignore ES5 transpilation in this article: we're dealing with the future here, Chrome has [97% ES6 coverage](https://kangax.github.io/compat-table/es6/#chrome59) and Safari Preview has already fabulous [100% ES6 coverage since version 10](https://kangax.github.io/compat-table/es6/#safari10_1).
+
+I asked the Twittersphere if there is a minifier available that can deal with ES6, and [Lars Graubner](https://twitter.com/larsgraubner) pointed me towards [Babili](https://github.com/babel/babili). Using Babili, we can easily minify the ES6 modules.
+
+
+```
+// app/dep-2.js
+
+export default function() {
+ return 'Hello World. dependencies loaded.';
+}
+
+// dist/modules/dep-2.js
+export default function(){return 'Hello World. dependencies loaded.'}
+```
+
+With the Babili CLI tool, it's almost too easy to minify all the files separately.
+
+```
+$ babili app -d dist/modules
+app/dep-1.js -> dist/modules/dep-1.js
+app/dep-2.js -> dist/modules/dep-2.js
+app/index.js -> dist/modules/index.js
+```
+
+The result looks then as follows.
+
+```
+$ ll dist
+-rw-r--r-- 1 stefanjudis staff 856B Mar 16 22:32 bundle.js
+
+$ ll dist/modules
+-rw-r--r-- 1 stefanjudis staff 69B Mar 16 22:32 dep-1.js
+-rw-r--r-- 1 stefanjudis staff 68B Mar 16 22:32 dep-2.js
+-rw-r--r-- 1 stefanjudis staff 161B Mar 16 22:32 index.js
+```
+
+The bundle is still roughly around 850B, and all the files are around 300B in total. I'm ignoring GZIP compression here as [it doesn't work well on such small file sizes](http://webmasters.stackexchange.com/questions/31750/what-is-recommended-minimum-object-size-for-gzip-performance-benefits) (we'll get back to that later).
+
+## Speeding up ES6 modules with rel=preload? ##
+
+The minification of the single JS files is a huge success. It's 298B vs. 856B, but we could even go further and speed things up more. Using ES6 modules we are now able to ship less code, but looking at the waterfall again we'll see that the requests are made sequentially because of the defined dependency chain of the modules.
+
+What if we could throw `` elements in the mix which can be used to tell the browser upfront that additionally requests will be made soon? We have build tool plugins like Addy Osmani's [Webpack preload plugin](https://github.com/GoogleChrome/preload-webpack-plugin) for code splitting already – is something like this possible for ES6 modules? In case you don't know how `rel="preload"` works, you should check out the [article on this topic](https://www.smashingmagazine.com/2016/02/preload-what-is-it-good-for/) by Yoav Weiss on Smashing Magazine.
+
+Unfortunately, preloading of ES6 modules is not so easy because they behave differently than normal scripts. The question is how a link element with a set `rel="preload"` attribute should treat an ES6 module? Should it fetch all the dependent files, too? This is an obvious question to answer, but there are more browser internal problems to solve, too, if module treatment should go into the `preload` directive. In case you're interested in this topic [Domenic Denicola](https://twitter.com/domenic) discusses these problems [in a GitHub issue](https://github.com/whatwg/fetch/issues/486), but it turns out that there are too many differences between scripts and modules to implement ES6 module treatment in the `rel="preload"` directive. The solution might be another `rel="modulepreload"` directive to clearly separate functionalities, with [the spec pull request](https://github.com/whatwg/html/pull/2383) pending at the time of writing, so let's see how we'll preload modules in the future.
+
+## Bringing in real dependencies ##
+
+Three files don't make a real app, so let's add a real dependency. Fortunately, [Lodash](https://lodash.com/) offers all of its functionality also in split ES6 modules, which I then minified using Babili. So let's modify the `index.js` file to also include a Lodash method.
+
+
+```
+import dep1 from './dep-1.js';
+import isEmpty from './lodash/isEmpty.js';
+
+function getComponent() {
+ const element = document.createElement('div');
+ element.innerHTML = dep1() + ' ' + isEmpty([]);
+
+ return element;
+}
+
+document.body.appendChild(getComponent());
+```
+
+The use of `isEmpty` is trivial in this case, but let's see what happens now after adding this dependency.
+
+![image07](http://images.contentful.com/256tjdsmm689/13F95Xpl32Mu0MgE0mgS2o/c9dbc002e53bf56ee0eeb0df40b55f9c/image07.png)
+
+The request count went up to over 40, the page load time went up from roughly 100ms to something between 400ms and 800ms on a decent wifi connection, and the shipped overall size increased to approximately 12KB without compression. Unfortunately, Safari Preview is not available on [WebPagetest](https://www.webpagetest.org/) to run some reliable benchmarks.
+
+Chrome receiving the bundled JavaScript, on the other hand, is at a slim ~8KB file size.
+
+![image05](http://images.contentful.com/256tjdsmm689/6xxfWBW9nqAeqQ8ck0MqU/62a74102e9247d785a61a84766356f51/image05.png)
+
+This 4KB difference is definitely something to check. You can find this example at [lodash-module-on.stefans-playground.rocks](https://lodash-module-on.stefans-playground.rocks/).
+
+### Compression works only well on larger files ###
+
+In case you looked closely at the screenshots of the Safari developer tools, you might have noticed that the transferred file size was actually bigger than the source. Especially in a large JavaScript app, including a lot of small chunks makes a big difference and that's because GZIP doesn't play well with small file sizes.
+
+Khan Academy [discovered the same thing](http://engineering.khanacademy.org/posts/js-packaging-http2.htm) a while ago when experimenting with HTTP/2. The idea of shipping smaller files is great to guarantee perfect cache hit ratios, but at the end, it's always a tradeoff and it's depending on several factors. For a large code base splitting the code into several chunks (a *vendor* and an *app* bundle) makes sense, but shipping thousands of tiny files that can't be compressed properly is not the right approach.
+
+### Tree shaking is the cool kid in town ###
+
+Another thing to point out is that thanks to the relatively new tree shaking mechanism, build processes can eliminate code that's not used and imported by any other module. The first build tool that supported this was Rollup, but now Webpack in version 2 supports it as well — [as long as we disable the `module` option in babel](https://medium.freecodecamp.com/tree-shaking-es6-modules-in-webpack-2-1add6672f31b#22c4).
+
+Let's say we changed `dep-2.js` to include things that won't be imported by `dep-1.js`.
+
+```
+export default function() {
+ return 'Hello World. dependencies loaded.';
+}
+
+export const unneededStuff = [
+ 'unneeded stuff'
+];
+```
+
+Babili will simply minify the file and Safari Preview, in this case, would receive several code lines that are not used. A Webpack or Rollup bundle, on the other hand, won't include `unneededStuff`. Tree shaking offers huge savings that definitely should be used in a real production code base.
+
+## The future looks bright, but build processes are here to stay ##
+
+So, ES6 modules are on their way, but it doesn't look like anything will change when they finally arrive in all the major browsers. We won't start shipping thousands of tiny files to guarantee good compression, and we won't abandon build processes to make use of tree shaking and dead code elimination. **Frontend development is and will be as complicated as always**.
+
+**The most important thing to remember is that measuring is the key to succees**. Don't split everything and assume that it will lead to an improvement. Just because we might have support for ES6 modules in browsers soon, it doesn't mean that we can get rid of a build process and a proper "bundle strategy". Here at Contentful we'll stick to our build processes, and continue to ship bundles including our [JavaScript SDKs](https://www.contentful.com/developers/docs/javascript/).
+
+Yet, I have to admit that Frontend development still feels great. JavaScript evolves, and we'll finally have a way to deal with modules baked into the language. I can't wait to see how and if this influences the JavaScript ecosystem and what the best practices will be in a couple of years.
+
+## Additional resources ##
+
+- [Article series on ES6 modules](https://blog.hospodarets.com/native-ecmascript-modules-the-first-overview) by Serg Hospodarets
+- [The modules chapter](http://exploringjs.com/es6/ch_modules.html) in "[Exploring ES6](http://exploringjs.com/)"
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 2161ad5a7ef2005ccb44401457ae4ea8adbf02be Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Mon, 17 Apr 2017 23:12:26 +0800
Subject: [PATCH 259/945] Create everyone-is-a-designer-get-over-it.md
---
TODO/everyone-is-a-designer-get-over-it.md | 85 ++++++++++++++++++++++
1 file changed, 85 insertions(+)
create mode 100644 TODO/everyone-is-a-designer-get-over-it.md
diff --git a/TODO/everyone-is-a-designer-get-over-it.md b/TODO/everyone-is-a-designer-get-over-it.md
new file mode 100644
index 00000000000..aee8b2663cd
--- /dev/null
+++ b/TODO/everyone-is-a-designer-get-over-it.md
@@ -0,0 +1,85 @@
+> * 原文地址:[Everyone is a designer. Get over it.](https://library.gv.com/everyone-is-a-designer-get-over-it-501cc9a2f434)
+> * 原文作者:[Daniel Burka](https://library.gv.com/@dburka)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+
+![](https://cdn-images-1.medium.com/max/2000/1*xIoFsnWI_2-1VOy00a2KrQ.jpeg)
+
+Photo by Alice Achterhof [via Unsplash](https://unsplash.com/search/designer-paint?photo=FwF_fKj5tBo)
+
+# Everyone is a designer. Get over it. #
+
+Recently, [Jared Spool](https://www.uie.com/about/) caught my attention with [an article](https://articles.uie.com/signup/) about how Netflix’s performance engineers are *actually designers*. It’s a provocative idea, but it makes sense. His argument is that everyone in your organization (including performance engineers) designs the product, not just the people with “design” in their job titles.
+
+![](https://cdn-images-1.medium.com/max/800/1*qLoczEHONP188zelJbn-6w@2x.png)
+
+From some of the reactions, you might think Jared had kidnapped someone’s baby for ritual sacrifice. What exactly did Jared write?
+
+> The members of this team are performance engineers. They are architecting, engineering, and maintaining the performance of a very complex system. It occupies all their time and then some. In systems engineering, there are few jobs more technical than these.
+
+> And yet, at the very moment that a Netflix viewer’s video stream stops and that spinning animation appears, indicating the player is now awaiting more data, these engineers make a dramatic change. **They become user experience designers.**
+
+I made that last sentence bold — because it’s really important. Some designers are uncomfortable with the idea that an engineer or a salesperson or a CFO could be a *designer*.
+
+![](https://cdn-images-1.medium.com/max/800/1*ErZDaGRy3mJ19jGdWqeJgA@2x.png)
+
+Common reactions
+
+Whether you like it or not, whether you approve it or not, people outside of your design team are making significant *design* choices that affect your customers in important ways. They are *designing* your product. They are *designers*.
+
+This shouldn’t be provocative — it’s just a statement of fact. I work with[ dozens of startups](http://www.gv.com/portfolio/) every year, and I see it happen at every one. A CFO makes a pricing decision and changes the product experience. An engineer makes a performance trade-off. A salesperson writes a script for talking to customers. In my view, people who fundamentally change the customer’s experience are *designers*.
+
+If this is so self-evident, why do Jared and I press the point? I keep beating the drum because I want designers to change the way they think about their role and become better stewards of good design.
+
+For a moment, consider how this shift in perspective could change the way you work.
+
+#### Everyone needs a design mindset ####
+
+When you accept the reality that design decisions are coming from outside your group, by people without “design” in their job titles, you approach your co-workers differently . Now they’re not just your co-workers — they’re your design team.
+
+The companies that produce great design, such as Apple and Airbnb, have learned this. Alex Schleifer, VP of design at Airbnb,[ tells *Wired*](https://www.wired.com/2015/01/airbnbs-new-head-design-believes-design-led-companies-dont-work/) how the company *isn’t* design-led:
+
+> *The solution [at Airbnb] actually deemphasizes the designers. The point… isn’t to create a “design-led culture,” because that tends to tell anyone who isn’t a designer that their insights take a backseat. It puts the entire organization in the position of having to react to one privileged point of view. Instead, Schleifer wants more people to appreciate what typically lies only within the realm of designers — the user viewpoint.*
+
+Does everyone need all the skills of a designer? Of course not. But each person needs to be armed with the tools to understand how their decisions affect the customer experience.
+
+When an engineer takes a shortcut and scrimps on performance, they need to understand how that damages the user experience. Likewise, when a designer pushes an engineer to make a change that affects performance, that engineer should help the designer make the best overall design decision — not just roll over and do what the designer asked. It’s this type of respectful collaboration that makes great design happen.
+
+One of the best ways to encourage empathy is to watch customer research studies with co-workers from across your company. When my colleague Michael Margolis runs a study with a GV company, we insist that the real team — not just the designers — watch those interviews and take notes. If it’s not possible for everyone to watch in real time, you can record the sessions and schedule a “viewing party” for later.
+
+#### Work outside your design team ####
+
+When you accept that design happens almost everywhere in your organization, you have to take responsibility for it. Your app is slow? Go sit with your engineering team. Your marketing team is poorly communicating your product to future customers? You’d better offer to work with them on the problem.
+
+Yes, doing design with everyone at your company is a lot of work. But it’s necessary if you want to be a truly great designer — otherwise, you’re simply papering over bad decisions. For example, imagine that your CEO created a complex pricing structure for your product. You could focus on making the pricing page as clear as possible using your interface and information design skills. But the harder and more important design opportunity is to work with your CEO on repricing your product so it’s clear to customers and compatible with the business goals.
+
+Focusing on the core business is what differentiates real product design from interface design or even user experience design. Fundamental product design is really hard and requires a lot of legwork, but this is what designers at the highest level do — and it’s why their work is better than yours.
+
+![](https://cdn-images-1.medium.com/max/600/1*czW-2nrN_3l50ZzgYQYqlw@2x.png)
+
+“The Disciplines of UX Design” by Dan Saffer and Thomas Gläser
+
+#### Grow your design team to include non-designers ####
+
+Design is a hard job. You’ll need a wide range of skills (look at all of those circles in the[ diagram of UX Disciplines](https://www.fastcodesign.com/1671735/infographic-the-intricate-anatomy-of-ux-design) by Dan Saffer) and years of practice to truly master design.
+
+Maybe that’s why so many designers are offended when non-designers do design work or get called “designers” by Jared and me. You can act offended if you want, but the reality is that other people are making design decisions with or without you. Embrace them . They don’t make your job less valuable. They don’t make your job title less meaningful.
+
+Having more people who *do design* is additive, not competitive. These designers make your team and your product stronger, because they’re contributing from their unique perspectives. Help them bolster their skills, and use their expertise to the advantage of your product and company.
+
+### **Let’s make a better future together** ###
+
+A few years ago, I met an executive from a Fortune 500 company. When I told her I’m a designer, her eyes lit up. “Oh, I love design!” she said. “My group is just down the hall from the design team. They do so much creative work in there.”
+
+My heart sank. The design team was just down the hall from this executive, but they didn’t work together. Instead, they sat behind soundproof glass walls and did special “creative work” in isolation.
+
+This executive made decisions every day that affected her customers. She bears some of the blame for not reaching out to the designers “in there.” But the design team is primarily at fault — for missing an opportunity to reach out and work together on some of the business’s most important challenges.
+
+
+Thanks to [Jared M. Spool](https://medium.com/@jmspool) for writing an excellent article, [John Zeratsky](https://medium.com/@jazer) and [Michael Margolis](https://medium.com/@mmargolis) for editing and suggestions, and [Alex Schleifer](https://medium.com/@alexoid) for the excellent *Wired* article.
+
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From da5a893d854e280b50844c21b6a86057e9b21e3e Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E6=A0=B9=E5=8F=B7=E4=B8=89?=
Date: Mon, 17 Apr 2017 23:41:34 +0800
Subject: [PATCH 260/945] Create test-driving-away-coupling-in-activities.md
---
...est-driving-away-coupling-in-activities.md | 199 ++++++++++++++++++
1 file changed, 199 insertions(+)
create mode 100644 TODO/test-driving-away-coupling-in-activities.md
diff --git a/TODO/test-driving-away-coupling-in-activities.md b/TODO/test-driving-away-coupling-in-activities.md
new file mode 100644
index 00000000000..346204681d7
--- /dev/null
+++ b/TODO/test-driving-away-coupling-in-activities.md
@@ -0,0 +1,199 @@
+> * 原文地址:[Test Driving away Coupling in Activities](https://www.philosophicalhacker.com/post/test-driving-away-coupling-in-activities/)
+> * 原文作者:[philosohacker](https://twitter.com/philosohacker)
+> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
+> * 译者:
+> * 校对者:
+
+# Test Driving away Coupling in Activities #
+
+`Activity`s and `Fragment`s, perhaps by [some strange historical accidents](/post/why-android-testing-is-so-hard-historical-edition/), have been seen as *the optimal* building blocks upon which we can build our Android applications for much of the time that Android has been around. Let’s call this idea – the idea that `Activity`s and `Fragment`s are the best building blocks for our apps – “android-centric” architecture.
+
+This series of posts is about the connection between the testability of android-centric architecture and the other problems that are now leading Android developers to reject it; it’s about how our unit tests are trying to tell us that `Activity`s and `Fragment`s – like the cracking bricks in the above image – don’t make the best building blocks for our apps because they tempt us to write code with *tight coupling* and *low cohesion*.
+
+
+[Last time](/post/what-unit-tests-are-trying-to-tell-us-about-activities-pt-2/), we saw `Activity`s and `Fragment`s tend to have low cohesion. This time, we’ll see how our tests can tell us that code within `Activity`s have tight coupling. We’ll also see how test driving the functionality leads to a design that has looser coupling, which makes it easier to change the app and also opens up opportunities for removing duplication. As with the the other posts in the series, we’ll be discussing all of this using the Google I/O app as an example.
+
+### The Target Code ###
+
+The code that we want to test, the “target code”, does the following: when the user navigates to the map view that shows where all the Google I/O sessions are, it asks for their location. If they reject the permission, we show a toast notifying the user that they’ve disabled an app permission. Here’s a screenshot of this:
+
+![permission denied toast](http:/images/permission-denied-toast.png)
+
+Here’s the code that accomplishes this:
+
+```
+@Override
+public void onRequestPermissionsResult(final int requestCode,
+ @NonNull final String[] permissions,
+ @NonNull final int[] grantResults) {
+
+ if (requestCode != REQUEST_LOCATION_PERMISSION) {
+ return;
+ }
+
+ if (permissions.length == 1 &&
+ LOCATION_PERMISSION.equals(permissions[0]) &&
+ grantResults[0] == PackageManager.PERMISSION_GRANTED) {
+ // Permission has been granted.
+ if (mMapFragment != null) {
+ mMapFragment.setMyLocationEnabled(true);
+ }
+ } else {
+ // Permission was denied. Display error message.
+ Toast.makeText(this, R.string.map_permission_denied,
+ Toast.LENGTH_SHORT).show();
+ }
+ super.onRequestPermissionsResult(requestCode, permissions,
+ grantResults);
+}
+```
+
+### The Test Code ### (#the-test-code)
+
+Let’s take a stab at testing this. Here’s what that would look like:
+
+```
+@Test
+public void showsToastIfPermissionIsRejected()
+ throws Exception {
+ MapActivity mapActivity = new MapActivity();
+
+ mapActivity.onRequestPermissionsResult(
+ MapActivity.REQUEST_LOCATION_PERMISSION,
+ new String[]{MapActivity.LOCATION_PERMISSION}, new int[]{
+ PackageManager.PERMISSION_DENIED});
+
+ assertToastDisplayed();
+}
+```
+
+Hopefully, you’re wondering what the implementation of `assertToastDisplayed()` looks like. Here’s the thing: there isn’t a straight forward implementation of that method. In order to implement without refactoring our code, we’d need to use a combination of roboelectric and powermock.
+
+However, since we are trying to listen to our tests and [change the way we write code, rather than change the way we write tests](/post/why-i-dont-use-roboletric/), we are going to stop for a moment and think about what this test is trying to tell us:
+
+> Our presentation logic that lives inside of `MapActivity` is tightly coupled with `Toast`.
+
+This coupling is what drives us to use roboelectric to give us mocked android behavior and powermock to mock the static `Toast.makeText` method. Instead, let’s listen to our test and remove the coupling.
+
+To guide our refactoring, let’s write our test first. This will ensure that our *new* classes are loosely coupled. We have to create a new class in this particular case in order to avoid Roboelectric, but ordinarily, we could just refactor already existing classes to reduce coupling.
+
+```
+@Test
+public void displaysErrorWhenPermissionRejected() throws Exception {
+
+ OnPermissionResultListener onPermissionResultListener =
+ new OnPermissionResultListener(mPermittedView);
+
+ onPermissionResultListener.onPermissionResult(
+ MapActivity.REQUEST_LOCATION_PERMISSION,
+ new String[]{MapActivity.LOCATION_PERMISSION},
+ new int[]{PackageManager.PERMISSION_DENIED});
+
+ verify(mPermittedView).displayPermissionDenied();
+}
+```
+
+We’ve introduced a `OnPermissionResultListener` whose job is just to handle the result of request permission from a user. Here’s the code for that:
+
+```
+void onPermissionResult(final int requestCode,
+ final String[] permissions, final int[] grantResults) {
+ if (requestCode != MapActivity.REQUEST_LOCATION_PERMISSION) {
+ return;
+ }
+
+ if (permissions.length == 1 &&
+ MapActivity.LOCATION_PERMISSION.equals(permissions[0]) &&
+ grantResults[0] == PackageManager.PERMISSION_GRANTED) {
+ // Permission has been granted.
+ mPermittedView.displayPermittedView();
+
+ } else {
+ // Permission was denied. Display error message.
+ mPermittedView.displayPermissionDenied();
+ }
+}
+```
+
+The calls to `MapFragment` and `Toast` have been replaced with method calls on the `PermittedView`, an object that gets passed in through the constructor. `PermittedView` is an interface:
+
+```
+interface PermittedView {
+ void displayPermissionDenied();
+
+ void displayPermittedView();
+}
+```
+
+And it gets implemented by the `MapActivity`:
+
+```
+public class MapActivity extends BaseActivity
+ implements SlideableInfoFragment.Callback, MapFragment.Callbacks,
+ ActivityCompat.OnRequestPermissionsResultCallback,
+ OnPermissionResultListener.PermittedView {
+ @Override
+ public void displayPermissionDenied() {
+ Toast.makeText(MapActivity.this, R.string.map_permission_denied,
+ Toast.LENGTH_SHORT).show();
+ }
+}
+```
+
+This may not the *best* solution, but it gets us to a point where we can test things. This *required* that `OnPermissionResultListener` be loosely coupled with its `PermittedView`. Loose coupling == definitely an improvement.
+
+### Who cares? ###
+
+At this point, some readers might be skeptical. “Is this definitely an improvement?,” they may wonder to themselves. Here are two reasons why this *design* is better.
+
+(Neither reason I give, you’ll notice is “the design is better because its testable.” That would be circular reasoning.)
+
+#### Easier Changes ####
+
+First, its going to be easier to change this code now that it consists of loosely coupled components, and here’s the kicker: the code that we’ve just tested from the Google I/O app *actually did change*, and with the tests that we have in place, making those changes will be easier. The code I tested was from [an older commit](https://github.com/google/iosched/blob/bd31a838ce4ddc123c71025c859959517c7ae178/android/src/main/java/com/google/samples/apps/iosched/map/MapActivity.java). Later on, the folks working on the I/O app decided to replace the `Toast` with a `Snackbar`:
+
+![snackbar permission rejected](http:/images/permission-denied-snackbar.png)
+
+Its a small change, but because we’ve separated `OnPermissionResultListener` from `PermittedView`, we can make the change on the `MapActivity`s implementation of `PermittedView` without having to think at all about the `OnPermissionResultListener`.
+
+Here’s what that change would have looked like, using their little `PermissionUtils` class they wrote for displaying `SnackBar`s.
+
+```
+@Override
+public void displayPermissionDenied() {
+ PermissionsUtils.displayConditionalPermissionDenialSnackbar(this,
+ R.string.map_permission_denied, new String[]{LOCATION_PERMISSION},
+ REQUEST_LOCATION_PERMISSION);
+}
+```
+
+Again, notice that we can make this change without thinking about the `OnPermissionResultListener` at all. This is actually exactly what Larry Constantine was talking about when he first defined the concept of coupling back in the 70s:
+
+> what we are striving for is loosely coupled systems…in which one can study (or debug, or maintain) any one module without having to know very much about any other modules in the system
+>
+> –Edward Yourdon and Larry Constantine, Structured Design
+
+#### Reducing Duplication ####
+
+Here’s another interesting reason to why the fact that our tests have forced us to remove coupling is a good thing: coupling often leads to duplication. Here’s Kent Beck on this:
+
+> Dependency is the key problem in software development at all scales…if dependency is the problem, duplication is the symptom.
+>
+> -Kent Beck, TDD By Example, pg 7.
+
+If this is true, when we remove coupling, we will often see opportunities to reduce duplication. Indeed, this is precisely what we find in this case. It turns out that there is another classes whose `onRequestPermissionsResult` is nearly identical to the one in `MapActivity`: [`AccountFragment`](https://github.com/google/iosched/blob/bd31a838ce4ddc123c71025c859959517c7ae178/android/src/main/java/com/google/samples/apps/iosched/welcome/AccountFragment.java#L139). Our tests drove us to create two classes `OnPermissionResultListener` and `PermittedView` that – without much modification – can be reused in these other classes.
+
+### Conclusion ###
+
+So, when we have a hard time testing our `Activity`s and `Fragment`s, its often because our tests are trying to tell us that our code is tightly coupled. The test’s warning about coupling often come in the form of an inability to make an assertion against the code we’re trying to test.1
+
+When we listen to our tests, instead of changing them by using Roboelectric our powermock, we’re lead to change in our code in a way that makes it less coupled, which makes it easier to make changes and opens up opportunities to reduce duplication.
+
+### Notes ###
+
+1. It could also show up as an inability to get your target code into the right state for testing. That’s what we saw [in this post](in this post), for example.
+
+### We're hiring mid-senior Android developers at [Unikey](http://www.unikey.com/). Email me if you want to work for a Startup in the smart lock space in Orlando ###
+---
+
+> [掘金翻译计划](https://github.com/xitu/gold-miner) 是一个翻译优质互联网技术文章的社区,文章来源为 [掘金](https://juejin.im) 上的英文分享文章。内容覆盖 [Android](https://github.com/xitu/gold-miner#android)、[iOS](https://github.com/xitu/gold-miner#ios)、[React](https://github.com/xitu/gold-miner#react)、[前端](https://github.com/xitu/gold-miner#前端)、[后端](https://github.com/xitu/gold-miner#后端)、[产品](https://github.com/xitu/gold-miner#产品)、[设计](https://github.com/xitu/gold-miner#设计) 等领域,想要查看更多优质译文请持续关注 [掘金翻译计划](https://github.com/xitu/gold-miner)。
From 0dbf71b08bba65f45d36a1af644dfc8018e1a4af Mon Sep 17 00:00:00 2001
From: xiaoyusilen
Date: Mon, 17 Apr 2017 23:41:52 +0800
Subject: [PATCH 261/945] Change translation details and add proofreader
informations
---
TODO/anatomy-of-a-function-call-in-go.md | 14 +++++++-------
TODO/go-function-calls-redux.md | 6 +++---
2 files changed, 10 insertions(+), 10 deletions(-)
diff --git a/TODO/anatomy-of-a-function-call-in-go.md b/TODO/anatomy-of-a-function-call-in-go.md
index 4fa70afdb03..c5df8c95dab 100644
--- a/TODO/anatomy-of-a-function-call-in-go.md
+++ b/TODO/anatomy-of-a-function-call-in-go.md
@@ -2,11 +2,11 @@
> * 原文作者:[Phil Pearl](https://syslog.ravelin.com/@philpearl?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[xiaoyusilen](http://xiaoyu.world)
-> * 校对者:[1992chenlu](https://github.com/1992chenlu)
+> * 校对者:[1992chenlu](https://github.com/1992chenlu),[Zheaoli](https://github.com/Zheaoli)
# 解析 Go 中的函数调用 #
-让我们来看一些简单的 Go 的函数,然后看看我们能否明白函数调用是怎么回事。我们将通过分析 Go 编译器根据函数生成的汇编语言来完成这件事。对于一个小小的博客来说这可能有一点野心,但是别担心,汇编语言很简单。哪怕是 CPU 都能读懂。
+让我们来看一些简单的 Go 的函数,然后看看我们能否明白函数调用是怎么回事。我们将通过分析 Go 编译器根据函数生成的汇编来完成这件事。对于一个小小的博客来讲,这样的目标可能有点不切实际,但是别担心,汇编语言很简单。哪怕是 CPU 都能读懂。
@@ -20,9 +20,9 @@ func add(a, b int) int {
}
```
-我们编译的时候需要关闭优化,这样方便理解每个部分。我们用 `go build -gcflags 'N -l'` 这个命令来完成上述操作。然后我们可以用 `go tool objdump -s main.add func` 输出我们函数的具体细节(这里的 func 是我们的包名,也就是我们刚刚用 go build 编译出的可执行文件)。
+我们编译的时候需要关闭优化,这样方便我们去理解生成的汇编代码。我们用 `go build -gcflags 'N -l'` 这个命令来完成上述操作。然后我们可以用 `go tool objdump -s main.add func` 输出我们函数的具体细节(这里的 func 是我们的包名,也就是我们刚刚用 go build 编译出的可执行文件)。
-如果你以前从来没有看过汇编语言,那么恭喜啦,你今天看到的内容对你来说是全新的。我将在 Mac 上完成这些事情,所以配置是 Intel 64位。
+如果你之前没有学过汇编,那么恭喜你,你将接触到一个全新的事物。另外我会在 Mac 上完成这篇博客的代码,因此所生成的是 Intel 64-bit 汇编。
```
main.go:20 0x22c0 48c744241800000000 MOVQ $0x0, 0x18(SP)
@@ -33,7 +33,7 @@ func add(a, b int) int {
main.go:21 0x22db c3 RET
```
-我们看到了什么?每一行如下所示被分为了4部分:
+现在我们看到了什么?如下所示,每一行被分为了4部分:
- 源文件的名称和行号(main.go:15)。这行的源代码会被转换为标有代码行号的说明。Go 的一行可能被转换成多行程序集。
- 目标文件中的偏移量(例如 0x22C0)。
@@ -51,8 +51,8 @@ func add(a, b int) int {
现在让我们从第一条指令开始看每一条内容。别忘了我们需要从内存中加载两个参数 `a` 和 `b`,把它们相加,然后返回至调用函数。
-1. `MOVQ $0x0, 0x18(SP)` 将 0 置于存储单元 SP+0x18 中。 这句看起来有点神秘。
-2. `MOVQ 0x8(SP), AX` 将存储单元 SP+0x8 中的内容放到 CPU 寄存器 AX 中。也许这就是从内存中加载的我们的参数之一?
+1. `MOVQ $0x0, 0x18(SP)` 将 0 置于存储单元 SP+0x18 中。 这句代码看起来有点抽象。
+2. `MOVQ 0x8(SP), AX` 将存储单元 SP+0x8 中的内容放到 CPU 寄存器 AX 中。也许这就是从内存中加载的我们所使用的参数之一?
3. `MOVQ 0x10(SP), CX` 将存储单元 SP+0x10 的内容置于 CPU 寄存器 CX 中。 这可能就是我们所需的另一个参数。
4. `ADDQ CX, AX` 将 CX 与 AX 相加,将结果存到 AX 中。好,现在已经把两个参数相加了。
5. `MOVQ AX, 0x18(sp)` 将寄存器 AX 的内容存储在存储单元 SP+0x18 中。这就是在存储相加的结果。
diff --git a/TODO/go-function-calls-redux.md b/TODO/go-function-calls-redux.md
index b8717411ff1..dad571a5324 100644
--- a/TODO/go-function-calls-redux.md
+++ b/TODO/go-function-calls-redux.md
@@ -2,13 +2,13 @@
> * 原文作者:[Phil Pearl](https://hackernoon.com/@philpearl?source=post_header_lockup)
> * 译文出自:[掘金翻译计划](https://github.com/xitu/gold-miner)
> * 译者:[xiaoyusilen](http://xiaoyu.world)
-> * 校对者:[1992chenlu](https://github.com/1992chenlu)
+> * 校对者:[1992chenlu](https://github.com/1992chenlu),[Zheaoli](https://github.com/Zheaoli)
# Go 函数调用 Redux #
-前段时间在一篇[文章](https://syslog.ravelin.com/anatomy-of-a-function-call-in-go-f6fc81b80ecc#.gpqsgzmjc)中我答应写一篇进一步分析 Go 中如何进行函数调用和调用堆栈在 Go 中如何工作的文章。现在我找到了一种简洁的方式来向大家展示上述内容,所以有了现在这篇文章。
+前段时间在一篇[文章](https://syslog.ravelin.com/anatomy-of-a-function-call-in-go-f6fc81b80ecc#.gpqsgzmjc)中我答应写一篇进一步分析 Go 中如何进行函数调用和堆栈调用在 Go 中如何工作的文章。现在我找到了一种简洁的方式来向大家展示上述内容,所以有了现在这篇文章。
-什么是调用堆栈?它是一个用于保存局部变量和调用参数的内存区域,并且跟踪每个函数应该返回到哪里去。每个 goroutine 都有它自己的堆栈。你甚至可以说每个 goroutine 就是它自己的堆栈。
+什么是堆栈调用?它是一个用于保存局部变量和调用参数的内存区域,并且跟踪每个函数应该返回到哪里去。每个 goroutine 都有它自己的堆栈。你甚至可以说每个 goroutine 就是它自己的堆栈。
下面是我用于演示堆栈的代码。就是一系列简单的函数调用,main() 函数调用 [f1(0xdeadbeef)](https://en.wikipedia.org/wiki/Hexspeak),然后调用 `f2(0xabad1dea)`,再调用 `f3(0xbaddcafe)`。然后 `f3()` 将其中一个作为它的参数,并且将它存储在名为 `local` 的本地变量中。然后获取 `local` 的内存地址并且从那里开始输出。因为 `local` 在栈内,所以输出的就是栈。
From c0c66b4c26a1b06011085d66adc994db9a6a700d Mon Sep 17 00:00:00 2001
From: zhuzi
Date: Tue, 18 Apr 2017 00:00:52 +0800
Subject: [PATCH 262/945] proofreading
---
TODO/secure-web-app-http-headers.md | 8 ++++----
1 file changed, 4 insertions(+), 4 deletions(-)
diff --git a/TODO/secure-web-app-http-headers.md b/TODO/secure-web-app-http-headers.md
index 5311d35ff7a..b707e2397a9 100644
--- a/TODO/secure-web-app-http-headers.md
+++ b/TODO/secure-web-app-http-headers.md
@@ -6,7 +6,7 @@
## 如何使用 HTTP Headers 来保护你的 Web 应用 ##
-众所周知,无论是简单的小网页还是复杂的单页应用,Web 应用都是网络攻击的目标。2016年,这种最主要的攻击模式 —— 攻击 web 应用,造成了大约 [40% 的数据泄露](http://www.verizonenterprise.com/verizon-insights-lab/dbir/2016/)。事实上,现在来说,了解网络安全并不是锦上添花, 而是 **Web 开发者的必需任务**,特别对于构建面向消费者的产品的开发人员。
+众所周知,无论是简单的小网页还是复杂的单页应用,Web 应用都是网络攻击的目标。2016 年,这种最主要的攻击模式 —— 攻击 web 应用,造成了大约 [40% 的数据泄露](http://www.verizonenterprise.com/verizon-insights-lab/dbir/2016/)。事实上,现在来说,了解网络安全并不是锦上添花,而是 **Web 开发者的必需任务**,特别对于构建面向消费者的产品的开发人员。
开发者可以利用 HTTP 响应头来加强 Web 应用程序的安全性,通常只需要添加几行代码即可。本文将介绍 web 开发者如何利用 HTTP Headers 来构建安全的应用。虽然本文的示例代码是 Node.js,但基本所有主流的服务端语言都支持设置 HTTP 响应头,并且都可以简单地对其进行配置。
@@ -85,7 +85,7 @@ HSTS 的指令如下:
这是一个强大的指令,强制浏览器**始终**安全加载你的 web 应用程序,即使是第一次收到响应之前加载!这是通过将启用 HSTS 预加载域的列表硬编码到浏览器的代码中实现的。要启用预加载功能,你需要在 Google Chrome 团队维护的网站 [HSTS 预加载列表提交](https://hstspreload.org)注册你的域。
-注意谨慎使用 `preload`,因为这意味着它不能轻易撤销,并可能更新延迟数个月。虽然预加载肯定会加强应用程序的安全性,但也意味着你需要充分确信你的应用程序可以支持仅 HTTPS!
+注意谨慎使用 `preload`,因为这意味着它不能轻易撤销,并可能更新延迟数个月。虽然预加载肯定会加强应用程序的安全性,但也意味着你需要充分确信你的应用程序仅支持 HTTPS!
我建议的用法是 `Strict-Transport-Security: max-age=31536000; includeSubDomains;`,这样指示了浏览器强制通过 HTTPS 连接到源主机并且有效期为一年。如果你对你的 app 仅处理 HTTPS 很有信心,我也推荐加上 `preload` 指令,当然别忘记去前面提到的预加载列表注册你的网站。
@@ -133,7 +133,7 @@ https://mywebapp.com/search?