2015-10-30 15:53:40 +00:00
|
|
|
|
---
|
|
|
|
|
contributors:
|
2020-02-13 00:08:29 +00:00
|
|
|
|
- ["Leigh Brenecki", "https://leigh.net.au"]
|
2015-10-30 15:53:40 +00:00
|
|
|
|
- ["Ariel Krakowski", "http://www.learneroo.com"]
|
2016-05-14 11:58:16 +00:00
|
|
|
|
- ["clearsense", "https://github.com/clearsense"]
|
2015-10-30 15:53:40 +00:00
|
|
|
|
translators:
|
2015-11-27 10:36:10 +00:00
|
|
|
|
- ["Ivan", "https://github.com/IvanEh"]
|
|
|
|
|
- ["Serhii Maksymchuk", "https://github.com/Serg-Maximchuk"]
|
2015-10-30 15:53:40 +00:00
|
|
|
|
---
|
|
|
|
|
|
2016-10-23 17:00:57 +00:00
|
|
|
|
JavaScript було створено в 1995 році Бренданом Айком, який працював у компанії Netscape.
|
2015-10-30 15:53:40 +00:00
|
|
|
|
Він був задуманий як проста мова сценаріїв для веб-сайтів, який би доповнював Java
|
|
|
|
|
для більш складних веб-застосунків. Але тісна інтеграція з веб-сторінками і
|
|
|
|
|
вбудована підтримка браузерами призвела до того, що JavaScript став популярніший
|
|
|
|
|
за власне Java.
|
|
|
|
|
|
2017-10-11 18:50:51 +00:00
|
|
|
|
Зараз JavaScript не обмежується тільки веб-браузером. Наприклад, Node.js,
|
2015-10-30 15:53:40 +00:00
|
|
|
|
програмна платформа, що дозволяє виконувати JavaScript код з використанням
|
|
|
|
|
рушія V8 від браузера Google Chrome, стає все більш і більш популярною.
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
// С-подібні коментарі. Однорядкові коментарі починаються з двох символів /(слеш)
|
|
|
|
|
/* а багаторядкові коментарі починаються з послідовності слеша та зірочки і
|
|
|
|
|
закінчуються символами зірочка-слеш */
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
//Інструкції можуть закінчуватися крапкою з комою ;
|
2015-10-30 15:53:40 +00:00
|
|
|
|
doStuff();
|
|
|
|
|
|
|
|
|
|
// ... але не обов’язково, тому що крапка з комою автоматично вставляється на
|
|
|
|
|
// місці символу нового рядка, крім деяких випадків.
|
|
|
|
|
doStuff()
|
|
|
|
|
|
|
|
|
|
// Ми завжди будемо використовувати крапку з комою в цьому посібнику, тому що ці
|
|
|
|
|
// винятки можуть призвести до неочікуваних результатів
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////
|
|
|
|
|
// 1. Числа, Рядки і Оператори
|
|
|
|
|
|
|
|
|
|
// В JavaScript числа зберігаються тільки в одному форматі (64-bit IEEE 754 double)
|
|
|
|
|
// Цей тип має 52-бітну мантису, якої достатньо для збереження чисел з
|
|
|
|
|
// точністю до 9✕10¹⁵.
|
|
|
|
|
3; // = 3
|
|
|
|
|
1.5; // = 1.5
|
|
|
|
|
|
2018-07-27 20:48:58 +00:00
|
|
|
|
// Деякі прості арифметичні операції працюють так, як ми очікуємо.
|
2015-10-30 15:53:40 +00:00
|
|
|
|
1 + 1; // = 2
|
|
|
|
|
0.1 + 0.2; // = 0.30000000000000004 (а деякі - ні)
|
|
|
|
|
8 - 1; // = 7
|
|
|
|
|
10 * 2; // = 20
|
|
|
|
|
35 / 5; // = 7
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// В тому числі ділення з остачею
|
2015-10-30 15:53:40 +00:00
|
|
|
|
5 / 2; // = 2.5
|
|
|
|
|
|
|
|
|
|
// В JavaScript є побітові операції; коли ви виконуєте таку операцію,
|
|
|
|
|
// число з плаваючою точкою переводиться в ціле зі знаком
|
|
|
|
|
// довжиною *до* 32 розрядів.
|
|
|
|
|
1 << 2; // = 4
|
|
|
|
|
|
|
|
|
|
// Пріоритет у виразах можна задати явно круглими дужками
|
|
|
|
|
(1 + 3) * 2; // = 8
|
|
|
|
|
|
|
|
|
|
// Є три спеціальні значення, які не є реальними числами:
|
|
|
|
|
Infinity; // "нескінченність", наприклад, як результат ділення на 0
|
|
|
|
|
-Infinity; // "мінус нескінченність", як результат ділення від’ємного числа на 0
|
|
|
|
|
NaN; // "не число", наприклад, ділення 0/0
|
|
|
|
|
|
|
|
|
|
// Логічні типи
|
|
|
|
|
true;
|
|
|
|
|
false;
|
|
|
|
|
|
|
|
|
|
// Рядки створюються за допомогою подвійних та одинарних лапок
|
|
|
|
|
'абв';
|
2015-11-27 10:36:10 +00:00
|
|
|
|
"Привіт, світе!";
|
2015-10-30 15:53:40 +00:00
|
|
|
|
|
|
|
|
|
// Для логічного заперечення використовується знак оклику.
|
|
|
|
|
!true; // = false
|
|
|
|
|
!false; // = true
|
|
|
|
|
|
|
|
|
|
// Строга рівність ===
|
|
|
|
|
1 === 1; // = true
|
|
|
|
|
2 === 1; // = false
|
|
|
|
|
|
|
|
|
|
// Строга нерівність !==
|
|
|
|
|
1 !== 1; // = false
|
|
|
|
|
2 !== 1; // = true
|
|
|
|
|
|
|
|
|
|
// Інші оператори порівняння
|
|
|
|
|
1 < 10; // = true
|
|
|
|
|
1 > 10; // = false
|
|
|
|
|
2 <= 2; // = true
|
|
|
|
|
2 >= 2; // = true
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// Рядки об’єднуються за допомогою оператора +
|
2015-10-30 15:53:40 +00:00
|
|
|
|
"hello, " + "world!"; // = "hello, world!"
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// І порівнюються за допомогою > та <
|
2015-10-30 15:53:40 +00:00
|
|
|
|
"a" < "b"; // = true
|
|
|
|
|
|
|
|
|
|
// Перевірка на рівність з приведнням типів здійснюється оператором ==
|
|
|
|
|
"5" == 5; // = true
|
|
|
|
|
null == undefined; // = true
|
|
|
|
|
|
|
|
|
|
// ... але приведення не виконується при ===
|
|
|
|
|
"5" === 5; // = false
|
2018-07-27 20:48:58 +00:00
|
|
|
|
null === undefined; // = false
|
2015-10-30 15:53:40 +00:00
|
|
|
|
|
|
|
|
|
// ... приведення типів може призвести до дивних результатів
|
|
|
|
|
13 + !0; // 14
|
|
|
|
|
"13" + !0; // '13true'
|
|
|
|
|
|
|
|
|
|
// Можна отримати доступ до будь-якого символа рядка за допомгою charAt
|
2015-11-11 06:58:13 +00:00
|
|
|
|
"Це рядок".charAt(0); // = 'Ц'
|
2015-10-30 15:53:40 +00:00
|
|
|
|
|
|
|
|
|
// ... або використати метод substring, щоб отримати більший кусок
|
|
|
|
|
"Hello, world".substring(0, 5); // = "Hello"
|
|
|
|
|
|
|
|
|
|
// length - це не метод, а поле
|
|
|
|
|
"Hello".length; // = 5
|
|
|
|
|
|
|
|
|
|
// Типи null и undefined
|
|
|
|
|
null; // навмисна відсутність результату
|
|
|
|
|
undefined; // використовується для позначення відсутності присвоєного значення
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// false, null, undefined, NaN, 0 та "" — хиба; все інше - істина.
|
|
|
|
|
// Потрібно відмітити, що 0 — це хиба, а "0" — істина, не зважаючи на те що:
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// 0 == "0".
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////
|
|
|
|
|
// 2. Змінні, Масиви, Об’єкти
|
|
|
|
|
|
|
|
|
|
// Змінні оголошуються за допомогою ключового слова var. JavaScript — мова з
|
|
|
|
|
// динамічною типізацією, тому не потрібно явно вказувати тип. Для присвоєння
|
|
|
|
|
// значення змінної використовується символ =
|
|
|
|
|
var someVar = 5;
|
|
|
|
|
|
|
|
|
|
// якщо пропустити слово var, ви не отримаєте повідомлення про помилку, ...
|
|
|
|
|
someOtherVar = 10;
|
|
|
|
|
|
2015-11-27 10:36:10 +00:00
|
|
|
|
// ... але вашу змінну буде створено в глобальному контексті, а не там, де
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// ви її оголосили
|
|
|
|
|
|
|
|
|
|
// Змінні, які оголошені без присвоєння, автоматично приймають значення undefined
|
|
|
|
|
var someThirdVar; // = undefined
|
|
|
|
|
|
|
|
|
|
// У математичних операцій є скорочені форми:
|
|
|
|
|
someVar += 5; // як someVar = someVar + 5;
|
|
|
|
|
someVar *= 10; // тепер someVar = 100
|
|
|
|
|
|
|
|
|
|
// Інкремент і декремент
|
|
|
|
|
someVar++; // тепер someVar дорівнює 101
|
|
|
|
|
someVar--; // а зараз 100
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// Масиви — це нумеровані списки, які зберігають значення будь-якого типу.
|
|
|
|
|
var myArray = ["Привіт", 45, true];
|
2015-10-30 15:53:40 +00:00
|
|
|
|
|
|
|
|
|
// Доступ до елементів можна отримати за допомогою синтаксиса з квадратними дужками
|
|
|
|
|
// Індексація починається з нуля
|
|
|
|
|
myArray[1]; // = 45
|
|
|
|
|
|
2015-11-27 10:39:27 +00:00
|
|
|
|
// Масиви в JavaScript змінюють свою довжину при додаванні нових елементів
|
2015-11-11 06:58:13 +00:00
|
|
|
|
myArray.push("Привіт");
|
2015-10-30 15:53:40 +00:00
|
|
|
|
myArray.length; // = 4
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// Додавання і редагування елементів
|
|
|
|
|
myArray[3] = "світ";
|
2015-10-30 15:53:40 +00:00
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// Об’єкти в JavaScript схожі на словники або асоціативні масиви в інших мовах
|
2015-10-30 15:53:40 +00:00
|
|
|
|
var myObj = {key1: "Hello", key2: "World"};
|
|
|
|
|
|
2018-07-27 20:48:58 +00:00
|
|
|
|
// Ключі - це рядки, але лапки не обов’язкові, якщо ключ задовольняє
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// правилам формування назв змінних. Значення можуть бути будь-яких типів.
|
|
|
|
|
var myObj = {myKey: "myValue", "my other key": 4};
|
|
|
|
|
|
|
|
|
|
// Атрибути можна отримати використовуючи квадратні дужки
|
|
|
|
|
myObj["my other key"]; // = 4
|
|
|
|
|
|
|
|
|
|
// Або через точку, якщо ключ є правильним ідентифікатором
|
|
|
|
|
myObj.myKey; // = "myValue"
|
|
|
|
|
|
|
|
|
|
// Об’єкти можна динамічно змінювати й додавати нові поля
|
|
|
|
|
myObj.myThirdKey = true;
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// Коли ви звертаєтесь до поля, що не існує, ви отримуєте значення undefined
|
2015-10-30 15:53:40 +00:00
|
|
|
|
myObj.myFourthKey; // = undefined
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// 3. Керуючі конструкції
|
2015-10-30 15:53:40 +00:00
|
|
|
|
|
|
|
|
|
// Синтаксис для цього розділу майже такий самий, як у Java
|
|
|
|
|
|
|
|
|
|
// Умовна конструкція
|
|
|
|
|
var count = 1;
|
|
|
|
|
if (count == 3) {
|
|
|
|
|
// виконується, якщо count дорівнює 3
|
|
|
|
|
} else if (count == 4) {
|
|
|
|
|
// ..
|
|
|
|
|
} else {
|
|
|
|
|
// ...
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ... цикл while.
|
|
|
|
|
while (true){
|
|
|
|
|
// Нескінченний цикл!
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Цикл do-while такий самий, як while, але завжди виконується принаймні один раз.
|
|
|
|
|
var input
|
|
|
|
|
do {
|
|
|
|
|
input = getInput();
|
|
|
|
|
} while (!isValid(input))
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// цикл for такий самий, як в C і Java:
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// ініціалізація; умова; крок.
|
|
|
|
|
for (var i = 0; i < 5; i++) {
|
|
|
|
|
// виконається 5 разів
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// && — логічне І, || — логічне АБО
|
|
|
|
|
if (house.size == "big" && house.color == "blue") {
|
|
|
|
|
house.contains = "bear";
|
|
|
|
|
}
|
|
|
|
|
if (color == "red" || color == "blue") {
|
|
|
|
|
// колір червоний або синій
|
|
|
|
|
}
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// && та || використовують скорочене обчислення
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// тому їх можна використовувати для задання значень за замовчуванням.
|
|
|
|
|
var name = otherName || "default";
|
|
|
|
|
|
|
|
|
|
// Оператор switch виконує перевірку на рівність за допомогою ===
|
|
|
|
|
// використовуйте break, щоб призупити виконання наступного case,
|
|
|
|
|
grade = 4;
|
|
|
|
|
switch (grade) {
|
|
|
|
|
case 5:
|
|
|
|
|
console.log("Відмінно");
|
|
|
|
|
break;
|
|
|
|
|
case 4:
|
|
|
|
|
console.log("Добре");
|
|
|
|
|
break;
|
|
|
|
|
case 3:
|
|
|
|
|
console.log("Можна краще");
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
console.log("Погано!");
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////
|
|
|
|
|
// 4. Функції, область видимості і замикання
|
|
|
|
|
|
|
|
|
|
// Функції в JavaScript оголошуються за допомогою ключового слова function.
|
|
|
|
|
function myFunction(thing) {
|
|
|
|
|
return thing.toUpperCase();
|
|
|
|
|
}
|
|
|
|
|
myFunction("foo"); // = "FOO"
|
2018-07-27 20:48:58 +00:00
|
|
|
|
|
2015-11-27 10:36:10 +00:00
|
|
|
|
// Зверніть увагу, що значення яке буде повернено, повинно починатися на тому ж
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// рядку, що і ключове слово return, інакше завжди буде повертатися значення undefined
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// через автоматичну вставку крапки з комою
|
2015-10-30 15:53:40 +00:00
|
|
|
|
function myFunction()
|
|
|
|
|
{
|
|
|
|
|
return // <- крапка з комою вставляється автоматично
|
|
|
|
|
{
|
|
|
|
|
thisIsAn: 'object literal'
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
myFunction(); // = undefined
|
|
|
|
|
|
|
|
|
|
// В JavaScript функції - це об`єкти першого класу, тому вони можуть присвоюватися
|
|
|
|
|
// іншим змінним і передаватися іншим функціям, наприклад, щоб визначити обробник
|
|
|
|
|
// події.
|
|
|
|
|
function myFunction() {
|
|
|
|
|
// код буде виконано через 5 сек.
|
|
|
|
|
}
|
|
|
|
|
setTimeout(myFunction, 5000);
|
|
|
|
|
// setTimeout не є частиною мови, але реалізований в браузерах і Node.js
|
|
|
|
|
|
|
|
|
|
// Функції не обов’язково мають мати ім’я при оголошенні — ви можете написати
|
2015-11-27 10:36:10 +00:00
|
|
|
|
// анонімну функцію як аргумент іншої функції
|
2015-10-30 15:53:40 +00:00
|
|
|
|
setTimeout(function() {
|
|
|
|
|
// Цей код буде виконано через п’ять секунд
|
|
|
|
|
}, 5000);
|
|
|
|
|
|
|
|
|
|
// В JavaScript реалізована концепція області видимості; функції мають свою
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// область видимості, а інші блоки не мають
|
2015-10-30 15:53:40 +00:00
|
|
|
|
if (true) {
|
|
|
|
|
var i = 5;
|
|
|
|
|
}
|
2015-11-11 06:58:13 +00:00
|
|
|
|
i; // = 5, а не undefined, як це звичайно буває в інших мовах
|
2015-10-30 15:53:40 +00:00
|
|
|
|
|
|
|
|
|
// Така особливість призвела до шаблону "анонімних функцій, які викликають самих себе"
|
|
|
|
|
// що дозволяє уникнути проникнення змінних в глобальну область видимості
|
|
|
|
|
(function() {
|
|
|
|
|
var temporary = 5;
|
|
|
|
|
// об’єкт window зберігає глобальний контекст; таким чином ми можемо також додавати
|
|
|
|
|
// змінні до глобальної області
|
|
|
|
|
window.permanent = 10;
|
|
|
|
|
})();
|
|
|
|
|
temporary; // повідомлення про помилку ReferenceError
|
|
|
|
|
permanent; // = 10
|
|
|
|
|
|
2015-11-27 10:36:10 +00:00
|
|
|
|
// Замикання - один з найпотужніших інструментів JavaScript. Якщо функція визначена
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// всередині іншої функції, то внутрішня функція має доступ до змінних зовнішньої
|
|
|
|
|
// функції навіть після того, як код буде виконуватися поза контекстом зовнішньої функції
|
|
|
|
|
function sayHelloInFiveSeconds(name) {
|
2015-11-11 06:58:13 +00:00
|
|
|
|
var prompt = "Привіт, " + name + "!";
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// Внутрішня функція зберігається в локальній області так,
|
|
|
|
|
// ніби функція була оголошена за допомогою ключового слова var
|
|
|
|
|
function inner() {
|
|
|
|
|
alert(prompt);
|
|
|
|
|
}
|
|
|
|
|
setTimeout(inner, 5000);
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// setTimeout асинхронна, тому функція sayHelloInFiveSeconds одразу завершиться,
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// після чого setTimeout викличе функцію inner. Але функція inner
|
|
|
|
|
// «замкнута» кругом sayHelloInFiveSeconds, вона все рівно має доступ до змінної prompt
|
|
|
|
|
}
|
2015-11-11 06:58:13 +00:00
|
|
|
|
sayHelloInFiveSeconds("Адам"); // Через 5 с відкриється вікно «Привіт, Адам!»
|
2015-10-30 15:53:40 +00:00
|
|
|
|
|
|
|
|
|
///////////////////////////////////
|
|
|
|
|
// 5. Об’єкти: конструктори і прототипи
|
|
|
|
|
|
|
|
|
|
// Об’єкти можуть містити функції
|
|
|
|
|
var myObj = {
|
|
|
|
|
myFunc: function() {
|
|
|
|
|
return "Hello, world!";
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
myObj.myFunc(); // = "Hello, world!"
|
|
|
|
|
|
2018-07-27 20:48:58 +00:00
|
|
|
|
// Функції, що прикріплені до об’єктів мають доступ до поточного об’єкта за
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// допомогою ключового слова this.
|
|
|
|
|
myObj = {
|
|
|
|
|
myString: "Hello, world!",
|
|
|
|
|
myFunc: function() {
|
|
|
|
|
return this.myString;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
myObj.myFunc(); // = "Hello, world!"
|
|
|
|
|
|
|
|
|
|
// Значення this залежить від того, як функція викликається
|
|
|
|
|
// а не від того, де вона визначена. Таким чином наша функція не працює, якщо
|
|
|
|
|
// вона викликана не в контексті об’єкта
|
|
|
|
|
var myFunc = myObj.myFunc;
|
|
|
|
|
myFunc(); // = undefined
|
|
|
|
|
|
2018-07-27 20:48:58 +00:00
|
|
|
|
// Функція може бути присвоєна іншому об’єкту. Тоді вона матиме доступ до
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// цього об’єкта через this
|
|
|
|
|
var myOtherFunc = function() {
|
2016-05-14 11:35:38 +00:00
|
|
|
|
return this.myString.toUpperCase();
|
2015-10-30 15:53:40 +00:00
|
|
|
|
}
|
|
|
|
|
myObj.myOtherFunc = myOtherFunc;
|
|
|
|
|
myObj.myOtherFunc(); // = "HELLO, WORLD!"
|
|
|
|
|
|
|
|
|
|
// Контекст виконання функції можна задати за допомогою сall або apply
|
|
|
|
|
var anotherFunc = function(s) {
|
|
|
|
|
return this.myString + s;
|
|
|
|
|
}
|
|
|
|
|
anotherFunc.call(myObj, " Hello!"); // = "Hello, world! Hello!"
|
|
|
|
|
|
|
|
|
|
// Функцiя apply приймає в якості аргументу масив
|
|
|
|
|
anotherFunc.apply(myObj, [" Hello!"]); // = "Hello, world! Hello!"
|
|
|
|
|
|
|
|
|
|
// apply можна використати, коли функція працює послідовністю аргументів, а
|
|
|
|
|
// ви хочете передати масив
|
|
|
|
|
Math.min(42, 6, 27); // = 6
|
|
|
|
|
Math.min([42, 6, 27]); // = NaN (Ой-ой!)
|
|
|
|
|
Math.min.apply(Math, [42, 6, 27]); // = 6
|
|
|
|
|
|
2018-07-27 20:48:58 +00:00
|
|
|
|
// Але call і apply — тимчасові. Коли ми хочемо зв’язати функцію і об’єкт
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// використовують bind
|
|
|
|
|
var boundFunc = anotherFunc.bind(myObj);
|
|
|
|
|
boundFunc(" Hello!"); // = "Hello world, Hello!"
|
|
|
|
|
|
|
|
|
|
// Bind можна використати для задання аргументів
|
|
|
|
|
var product = function(a, b) { return a * b; }
|
|
|
|
|
var doubler = product.bind(this, 2);
|
|
|
|
|
doubler(8); // = 16
|
|
|
|
|
|
|
|
|
|
// Коли ви викликаєте функцію за допомогою ключового слова new, створюється новий об’єкт,
|
|
|
|
|
// доступний функції за допомогою this. Такі функції називають конструкторами.
|
|
|
|
|
var MyConstructor = function() {
|
|
|
|
|
this.myNumber = 5;
|
|
|
|
|
}
|
|
|
|
|
myNewObj = new MyConstructor(); // = {myNumber: 5}
|
|
|
|
|
myNewObj.myNumber; // = 5
|
|
|
|
|
|
|
|
|
|
// У кожного об’єкта є прототип. Коли ви звертаєтесь до поля, яке не існує в цьому
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// об’єкті, інтерпретатор буде шукати поле в прототипі
|
2015-10-30 15:53:40 +00:00
|
|
|
|
|
|
|
|
|
// Деякі реалізації мови дозволяють отримати доступ до прототипа об’єкта через
|
|
|
|
|
// "магічну" властивість __proto__. Це поле не є частиною стандарта, але існують
|
|
|
|
|
// стандартні способи використання прототипів, які ми побачимо пізніше
|
|
|
|
|
var myObj = {
|
|
|
|
|
myString: "Hello, world!"
|
|
|
|
|
};
|
|
|
|
|
var myPrototype = {
|
|
|
|
|
meaningOfLife: 42,
|
|
|
|
|
myFunc: function() {
|
|
|
|
|
return this.myString.toLowerCase()
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
myObj.__proto__ = myPrototype;
|
|
|
|
|
myObj.meaningOfLife; // = 42
|
|
|
|
|
|
|
|
|
|
// Аналогічно для функцій
|
2016-05-14 11:35:38 +00:00
|
|
|
|
myObj.myFunc(); // = "hello, world!"
|
2015-10-30 15:53:40 +00:00
|
|
|
|
|
2015-11-27 10:36:10 +00:00
|
|
|
|
// Якщо інтерпретатор не знайде властивості в прототипі, то він продовжить пошук
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// в прототипі прототипа і так далі
|
|
|
|
|
myPrototype.__proto__ = {
|
|
|
|
|
myBoolean: true
|
|
|
|
|
};
|
|
|
|
|
myObj.myBoolean; // = true
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// Кожен об’єкт зберігає посилання на свій прототип. Це значить, що ми можемо змінити
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// наш прототип, і наші зміни будуть всюди відображені.
|
|
|
|
|
myPrototype.meaningOfLife = 43;
|
|
|
|
|
myObj.meaningOfLife; // = 43
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// Ми сказали, що властивість __proto__ нестандартна, і нема ніякого стандартного способу
|
|
|
|
|
// змінити прототип об’єкта, що вже існує. Але є два способи створити новий об’єкт із заданим
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// прототипом
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// Перший спосіб — це Object.create, який з’явився в JavaScript недавно,
|
|
|
|
|
// а тому в деяких реалізаціях може бути недоступним.
|
2015-10-30 15:53:40 +00:00
|
|
|
|
var myObj = Object.create(myPrototype);
|
|
|
|
|
myObj.meaningOfLife; // = 43
|
|
|
|
|
|
|
|
|
|
// Другий спосіб: у конструкторів є властивість з іменем prototype. Це *не*
|
|
|
|
|
// прототип функції-конструктора, це прототип для нових об’єктів, які будуть створені
|
2016-05-14 11:35:38 +00:00
|
|
|
|
// цим конструктором і ключовим словом new.
|
2015-10-30 15:53:40 +00:00
|
|
|
|
MyConstructor.prototype = {
|
|
|
|
|
myNumber: 5,
|
|
|
|
|
getMyNumber: function() {
|
|
|
|
|
return this.myNumber;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
var myNewObj2 = new MyConstructor();
|
|
|
|
|
myNewObj2.getMyNumber(); // = 5
|
|
|
|
|
myNewObj2.myNumber = 6
|
|
|
|
|
myNewObj2.getMyNumber(); // = 6
|
|
|
|
|
|
|
|
|
|
// У вбудованих типів(рядок, число) теж є конструктори, які створють еквівалентні
|
|
|
|
|
// об’єкти-обгортки
|
|
|
|
|
var myNumber = 12;
|
|
|
|
|
var myNumberObj = new Number(12);
|
|
|
|
|
myNumber == myNumberObj; // = true
|
|
|
|
|
|
|
|
|
|
// Але вони не ідентичні
|
|
|
|
|
typeof myNumber; // = 'number'
|
|
|
|
|
typeof myNumberObj; // = 'object'
|
|
|
|
|
myNumber === myNumberObj; // = false
|
|
|
|
|
|
|
|
|
|
// Об’єкти-обгортки і вбудовані типи мають спільні прототипи, тому
|
|
|
|
|
// ви можете розширити функціонал рядків:
|
|
|
|
|
String.prototype.firstCharacter = function() {
|
|
|
|
|
return this.charAt(0);
|
|
|
|
|
}
|
|
|
|
|
"abc".firstCharacter(); // = "a"
|
|
|
|
|
|
|
|
|
|
// Такий прийом часто використовуються в поліфілах, які реалізують нові можливості
|
|
|
|
|
// JavaScript в старій реалізації мови, так що вони можуть бути використані в старих
|
|
|
|
|
// середовищах
|
|
|
|
|
|
2015-11-11 06:58:13 +00:00
|
|
|
|
// Наприклад, Object.create доступний не у всіх реалізаціях, але ми можемо
|
2015-10-30 15:53:40 +00:00
|
|
|
|
// використати функції за допомогою наступного поліфіла:
|
2015-11-11 06:58:13 +00:00
|
|
|
|
if (Object.create === undefined) { // не перезаписуємо метод, якщо він існує
|
2015-10-30 15:53:40 +00:00
|
|
|
|
Object.create = function(proto) {
|
|
|
|
|
// Створюємо правильний конструктор з правильним прототипом
|
|
|
|
|
var Constructor = function(){};
|
|
|
|
|
Constructor.prototype = proto;
|
2018-07-27 20:48:58 +00:00
|
|
|
|
|
2015-10-30 15:53:40 +00:00
|
|
|
|
return new Constructor();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
## Що почитати
|
|
|
|
|
|
2024-12-03 23:56:27 +00:00
|
|
|
|
* https://developer.mozilla.org/en-US/docs/Web/JavaScript
|
|
|
|
|
* https://developer.mozilla.org/en-US/docs/Web/JavaScript/A_re-introduction_to_JavaScript
|
|
|
|
|
* https://developer.mozilla.org/en-US/docs/Using_the_W3C_DOM_Level_1_Core
|
|
|
|
|
* https://shamansir.github.io/JavaScript-Garden/
|
|
|
|
|
* http://www.amazon.com/gp/product/0596805527/
|
|
|
|
|
* https://developer.mozilla.org/en-US/docs/Web/JavaScript/A_re-introduction_to_JavaScript
|
|
|
|
|
* http://eloquentjavascript.net/
|
|
|
|
|
* http://jstherightway.org/
|