Publikuję pytania wielokrotnego wyboru dotyczące JavaScriptu na swoich Instagram stories, które również zamieszczę tutaj! Ostatnia aktualizacja: 27 Czerwca
Od podstawowych do zaawansowanych: sprawdź, jak dobrze znasz JavaScript, odśwież swoją wiedzę lub przygotuj się do rozmowy kwalifikacyjnej! 💪 🚀 Regularnie aktualizuję to repozytorium nowymi pytaniami. Odpowiedzi znajdują się w ukrytych zakładkach poniżej pytań - po prostu kliknij, aby je rozwinięć. To dla zabawy, powodzenia! ❤️
Nie krępuj się ze mną kontaktować! 😊
Instagram || Twitter || LinkedIn || Blog
Śmiało używaj ich w projekcie! 😃 Byłabym bardzo wdzięczna za referencje do tego repozytorium, tworzę pytania i wyjaśnienia (tak, jestem smutna lol) i społeczność bardzo mi pomaga w utrzymaniu i ulepszaniu go! 💪🏼 Dziękuję i baw się dobrze! |
---|
Zobacz 20 dostępnych tłumaczeń 🇸🇦🇪🇬🇧🇦🇩🇪🇪🇸🇫🇷🇮🇩🇯🇵🇰🇷🇳🇱🇧🇷🇷🇺🇹🇭🇹🇷🇺🇦🇻🇳🇨🇳🇹🇼🇽🇰
function sayHi() {
console.log(name);
console.log(age);
var name = 'Lydia';
let age = 21;
}
sayHi();
- A:
Lydia
andundefined
- B:
Lydia
andReferenceError
- C:
ReferenceError
and21
- D:
undefined
andReferenceError
Odpowiedź
Wewnątrz funkcji najpierw deklarujemy zmienną name
za pomocą słowa kluczowego var
. Oznacza to, że zmienna jest "wyciągana" (przestrzeń pamięci jest tworzona) z domyślną wartością undefined
podczas fazy tworzenia, aż do momentu, gdy naprawdę definiujemy zmienną. W linii, w której próbujemy wyświetlić w konsoli zmienną name
, jeszcze jej nie zdefiniowaliśmy, więc nadal przechowuje wartość undefined
.
Zmienne zadeklarowane za pomocą słowa kluczowego let
(i const
) są wyciągane, ale w przeciwieństwie do var
, nie są inicjalizowane. Nie są dostępne przed linią, na której je deklarujemy (inicjalizujemy). Nazywa się to "czasową strefą martwą" (temporal dead zone). Gdy próbujemy uzyskać dostęp do zmiennych przed ich zadeklarowaniem, JavaScript generuje błąd ReferenceError
.
for (var i = 0; i < 3; i++) {
setTimeout(() => console.log(i), 1);
}
for (let i = 0; i < 3; i++) {
setTimeout(() => console.log(i), 1);
}
- A:
0 1 2
and0 1 2
- B:
0 1 2
and3 3 3
- C:
3 3 3
and0 1 2
Odpowiedź
Ze względu na kolejkę zdarzeń w JavaScript, funkcja zwrotna setTimeout
jest wywoływana po wykonaniu pętli. Ponieważ zmienna i
w pierwszej pętli została zadeklarowana za pomocą słowa kluczowego var
, jej wartość była globalna. Podczas pętli inkrementowaliśmy wartość i
o 1
za każdym razem, używając operatora jednoargumentowego ++
. W momencie wywołania funkcji zwrotnej setTimeout
, i
miało wartość 3
w pierwszym przykładzie.
W drugiej pętli zmienna i
została zadeklarowana za pomocą słowa kluczowego let
: zmienne zadeklarowane za pomocą słowa kluczowego let
(i const
) mają zakres blokowy (blokiem jest cokolwiek między { }
). Podczas każdej iteracji i
będzie miało nową wartość, a każda wartość będzie miała zakres wewnątrz pętli.
const shape = {
radius: 10,
diameter() {
return this.radius * 2;
},
perimeter: () => 2 * Math.PI * this.radius,
};
console.log(shape.diameter());
console.log(shape.perimeter());
- A:
20
and62.83185307179586
- B:
20
andNaN
- C:
20
and63
- D:
NaN
and63
Odpowiedź
Zwróć uwagę, że wartość diameter
jest zwykłą funkcją, podczas gdy wartość perimeter
jest funkcją strzałkową.
W przypadku funkcji strzałkowych, słowo kluczowe this
odnosi się do bieżącego otaczającego zakresu, w przeciwieństwie do zwykłych funkcji! Oznacza to, że gdy wywołujemy perimeter
, nie odnosi się ono do obiektu shape, ale do swojego otaczającego zakresu (np. okna).
Na tym obiekcie nie ma wartości radius
, co powoduje zwrócenie NaN
(Not a Number).
+true;
!'Lydia';
- A:
1
andfalse
- B:
false
andNaN
- C:
false
andfalse
Odpowiedź
Operator jednoargumentowy plus próbuje przekonwertować operand na liczbę. true
jest równoważne 1
, a false
jest równoważne 0
.
Łańcuch znaków 'Lydia'
jest wartością prawdziwą. Tak naprawdę pytamy, "czy ta wartość prawdziwa jest fałszywa?". To zwraca false
.
const bird = {
size: 'small',
};
const mouse = {
name: 'Mickey',
small: true,
};
- A:
mouse.bird.size
is not valid - B:
mouse[bird.size]
is not valid - C:
mouse[bird["size"]]
is not valid - D: All of them are valid
Odpowiedź
W JavaScript wszystkie klucze obiektów są stringami (chyba że są to symbole). Nawet jeśli nie wpisujemy ich jako stringi, zawsze są one konwertowane na stringi wewnątrz.
JavaScript interpretuje (lub "odpakuowuje") instrukcje. Gdy używamy notacji nawiasów kwadratowych, interpreter widzi pierwszy otwierający nawias [
i kontynuuje do momentu znalezienia zamykającego nawiasu ]
. Dopiero wtedy ocenia tę instrukcję.
mouse[bird.size]
: Najpierw ocenia bird.size
, które wynosi "small"
. mouse["small"]
zwraca true
.
Jednakże, w przypadku notacji kropkowej, to się nie dzieje. mouse
nie ma klucza o nazwie bird
, co oznacza, że mouse.bird
jest undefined
. Następnie pytamy o size
używając notacji kropkowej: mouse.bird.size
. Ponieważ mouse.bird
jest undefined
, tak naprawdę pytamy o undefined.size
. To nie jest poprawne i spowoduje błąd podobny do Cannot read property "size" of undefined
(Nie można odczytać właściwości "size" z undefined).
let c = { greeting: 'Hey!' };
let d;
d = c;
c.greeting = 'Hello';
console.log(d.greeting);
- A:
Hello
- B:
Hey!
- C:
undefined
- D:
ReferenceError
- E:
TypeError
Odpowiedź
W JavaScript wszystkie obiekty komunikują się ze sobą przez referencje, gdy są sobie przypisywane.
Na początku zmienna c
przechowuje referencję do obiektu. Później przypisujemy zmiennej d
tę samą referencję, którą ma c
, do tego obiektu.
Kiedy zmieniasz jeden obiekt, zmieniasz je wszystkie.
let a = 3;
let b = new Number(3);
let c = 3;
console.log(a == b);
console.log(a === b);
console.log(b === c);
- A:
true
false
true
- B:
false
false
true
- C:
true
false
false
- D:
false
true
true
Odpowiedź
new Number()
jest wbudowanym konstruktorem funkcji. Chociaż wygląda jak liczba, nie jest faktycznie liczbą: ma wiele dodatkowych funkcji i jest obiektem.
Gdy używamy operatora ==
(operator równości), sprawdza on jedynie, czy mają tą samą wartość. Oba mają wartość 3
, więc zwraca true
.
Jednak gdy używamy operatora ===
(operator ścisłej równości), zarówno wartość, jak i typ powinny być takie same. Tutaj nie są: new Number()
nie jest liczbą, lecz obiektem. Oba zwracają false
.
class Chameleon {
static colorChange(newColor) {
this.newColor = newColor;
return this.newColor;
}
constructor({ newColor = 'green' } = {}) {
this.newColor = newColor;
}
}
const freddie = new Chameleon({ newColor: 'purple' });
console.log(freddie.colorChange('orange'));
- A:
orange
- B:
purple
- C:
green
- D:
TypeError
Odpowiedź
Funkcja colorChange
jest statyczna. Metody statyczne są zaprojektowane tak, aby istniały tylko w konstruktorze, w którym zostały utworzone, i nie mogą być przekazywane do żadnych potomków (children) ani wywoływane na instancjach klasy. Ponieważ freddie
jest instancją klasy Chameleon, funkcja nie może być na niej wywołana. Otrzymujemy błąd TypeError
.
let greeting;
greetign = {}; // Celowa Literówka!
console.log(greetign);
- A:
{}
- B:
ReferenceError: greetign is not defined
- C:
undefined
Odpowiedź
Kod wypisuje w konsoli obiekt, ponieważ właśnie utworzyliśmy pusty obiekt w obiekcie globalnym! Gdy pomyłkowo wpisaliśmy greeting
jako greetign
, interpreter JavaScript faktycznie zobaczył to jako:
global.greetign = {}
w Node.js.window.greetign = {}
,frames.greetign = {}
iself.greetign
w przeglądarkach.self.greetign
w web workerach.globalThis.greetign
we wszystkich środowiskach.
Aby temu zapobiec, możemy użyć "use strict"
. Powoduje to, że musisz zadeklarować zmienną przed jej przypisaniem.
function bark() {
console.log('Woof!');
}
bark.animal = 'dog';
- A: Nothing, this is totally fine!
- B:
SyntaxError
. You cannot add properties to a function this way. - C:
"Woof"
gets logged. - D:
ReferenceError
Odpowiedź
Jest to możliwe w JavaScript, ponieważ funkcje są obiektami! (Wszystko oprócz typów prymitywnych jest obiektem)
Funkcja jest specjalnym rodzajem obiektu. Kod, który sam piszesz, nie jest właściwą funkcją. Funkcja jest obiektem posiadającym właściwość, która jest wywoływalna.
function Person(firstName, lastName) {
this.firstName = firstName;
this.lastName = lastName;
}
const member = new Person('Lydia', 'Hallie');
Person.getFullName = function() {
return `${this.firstName} ${this.lastName}`;
};
console.log(member.getFullName());
- A:
TypeError
- B:
SyntaxError
- C:
Lydia Hallie
- D:
undefined
undefined
Odpowiedź
W JavaScript funkcje są obiektami, więc metoda getFullName
jest dodawana do samego obiektu funkcji konstruktora. Dlatego możemy wywołać Person.getFullName()
, ale member.getFullName
zwraca błąd TypeError
.
Jeśli chcesz, aby metoda była dostępna dla wszystkich instancji obiektów, musisz dodać ją do właściwości prototype:
Person.prototype.getFullName = function() {
return `${this.firstName} ${this.lastName}`;
};
function Person(firstName, lastName) {
this.firstName = firstName;
this.lastName = lastName;
}
const lydia = new Person('Lydia', 'Hallie');
const sarah = Person('Sarah', 'Smith');
console.log(lydia);
console.log(sarah);
- A:
Person {firstName: "Lydia", lastName: "Hallie"}
andundefined
- B:
Person {firstName: "Lydia", lastName: "Hallie"}
andPerson {firstName: "Sarah", lastName: "Smith"}
- C:
Person {firstName: "Lydia", lastName: "Hallie"}
and{}
- D:
Person {firstName: "Lydia", lastName: "Hallie"}
andReferenceError
Odpowiedź
Dla sarah
, nie użyliśmy słowa kluczowego new
. Kiedy używamy new
, this
odwołuje się do nowego pustego obiektu, który tworzymy. Jednak jeśli nie dodajemy new
, this
odwołuje się do globalnego obiektu!
Mówiliśmy, że this.firstName
równa się "Sarah"
, a this.lastName
równa się "Smith"
. Czyli faktycznie zdefiniowaliśmy global.firstName = 'Sarah'
i global.lastName = 'Smith'
. sarah
pozostaje undefined
, ponieważ nie zwracaliśmy żadnej wartości z funkcji Person
.
- A: Target > Capturing > Bubbling
- B: Bubbling > Target > Capturing
- C: Target > Bubbling > Capturing
- D: Capturing > Target > Bubbling
Odpowiedź
W fazie capturing (przechwytywanie), zdarzenie przechodzi przez elementy nadrzędne w doł do elementu docelowego. Następnie dociera do elementu target (cel) i rozpoczyna się bubbling (bąbelkowanie).
- A: true
- B: false
Odpowiedź
Wszystkie obiekty mają prototypy, z wyjątkiem obiektu bazowego. Obiekt bazowy jest obiektem utworzonym przez użytkownika lub obiektem utworzonym przy użyciu słowa kluczowego new
. Obiekt bazowy ma dostęp do niektórych metod i właściwości, takich jak .toString
. Jest to powód, dla którego można używać wbudowanych metod JavaScript! Wszystkie takie metody są dostępne w prototypie. Chociaż JavaScript nie może znaleźć ich bezpośrednio w twoim obiekcie, przechodzi w dół łańcucha prototypów i je tam znajduje, co czyni je dostępnymi dla ciebie.
function sum(a, b) {
return a + b;
}
sum(1, '2');
- A:
NaN
- B:
TypeError
- C:
"12"
- D:
3
Odpowiedź
JavaScript jest językiem dynamicznie typowanym: nie określamy typów niektórych zmiennych. Wartości mogą być automatycznie konwertowane na inny typ bez wiedzy użytkownika, co nazywa się implicit type coercion. Koercja (Wymuszenie) to konwersja z jednego typu na inny.
W tym przykładzie JavaScript konwertuje liczbę 1
na string, aby funkcja miała sens i zwróciła wartość. Podczas dodawania typu liczbowego (1
) i typu łańcuchowego ('2'
), liczba traktowana jest jako string. Możemy łączyć stringi takie jak "Hello" + "World"
, więc to co się tutaj dzieje to "1" + "2"
, które zwraca "12"
.
let number = 0;
console.log(number++);
console.log(++number);
console.log(number);
- A:
1
1
2
- B:
1
2
2
- C:
0
2
2
- D:
0
1
2
Odpowiedź
Operator jednoargumentowy Postfix ++
:
- Zwraca wartość (ten zwraca
0
) - Zwiększa wartość (liczba wynosi teraz
1
)
Operator jednoargumentowy Prefix ++
:
- Zwiększa wartość (liczba wynosi teraz
2
) - Zwraca wartość (to zwraca
2
)
number zwraca 0 2 2
.
function getPersonInfo(one, two, three) {
console.log(one);
console.log(two);
console.log(three);
}
const person = 'Lydia';
const age = 21;
getPersonInfo`${person} is ${age} years old`;
- A:
"Lydia"
21
["", " is ", " years old"]
- B:
["", " is ", " years old"]
"Lydia"
21
- C:
"Lydia"
["", " is ", " years old"]
21
Odpowiedź
W przypadku użycia template strings
, wartością pierwszego argumentu jest zawsze tablica wartości łańcuchowych (string). Pozostałe argumenty otrzymują wartości przekazanych wyrażeń!
function checkAge(data) {
if (data === { age: 18 }) {
console.log('You are an adult!');
} else if (data == { age: 18 }) {
console.log('You are still an adult.');
} else {
console.log(`Hmm.. You don't have an age I guess`);
}
}
checkAge({ age: 18 });
- A:
You are an adult!
- B:
You are still an adult.
- C:
Hmm.. You don't have an age I guess
Odpowiedź
Podczas testowania równości, liczby i ciągi znaków są porównywane przez ich wartości, a obiekty są porównywane przez ich referencję. JavaScript sprawdza, czy obiekty mają odwołanie do tej samej lokalizacji w pamięci.
Dwa obiekty, które porównujemy, nie mają tej samej lokalizacji w pamięci: obiekt, który przekazujemy jako parametr, odwołuje się do innej lokalizacji w pamięci niż obiekt, którego użyliśmy do sprawdzenia równości.
Dlatego też zarówno { age: 18 } == { age: 18 }
i { age: 18 } == { age: 18 }
zwracają false
.
function getAge(...args) {
console.log(typeof args);
}
getAge(21);
- A:
"number"
- B:
"array"
- C:
"object"
- D:
"NaN"
Odpowiedź
Parametr reszty (...args
) pozwala nam "zbierać" wszystkie pozostałe argumenty do tablicy. Tablica to obiekt, więc typeof args
zwraca "object"
.
function getAge() {
'use strict';
age = 21;
console.log(age);
}
getAge();
- A:
21
- B:
undefined
- C:
ReferenceError
- D:
TypeError
Odpowiedź
Dzięki "use strict"
możesz upewnić się, że przypadkowo nie zadeklarujesz zmiennych globalnych. Nigdy nie zadeklarowaliśmy zmiennej age
, a ponieważ używamy "use strict"
, zostanie zgłoszony błąd referencji. Gdybyśmy nie użyli "use strict"
, to by zadziałało, ponieważ właściwość age
zostałaby dodana do obiektu globalnego.
const sum = eval('10*10+5');
- A:
105
- B:
"105"
- C:
TypeError
- D:
"10*10+5"
Odpowiedź
eval
oblicza kod, który przekazywany jest jako ciąg znaków. Jeśli jest to wyrażenie, tak jak w tym przypadku, oblicza ono wyrażenie. Wyrażenie to 10 * 10 + 5
. Zwraca liczbę 105
.
sessionStorage.setItem('cool_secret', 123);
- A: Dane nigdy nie zostaną utracone.
- B: Gdy użytkownik zamyka kartę.
- C: Gdy użytkownik zamyka cały przeglądarkę, a nie tylko kartę.
- D: Gdy użytkownik wyłącza swój komputer.
Odpowiedź
Dane przechowywane w sessionStorage
są usuwane po zamknięciu zakładki.
Gdybyś użył localStorage
, dane pozostałyby tam na zawsze, chyba że na przykład wywołano by localStorage.clear()
.
var num = 8;
var num = 10;
console.log(num);
- A:
8
- B:
10
- C:
SyntaxError
- D:
ReferenceError
Odpowiedź
Za pomocą słowa kluczowego var
można zadeklarować wiele zmiennych o tej samej nazwie. Zmienna będzie wtedy przechowywać najnowszą wartość.
Nie można tego zrobić za pomocą let
lub const
, ponieważ są one blokowe.
const obj = { 1: 'a', 2: 'b', 3: 'c' };
const set = new Set([1, 2, 3, 4, 5]);
obj.hasOwnProperty('1');
obj.hasOwnProperty(1);
set.has('1');
set.has(1);
- A:
false
true
false
true
- B:
false
true
true
true
- C:
true
true
false
true
- D:
true
true
true
true
Odpowiedź
Wszystkie klucze obiektów (z wyjątkiem symboli) są łańcuchami znaków pod względem samego obiektu, nawet jeśli nie napiszesz ich samodzielnie jako łańcucha znaków. Dlatego obj.hasOwnProperty('1')
również zwraca true.
Nie działa to w ten sam sposób dla zbioru. W zbiorze nie ma klucza '1'
:set.has('1')
, dlatego zwraca wartość false. Zawiera on liczbę całkowitą 1
, set.has(1)
zwraca wartość true.
const obj = { a: 'one', b: 'two', a: 'three' };
console.log(obj);
- A:
{ a: "one", b: "two" }
- B:
{ b: "two", a: "three" }
- C:
{ a: "three", b: "two" }
- D:
SyntaxError
Odpowiedź
Jeśli masz dwa klucze o takiej samej nazwie, zostanie on zastąpiony. Nadal będzie umieszczony na pierwszej pozycji, ale z ostatnią zdefiniowaną wartością.
26. Globalny kontekst wykonania JavaScript tworzy dwie rzeczy: obiekt globalny i słowo kluczowe "this".
- A: true
- B: false
- C: it depends
for (let i = 1; i < 5; i++) {
if (i === 3) continue;
console.log(i);
}
- A:
1
2
- B:
1
2
3
- C:
1
2
4
- D:
1
3
4
String.prototype.giveLydiaPizza = () => {
return 'Just give Lydia pizza already!';
};
const name = 'Lydia';
console.log(name.giveLydiaPizza())
- A:
"Just give Lydia pizza already!"
- B:
TypeError: not a function
- C:
SyntaxError
- D:
undefined
Odpowiedź
String
jest wbudowanym konstruktorem, do którego możemy dodawać właściwości. Dodana została metoda do jego prototypu. Prymitywne ciągi znaków są automatycznie konwertowane na obiekt typu string, generowany przez funkcję prototypu ciągu znaków. Tak więc wszystkie ciągi (obiekty typu string) mają dostęp do tej metody!
const a = {};
const b = { key: 'b' };
const c = { key: 'c' };
a[b] = 123;
a[c] = 456;
console.log(a[b]);
- A:
123
- B:
456
- C:
undefined
- D:
ReferenceError
Odpowiedź
Klucze obiektów są automatycznie konwertowane na ciągi znaków. Próbujemy ustawić obiekt jako klucz do obiektu a
, z wartością 123
.
Jednakże, kiedy stringujemy obiekt, staje się on "[obiekt Object]"
. Mówimy więc, że a["[obiekt Object]"] = 123
. Następnie próbujemy zrobić to samo. c
jest kolejnym obiektem, który niejawnie stringujemy. Zatem a["[obiekt Object]"] = 456
.
Następnie wyświetlamy w konsoli a[b]
, co w rzeczywistości jest a["[obiekt Object]"]
, ustawiony wcześniej na 456
, więc zwraca 456
.
const foo = () => console.log('First');
const bar = () => setTimeout(() => console.log('Second'));
const baz = () => console.log('Third');
bar();
foo();
baz();
- A:
First
Second
Third
- B:
First
Third
Second
- C:
Second
First
Third
- D:
Second
Third
First
Odpowiedź
Mamy funkcję setTimeout
i wywołaliśmy ją jako pierwszą. Została jednak wyświetlona jako ostatnia.
Dzieje się tak, ponieważ w przeglądarkach mamy nie tylko silnik wykonawczy, ale także coś, co nazywa się WebAPI
. Interfejs WebAPI
daje nam na początek funkcję setTimeout
.
Po przesłaniu callback do WebAPI, sama funkcja setTimeout
(ale nie callback!) jest usuwana ze stosu.
Teraz, foo
jest wywoływane, a "First"
jest wyświetlane.
foo
jest zdejmowane ze stosu, a baz
jest wywoływane. "Third" zostaje wyświetlony.
WebAPI nie może dodawać rzeczy do stosu, gdy jest gotowy. Zamiast tego przesuwa funkcję zwrotną do czegoś zwanego kolejką.
W tym miejscu zaczyna działać pętla zdarzeń. Pętla zdarzeń patrzy na stos i kolejkę zadań. Jeśli stos jest pusty, pobiera pierwszą rzecz z kolejki i przesuwa ją na stos.
bar
zostaje wywołany, "Second"
zostaje wyświetlony i zdjęty ze stosu.
<div onclick="console.log('first div')">
<div onclick="console.log('second div')">
<button onclick="console.log('button')">
Click!
</button>
</div>
</div>
- A: Zewnętrzny
div
- B: Wewnętrzny
div
- C:
button
- D: Tablica wszystkich zagnieżdżonych elementów.
Odpowiedź
Najgłębiej zagnieżdżony element, który spowodował zdarzenie jest celem zdarzenia. Możesz zatrzymać bąbelkowanie poprzez event.stopPropagation
<div onclick="console.log('div')">
<p onclick="console.log('p')">
Click here!
</p>
</div>
- A:
p
div
- B:
div
p
- C:
p
- D:
div
Odpowiedź
Jeśli klikniemy p
, zobaczymy dwa logi: p
i div
. Podczas propagacji zdarzeń istnieją 3 fazy: przechwytywanie, cel i bąbelkowanie (capturing, target, and bubbling). Domyślnie, event handlery są wykonywane w fazie bąbelkowania (chyba że ustawisz useCapture
na true
). Przebiega ona od najgłębiej zagnieżdżonego elementu na zewnątrz.
const person = { name: 'Lydia' };
function sayHi(age) {
return `${this.name} is ${age}`;
}
console.log(sayHi.call(person, 21));
console.log(sayHi.bind(person, 21));
- A:
undefined is 21
Lydia is 21
- B:
function
function
- C:
Lydia is 21
Lydia is 21
- D:
Lydia is 21
function
Odpowiedź
W obu przypadkach możemy przekazać obiekt, do którego ma się odnosić słowo kluczowe this
. Jednakże, .call
jest wykonywane natychmiast!
.bind.
zwraca kopię funkcji, ale z powiązanym kontekstem! Nie jest ona wykonywana natychmiast.
function sayHi() {
return (() => 0)();
}
console.log(typeof sayHi());
- A:
"object"
- B:
"number"
- C:
"function"
- D:
"undefined"
Odpowiedź
Funkcja sayHi
zwraca zwróconą wartość natychmiast wywołanego wyrażenia funkcyjnego (IIFE). Ta funkcja zwróciła wartość 0
, która jest typu "number"
.
FYI: typeof
może zwrócić następującą listę wartości: undefined
, boolean
, number
, bigint
, string
, symbol
, function
i object
. Zauważ, że typeof null
zwraca "object"
.
0;
new Number(0);
('');
(' ');
new Boolean(false);
undefined;
- A:
0
,''
,undefined
- B:
0
,new Number(0)
,''
,new Boolean(false)
,undefined
- C:
0
,''
,new Boolean(false)
,undefined
- D: Wszystkie są fałszywe
Odpowiedź
Istnieje 8 fałszywych wartości:
undefined
null
NaN
false
''
(pusty ciąg)0
-0
0n
(BigInt(0))
Konstruktory funkcji, takie jak new Number
i new Boolean
są prawdziwe.
console.log(typeof typeof 1);
- A:
"number"
- B:
"string"
- C:
"object"
- D:
"undefined"
const numbers = [1, 2, 3];
numbers[10] = 11;
console.log(numbers);
- A:
[1, 2, 3, null x 7, 11]
- B:
[1, 2, 3, 11]
- C:
[1, 2, 3, empty x 7, 11]
- D:
SyntaxError
Odpowiedź
Po ustawieniu wartości elementu w tablicy, która przekracza długość tablicy, JavaScript tworzy coś, co nazywa się "pustymi slotami". W rzeczywistości mają one wartość undefined
, ale zobaczysz coś takiego jak:
[1, 2, 3, puste x 7, 11]
.
w zależności od tego, gdzie go uruchomisz (jest inny dla każdej przeglądarki, node itp.).
(() => {
let x, y;
try {
throw new Error();
} catch (x) {
(x = 1), (y = 2);
console.log(x);
}
console.log(x);
console.log(y);
})();
- A:
1
undefined
2
- B:
undefined
undefined
undefined
- C:
1
1
2
- D:
1
undefined
undefined
Odpowiedź
Blok catch
otrzymuje argument x
. Nie jest to ten sam x
co zmienna, gdy przekazujemy argumenty. Ta zmienna x
jest blokowa.
Później, ustawiamy tę blokową zmienną równą 1
i ustawiamy wartość zmiennej y
. Teraz wyświetlamy w konsoli zmienną blokową x
, która jest równa 1
.
Poza blokiem catch
, x
jest wciąż undefined
, a y
wynosi 2
. Gdy chcemy wykonać console.log(x)
poza blokiem catch
, zwraca on undefined
, a y
zwraca 2
.
- A: prymitywem lub obiektem
- B: funkcją lub obiektem
- C: podchwytliwe pytanie! tylko obiektem
- D: numerem lub obiektem
Odpowiedź
JavaScript ma tylko prymitywne typy i obiekty.
Typy prymitywne to boolean
, null
, undefined
, bigint
, number
, string
i symbol
.
To, co odróżnia prymityw od obiektu, to fakt, że prymitywy nie mają żadnych właściwości ani metod; zauważysz jednak, że 'foo'.toUpperCase()
wylicza 'FOO'
i nie powoduje TypeError
. Dzieje się tak dlatego, że gdy próbujesz uzyskać dostęp do właściwości lub metody na prymitywie takim jak ciąg znaków, JavaScript niejawnie opakuje prymitywny typ za pomocą jednej z klas opakowujących, tj. String
, a następnie natychmiast odrzuci opakowanie po ocenie wyrażenia. Wszystkie prymitywy z wyjątkiem null
i undefined
wykazują to zachowanie.
[[0, 1], [2, 3]].reduce(
(acc, cur) => {
return acc.concat(cur);
},
[1, 2],
);
- A:
[0, 1, 2, 3, 1, 2]
- B:
[6, 1, 2]
- C:
[1, 2, 0, 1, 2, 3]
- D:
[1, 2, 6]
Odpowiedź
[1, 2]
jest naszą wartością początkową. Jest to wartość, z którą zaczynamy i wartość pierwszego acc
.Podczas pierwszej rundy, acc
to [1, 2]
, a cur
to [0, 1]
.Łączymy je, co daje [1, 2, 0, 1]
.
Następnie [1, 2, 0, 1]
to acc
, a [2, 3]
to cur
. Łączymy je i otrzymujemy [1, 2, 0, 1, 2, 3]
.
!!null;
!!'';
!!1;
- A:
false
true
false
- B:
false
false
true
- C:
false
true
true
- D:
true
true
false
Odpowiedź
null
jest fałszywe. !null
zwraca true
. !true
zwraca false
.
""
jest fałszywe. !""
zwraca true
. !true
zwraca false
.
1
jest prawdziwe. !1
zwraca false
. !false
zwraca true
.
setInterval(() => console.log('Hi'), 1000);
- A: unikalny identyfikator
- B: określona ilość milisekund
- C: przekazana funkcja
- D:
undefined
Odpowiedź
Zwraca unikalny identyfikator. Ten identyfikator może być użyty do wyczyszczenia tego interwału za pomocą funkcji clearInterval()
.
[...'Lydia'];
- A:
["L", "y", "d", "i", "a"]
- B:
["Lydia"]
- C:
[[], "Lydia"]
- D:
[["L", "y", "d", "i", "a"]]
Odpowiedź
Łańcuch znaków jest iterowalny. Operator spread odwzorowuje każdy znak iterable na jeden element.
function* generator(i) {
yield i;
yield i * 2;
}
const gen = generator(10);
console.log(gen.next().value);
console.log(gen.next().value);
- A:
[0, 10], [10, 20]
- B:
20, 20
- C:
10, 20
- D:
0, 10 and 10, 20
Odpowiedź
Zwykłe funkcje nie mogą zostać zatrzymane w połowie wywoływania. Jednak funkcja generatora może zostać "zatrzymana" w połowie, a następnie kontynuować od miejsca, w którym się zatrzymała. Za każdym razem, gdy funkcja generująca napotka słowo kluczowe yield
, funkcja zwraca wartość określoną po nim.
Najpierw inicjalizujemy funkcję generatora z i
równym 10
. Wywołujemy funkcję generatora za pomocą metody next()
. Przy pierwszym wywołaniu funkcji generatora, i
jest równe 10
. Funkcja napotyka pierwsze słowo kluczowe yield
: zwraca wartość i
. Generator jest teraz "wstrzymany", a wartość 10
zostaje zarejestrowana.
Następnie ponownie wywołujemy funkcję za pomocą metody next()
. Kontynuuje ona tam, gdzie zatrzymała się poprzednio, wciąż z i
równym 10
. Teraz napotyka następne słowo kluczowe yield
i zwraca i * 2
. i
jest równe 10
, więc zwraca 10 * 2
, czyli 20
. Wynikiem jest 10, 20
.
const firstPromise = new Promise((res, rej) => {
setTimeout(res, 500, 'one');
});
const secondPromise = new Promise((res, rej) => {
setTimeout(res, 100, 'two');
});
Promise.race([firstPromise, secondPromise]).then(res => console.log(res));
- A:
"one"
- B:
"two"
- C:
"two" "one"
- D:
"one" "two"
Odpowiedź
Kiedy przekazujemy wiele 'promise' do metody Promise.race
, rozwiązuje ona/odrzuca pierwszą 'promise'. Do metody setTimeout
przekazujemy timer: 500ms dla firstPromise
i 100ms dla secondPromise
. Oznacza to, że secondPromise
zostanie rozwiązana jako pierwsza z wartością 'two'
. res
przechowuje teraz wartość 'two'
, która jest wyświetlona w konsoli.
let person = { name: 'Lydia' };
const members = [person];
person = null;
console.log(members);
- A:
null
- B:
[null]
- C:
[{}]
- D:
[{ name: "Lydia" }]
Odpowiedź
Najpierw deklarujemy zmienną person
z wartością obiektu, który ma właściwość name
.
Następnie deklarujemy zmienną o nazwie members
. Ustawiamy pierwszy element tej tablicy równy wartości zmiennej person
. Obiekty oddziałują na siebie poprzez referencję, gdy ustawiamy je równe sobie. Kiedy przypisujesz referencję z jednej zmiennej do drugiej, tworzysz kopię tej referencji. (Zauważ, że nie mają one tej samej referencji!).
Następnie ustawiamy zmienną person
równą null
.
Modyfikujemy tylko wartość zmiennej person
, a nie pierwszy element w tablicy, ponieważ ten element ma inną (skopiowaną) referencję do obiektu. Pierwszy element w members
wciąż posiada referencję do oryginalnego obiektu. Kiedy wyświetlamy tablicę members
, pierwszy element nadal przechowuje wartość obiektu, który jest wyświetlany.
const person = {
name: 'Lydia',
age: 21,
};
for (const item in person) {
console.log(item);
}
- A:
{ name: "Lydia" }, { age: 21 }
- B:
"name", "age"
- C:
"Lydia", 21
- D:
["name", "Lydia"], ["age", 21]
Odpowiedź
Za pomocą pętli for-in
możemy iterować po kluczach obiektów, w tym przypadku name
i age
. Klucze obiektów są łańcuchami (jeśli nie są symbolami). W każdej pętli ustawiamy wartość item
równą bieżącemu kluczowi, który iterujemy. Najpierw item
jest równy name
. Następnie, item
jest równy age
.
console.log(3 + 4 + '5');
- A:
"345"
- B:
"75"
- C:
12
- D:
"12"
Odpowiedź
Asocjatywność operatorów to kolejność, w jakiej kompilator ocenia wyrażenia, od lewej do prawej lub od prawej do lewej. Dzieje się tak tylko wtedy, gdy wszystkie operatory mają takie samo pierwszeństwo. Mamy tylko jeden typ operatora: +
. Dla dodawania, asocjatywność jest od lewej do prawej.
3 + 4
jest obliczane jako pierwsze. Wynikiem jest liczba 7
.
7 + '5'
skutkuje "75"
z powodu przymusu. JavaScript konwertuje liczbę 7
na ciąg znaków, patrz pytanie 15. Możemy połączyć dwa ciągi znaków za pomocą operatora +
. "7" + "5" daje w wyniku "75"
.
const num = parseInt('7*6', 10);
- A:
42
- B:
"42"
- C:
7
- D:
NaN
Odpowiedź
Zwracana jest tylko pierwsza liczba w łańcuchu. W oparciu o radix (drugi argument w celu określenia typu liczby, którą chcemy przetworzyć: podstawa 10, szesnastkowy, ósemkowy, binarny itp.), parseInt
sprawdza, czy znaki w łańcuchu są prawidłowe. Gdy napotka znak, który nie jest prawidłową liczbą w radix, zatrzymuje parsowanie i ignoruje następujące znaki.
*
nie jest prawidłową liczbą. Przetwarza tylko "7"
na dziesiętne 7
. num
posiada teraz wartość 7
.
[1, 2, 3].map(num => {
if (typeof num === 'number') return;
return num * 2;
});
- A:
[]
- B:
[null, null, null]
- C:
[undefined, undefined, undefined]
- D:
[ 3 x empty ]
Odpowiedź
Podczas mapowania tablicy, wartość num
jest równa elementowi, nad którym aktualnie wykonywana jest pętla. W tym przypadku elementami są liczby, więc warunek instrukcji if typeof num == "number"
zwraca true
. Funkcja map tworzy nową tablicę i wstawia do niej wartości zwrócone przez funkcję.
Nie zwracamy jednak żadnej wartości. Gdy nie zwracamy wartości z funkcji, funkcja zwraca undefined
. Dla każdego elementu w tablicy wywoływany jest blok funkcji, więc dla każdego elementu zwracamy undefined
.