Je poste quotidiennement des questions à choix multiple sur mon Instagram, que je posterai aussi ici !
De la base aux subtilités du langage : testez votre compréhension de JavaScript, rafraîchissez vos connaissances, ou préparez-vous pour un entretien technique ! 💪 🚀 Je mets à jour ce dépôt chaque semaine avec des nouvelles questions. Dernière mise à jour : 29 juin
Les réponses se trouvent dans les sections repliées en dessous des questions, cliquez simplement dessus pour les faire apparaître. Bonne chance ❤️
- 🇸🇦 العربية
- 🇪🇬 اللغة العامية
- 🇧🇦 Bosanski
- 🇩🇪 Deutsch
- 🇬🇧 English
- 🇪🇸 Español
- 🇮🇩 Indonesia
- 🇮🇹 Italiano
- 🇯🇵 日本語
- 🇰🇷 한국어
- 🇳🇱 Nederlands
- 🇧🇷 Português Brasil
- 🇷🇺 Русский
- 🇹🇭 ไทย
- 🇹🇷 Türkçe
- 🇺🇦 Українська мова
- 🇻🇳 Tiếng Việt
- 🇨🇳 简体中文
- 🇹🇼 繁體中文
function sayHi() {
console.log(name);
console.log(age);
var name = "Lydia";
let age = 21;
}
sayHi();
- A:
Lydia
etundefined
- B:
Lydia
etReferenceError
- C:
ReferenceError
et21
- D:
undefined
etReferenceError
Réponse
Dans la fonction, nous déclarons en premier la variable name
grâce au mot clé var
. Cela signifie que la variable est "levée" (hoisted) (l'espace mémoire est définie à la phase de création) avec pour valeur par défaut undefined
, jusqu'à ce que le script atteigne la ligne de définition de la variable. Nous n'avons pas encore défini la variable lorsque nous essayons d'afficher la variable name
, donc elle a toujours la valeur undefined
.
Les variables avec le mot clé let
(et const
) sont "levées" (hoisted), mais contrairement à var
, elle n'est pas initialisée. Elles ne sont pas accessible avant la ligne qui les déclare (initialise). C'est appelé la "zone morte temporaire". Lorsque nous essayons d'accéder aux variables avant leur déclaration, JavaScript renvoie une 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
et0 1 2
- B:
0 1 2
et3 3 3
- C:
3 3 3
et0 1 2
Réponse
À cause du système de queue dans JavaScript, la fonction de rappel (callback) du setTimeout
est appelée après que la boucle soit exécutée. Comme la variable i
dans la première boucle est déclarée avec le mot-clé var
, c'est une variable globale. Pendant la boucle, nous incrémentons la valeur de i
de 1
à chaque fois, en utilisant l'opérateur arithmétique ++
. Lorsque la fonction de rappel (callback) du setTimeout
est invoquée, i
est égal à 3
dans le premier exemple.
Dans la seconde boucle, la variable i
est déclarée avec le mot clé let
: les variables déclarées avec let
(et const
) ont une portée de bloc (tout ce qui est entre { }
est considéré comme un bloc). Pendant chaque itération, i
aura une nouvelle valeur, et chaque valeur sera définie dans la boucle.
const shape = {
radius: 10,
diameter() {
return this.radius * 2;
},
perimeter: () => 2 * Math.PI * this.radius
};
shape.diameter();
shape.perimeter();
- A:
20
et62.83185307179586
- B:
20
etNaN
- C:
20
et63
- D:
NaN
et63
Réponse
Notez que la valeur de diameter
est une fonction régulière, alors que celle de perimeter
est une fonction fléchée.
Avec les fonctions fléchée, le mot clé this
réfère à son périmètre actuel, contrairement aux fonctions régulières ! Cela signifie que lorsque nous appelons perimeter
, elle ne réfère pas à l'objet shape
, mais à son périmètre actuel (window
par exemple).
Il n'y a pas de valeur radius
dans cet objet, on retournera undefined
.
+true;
!"Lydia";
- A:
1
etfalse
- B:
false
etNaN
- C:
false
etfalse
Réponse
L'opérateur arithmétique +
essait de convertir un opérande en une valeur numérique. true
devient 1
, et false
devient 0
.
La chaîne de caractère 'Lydia'
est une valeur considérée comme vraie (truthy). Ce que nous sommes actuellement en train de demander, c'est "est-ce que cette valeur considérée comme vraie est fausse ?". Ce qui retournera false
.
const bird = {
size: "small"
};
const mouse = {
name: "Mickey",
small: true
};
- A:
mouse.bird.size
n'est pas valide - B:
mouse[bird.size]
n'est pas valide - C:
mouse[bird["size"]]
n'est pas valide - D: Toutes sont valides
Réponse
En JavaScript, toutes les clés d'objet sont des chaînes de caractères (sauf si c'est un Symbol). Bien que nous ne puissions pas les typer comme des chaînes de caractères, elles sont converties en chaînes de caractères sous le capot.
JavaScript interprète (ou décompresse) les instructions. Lorsque nous utilisons la notation par crochet, il voit le premier crochet [
et continue jusqu'à ce qu'il trouve le crochet fermant ]
. Seulement après, il évalue l'instruction.
mouse[bird.size]
: Premièrement, il évalue bird.size
, qui est "small"
. mouse["small"]
retourne true
.
Cependant, avec la notation par points, cela n'arrive pas. mouse
n'a pas de clé appelée bird
, ce qui signifie que mouse.bird
est undefined
. Puis, on demande size
en utilisant la notation par point : mouse.bird.size
. Comme mouse.bird
est undefined
, on demande undefined.size
. Cela n'est pas valide, et nous aurons une erreur similaire à Impossible de lire la propriété "size" de 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
Réponse
En JavaScript, tous les objets interagissent par référence lorsqu'on les définit égaux les uns aux autres.
Premièrement, la variable c
contaient une valeur d'objet. Plus tard, nous assignons d
avec la même référence que c
à l'objet.
Quand on modifie un objet, on les modifie donc tous.
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
Réponse
new Number()
est une fonction globale. Bien qu'il ressemble à un nombre, ce n'en est pas vraiment un : il a une poignée de fonctionnalités supplémentaire et est un objet.
Quand nous utilisons l'opérateur ==
, il vérifie seulement qu'il s'agisse de la même valeur. Les deux ont pour valeur 3
, donc il retourne true
.
Cependant, quand on utilise l'opérateur ===
, les 2 valeurs et types doivent être les mêmes. new Number()
n'est pas un nombre, c'est un objet, il retourne false
.
class Chameleon {
static colorChange(newColor) {
this.newColor = newColor;
return this.newColor;
}
constructor({ newColor = "green" } = {}) {
this.newColor = newColor;
}
}
const freddie = new Chameleon({ newColor: "purple" });
freddie.colorChange("orange");
- A:
orange
- B:
purple
- C:
green
- D:
TypeError
Réponse
La fonction colorChange
est statique. Les méthodes statiques sont désignées pour vivre seulement dans le constructeur qui les a créé et ne peuvent pas être transférer aux enfants. Comme freddie
est un enfant, la fonction n'est pas transférée et n'est pas disponible dans l'instance de freddie
: une erreur TypeError
est renvoyée.
let greeting;
greetign = {}; // Typo!
console.log(greetign);
- A:
{}
- B:
ReferenceError: greetign is not defined
- C:
undefined
Réponse
Il affiche l'objet, car on a juste créé un objet vide dans l'objet global ! Quand on écrit mal greeting
en greetign
, JavaScript l'interprète comme il le voit global.greetign = {}
(ou window.greetign = {}
dans le navigateur).
Pour éviter cela, on peut utiliser "use strict"
. Cela nous assure de devoir déclarer la variable avant de lui assigner une valeur.
function bark() {
console.log("Woof!");
}
bark.animal = "dog";
- A: Rien, c'est tout à fait bon !
- B:
SyntaxError
. Vous ne pouvez pas ajouter de propriétés à une fonction de cette façon. - C:
undefined
- D:
ReferenceError
Réponse
C'est possible en JavaScript, car les fonctions sont des objets ! (Tout ce qui n'est pas de type primitif est un objet)
Une fonction est un type spécial d'objet. Le code que vous écrivez vous-même n'est pas la fonction. La fonction est un objet avec des propriétés. Cette propriété est invocable.
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
Réponse
Vous ne pouvez pas ajouter de propriétés à un constructeur comme pour des objets normaux. Si vous voulez ajouter une fonctionnalité pour tous les objets en une fois, vous devez utiliser le prototype. Donc dans ce cas,
Person.prototype.getFullName = function() {
return `${this.firstName} ${this.lastName}`;
};
rendra fonctionnel member.getFullName
. Pourquoi est-ce bénéfique ? Disons que nous ajoutons cette méthode au constructeur directement. Peut-être que toutes les instances de Person
n'ont pas besoin de cette méthode. Cela fera perdre de l'espace mémoire, car elles auront tous cette propriété, ce qui prendra de l'espace mémoire pour chaque instance. Alors que, si nous ajoutons la méthode au prototype uniquement, nous n'utilisons qu'un seul slot mémoire, et ils y auront tous accès !
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"}
etundefined
- B:
Person {firstName: "Lydia", lastName: "Hallie"}
etPerson {firstName: "Sarah", lastName: "Smith"}
- C:
Person {firstName: "Lydia", lastName: "Hallie"}
et{}
- D:
Person {firstName: "Lydia", lastName: "Hallie"}
etReferenceError
Réponse
Pour sarah
, nous n'avons pas utilisé le mot clé new
. Quand nous utilisons new
, il fait référence à un nouvel objet vide que nous créons. Cependant, nous n'ajoutons pas new
. Il réfère à l'objet global !
Nous disons que this.firstName
est égal à "Sarah"
et que this.lastName
est égal à Smith
. Ce que nous faisons c'est définir global.firstName = 'Sarah'
et global.lastName = 'Smith'
. La variable sarah
elle-même reste à undefined
.
- A: Target > Capturing > Bubbling
- B: Bubbling > Target > Capturing
- C: Target > Bubbling > Capturing
- D: Capturing > Target > Bubbling
Réponse
Durant la phase de capture (capturing), l'événement passe par les éléments parents jusqu'à l'élément ciblé. Il atteint ensuite l'élément ciblé (target), et commence à bouillonner (bubbling).
- A: vrai
- B: faux
Réponse
Tous les objets ont des prototypes, excepté pour les objets standards. Les objets standards ont accès à certaines méthodes et propriétés, comme .toString
. C'est pour cette raison que vous pouvez utiliser les méthodes natives de JavaScript ! Toutes ces méthodes sont disponibles dans le prototype. Bien que JavaScript ne trouve pas la fonction dans l'objet, il parcourt le prototype et la méthode afin de la rendre accessible.
function sum(a, b) {
return a + b;
}
sum(1, "2");
- A:
NaN
- B:
TypeError
- C:
"12"
- D:
3
Réponse
JavaScript est un langage à types dynamiques : nous n'avons pas besoin de spécifier le types des variables. Les valeurs peuvent être automatiquement converties vers les autres types sans que vous le sachiez, c'est ce que l'on appelle la conversion de types implicites (implicit type coercion).
Dans cette exemple, JavaScript convertit le nombre 1
en une chaîne de caractère, afin que la fonction ait du sens et puisse renvoyer une valeur. Durant l'addition d'un type numérique (1
) et d'un type chaîne de caractère ('2'
), le nombre est traité comme une chaîne de caractère. Nous pouvons concaténer les chaînes de caractères comme "Hello" + "World"
, c'est donc ce qui arrive ici avec "1" + "2"
qui retourne "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
Réponse
L'opérateur arithmétique postfix ++
:
- Retourne la valeur (ici il retourne
0
) - Incrémente la valeur (le nombre est maintenant égal à
1
)
L'opérateur arithmétique préfix ++
:
- Incrémente la valeur (le nombre est maintenant égal à
2
) - Retourne la valeur (ici il retourne
2
)
Cela retourne donc 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
Réponse
Si vous utilisez les littéraux de gabarits, la valeur du premier argument sera toujours un tableau de valeurs des chaînes de caractère. Le reste des arguments seront les valeurs des expressions utilisées !
function checkAge(data) {
if (data === { age: 18 }) {
console.log("Vous êtes un adulte !");
} else if (data == { age: 18 }) {
console.log("Vous êtes toujours un adulte.");
} else {
console.log(`Hmm.. Vous n'avez pas l'âge, je suppose.`);
}
}
checkAge({ age: 18 });
- A:
Vous êtes un adulte !
- B:
Vous êtes toujours un adulte.
- C:
Hmm.. Vous n'avez pas l'âge, je suppose.
Réponse
Lorsque l'on teste une égalité, les primitifs sont comparés par leur valeur, alors que les objets sont comparés par leur référence. JavaScript vérifie si les objets ont une référence à la même zone de la mémoire.=
Les 2 objets que nous comparons n'ont pas ça : l'objet passé en paramètre fait référence à une zone mémoire différente que l'objet que nous utilisons pour faire la comparaison.
C'est pourquoi les 2 conditions { age: 18 } === { age: 18 }
et { age: 18 } == { age: 18 }
retournent false
.
function getAge(...args) {
console.log(typeof args);
}
getAge(21);
- A:
"number"
- B:
"array"
- C:
"object"
- D:
"NaN"
Réponse
La syntaxe des paramètres du reste (rest parameters) (...args
) retourne un tableau avec les arguments. Un tableau est un objet, donc typeof args
retournera "object"
.
function getAge() {
"use strict";
age = 21;
console.log(age);
}
getAge();
- A:
21
- B:
undefined
- C:
ReferenceError
- D:
TypeError
Réponse
Avec "use strict"
, vous pouvez êtes sûr de ne pas déclarer accidentellement des variables gloables. Nous ne déclarerons jamais la variable age
, et temps que nous utiliserons "use strict"
, cela créera une erreur de référence. Si nous n'utilisons pas "use strict"
, cela fonctionnera et la variable age
sera attribué à l'objet global.
const sum = eval("10*10+5");
- A:
105
- B:
"105"
- C:
TypeError
- D:
"10*10+5"
Réponse
eval
évalue les codes que nous passons en paramètre de type chaîne de caractères. Si c'est une expression, comme dans notre cas, il évaluera l'expression. L'expression est 10 * 10 + 5
, ce qui retourne le nombre 105
.
sessionStorage.setItem("cool_secret", 123);
- A: Pour toujours, les données ne seront pas perdues.
- B: Jusqu'à ce que l'utilisateur ferme l'onglet.
- C: Jusqu'à ce que l'utilisateur ferme son navigateur en entier, pas seulement son onglet.
- D: Jusqu'à ce que l'utilisateur éteindra son ordinateur.
Réponse
La donnée stockée dans le sessionStorage
est supprimée après la fermeture de l'onglet.
Si vous utilisez le localStorage
, la donnée sera là pour toujours, jusqu'à ce que, par exemple, localStorage.clear()
soit invoquée.
var num = 8;
var num = 10;
console.log(num);
- A:
8
- B:
10
- C:
SyntaxError
- D:
ReferenceError
Réponse
Avec le mot clé var
, vous pouvez déclarer plusieurs variables avec le même nom. La variable aura pour valeur la dernière assignée.
Vous ne pouvez pas faire cela avec let
ou const
puisqu'ils ont une portée de bloc.
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
Réponse
Toutes les clés d'objet (à l'exception des symboles) sont des chaînes de caractères sous le capot, même si vous ne les tapez pas vous-même en tant que chaîne. C'est pourquoi obj.hasOwnProperty("1")
renvoie également la valeur true
.
Ça ne marche pas comme ça pour un set. Il n'y a pas de '1'
dans notre ensemble : set.has('1')
renvoie false
. Il a le type numérique 1
, set.has(1)
renvoie true
.
const obj = { a: "un", b: "deux", a: "trois" };
console.log(obj);
- A:
{ a: "un", b: "deux" }
- B:
{ b: "deux", a: "trois" }
- C:
{ a: "trois", b: "deux" }
- D:
SyntaxError
Réponse
Si vous avez deux clés portant le même nom, la clé sera remplacée. Elle sera toujours dans sa première position, mais avec la dernière valeur spécifiée.
26. Le contexte global d'exécution de JavaScript crée 2 choses pour vous : l'objet global and le mot-clé this
.
- A: Vrai
- B: Faux
- C: Ça dépend
Réponse
Le contexte d'exécution de base est le contexte d'exécution global : c'est ce qui est accessible partout dans votre code.
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
Réponse
L'instruction continue
ignore une itération si une condition donnée renvoie true
.
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
Réponse
String
est un constructeur intégré, auquel nous pouvons ajouter des propriétés. Je viens d'ajouter une méthode à son prototype. Les chaînes de caractère primitives sont automatiquement converties en un objet chaîne, généré par la fonction prototype de chaîne. Ainsi, toutes les chaînes (objets de chaîne) ont accès à cette méthode !
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
Réponse
Les clés d'objet sont automatiquement converties en chaînes de caractères. Nous essayons de définir un objet en tant que clé de l'objet a
, avec la valeur 123
.
Cependant, lorsque nous transformons un objet en chaîne de caractère, il devient "[Objet objet]"
. Donc, ce que nous disons ici, c'est que un a["Objet objet"] = 123
. Ensuite, nous pouvons essayer de refaire la même chose. c
est un autre objet que nous sommes implicitement en train de transformer en chaîne de caractère. Donc, a["Objet objet"] = 456
.
Ensuite, nous affichons a[b]
, qui est en fait a["Objet objet"]
. Que nous venons de définir à 456
, nous renvoyons donc 456
.
const foo = () => console.log("Premier");
const bar = () => setTimeout(() => console.log("Second"));
const baz = () => console.log("Troisième");
bar();
foo();
baz();
- A:
Premier
Second
Troisième
- B:
Premier
Troisième
Second
- C:
Second
Premier
Troisième
- D:
Second
Troisième
Premier
Réponse
Nous avons une fonction setTimeout
et nous l'avons d'abord appelée. Pourtant, il a été affiché en dernier.
En effet, dans les navigateurs, nous n’avons pas seulement le moteur d’exécution, nous avons aussi quelque chose appelé WebAPI
. WebAPI
nous donne la fonction setTimeout
pour commencer, et par exemple le DOM.
Une fois que la fonction de rappel (callback) est poussée via la WebAPI, la fonction setTimeout
elle-même (mais pas la fonction de rappel !) est extraite de la pile.
Maintenant, foo
est invoqué et "Premier"
est affiché.
foo
est extrait de la pile et baz
est invoqué. "Troisième"
est affiché.
WebAPI ne peut simplement pas ajouter des éléments à la pile dès qu’elle est prête. Au lieu de cela, elle pousse la fonction de rappel vers quelque chose appelé la file d'attente.
C'est ici qu'une boucle d'événement commence à fonctionner. La boucle d'événement examine la pile et la file d'attente des tâches. Si la pile est vide, il prend la première chose dans la file d'attente et la pousse sur la pile.
bar
est invoqué, "Second"
est affiché et il est sorti de la pile.
<div onclick="console.log('first div')">
<div onclick="console.log('second div')">
<button onclick="console.log('button')">
Click!
</button>
</div>
</div>
- A: La
div
extérieure - B: La
div
intérieure - C:
button
- D: Un tableau de tous les éléments imbriqués.
Réponse
L'élément imbriqué le plus profond qui a provoqué l'événement est la cible de l'événement. Vous pouvez arrêter le bouillonnement (bubbling) en utilisant 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
Réponse
Si nous cliquons sur p
, nous verrons deux lignes : p
et div
. Lors de la propagation d'un événement, il y a 3 phases: capture, cible et bouillonnement (bubbling). Par défaut, les gestionnaires d'événements sont exécutés dans la phase de bouillonnement (sauf si vous définissez useCapture
sur true
). Il va de l'élément imbriqué le plus profond vers l'extérieur.
const person = { name: "Lydia" };
function sayHi(age) {
console.log(`${this.name} is ${age}`);
}
sayHi.call(person, 21);
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
Réponse
Avec les deux, nous pouvons transmettre l'objet auquel nous voulons que le mot clé this
fasse référence. Cependant, .call
est aussi exécuté immédiatement !
.bind.
renvoie une copie de la fonction, mais avec un contexte lié ! Elle n'est pas exécutée immédiatement.
function sayHi() {
return (() => 0)();
}
typeof sayHi();
- A:
"object"
- B:
"number"
- C:
"function"
- D:
"undefined"
Réponse
La fonction sayHi
renvoie la valeur renvoyée par la fonction immédiatement appelée (IIFE). Cette fonction a renvoyé 0
, qui est du type "nombre"
.
Pour info : il n'y a que 7 types natifs : null
, undefined
, boolean
, number
, string
, object
, symbol
et bigint
. "function"
n'est pas un type, puisque les fonctions sont des objets, il est de type "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: Toutes sont fausses
Réponse
Il n'y a que six valeurs de fausses :
undefined
null
NaN
0
''
(chaîne de caractère vide)false
Les constructeurs de fonctions, comme new Number
et new Boolean
sont la vraies.
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, 7 x null, 11]
- B:
[1, 2, 3, 11]
- C:
[1, 2, 3, 7 x empty, 11]
- D:
SyntaxError
Réponse
Lorsque vous définissez une valeur sur un élément d'un tableau qui dépasse la longueur du tableau, JavaScript crée un quelque chose appelé "emplacements vides". Ceux-ci ont en fait la valeur undefined
, mais vous verrez quelque chose comme :
[1, 2, 3, 7 x empty, 11]
en fonction de l'endroit où vous l'exécutez (différent pour chaque navigateur, nœud, etc.).
(() => {
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
Réponse
Le bloc catch
reçoit l'argument x
. Ce n'est pas le même x
que la variable que nous passons en arguments. Cette variable x
a une portée de bloc.
Plus tard, nous définissons cette variable de bloc égale à 1
et définissons la valeur de la variable y
. Maintenant, nous affichons la variable x
de portée de bloc, dont la valeur est égale à 1
.
En dehors du bloc catch
, x
est toujours undefined
et y
est égal à 2
. Lorsque nous voulons console.log(x)
en dehors du bloc catch
, il renvoie undefined
, et y
renvoie 2
.
- A: primitif ou objet
- B: fonction ou objet
- C: question délicate ! Seulement des objets
- D: nombre ou objet
Réponse
JavaScript n'a que des types et des objets primitifs.
Les types primitifs sont boolean
, null
, undefined
, bigint
, number
, string
et symbol
.
Ce qui différencie une primitive d'un objet, c'est que les primitives n'ont aucune propriété ou méthode. Cependant, vous remarquerez que 'foo'.toUpperCase()
est évalué à 'FOO'
et n'entraîne pas de TypeError
. En effet, lorsque vous essayez d'accéder à une propriété ou à une méthode sur une primitive telle qu'une chaîne, JavaScript encapsule implicitement l'objet à l'aide de l'une des classes d'encapsulation, à savoir String
, puis supprime immédiatement l'encapsulation après l'évaluation de l'expression. Toutes les primitives à l'exception de null
et undefined
présentent ce comportement.
[[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]
Réponse
[1, 2]
est notre valeur initiale. C'est la valeur avec laquelle nous commençons et la valeur du tout premier acc
. Au premier tour, acc
est [1,2]
et cur
est [0, 1]
. Nous les concaténons, ce qui donne [1, 2, 0, 1]
.
Ensuite, acc
est [1, 2, 0, 1]
et cur
est [2, 3]
. Nous les concaténons et obtenons [1, 2, 0, 1, 2, 3]
!!null;
!!"";
!!1;
- A:
false
true
false
- B:
false
false
true
- C:
false
true
true
- D:
true
true
false
Réponse
null
est faux. !null
retourne true
. !true
retourne false
.
""
est faux. !""
retourne true
. !true
retourne false
.
1
est vrai. !1
retourne false
. !false
retourne true
.
setInterval(() => console.log("Hi"), 1000);
- A: un identifiant unique
- B: le temps de millisecondes spécifié
- C: la fonction passée en paramètre
- D:
undefined
Réponse
Il retourne un identifiant unique. Cet identifiant peut être utilisé pour effacer cet interval avec la fonction clearInterval()
.
[..."Lydia"];
- A:
["L", "y", "d", "i", "a"]
- B:
["Lydia"]
- C:
[[], "Lydia"]
- D:
[["L", "y", "d", "i", "a"]]
Réponse
Une chaîne de caractère est itérable. L'opérateur de déconstruction transforme chaque caractère d'un itérable en un élément.
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
Réponse
Les fonctions régulières ne peuvent pas être arrêtées à mi-parcours après l'invocation. Cependant, une fonction de générateur peut être "arrêtée" à mi-chemin et continuer plus tard à partir de son point d'arrêt. Chaque fois qu'une fonction génératrice rencontre un mot-clé yield
, elle renvoie la valeur spécifiée après celui-ci. Notez que la fonction du générateur dans ce cas ne return pas la valeur, il yields la valeur.
Premièrement, nous initialisons la fonction du générateur avec i
égal à 10
. Nous appelons la fonction génératrice en utilisant la méthode next()
. La première fois que nous appelons la fonction de générateur, i
est égal à 10
. Il rencontre le premier mot-clé yield
: il donne la valeur de i
. Le générateur est maintenant "en pause" et 10
est affiché.
Ensuite, nous appelons à nouveau la fonction avec la méthode next()
. Il commence à continuer là où il s’était arrêté précédemment, toujours avec i
égal à 10
. Maintenant, il rencontre le prochain mot-clé yield
et donne i * 2
. i
est égal à 10
, donc il renvoie 10 * 2
, ce qui correspond à 20
. Cela donne 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"
Réponse
Lorsque nous passons plusieurs promesses à la méthode Promise.race
, elle résout/rejette la promesse first qui résout/rejette. Nous passons, à la méthode setTimeout
, un timer: 500 ms pour la première promesse (firstPromise
), et 100 ms pour la deuxième promesse (secondPromise
). Cela signifie que le secondPromise
se résout en premier avec la valeur de 'two'
. res
contient maintenant la valeur de 'deux'
, qui est affiché.
let person = { name: "Lydia" };
const members = [person];
person = null;
console.log(members);
- A:
null
- B:
[null]
- C:
[{}]
- D:
[{ name: "Lydia" }]
Réponse
Tout d'abord, nous déclarons une variable person
avec la valeur d'un objet possédant une propriété name
.
Ensuite, nous déclarons une variable appelée membres
. Nous définissons le premier élément de ce tableau égal à la valeur de la variable person
. Les objets interagissent par référence quand ils sont égaux. Lorsque vous affectez une référence d'une variable à une autre, vous créez une copie de cette référence. (notez qu'ils n'ont pas la même référence !)
Ensuite, nous définissons la variable person
égale à null
.
Nous modifions seulement la valeur de la variable person
, et non le premier élément du tableau, car cet élément a une référence (copiée) différente de l'objet. Le premier élément de members
conserve sa référence à l'objet d'origine. Lorsque nous affichons le tableau members
, le premier élément contient toujours la valeur de l'objet, qui est affiché.
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]
Réponse
Avec une boucle for-in
, nous pouvons parcourir les clés d'objet, dans ce cas name
et age
. Sous le capot, les clés d'objet sont des chaînes (si elles ne sont pas un symbole). A chaque boucle, nous définissons la valeur de item
égal à la clé courante sur laquelle elle est entrain d'itéré. Premièrement, item
est égal à name
et est affiché. Ensuite, item
est égal à age
, qui est affiché.
console.log(3 + 4 + "5");
- A:
"345"
- B:
"75"
- C:
12
- D:
"12"
Réponse
L'associativité des opérateurs correspond à l'ordre dans lequel le compilateur évalue les expressions, de gauche à droite ou de droite à gauche. Cela se produit uniquement si tous les opérateurs ont la même priorité. Nous n'avons qu'un type d'opérateur : +
. De plus, l'associativité est de gauche à droite.
3 + 4
est évalué en premier. Cela donne le nombre 7
.
7 + '5'
donne '75'
à cause de la conversion. JavaScript convertit le nombre 7
en une chaîne, voir question 15. Nous pouvons concaténer deux chaînes en utilisant l'opérateur +
. "7" + "5"
a pour résultat "75"
.
const num = parseInt("7*6", 10);
- A:
42
- B:
"42"
- C:
7
- D:
NaN
Réponse
Seuls les premiers nombres de la chaîne sont renvoyés. Basé sur la base dix (le deuxième argument permettant de spécifier le type de nombre que nous voulons analyser: base 10, hexadécimal, octal, binaire, etc.), parseInt
vérifie si les caractères de la chaîne sont valides. Une fois qu'il rencontre un caractère qui n'est pas un nombre valide dans la base, il arrête l'analyse et ignore les caractères suivants.
*
n'est pas un nombre valide. Il analyse seulement 7
dans la décimale 7
. num
contient maintenant la valeur de 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 ]
Réponse
Lors du l'utilisation de map
sur le tableau, la valeur de num
est égale à l’élément sur lequel elle est en train de boucler. Dans ce cas, les éléments sont des nombres. La condition de l'instruction si typeof num === "numéro"
renvoie true
. La fonction map crée un nouveau tableau et insère les valeurs renvoyées par la fonction.
Cependant, nous ne renvoyons pas de valeur. Lorsque nous ne renvoyons pas de valeur à partir de la fonction, la fonction renvoie undefined
. Le bloc de fonction est appelé pour chaque élément du tableau. Ainsi, pour chaque élément, nous renvoyons undefined
.
function getInfo(member, year) {
member.name = "Lydia";
year = "1998";
}
const person = { name: "Sarah" };
const birthYear = "1997";
getInfo(person, birthYear);
console.log(person, birthYear);
- A:
{ name: "Lydia" }, "1997"
- B:
{ name: "Sarah" }, "1998"
- C:
{ name: "Lydia" }, "1998"
- D:
{ name: "Sarah" }, "1997"
Réponse
Les arguments sont passés par valeur, à moins que leur valeur ne soit un objet, ils sont passés par référence. birthYear
est passée par valeur, car c'est une chaîne, pas un objet. Lorsque nous passons des arguments par valeur, une copie de cette valeur est créée (voir question 46).
La variable birthYear
a une référence à la valeur "1997"
. L'argument year
fait également référence à la valeur "1997"
, mais il ne s'agit pas de la même valeur que celle de birthYear
. Lorsque nous mettons à jour la valeur de year
en plaçant year
égal à "1998"
, nous ne mettons à jour que la valeur de year
. birthYear
est toujours égal à "1997"
.
La valeur de person
est un objet. L'argument membre
a une référence (copiée) au même objet. Lorsque nous modifions une propriété de l'objet, membre
a une référence à la valeur de person
sera également modifiée, puisqu'elles ont toutes les deux une référence au même objet. La propriété name
de person
est maintenant égale à la valeur "Lydia"
.
function greeting() {
throw "Hello world!";
}
function sayHi() {
try {
const data = greeting();
console.log("It worked!", data);
} catch (e) {
console.log("Oh no an error!", e);
}
}
sayHi();
- A:
"It worked! Hello world!"
- B:
"Oh no an error: undefined
- C:
SyntaxError: can only throw Error objects
- D:
"Oh no an error: Hello world!
Réponse
Avec l'instruction throw
, nous pouvons créer des erreurs personnalisées. Avec cette déclaration, vous pouvez lancer des exceptions. Une exception peut être une chaîne, un numéro, un booléen ou un objet. Dans ce cas, notre exception est la chaîne 'Hello world'
.
Avec l'instruction catch
, nous pouvons spécifier quoi faire si une exception est levée dans le bloc try
. Une exception est levée : la chaîne 'Hello world'
. e
est maintenant égal à cette chaîne, que nous affichons. Il en résulte 'Oh an error: Hello world'
.
function Car() {
this.make = "Lamborghini";
return { make: "Maserati" };
}
const myCar = new Car();
console.log(myCar.make);
- A:
"Lamborghini"
- B:
"Maserati"
- C:
ReferenceError
- D:
TypeError
Réponse
Lorsque vous retournez une propriété, la valeur de la propriété est égale à la valeur retournée et non à la valeur définie dans la fonction constructeur. Nous renvoyons la chaîne "Maserati"
, donc myCar.make
est égal à "Maserati"
.
(() => {
let x = (y = 10);
})();
console.log(typeof x);
console.log(typeof y);
- A:
"undefined", "number"
- B:
"number", "number"
- C:
"object", "number"
- D:
"number", "undefined"
Réponse
let x = y = 10;
est en réalité un raccourci pour :
y = 10;
let x = y;
Lorsque nous définissons y
égal à 10
, nous ajoutons en fait une propriété y
à l'objet global (window
dans le navigateur, global
dans NodeJS). Dans un navigateur, window.y
est égal à 10
.
Ensuite, nous déclarons une variable x
avec la valeur de y
, qui est 10
. Les variables déclarées avec le mot-clé let
ont une portée de bloc, elles ne sont définies que dans le bloc dans lequel elles sont déclarées; la fonction immédiatement invoquée (IIFE) dans ce cas. Lorsque nous utilisons l'opérateur typeof
, l'opérande x
n'est pas défini: nous essayons d'accéder à x
en dehors du bloc dans lequel il est déclaré. Cela signifie que x
n'est pas défini. Les valeurs auxquelles aucune valeur n'a été attribuée ni déclarée sont du type "undefined"
. console.log (typeof x)
renvoie "undefined"
.
Cependant, nous avons créé une variable globale y
lorsque y
est égal à 10
. Cette valeur est accessible n'importe où dans notre code. y
est défini et contient une valeur de type "number"
. console.log (typeof y)
renvoie "number"
.
class Dog {
constructor(name) {
this.name = name;
}
}
Dog.prototype.bark = function() {
console.log(`Woof I am ${this.name}`);
};
const pet = new Dog("Mara");
pet.bark();
delete Dog.prototype.bark;
pet.bark();
- A:
"Woof I am Mara"
,TypeError
- B:
"Woof I am Mara"
,"Woof I am Mara"
- C:
"Woof I am Mara"
,undefined
- D:
TypeError
,TypeError
Réponse
Nous pouvons supprimer des propriétés d'objets en utilisant le mot-clé delete
, également dans le prototype. En supprimant une propriété dans le prototype, elle n’est plus disponible dans la chaîne de prototypes. Dans ce cas, la fonction bark
n'est plus disponible dans le prototype après delete Dog.prototype.bark
, mais nous essayons toujours d'y accéder.
Lorsque nous essayons d'appeler quelque chose qui n'est pas une fonction, un TypeError
est levé. Dans ce cas, TypeError: pet.bark n'est pas une fonction
, puisque pet.bark
est undefined
.
const set = new Set([1, 1, 2, 3, 4]);
console.log(set);
- A:
[1, 1, 2, 3, 4]
- B:
[1, 2, 3, 4]
- C:
{1, 1, 2, 3, 4}
- D:
{1, 2, 3, 4}
Réponse
L'objet Set
est une collection de valeurs uniques : une valeur ne peut apparaître qu'une seule fois dans un ensemble.
Nous avons passé l'itérable [1, 1, 2, 3, 4]
avec une valeur dupliquée 1
. Puisque nous ne pouvons pas avoir deux valeurs identiques dans un ensemble, l'une d'entre elles est supprimée. Cela donne {1, 2, 3, 4}
.
// counter.js
let counter = 10;
export default counter;
// index.js
import myCounter from "./counter";
myCounter += 1;
console.log(myCounter);
- A:
10
- B:
11
- C:
Error
- D:
NaN
Réponse
Un module importé est en lecture seule : vous ne pouvez pas modifier le module importé. Seul le module qui les exporte peut en changer la valeur.
Lorsque nous essayons d'incrémenter la valeur de myCounter
, une erreur est générée : myCounter
est en lecture seule et ne peut pas être modifié.
const name = "Lydia";
age = 21;
console.log(delete name);
console.log(delete age);
- A:
false
,true
- B:
"Lydia"
,21
- C:
true
,true
- D:
undefined
,undefined
Réponse
L'opérateur delete
renvoie une valeur booléenne : true
en cas de suppression réussie, sinon il renvoie false
. Toutefois, les variables déclarées avec les mots clés var
, const
ou let
ne peuvent pas être supprimées à l'aide de l'opérateur delete
.
La variable name
a été déclarée avec un mot-clé const
. Par conséquent, sa suppression a échoué : false
est renvoyé. Lorsque nous définissons age
égal à 21
, nous avons en fait ajouté une propriété appelée age
à l'objet global. De cette façon, vous pouvez supprimer des propriétés d’objets, ainsi que de l’objet global, pour que delete age
renvoie true
.
const numbers = [1, 2, 3, 4, 5];
const [y] = numbers;
console.log(y);
- A:
[[1, 2, 3, 4, 5]]
- B:
[1, 2, 3, 4, 5]
- C:
1
- D:
[1]
Réponse
Nous pouvons décompresser les valeurs des tableaux ou les propriétés des objets en les détruisant. Par exemple :
[a, b] = [1, 2];
La valeur de a
est maintenant 1
et la valeur de b
est maintenant 2
. Ce que nous avons réellement fait dans la question, c'est :
[y] = [1, 2, 3, 4, 5];
Cela signifie que la valeur de y
est égale à la première valeur du tableau, qui correspond au nombre 1
. Lorsque nous affichons y
, 1
est renvoyé.
const user = { name: "Lydia", age: 21 };
const admin = { admin: true, ...user };
console.log(admin);
- A:
{ admin: true, user: { name: "Lydia", age: 21 } }
- B:
{ admin: true, name: "Lydia", age: 21 }
- C:
{ admin: true, user: ["Lydia", 21] }
- D:
{ admin: true }
Réponse
Il est possible de combiner des objets en utilisant l'opérateur de déconstruction ...
. Il vous permet de créer des copies des paires clé / valeur d'un objet et de les ajouter à un autre objet. Dans ce cas, nous créons des copies de l'objet user
et nous les ajoutons à l'objet admin
. L'objet admin
contient maintenant les paires clé / valeur copiées, ce qui donne {admin: true, nom: "Lydia", age: 21}
.
const person = { name: "Lydia" };
Object.defineProperty(person, "age", { value: 21 });
console.log(person);
console.log(Object.keys(person));
- A:
{ name: "Lydia", age: 21 }
,["name", "age"]
- B:
{ name: "Lydia", age: 21 }
,["name"]
- C:
{ name: "Lydia"}
,["name", "age"]
- D:
{ name: "Lydia"}
,["age"]
Réponse
Avec la méthode defineProperty
, nous pouvons ajouter de nouvelles propriétés à un objet ou modifier celles existantes. Lorsque nous ajoutons une propriété à un objet en utilisant la méthode defineProperty
, il s’agit par défaut d'une propriété non énumérable. La méthode Object.keys
renvoie tous les noms de propriétés énumérable à partir d'un objet, dans ce cas uniquement "name"
.
Les propriétés ajoutées à l'aide de la méthode defineProperty
sont immuables par défaut. Vous pouvez remplacer ce comportement en utilisant les propriétés writeable
, configurable
et enumerable
. De cette façon, la méthode defineProperty
vous donne beaucoup plus de contrôle sur les propriétés que vous ajoutez à un objet.
const settings = {
username: "lydiahallie",
level: 19,
health: 90
};
const data = JSON.stringify(settings, ["level", "health"]);
console.log(data);
- A:
"{"level":19, "health":90}"
- B:
"{"username": "lydiahallie"}"
- C:
"["level", "health"]"
- D:
"{"username": "lydiahallie", "level":19, "health":90}"
Réponse
Le second argument de JSON.stringify
est le replaçant. Le remplaçant peut être une fonction ou un tableau, et vous permet de contrôler quoi et comment les valeurs doivent être stringifiées.
Si le remplaçant est un tableau, seules les propriétés dont les noms sont inclus dans le tableau seront ajoutées à la chaîne JSON. Dans ce cas, seules les propriétés avec les noms "level"
et "health"
sont incluses, "username"
est exclu. data
est maintenant égal à "{"level":19, "health":90}"
.
Si le remplaçant est une fonction, cette fonction est appelée sur chaque propriété de l'objet que vous personnalisez. La valeur renvoyée par cette fonction sera la valeur de la propriété lorsqu'elle sera ajoutée à la chaîne JSON. Si la valeur est undefined
, cette propriété est exclue de la chaîne JSON.
let num = 10;
const increaseNumber = () => num++;
const increasePassedNumber = number => number++;
const num1 = increaseNumber();
const num2 = increasePassedNumber(num1);
console.log(num1);
console.log(num2);
- A:
10
,10
- B:
10
,11
- C:
11
,11
- D:
11
,12
Réponse
L'opérateur arithmétique ++
renvoie en premier la valeur de l'opérande, puis incrémente la valeur de l'opérande. La valeur de num1
est égale à 10
, puisque la fonction increaseNumber
renvoie d'abord la valeur de num
, qui correspond à 10
, et augmente la valeur de num
par la suite.
num2
est égal à 10
, puisque nous avons passé num1
à la commande increasePassedNumber
. number
est égal à 10
(la valeur de num1
). Encore une fois, l'opérateur arithmétique ++
renvoie d'abord la valeur de l'opérande, puis incrémente_ la valeur de l'opérande. La valeur de nombre
est 10
, donc num2
est égal à 10
.
const value = { number: 10 };
const multiply = (x = { ...value }) => {
console.log((x.number *= 2));
};
multiply();
multiply();
multiply(value);
multiply(value);
- A:
20
,40
,80
,160
- B:
20
,40
,20
,40
- C:
20
,20
,20
,40
- D:
NaN
,NaN
,20
,40
Répondre
Dans ES6, nous pouvons initialiser les paramètres avec une valeur par défaut. La valeur du paramètre sera la valeur par défaut, si aucune autre valeur n'a été passée à la fonction, ou si la valeur du paramètre est "undefined"
. Dans ce cas, nous répartissons les propriétés de l'objet value
dans un nouvel objet, donc x
a la valeur par défaut {number: 10}
.
L'argument par défaut est évalué at call time! Chaque fois que nous appelons la fonction, un a new object créé. Nous invoquons la fonction multiply
les deux premières fois sans passer de valeur: x
a la valeur par défaut {number: 10}
. Nous enregistrons ensuite la valeur multipliée de ce nombre, qui est 20
.
La troisième fois que nous invoquons multiplier, nous passons un argument: l'objet appelé value
. L'opérateur * =
est en fait un raccourci pour x.number = x.number * 2
: nous modifions la valeur de x.number
, et enregistrons la valeur multipliée 20
.
La quatrième fois, nous passons à nouveau l'objet value
. x.number
a été précédemment modifié en 20
, donc x.number * = 2
enregistre «40».