Pourquoi le résultat de ('b' + 'a' + + 'a' + 'a'). ToLowerCase () 'banana'?


575

Je pratiquais du JavaScript lorsque l'un de mes amis est tombé sur ce code JavaScript:

document.write(('b' + 'a' + + 'a' + 'a').toLowerCase());

Le code ci-dessus répond "banana"! Quelqu'un peut-il expliquer pourquoi?


22
Ce deuxième plus est un opérateur unaire: +"a"est NaN.
Gerardo Furtado

8
Dans une console, écrivez tout +'a'seul et voyez ce qui se passe.
Un programmeur mec du

23
Et pour ceux qui en veulent plus. Voir la liste complète des divertissements
Giddy Naya

4
Fortement lié: stackoverflow.com/q/9032856
Kyll

Réponses:


566

+'a'se résout en NaN("Pas un nombre") car il contraint une chaîne à un nombre, tandis que le caractère ane peut pas être analysé comme un nombre.

document.write(+'a');
En minuscules, cela devient banana.

L'ajout NaNde "ba"tours NaNà la chaîne en "NaN"raison de la conversion de type donne baNaN. Et puis il y a un aderrière, donnant baNaNa.

L'espace entre les deux + +est de faire de la première concaténation de chaîne et de la seconde un opérateur unaire plus (c'est-à-dire "positif"). Vous obtenez le même résultat si vous utilisez 'ba'+(+'a')+'a', résolu comme 'ba'+NaN+'a', ce qui équivaut 'ba'+'NaN'+'a'à un jonglage de type.

document.write('ba'+(+'a')+'a');


90
'b' + 'a' + + 'a' + 'a'

... est évalué comme ....

('b') + ('a') + (+'a') + ('a')

