{}[true]
adalah [true]
dan ![true]
seharusnya false
.
Jadi mengapa !{}[true]
harus dievaluasi true
?
javascript
pengguna2430508
sumber
sumber
var o = {}; o[true] === undefined
.{}[true] === [true]
dari konsol, itu karena itu memperlakukan{}
sebagai blok kode kosong, bukan objek.{}
dan({})
di konsol Anda (atau{}[true]
dan({})[true]
). Juga, karena tidak ada yang menyebutkannya, objek [benar] dievaluasi ke objek ["benar"].Jawaban:
Saya percaya itu karena plain
{}[true]
diuraikan sebagai blok pernyataan kosong (bukan objek literal) diikuti oleh array yang berisitrue
, yaitutrue
.Di sisi lain, menerapkan
!
operator membuat parser mengartikan{}
sebagai objek literal, sehingga berikut ini{}[true]
menjadi akses anggota yang kembaliundefined
, dan!{}[true]
memangtrue
(sebagaimana!undefined
adanyatrue
).sumber
undefined
adalah falsy (sesuatu yang sering kita andalkan -if (obj.maybeExists) ...
), sehingga masuk akal logis yang!undefined
benar.null
dalam beberapa bahasa, dengan!undefined
menjadi sama denganundefined
. Tapi itu tidak terjadi di Javascript.not undefined
(!undefined
) harus didefinisikan. Jika sesuatu didefinisikan maka biasanya ditafsirkan sebagaitrue
.Karena
{}[true]
tidak kembalitrue
, tetapiundefined
, danundefined
dievaluasi sebagaifalse
:http://jsfiddle.net/67GEu/
sumber
{}[true]
di konsol, Anda mendapatkan[true]
, karena{}
ditafsirkan sebagai blok kode kosong, bukan objek. Ini semua tentang konteks dan ambiguitas{}
.{key:"value"}[1,2,3];
juga mengevaluasi[1,2,3]
?key:
) dan string literal ("value"
), diikuti oleh array. Parser masih tidak melihat objek literal.alert()
atauconsole.log()
, atau menetapkannya ke variabel, Anda mengubah konteksnya, itulah sebabnya ia tidak berperilaku sama seperti mengetiknya sendiri di konsol.Karena
mengevaluasi untuk
undefined
, dan!undefined
adalahtrue
.Dari @schlingel:
true
digunakan sebagai kunci dan{}
sebagai peta hash. Tidak ada properti dengan kuncitrue
sehingga kembaliundefined
. Tidakundefined
initrue
, seperti yang diharapkan.Sesi konsol ( Node.js
[0.10.17]
):Namun, di konsol Google Chrome :
Jadi, tidak ada inkonsistensi. Anda mungkin menggunakan versi lama JavaScript VM. Bagi mereka yang membutuhkan bukti lebih lanjut:
MEMPERBARUI
Dengan Firefox , itu juga mengevaluasi untuk
true
:sumber
eval('{}[true]')
atau mengetiknya di konsol. Kemudian misalnya als{}"test"
adalahtest
atau bahkan{key:"value"}"test"
adalahtest
.{}[true];
(dengan;
) kembali[true]
untuk Anda, karena ini dia?Alasan kebingungan adalah kesalahpahaman dari pernyataan pertama Anda:
{}[true]
adalah[true]
Apa yang Anda lihat ketika Anda menjalankannya adalah hasil dari ambiguitas. Javascript memiliki seperangkat aturan yang ditentukan tentang cara menangani ambiguitas seperti ini, dan dalam hal ini, Javascript memecah apa yang Anda lihat sebagai pernyataan signle menjadi dua pernyataan terpisah.
Jadi Javascript melihat kode di atas sebagai dua pernyataan terpisah: Pertama, ada
{}
, dan kemudian ada yang sepenuhnya terpisah[true]
. Pernyataan kedua adalah apa yang memberi Anda hasilnya[true]
. Pernyataan pertama secara{}
efektif diabaikan sepenuhnya.Anda dapat membuktikan ini dengan mencoba yang berikut:
yaitu membungkus semuanya dalam tanda kurung untuk memaksa penerjemah untuk membacanya sebagai satu pernyataan.
Sekarang Anda akan melihat bahwa nilai sebenarnya dari pernyataan Anda adalah
undefined
. (ini juga akan membantu kita nanti untuk memahami bagian selanjutnya)Sekarang kita tahu bahwa bagian awal dari pertanyaan Anda adalah ikan haring merah, jadi mari kita lanjutkan ke bagian akhir dari pertanyaan:
Di sini, kita memiliki pernyataan yang sama, tetapi dengan
!
menambahkan di depannya.Dalam hal ini, aturan Javascript memberitahu untuk mengevaluasi semuanya sebagai satu pernyataan.
Lihat kembali apa yang terjadi ketika kami membungkus pernyataan sebelumnya dalam tanda kurung; kami punya
undefined
. Kali ini, kita secara efektif melakukan hal yang sama, tetapi menempatkannya!
di depannya. Jadi kode Anda dapat disederhanakan!undefined
, yaitutrue
.Semoga itu menjelaskan sedikit.
Ini adalah binatang yang kompleks, tetapi pelajaran untuk belajar di sini adalah menggunakan tanda kurung di sekitar pernyataan Anda saat mengevaluasi mereka di konsol, untuk menghindari hasil palsu seperti ini.
sumber
{}[true]
itu tidak benar, hanya ambigu . Ini dapat diartikan sebagai "blok kode kosong diikuti oleh array literal" atau "objek literal tanpa properti, yang properti sedang diakses". Saya tidak tahu apakah yang pertama secara teknis adalah kasus ASI (banyak bahasa tidak akan menempatkan titik koma di sana), tetapi interpretasi yang peka konteks itulah yang menjadi inti masalah.{}[true]
adalahtrue
" kata mereka "{}[true]
adalah[true]
", yang merupakan salah satu dari dua interpretasi valid dari pernyataan ambigu.{}[true]
adalahundefined
. Untuk menemukan yang menulis ini:atau hanya:
Kita tahu bahwa
!undefined
adalahtrue
.Dari jawaban @Benjamin Gruenbaum :
Informasi lebih lanjut dapat ditemukan dalam pertanyaan ini .
sumber
Jawaban di sini baik, berikut adalah rincian dalam pseudo-code:
{}['whatever']
= blok kosong, NewArray ('apa pun') = NewArray ('apa pun'){}[true]
= blok kosong, NewArray (true) = NewArray (true)!{}['whatever']
= LogicalNOT (convertToBool (NewObject.whthing)) = LogicalNOT (convertToBool (undefined)) = LogicalNOT (false) = true({}['whatever'])
= Pengelompokan (NewObject.whthing) = Pengelompokan (tidak terdefinisi) = tidak terdefinisisumber
Ini terjadi karena
{}
dalam arti Anda bukan presentasi literalObject
, tetapi ruang lingkup kosong (atau blok kode kosong):Itu hanya mengevaluasi kode di dalam lingkup dan kemudian menunjukkan array Anda.
Dan dari Anda
Konversi ke int lingkup ini dan mengembalikan array yang sama benar. Tidak ada pemeriksaan bool dalam kode ini.
Dan jika Anda akan mencoba untuk memeriksa hasil dari
{}[true]
Anda akan mendapatkanfalse
:Karena tidak ada lagi ruang lingkup.
Jadi,
!
dalam pertanyaan Anda lakukan hal yang sama seperti:sumber
var x = {}; x[true]
.!
itu adalah ditafsirkan sebagai objek kosong, tidak ruang lingkup, dan ini adalah perbedaan tersebut.{}
adalah objek tanpa properti.[]
segera mengikuti objek, itu berarti "Akses properti dari nama ini" dan bukan "Buat array"true
adalah boolean, tetapi digunakan sebagai nama properti sehingga dilemparkan ke string ("true"
)true
(karena tidak memiliki properti) begitu{}['true']
pulaundefined
!undefined
dilemparkanundefined
ke boolean (false
)false
menjaditrue
.sumber
{}[true]
(tanpa konteks lain),{}
adalah bukan suatu objek tanpa sifat, itu sebuah blok kode yang kosong.Anda tidak membalikkan nilainya.
Lihat ini: Mengapa! Benar? 'false': 'benar' mengembalikan 'benar'?
sumber
Ayo Main Sedikit Lagi!
Pertama, mari bersenang-senang !:
Ok, mari kita coba memahami perilaku gila ini, satu per satu:
1) Di sini,
{}
diuraikan sebagai blok kode kosong. Tanpa penetapan, negasi, pengelompokan (dengan tanda kurung) atau sintaks apa pun yang menunjukkan kepada parser bahwa ini{}
adalah objek literal, asumsi default adalah menganggapnya hanyalah blok kosong yang tidak berguna.Ini adalah bukti dari perilaku ini:
Kode di atas akan menampilkan peringatan itu secara normal, dan akan dievaluasi seperti
[true]
halnya dengan cara yang sama{}[true]
.Blokir Pernyataan Tanpa Titik Koma
Pernyataan tipe blok tidak perlu titik koma setelahnya.
Misalnya:
Kedua peringatan ditampilkan.
Jadi, kita dapat melihat bahwa pernyataan blok kosong, tanpa titik koma, valid dan tidak melakukan apa-apa. Dengan cara ini, ketika Anda memasukkan
{}[true]
di Konsol Alat Pengembang (atau Firebug), nilai yang dievaluasi akan menjadi nilai pernyataan ekspresi terakhir . Dalam hal ini, pernyataan ekspresi terakhir adalah[true]
.2) Dalam konteks penugasan, parser akan memastikan bahwa itu
{}
adalah objek literal. Ketika Anda melakukan var a ={}[true]
, Anda menghapus ambiguitas dan memberi tip parser yang{}
bukan pernyataan blok.Jadi, di sini, Anda mencoba untuk mendapatkan nilai dengan kunci
"true"
dari objek kosong. Jelas, tidak ada pasangan nilai kunci dengan nama kunci ini. Dengan cara ini, variabel tidak terdefinisi.Kata-kata yang dicadangkan sebagai kunci Objek
ECMAScript 5 memungkinkan kunci objek untuk dilindungi undang-undang. Jadi, kunci-kunci berikut ini legal:
3) Penjelasan yang sama dari contoh 1 . Tapi ... Jika
{ b: 12345 }
bagian itu diperlakukan sebagai pernyataan blok, apa jenisb: 12345
pernyataannya ??... (?????)
Ini pernyataan label , Anda sudah melihatnya sebelumnya ... Ini digunakan dalam loop dan in
switch
. Berikut adalah beberapa tautan menarik tentang pernyataan label: 1 , (2) [ Cara terbaik untuk melepaskan diri dari loop bersarang di Javascript? , (3) [ Bagaimana cara memecah loop bersarang di javascript? .CATATAN: Coba saja evaluasi ini:
Pernyataan label tidak dapat dipisahkan oleh operator koma , Anda harus memisahkannya dengan tanda titik koma. Jadi ini valid:
{a: 1; b: 2}
4) Lihat penjelasan untuk contoh 1 dan 3 ...
5) Sekali lagi, kami
{ b: 12345 }
diperlakukan sebagai blok kode, dan Anda mencoba mengakses properti blok kode dengan menggunakan notasi titik , dan jelas, ini tidak diizinkan, dan parser melempar"Unexpected token :"
pengecualian.6) Kode hampir identik dengan contoh di atas, tetapi dengan mengelilingi
{ b: 12345 }
pernyataan dengan operator pengelompokan ekspresi , parser akan tahu bahwa itu adalah objek. Dengan cara ini, Anda dapat mengakses"b"
properti secara normal.7) Ingat contoh 2 , kami punya tugas di sini, parser tahu itu
{ b: 12345 }
adalah objek.8) Identik dengan contoh di atas, tetapi alih-alih notasi titik, di sini kita menggunakan notasi braket .
9) Saya sudah mengatakan bahwa
"identifier: value"
sintaks ini di dalam pernyataan blok adalah label. Tapi, Anda juga harus tahu bahwa nama label tidak boleh kata kunci yang dipesan (kebalikan dari nama properti objek). Ketika kami mencoba mendefinisikan label yang disebut"true"
, kami mendapat aSyntaxError
.10) Sekali lagi, kita berurusan dengan suatu objek. Tidak ada masalah menggunakan kata-kata yang dipesan di sini. =)
11) Akhirnya, kami memiliki ini:
!{}[true]
Mari kita pisahkan hal-hal di sini:
a) Dengan melakukan negasi, kami memberi tahu parser bahwa objek
{}
adalah objek .b) Seperti yang ditunjukkan pada contoh 2 ,
{}
objek tidak memiliki properti yang dipanggiltrue
, jadi ungkapan ini akan dievaluasiundefined
.c) Hasil akhirnya adalah negasi
undefined
nilai. Javascript melakukan konversi tipe implikasi , danundefined
nilainya falsy .d) Jadi, negasi dari
false
adalah ...true
!sumber