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.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Copy file name to clipboardexpand all lines: 1-js/02-first-steps/03-strict-mode/article.md
+10-2
Original file line number
Diff line number
Diff line change
@@ -53,9 +53,17 @@ For the future, when you use a browser console to test features, please note tha
53
53
54
54
Sometimes, when `use strict` makes a difference, you'll get incorrect results.
55
55
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:
57
57
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:
Copy file name to clipboardexpand all lines: 1-js/03-code-quality/04-ninja-code/article.md
+4-5
Original file line number
Diff line number
Diff line change
@@ -137,7 +137,7 @@ Instead, reuse existing names. Just write new values into them.
137
137
138
138
In a function try to use only variables passed as parameters.
139
139
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.
141
141
142
142
**An advanced variant of the approach is to covertly (!) replace the value with something alike in the middle of a loop or a function.**
143
143
@@ -155,7 +155,7 @@ function ninjaFunction(elem) {
155
155
156
156
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!
157
157
158
-
Seen in code regularly. Deadly effective even against an experienced ninja.
158
+
Seen in code regularly. Deadly effective even against an experienced ninja.
159
159
160
160
## Underscores for fun
161
161
@@ -169,8 +169,7 @@ A smart ninja puts underscores at one spot of code and evades them at other plac
169
169
170
170
Let everyone see how magnificent your entities are! Names like `superElement`, `megaFrame` and `niceItem` will definitely enlighten a reader.
171
171
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.
174
173
175
174
176
175
## Overlap outer variables
@@ -180,7 +179,7 @@ When in the light, can't see anything in the darkness.<br>
180
179
When in the darkness, can see everything in the light.
181
180
```
182
181
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.
Copy file name to clipboardexpand all lines: 1-js/04-object-basics/02-garbage-collection/article.md
+1-1
Original file line number
Diff line number
Diff line change
@@ -207,6 +207,6 @@ A general book "The Garbage Collection Handbook: The Art of Automatic Memory Man
207
207
208
208
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).
209
209
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.
211
211
212
212
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.
Copy file name to clipboardexpand all lines: 1-js/04-object-basics/04-object-methods/article.md
+13-9
Original file line number
Diff line number
Diff line change
@@ -63,7 +63,7 @@ user.sayHi(); // Hello!
63
63
```smart header="Object-oriented programming"
64
64
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".
65
65
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.
67
67
```
68
68
### Method shorthand
69
69
@@ -72,14 +72,14 @@ There exists a shorter syntax for methods in an object literal:
72
72
```js
73
73
// these objects do the same
74
74
75
-
letuser = {
75
+
user = {
76
76
sayHi:function() {
77
77
alert("Hello");
78
78
}
79
79
};
80
80
81
81
// method shorthand looks better, right?
82
-
letuser = {
82
+
user = {
83
83
*!*
84
84
sayHi() { // same as "sayHi: function()"
85
85
*/!*
@@ -166,7 +166,7 @@ If we used `this.name` instead of `user.name` inside the `alert`, then the code
166
166
167
167
## "this" is not bound
168
168
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.
170
170
171
171
There's no syntax error in the code like that:
172
172
@@ -176,9 +176,9 @@ function sayHi() {
176
176
}
177
177
```
178
178
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.
180
180
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:
admin['f'](); // Admin (dot or square brackets access the method – doesn't matter)
203
203
```
204
204
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:
206
209
207
210
```js run
208
211
functionsayHi() {
@@ -216,7 +219,8 @@ In this case `this` is `undefined` in strict mode. If we try to access `this.nam
216
219
217
220
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.
218
221
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
+
````
220
224
221
225
```smart header="The consequences of unbound `this`"
222
226
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.
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.
226
226
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.
228
228
229
229
The same methods `Map` has for iterators are also supported:
Copy file name to clipboardexpand all lines: 1-js/05-data-types/11-json/article.md
+1-1
Original file line number
Diff line number
Diff line change
@@ -66,7 +66,7 @@ alert(json);
66
66
67
67
The method `JSON.stringify(student)` takes the object and converts it into a string.
68
68
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.
70
70
71
71
72
72
Please note that a JSON-encoded object has several important differences from the object literal:
Copy file name to clipboardexpand all lines: 1-js/06-advanced-functions/11-currying-partials/article.md
+10-15
Original file line number
Diff line number
Diff line change
@@ -170,7 +170,7 @@ To understand the benefits we definitely need a worthy real-life example.
170
170
171
171
Advanced currying allows the function to be both callable normally and partially.
172
172
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`:
174
174
175
175
```js
176
176
functionlog(date, importance, message) {
@@ -184,34 +184,29 @@ Let's curry it!
184
184
log =_.curry(log);
185
185
```
186
186
187
-
After that `log` still works the normal way:
188
-
189
-
```js
190
-
log(newDate(), "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:
Copy file name to clipboardexpand all lines: 1-js/08-prototypes/02-function-prototype/article.md
+1-1
Original file line number
Diff line number
Diff line change
@@ -162,7 +162,7 @@ Everything is quite simple, just few notes to make things clear:
162
162
163
163
- 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.
164
164
- 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`.
166
166
167
167
On regular objects the `prototype` is nothing special:
Copy file name to clipboardexpand all lines: 1-js/09-classes/01-class/article.md
+5-4
Original file line number
Diff line number
Diff line change
@@ -64,7 +64,7 @@ The notation here is not to be confused with object literals. Within the class,
64
64
65
65
## What is a class?
66
66
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.
68
68
69
69
Let's unveil any magic and see what a class really is. That'll help in understanding many complex aspects.
70
70
@@ -91,7 +91,7 @@ What `class User {...}` construct really does is:
91
91
92
92
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.
93
93
94
-
We can illustrate the result of `class User` as:
94
+
We can illustrate the result of `class User`declaration as:
95
95
96
96

97
97
@@ -198,7 +198,8 @@ Similar to Named Function Expressions, class expressions may or may not have a n
198
198
If a classexpression has a name, it's visible inside the class only:
199
199
200
200
```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)
202
203
let User =class*!*MyClass*/!* {
203
204
sayHi() {
204
205
alert(MyClass); // MyClass is visible only inside the class
@@ -268,7 +269,7 @@ alert(user.name); // John
268
269
user =newUser(""); // Name too short.
269
270
```
270
271
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:
Copy file name to clipboardexpand all lines: 1-js/10-error-handling/2-custom-errors/article.md
+6-6
Original file line number
Diff line number
Diff line change
@@ -6,7 +6,7 @@ Our errors should support basic error properties like `message`, `name` and, pre
6
6
7
7
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.
8
8
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.
10
10
11
11
## Extending Error
12
12
@@ -126,7 +126,7 @@ We could also look at `err.name`, like this:
126
126
127
127
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.
128
128
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.
130
130
131
131
## Further inheritance
132
132
@@ -185,7 +185,7 @@ try {
185
185
186
186
The new class `PropertyRequiredError` is easy to use: we only need to pass the property name: `newPropertyRequiredError(property)`. The human-readable `message` is generated by the constructor.
187
187
188
-
Please note that `this.name` in `PropertyRequiredError` constructor is again assigned manually. That may become a bit tedious -- to assign `this.name=<classname>`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=<classname>`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.
189
189
190
190
Let's call it `MyError`.
191
191
@@ -218,7 +218,7 @@ Now custom errors are much shorter, especially `ValidationError`, as we got rid
218
218
219
219
## Wrapping exceptions
220
220
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.
222
222
223
223
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`?
224
224
@@ -303,5 +303,5 @@ The approach is called "wrapping exceptions", because we take "low level excepti
303
303
## Summary
304
304
305
305
- 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.
0 commit comments