(voir: priorité de l'opérateur )

(+'a')tente de convertir 'a'un nombre en utilisant l' opérateur unaire plus . Parce qu'il 'a'ne s'agit pas d'un nombre, le résultat est NaN ( "Not-A-Number" ):

'b'  +  'a'  +  NaN  + 'a'

Bien que NaNsignifie «pas un nombre», c'est toujours un type numérique ; lorsqu'il est ajouté à des chaînes, il concatène comme tout autre nombre:

'b'  +  'a'  +  NaN  + 'a'  =>  'baNaNa'

Enfin, il est en minuscule:

'baNaNa'.toLowerCase()      =>  'banana'

36
('b' + 'a' + + 'a' + 'a').toLowerCase()

Pour plus de clarté, décomposons cela en deux étapes. Tout d'abord, nous obtenons la valeur de l'expression entre parenthèses, puis nous appliquons la toLowerCase()fonction au résultat.

La première étape

'b' + 'a' + + 'a' + 'a'

Going LR , nous avons:

  • 'b' + 'a'renvoie ba , c'est une concaténation régulière.
  • ba + + 'a'tente de concaténer ba avec + 'a'. Cependant, puisque l'opérateur unaire +tente de convertir son opérande en nombre, la valeur NaN est renvoyée, qui est ensuite convertie en chaîne lorsqu'elle est concaténée avec le ba d' origine - résultant ainsi en baNaN .
  • baNaN+ 'a' renvoie baNaNa . Encore une fois, c'est une concaténation régulière.

À ce stade, le résultat de la première étape est baNaNa .

Deuxième étape

L'application .toLowerCase()de la valeur renvoyée à la première étape donne:

banane

Il existe de nombreux jeux de mots similaires en JavaScript que vous pouvez consulter.


24

C'est juste à cause de l' opérateur + .

Nous pouvons obtenir davantage de connaissances en le découpant.

=> ( ('b') + ('a') + (++) + ('a') + ('a'))
=> ( ('b') + ('a') + (+) + ('a') + ('a')) // Here + + convert it to +operator 
Which later on try to convert next character to the number.

Par exemple

const string =  '10';

Vous pouvez convertir une chaîne en nombre de 2 manières:

  1. Numéro (chaîne);
  2. + chaîne;

Revenons donc à la requête d'origine; Ici, il essaie de convertir le prochain caractère ('a') en nombre, mais tout à coup, nous avons eu une erreur NaN,

( ('b') + ('a') + (+'a') + ('a'))
( ('b') + ('a') + NaN + ('a'))

Mais il traite comme une chaîne car le caractère précédent était dans la chaîne. Alors il sera

( ('b') + ('a') + 'NaN' + ('a'))

Et enfin, il le convertit enLowerCase (), donc ce serait la banane

Si vous mettez un numéro à côté, votre résultat sera modifié.

( 'b' + 'a' +  + '1' + 'a' ) 

Ce serait 'ba1a'

const example1 = ('b' + 'a' + + 'a' + 'a').toLowerCase(); // 'banana' 
const example2 = ('b' + 'a' + + '1' + 'a').toLowerCase(); // 'ba1a'
console.log(example1);
console.log(example2);


9

Cette ligne de code évalue une expression, puis appelle une méthode basée sur la valeur renvoyée.

L'expression ('b' + 'a' + + 'a' + 'a')est uniquement composée de littéraux de chaîne et d'opérateurs d'addition.

  • Littéraux de chaîne "Un littéral de chaîne est composé de zéro ou plusieurs caractères entre guillemets simples ou doubles."
  • L'opérateur d'addition (+) "L'opérateur d'addition effectue une concaténation de chaîne ou une addition numérique."

Une action implicite prise est l'appel de ToNumber sur une chaîne

  • ToNumber appliqué au type de chaîne "ToNumber appliqué aux chaînes applique la grammaire à la chaîne d'entrée. Si la grammaire ne peut pas interpréter la chaîne comme une expansion de StringNumericLiteral, le résultat de ToNumber est NaN."

L'interpréteur a des règles sur la façon d'analyser l'expression, en la décomposant en ses composants d'expressions de gauche et de droite.


Étape 1: 'b' + 'a'

Expression de gauche: 'b'
Valeur de gauche: «b»

Opérateur: + (l'un des côtés d'expression est une chaîne, donc concaténation de chaîne)

Bonne expression: 'a' bonne valeur: 'a'

Résultat: 'ba'


Étape 2: 'ba' + + 'a'

Expression gauche: 'ba'
Valeur de gauche: «ba»

Opérateur: + (l'un des côtés d'expression est une chaîne, donc concaténation de chaîne)

Expression de droite: + 'a'(ceci évalue la valeur mathématique du caractère 'a' en supposant qu'il s'agit d'un nombre positif du signe + - le signe moins aurait également fonctionné ici indiquant un nombre négatif - ce qui donne NaN)
Valeur de droite: NaN (car l'opérateur est une concaténation de chaînes, toString est appelé sur cette valeur pendant la concaténation)

Résultat: 'baNaN'


Étape 3: 'baNaN' + 'a'

Expression de gauche: 'baNaN'
Valeur de gauche: 'baNaN'

Opérateur: + (l'un des côtés d'expression est une chaîne, donc concaténation de chaîne)

Bonne expression: 'a'
bonne valeur: 'a'

Résultat: 'baNaNa'


Après cela, l'expression de regroupement a été évaluée, et toLowerCase est appelé, ce qui nous laisse avec la banane.


7

L'utilisation de + convertira n'importe quelle valeur en nombre en JavaScript!

Donc...

La principale chose à savoir en premier lieu et à en tirer des leçons est d'utiliser +avant toute valeur en JavaScript, convertira cette valeur en nombre , mais si cette valeur ne peut pas être convertie, le moteur JavaScript retournera NaN , ce qui signifie, pas un nombre (ne peut pas être converti en un nombre, mec!) et le reste de l'histoire comme ci-dessous:

Pourquoi le résultat de ('b' + 'a' + + 'a' + 'a'). ToLowerCase () 'banana'?



0

Voyez la magie ici. Le deuxième plus est un opérateur unaire qui donne «NaN»

console.log(('b' + 'a' + + 'a' + 'a').toLowerCase());
console.log(('b' + 'a' + + 'a' + 'a'));
console.log(('b' + 'a' + 'a' + 'a').toLowerCase());

En utilisant notre site, vous reconnaissez avoir lu et compris notre politique liée aux cookies et notre politique de confidentialité.
Licensed under cc by-sa 3.0 with attribution required.