Rumah >hujung hadapan web >tutorial js >Mengapa Gelung JavaScript Ini Menghasilkan Keputusan Tidak Dijangka dengan `setTimeout`?

Mengapa Gelung JavaScript Ini Menghasilkan Keputusan Tidak Dijangka dengan `setTimeout`?

DDD
DDDasal
2024-12-15 20:37:10189semak imbas

Why Does This JavaScript Loop Produce Unexpected Results with `setTimeout`?

Pelaksanaan Asynchronous Gelung dan setTimeout

Pertimbangkan skrip berikut:

for (var i = 1; i <= 2; i++) {
  setTimeout(function () { alert(i) }, 100);
}

Berbanding dengan nilai jangkaan 1 dan 2, skrip output 3 dua kali. Tingkah laku ini timbul disebabkan oleh sifat tak segerak bagi gelung peristiwa JavaScript.

Memahami Gelung Acara JavaScript

Gelung peristiwa JavaScript memproses pelaksanaan kod dalam dua peringkat: tindanan segerak dan baris gilir tak segerak. Kod segerak dilaksanakan serta-merta dalam timbunan, manakala kod tak segerak, seperti setTimeouts, diletakkan dalam baris gilir untuk dilaksanakan kemudian.

Isu dengan Gelung

Dalam skrip, setTimeout panggil balik fungsi menggunakan pembolehubah i. Walau bagaimanapun, i dikongsi antara semua lelaran gelung dan pada masa panggilan balik dilaksanakan, ia sentiasa merujuk kepada nilai akhir 2. Oleh itu, 3 dicetak kedua-dua kali.

Pendekatan yang Betul

Untuk memastikan nilai berturut-turut dicetak, cipta salinan i yang berbeza untuk setiap fungsi panggil balik. Ini boleh dicapai menggunakan penutupan fungsi, seperti yang ditunjukkan di bawah:

function doSetTimeout(i) {
  setTimeout(function () {
    alert(i);
  }, 100);
}

for (var i = 1; i <= 2; ++i) doSetTimeout(i);

Dalam skrip ini, fungsi doSetTimeout menangkap nilai i sebagai pembolehubah setempat untuk penutupannya, memastikan setiap fungsi panggil balik menggunakan yang betul nilai.

Atas ialah kandungan terperinci Mengapa Gelung JavaScript Ini Menghasilkan Keputusan Tidak Dijangka dengan `setTimeout`?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Kenyataan:
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn