Skip to content

Commit 88b8604

Browse files
authored
Merge pull request #1 from javascript-tutorial/master
update
2 parents e4300bb + e1c0d62 commit 88b8604

File tree

37 files changed

+202
-186
lines changed

37 files changed

+202
-186
lines changed

1-js/02-first-steps/03-strict-mode/article.md

+10-2
Original file line numberDiff line numberDiff line change
@@ -53,9 +53,17 @@ For the future, when you use a browser console to test features, please note tha
5353
5454
Sometimes, when `use strict` makes a difference, you'll get incorrect results.
5555
56-
Even if we press `key:Shift+Enter` to input multiple lines, and put `use strict` on top, it doesn't work. That's because of how the console executes the code internally.
56+
You can try to press `key:Shift+Enter` to input multiple lines, and put `use strict` on top, like this:
5757
58-
The reliable way to ensure `use strict` would be to input the code into console like this:
58+
```js
59+
'use strict'; <Shift+Enter for a newline>
60+
// ...your code
61+
<Enter to run>
62+
```
63+
64+
It works in most browsers, namely Firefox and Chrome.
65+
66+
If it doesn't, the most reliable way to ensure `use strict` would be to input the code into console like this:
5967

6068
```js
6169
(function() {

1-js/03-code-quality/04-ninja-code/article.md

+4-5
Original file line numberDiff line numberDiff line change
@@ -137,7 +137,7 @@ Instead, reuse existing names. Just write new values into them.
137137

138138
In a function try to use only variables passed as parameters.
139139

140-
That would make it really hard to identify what's exactly in the variable *now*. And also where it comes from. A person with weak intuition would have to analyze the code line-by-line and track the changes through every code branch.
140+
That would make it really hard to identify what's exactly in the variable *now*. And also where it comes from. The purpose is to develop the intuition and memory of a person reading the code. A person with weak intuition would have to analyze the code line-by-line and track the changes through every code branch.
141141

142142
**An advanced variant of the approach is to covertly (!) replace the value with something alike in the middle of a loop or a function.**
143143

@@ -155,7 +155,7 @@ function ninjaFunction(elem) {
155155

156156
A fellow programmer who wants to work with `elem` in the second half of the function will be surprised... Only during the debugging, after examining the code they will find out that they're working with a clone!
157157

158-
Seen in code regularly. Deadly effective even against an experienced ninja.
158+
Seen in code regularly. Deadly effective even against an experienced ninja.
159159

160160
## Underscores for fun
161161

@@ -169,8 +169,7 @@ A smart ninja puts underscores at one spot of code and evades them at other plac
169169

170170
Let everyone see how magnificent your entities are! Names like `superElement`, `megaFrame` and `niceItem` will definitely enlighten a reader.
171171

172-
Indeed, from one hand, something is written: `super..`, `mega..`, `nice..` But from the other hand -- that brings no details. A reader may decide to look for a hidden meaning and meditate for an hour or two.
173-
172+
Indeed, from one hand, something is written: `super..`, `mega..`, `nice..` But from the other hand -- that brings no details. A reader may decide to look for a hidden meaning and meditate for an hour or two of their paid working time.
174173

175174

176175
## Overlap outer variables
@@ -180,7 +179,7 @@ When in the light, can't see anything in the darkness.<br>
180179
When in the darkness, can see everything in the light.
181180
```
182181

183-
Use same names for variables inside and outside a function. As simple. No efforts required.
182+
Use same names for variables inside and outside a function. As simple. No efforts to invent new names.
184183

185184
```js
186185
let *!*user*/!* = authenticateUser();

1-js/04-object-basics/02-garbage-collection/article.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -207,6 +207,6 @@ A general book "The Garbage Collection Handbook: The Art of Automatic Memory Man
207207

208208
If you are familiar with low-level programming, the more detailed information about V8 garbage collector is in the article [A tour of V8: Garbage Collection](http://jayconrod.com/posts/55/a-tour-of-v8-garbage-collection).
209209

210-
[V8 blog](http://v8project.blogspot.com/) also publishes articles about changes in memory management from time to time. Naturally, to learn the garbage collection, you'd better prepare by learning about V8 internals in general and read the blog of [Vyacheslav Egorov](http://mrale.ph) who worked as one of V8 engineers. I'm saying: "V8", because it is best covered with articles in the internet. For other engines, many approaches are similar, but garbage collection differs in many aspects.
210+
[V8 blog](https://v8.dev/) also publishes articles about changes in memory management from time to time. Naturally, to learn the garbage collection, you'd better prepare by learning about V8 internals in general and read the blog of [Vyacheslav Egorov](http://mrale.ph) who worked as one of V8 engineers. I'm saying: "V8", because it is best covered with articles in the internet. For other engines, many approaches are similar, but garbage collection differs in many aspects.
211211

212212
In-depth knowledge of engines is good when you need low-level optimizations. It would be wise to plan that as the next step after you're familiar with the language.

1-js/04-object-basics/04-object-methods/article.md

+13-9
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,7 @@ user.sayHi(); // Hello!
6363
```smart header="Object-oriented programming"
6464
When we write our code using objects to represent entities, that's called an [object-oriented programming](https://en.wikipedia.org/wiki/Object-oriented_programming), in short: "OOP".
6565
66-
OOP is a big thing, an interesting science of its own. How to choose the right entities? How to organize the interaction between them? That's architecture, and there are great books on that topic, like "Design Patterns: Elements of Reusable Object-Oriented Software" by E.Gamma, R.Helm, R.Johnson, J.Vissides or "Object-Oriented Analysis and Design with Applications" by G.Booch, and more.
66+
OOP is a big thing, an interesting science of its own. How to choose the right entities? How to organize the interaction between them? That's architecture, and there are great books on that topic, like "Design Patterns: Elements of Reusable Object-Oriented Software" by E.Gamma, R.Helm, R.Johnson, J.Vissides or "Object-Oriented Analysis and Design with Applications" by G.Booch, and more.
6767
```
6868
### Method shorthand
6969

@@ -72,14 +72,14 @@ There exists a shorter syntax for methods in an object literal:
7272
```js
7373
// these objects do the same
7474

75-
let user = {
75+
user = {
7676
sayHi: function() {
7777
alert("Hello");
7878
}
7979
};
8080

8181
// method shorthand looks better, right?
82-
let user = {
82+
user = {
8383
*!*
8484
sayHi() { // same as "sayHi: function()"
8585
*/!*
@@ -166,7 +166,7 @@ If we used `this.name` instead of `user.name` inside the `alert`, then the code
166166

167167
## "this" is not bound
168168

169-
In JavaScript, "this" keyword behaves unlike most other programming languages. First, it can be used in any function.
169+
In JavaScript, "this" keyword behaves unlike most other programming languages. It can be used in any function.
170170

171171
There's no syntax error in the code like that:
172172

@@ -176,9 +176,9 @@ function sayHi() {
176176
}
177177
```
178178

179-
The value of `this` is evaluated during the run-time. And it can be anything.
179+
The value of `this` is evaluated during the run-time, depending on the context. And it can be anything.
180180

181-
For instance, the same function may have different "this" when called from different objects:
181+
For instance, here the same function is assigned to two different objects and has different "this" in the calls:
182182

183183
```js run
184184
let user = { name: "John" };
@@ -189,7 +189,7 @@ function sayHi() {
189189
}
190190

191191
*!*
192-
// use the same functions in two objects
192+
// use the same function in two objects
193193
user.f = sayHi;
194194
admin.f = sayHi;
195195
*/!*
@@ -202,7 +202,10 @@ admin.f(); // Admin (this == admin)
202202
admin['f'](); // Admin (dot or square brackets access the method – doesn't matter)
203203
```
204204

205-
Actually, we can call the function without an object at all:
205+
The rule is simple: if `obj.f()` is called, then `this` is `obj` during the call of `f`. So it's either `user` or `admin` in the example above.
206+
207+
````smart header="Calling without an object: `this == undefined`"
208+
We can even call the function without an object at all:
206209

207210
```js run
208211
function sayHi() {
@@ -216,7 +219,8 @@ In this case `this` is `undefined` in strict mode. If we try to access `this.nam
216219

217220
In non-strict mode the value of `this` in such case will be the *global object* (`window` in a browser, we'll get to it later in the chapter [](info:global-object)). This is a historical behavior that `"use strict"` fixes.
218221

219-
Please note that usually a call of a function that uses `this` without an object is not normal, but rather a programming mistake. If a function has `this`, then it is usually meant to be called in the context of an object.
222+
Usually such call is an programming error. If there's `this` inside a function, it expects to be called in an object context.
223+
````
220224
221225
```smart header="The consequences of unbound `this`"
222226
If you come from another programming language, then you are probably used to the idea of a "bound `this`", where methods defined in an object always have `this` referencing that object.

1-js/05-data-types/07-map-set-weakmap-weakset/article.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -222,9 +222,9 @@ set.forEach((value, valueAgain, set) => {
222222
});
223223
```
224224
225-
Note the funny thing. The `forEach` function in the `Set` has 3 arguments: a value, then *again a value*, and then the target object. Indeed, the same value appears in the arguments twice.
225+
Note the funny thing. The callback function passed in `forEach` has 3 arguments: a value, then *again a value*, and then the target object. Indeed, the same value appears in the arguments twice.
226226
227-
That's for compatibility with `Map` where `forEach` has three arguments. Looks a bit strange, for sure. But may help to replace `Map` with `Set` in certain cases with ease, and vice versa.
227+
That's for compatibility with `Map` where the callback passed `forEach` has three arguments. Looks a bit strange, for sure. But may help to replace `Map` with `Set` in certain cases with ease, and vice versa.
228228
229229
The same methods `Map` has for iterators are also supported:
230230

1-js/05-data-types/11-json/article.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -66,7 +66,7 @@ alert(json);
6666

6767
The method `JSON.stringify(student)` takes the object and converts it into a string.
6868

69-
The resulting `json` string is a called *JSON-encoded* or *serialized* or *stringified* or *marshalled* object. We are ready to send it over the wire or put into a plain data store.
69+
The resulting `json` string is called a *JSON-encoded* or *serialized* or *stringified* or *marshalled* object. We are ready to send it over the wire or put into a plain data store.
7070

7171

7272
Please note that a JSON-encoded object has several important differences from the object literal:

1-js/06-advanced-functions/11-currying-partials/article.md

+10-15
Original file line numberDiff line numberDiff line change
@@ -170,7 +170,7 @@ To understand the benefits we definitely need a worthy real-life example.
170170

171171
Advanced currying allows the function to be both callable normally and partially.
172172

173-
For instance, we have the logging function `log(date, importance, message)` that formats and outputs the information. In real projects such functions also have many other useful features like sending logs over the network:
173+
For instance, we have the logging function `log(date, importance, message)` that formats and outputs the information. In real projects such functions also have many other useful features like sending logs over the network, here we just use `alert`:
174174

175175
```js
176176
function log(date, importance, message) {
@@ -184,34 +184,29 @@ Let's curry it!
184184
log = _.curry(log);
185185
```
186186

187-
After that `log` still works the normal way:
188-
189-
```js
190-
log(new Date(), "DEBUG", "some debug");
191-
```
192-
193-
...But also can be called in the curried form:
187+
After that `log` work both the normal way and in the curried form:
194188

195189
```js
190+
log(new Date(), "DEBUG", "some debug"); // log(a,b,c)
196191
log(new Date())("DEBUG")("some debug"); // log(a)(b)(c)
197192
```
198193

199-
Let's get a convenience function for today's logs:
194+
Now we can easily make a convenience function for current logs:
200195

201196
```js
202-
// todayLog will be the partial of log with fixed first argument
203-
let todayLog = log(new Date());
197+
// currentLog will be the partial of log with fixed first argument
198+
let logNow = log(new Date());
204199

205200
// use it
206-
todayLog("INFO", "message"); // [HH:mm] INFO message
201+
logNow("INFO", "message"); // [HH:mm] INFO message
207202
```
208203

209-
And now a convenience function for today's debug messages:
204+
And here's a convenience function for current debug messages:
210205

211206
```js
212-
let todayDebug = todayLog("DEBUG");
207+
let debugNow = logNow("DEBUG");
213208

214-
todayDebug("message"); // [HH:mm] DEBUG message
209+
debugNow("message"); // [HH:mm] DEBUG message
215210
```
216211

217212
So:

1-js/08-prototypes/02-function-prototype/article.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -162,7 +162,7 @@ Everything is quite simple, just few notes to make things clear:
162162

163163
- The `F.prototype` property is not the same as `[[Prototype]]`. The only thing `F.prototype` does: it sets `[[Prototype]]` of new objects when `new F()` is called.
164164
- The value of `F.prototype` should be either an object or null: other values won't work.
165-
- The `"prototype"` property only has such a special effect when is set to a constructor function, and invoked with `new`.
165+
- The `"prototype"` property only has such a special effect when set on a constructor function, and invoked with `new`.
166166

167167
On regular objects the `prototype` is nothing special:
168168
```js

1-js/09-classes/01-class/article.md

+5-4
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,7 @@ The notation here is not to be confused with object literals. Within the class,
6464

6565
## What is a class?
6666

67-
So, what exactly is a `class`? That's not an entirely new language-level entity, as one might think.
67+
So, what exactly is a `class`? That's not an entirely new language-level entity, as one might think.
6868

6969
Let's unveil any magic and see what a class really is. That'll help in understanding many complex aspects.
7070

@@ -91,7 +91,7 @@ What `class User {...}` construct really does is:
9191

9292
Afterwards, for new objects, when we call a method, it's taken from the prototype, just as described in the chapter <info:function-prototype>. So `new User` object has access to class methods.
9393

94-
We can illustrate the result of `class User` as:
94+
We can illustrate the result of `class User` declaration as:
9595

9696
![](class-user.png)
9797

@@ -198,7 +198,8 @@ Similar to Named Function Expressions, class expressions may or may not have a n
198198
If a class expression has a name, it's visible inside the class only:
199199
200200
```js run
201-
// "Named Class Expression" (alas, no such term, but that's what's going on)
201+
// "Named Class Expression"
202+
// (no such term in the spec, but that's similar to Named Function Expression)
202203
let User = class *!*MyClass*/!* {
203204
sayHi() {
204205
alert(MyClass); // MyClass is visible only inside the class
@@ -268,7 +269,7 @@ alert(user.name); // John
268269
user = new User(""); // Name too short.
269270
```
270271

271-
Internally, getters and setters are created on `User.prototype`, like this:
272+
The class declaration creates getters and setters in `User.prototype`, like this:
272273

273274
```js
274275
Object.defineProperties(User.prototype, {

1-js/10-error-handling/2-custom-errors/article.md

+6-6
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ Our errors should support basic error properties like `message`, `name` and, pre
66

77
JavaScript allows to use `throw` with any argument, so technically our custom error classes don't need to inherit from `Error`. But if we inherit, then it becomes possible to use `obj instanceof Error` to identify error objects. So it's better to inherit from it.
88

9-
As we build our application, our own errors naturally form a hierarchy, for instance `HttpTimeoutError` may inherit from `HttpError`, and so on.
9+
As the application grows, our own errors naturally form a hierarchy, for instance `HttpTimeoutError` may inherit from `HttpError`, and so on.
1010

1111
## Extending Error
1212

@@ -126,7 +126,7 @@ We could also look at `err.name`, like this:
126126
127127
The `instanceof` version is much better, because in the future we are going to extend `ValidationError`, make subtypes of it, like `PropertyRequiredError`. And `instanceof` check will continue to work for new inheriting classes. So that's future-proof.
128128
129-
Also it's important that if `catch` meets an unknown error, then it rethrows it in the line `(**)`. The `catch` only knows how to handle validation and syntax errors, other kinds (due to a typo in the code or such) should fall through.
129+
Also it's important that if `catch` meets an unknown error, then it rethrows it in the line `(**)`. The `catch` block only knows how to handle validation and syntax errors, other kinds (due to a typo in the code or other unknown ones) should fall through.
130130
131131
## Further inheritance
132132
@@ -185,7 +185,7 @@ try {
185185
186186
The new class `PropertyRequiredError` is easy to use: we only need to pass the property name: `new PropertyRequiredError(property)`. The human-readable `message` is generated by the constructor.
187187
188-
Please note that `this.name` in `PropertyRequiredError` constructor is again assigned manually. That may become a bit tedious -- to assign `this.name = <class name>` when creating each custom error. But there's a way out. We can make our own "basic error" class that removes this burden from our shoulders by using `this.constructor.name` for `this.name` in the constructor. And then inherit from it.
188+
Please note that `this.name` in `PropertyRequiredError` constructor is again assigned manually. That may become a bit tedious -- to assign `this.name = <class name>` in every custom error class. But there's a way out. We can make our own "basic error" class that removes this burden from our shoulders by using `this.constructor.name` for `this.name` in its constructor. And then inherit all ours custom errors from it.
189189
190190
Let's call it `MyError`.
191191
@@ -218,7 +218,7 @@ Now custom errors are much shorter, especially `ValidationError`, as we got rid
218218
219219
## Wrapping exceptions
220220
221-
The purpose of the function `readUser` in the code above is "to read the user data", right? There may occur different kinds of errors in the process. Right now we have `SyntaxError` and `ValidationError`, but in the future `readUser` function may grow: the new code will probably generate other kinds of errors.
221+
The purpose of the function `readUser` in the code above is "to read the user data", right? There may occur different kinds of errors in the process. Right now we have `SyntaxError` and `ValidationError`, but in the future `readUser` function may grow and probably generate other kinds of errors.
222222
223223
The code which calls `readUser` should handle these errors. Right now it uses multiple `if` in the `catch` block to check for different error types and rethrow the unknown ones. But if `readUser` function generates several kinds of errors -- then we should ask ourselves: do we really want to check for all error types one-by-one in every code that calls `readUser`?
224224
@@ -303,5 +303,5 @@ The approach is called "wrapping exceptions", because we take "low level excepti
303303
## Summary
304304
305305
- We can inherit from `Error` and other built-in error classes normally, just need to take care of `name` property and don't forget to call `super`.
306-
- Most of the time, we should use `instanceof` to check for particular errors. It also works with inheritance. But sometimes we have an error object coming from the 3rd-party library and there's no easy way to get the class. Then `name` property can be used for such checks.
307-
- Wrapping exceptions is a widespread technique when a function handles low-level exceptions and makes a higher-level object to report about the errors. Low-level exceptions sometimes become properties of that object like `err.cause` in the examples above, but that's not strictly required.
306+
- We can use `instanceof` to check for particular errors. It also works with inheritance. But sometimes we have an error object coming from the 3rd-party library and there's no easy way to get the class. Then `name` property can be used for such checks.
307+
- Wrapping exceptions is a widespread technique: a function handles low-level exceptions and creates higher-level errors instead of various low-level ones. Low-level exceptions sometimes become properties of that object like `err.cause` in the examples above, but that's not strictly required.

1-js/11-async/02-promise-basics/article.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -278,7 +278,7 @@ The good thing is: a `.then` handler is guaranteed to run whether the promise ta
278278

279279
Next, let's see more practical examples of how promises can help us to write asynchronous code.
280280

281-
## Example: loadScript
281+
## Example: loadScript [#loadscript]
282282

283283
We've got the `loadScript` function for loading a script from the previous chapter.
284284

0 commit comments

Comments
 (0)