Очерёдность событий и синхронизация в JavaScript
Проблемы с очерёдностью исполнения являются источником некоторых наиболее коварных багов в JavaScript-приложениях. Ошибки, которые не проявлялись во время разработки, внезапно начинают приводить к проблемам, когда приложение используется конечным пользователем на старом компьютере или с медленным доступом в интернет. Эти проблемы могут проявляться случайным образом. В этом случае их довольно сложно воспроизвести.
Простой пример: рассмотрим кнопку с обработчиком нажатия мышью, которая изменяет некоторый элемент, находящийся ниже её. Если пользователь нажимает кнопку до того, как исходный текст элемента был распарсен (декодирован и преобразован во внутреннее представление), скрипт вызовет ошибку. Разработчик никогда не столкнётся с этой проблемой, поскольку он тестирует страницу на быстром компьютере с быстрым подключением, на котором вся страница загружается в одно мгновение.
Данная статья является попыткой описать всевозможные проблемы с очерёдностью исполнения в современных браузерах.
Окно браузера использует единственный поток исполнения, который производит разпознавание HTML-кода, управление событиями и выполнение кода на JavaScript.
Код на JavaScript исполняется в одном из двух режимов:
- Код высшего уровня (основная программа) выполняется в процессе загрузки страницы
- Функции-обработчики событий выполняются в процессе управления событиями
Оба типа исполнения инициируются браузером, и они выполняются в одном и том же потоке таким образом, что только одна единица кода выполняется в один момент времени.
Как правило, исполнение кода в браузере управляется событиями (т.е. является реакцией на пользовательский ввод), но во время загрузки страницы оно управляется ещё и потоком парсинга страницы.
Событие является сигналом от браузера, говорящим о каком-то изменении в окне страницы, которое уже произошло или должно произойти в ближайшем времени, если вы не предпримете каких-либо мер.
Обработчик события - функция на JavaScript, которая назначается в соответствие некоторой паре из объекта и названия события. Когда соответствующее событие происходит для данного объекта, выполняются все обработчики событий, назначенные этому узлу.
Все функции-обработчики событий выполняются последовательно, и каждое событие обрабатывается полностью (включая всплывание сквозь DOM и выполнение действия по умолчанию) перед тем, как будет выполнено следующее событие.
Действие по умолчанию является особенностью модели событий браузера: это действие, которое происходит, если JavaScript не перехватывает событие. Например, действие по умолчанию для события click на ссылке является переход на URL. Событием по умолчанию для нажатия мышью на чекбоксе является включение галочки, и так далее.
Действие по умолчанию не является само по себе обработчиком события, и мы не можем его удалить или переопределить, в отличие от пользвательских обработчиков событий. Однако, мы можем отменить его выполнение во время обработки события, используя preventDefault() (или event.returnValue в IE). Если действие по умолчанию отменено, все прочие обработчики событий всё равно будут выполнены, но действие по умолчанию после этого выполнено не будет.
Такие события как load возникают только для соответствующего объекта (в данном случае, window или document ). Однако, для событий, происходящих для отдельных элементов документа, возможно также выполнение обработчиков событий для элементов-предков.
До того как событие возникнет для целевого объекта, имеет место фаза перехвата, во время которой предки целевого узла могут перехватить событие. Однако, перехватывание событий работает неодинаково в разных браузерах.
Некоторые события всплывают, что означает, что после того как они возникли для целевого элемента, они по очереди возникают для каждого предка в DOM-дереве, вплоть до объекта document включительно. Этот процесс идентичен в различных браузерах.
Весь процесс возникновения события для всех соответствующих элементов и вызова действия по умолчанию называется управлением событиями.
Для невсплывающего события порядок управления событиями является таким:
- Фаза перехвата: все "перехватывающие" обработчики событий выполняются для всех элементов-предков, сверху вниз по иерархии.
- Событие возникает для целевого элемента, что означает, что выполняются все обработчики события, назначенные данному элементу для данного события (в неопределённом порядке).
- Выполняется действие по умолчанию (если оно не было отменено одним из обработчиков).
Для всплывающего события порядок таков:
- Фаза перехвата: все "перехватывающие" обработчики событий выполняются для всех элементов-предков, сверху вниз по иерархии.
- Событие возникает для целевого элемента.
- Фаза всплывания: событие возникает для всех событий-предков, начиная с целевого и выше.
- Выполняется действие по умолчанию (если оно не было отменено одним из обработчиков).
Всплывание события возможно отменить используя stopPropagation() (cancelBubble() в IE), однако действие по умолчанию всё равно будет выполнено. Таким образом, отмена всплывания и отмена действия по умолчанию являются отдельными и независимыми операциями.
Отдельные этапы работы модели событий объясняются гораздо более подробно в спецификации DOM 3 Events.
Существуют некоторые интересные ситуации, в которых действие по умолчанию на самом деле происходит до управления событиями - но всё равно может быть отменено. Например, когда происходит нажатие на чекбоксе, отображение галочки и установка атрибута checked происходят ещё до активации обработчиков события. Однако, если действие по умолчанию отменяется, эти изменения откатываются во время фазы действия по умолчанию. Галочка снова снимается, а атрибут checked переключается обратно.
Некоторые события активируются группами: т.е. одно действие пользователя приводит к вызову нескольких событий.
Например, когда фокус переходит от одного поля к другому, событие blur возникает для одного поля, а focus - для другого. В принципе, это происходит одновременно (поскольку это является реакцией на одно действие пользователя), однако события всё равно возникают по очереди.
Если событие всплывает, вся последовательность перехвата-всплывания событий и вызов действия по умолчанию происходят перед обработкой следующего события.
Примером такого случая является отпускание кнопки мыши над кнопкой на странице. В этом случае возникают события mouseup и click . Выполняются следующие этапы:
- Фаза перехвата для
mouseup – выполняются все перехватывающие обработчики.
- Событие возникает для целевого элемента.
- Фаза всплывания для
mouseup - событие возникает для всех родительских элементов
- (Для
mouseup нет действия по умолчанию).
- Фаза перехвата – выполняются все перехватывающие обработчики.
- Событие
click возникает для целевого элемента.
- Фаза всплывания - событие
click возникает для всех родительских элементов
- Выполняется действие по умолчанию для
click .
В каждом случае обработки события возможна отмена только текущего события. Например, в обработчике события mouseup отмена действия по умолчанию не возымеет никакого эффекта, поскольку для mouseup нет действия по умолчанию. Это не остановит возникновение события click, поскольку они являются различными событиями.
Однако действие по умолчанию может привести к возникновению другого события. В случае события click для кнопки отсылки формы действием по умолчанию является отсылка текущей формы, которая, в свою очередь, породит событие submit . В этом случае отмена события по умолчанию предотвратит возникновение следующего события.
События возникают в ответ на пользовательский ввод (с мыши или клавиатуры) или в ответ на внутренние события, например окончание загрузки страницы. Однако, вызов событий асинхронен породившему его вводу.
Пользовательский ввод может случиться в тот момент, когда обработчик другого события всё ещё выполняется. В таком случае действия пользователя буферизуются, и когда модуль обработки событий вновь свободен, происходит обработка событий, соответствующих буферизованным действиям. События всегда возникают в правильном порядке, но между действием и вызовом события может произойти заметная задержка, если код некоторого обработчика события займёт продолжительное время.
Internet Explorer и Mozilla совершенно не реагируют на пользовательские действия в течение выполнения скриптов-обработчиков. Даже панели инструментов браузеров блокируются. Хотя пользователь по-прежнему может, например, нажимать на кнопки, и эти действия заносятся в буфер, никакой наблюдаемой реакции они не вызывают. Это может дезориентировать пользователя, который, не осознав, что его действие было учтено, вероятно, попробует нажать кнопку несколько раз, что может привести к нежелательным последствиям. Пользователь даже может решить, что браузер завис вследствие ошибки.
Opera же продолжает визуально реагировать на действия пользователя (например, нажатия кнопок) даже во время исполнения другого скрипта. Однако действия по-прежнему буферизуются и обрабатываются последовательно, как и в других браузерах. Таким образом, действия по умолчанию для события не производятся до тех пор, пока обработчик событий не доберётся до него. Это тоже может озадачить пользователя, хотя, наверное, не так сильно, как полная блокировка IE и Mozilla.
Отсюда следует сделать вывод, что скрипты-обработчики событий никогда не должны занимать много времени. С особой осторожностью следует отнестись к синхронным запросам XMLHttpRequest , так как они могут вызвать заметную задержку, которая заблокирует браузер или окно документа.
Существует особый случай, когда события обрабатываются не последовательно, а "вложенно". Если событие активируется непосредственно из скрипта с использованием метода dispatchEvent() (fireEvent() в Internet Explorer), то событие обрабатывается немедленно. Исходный скрипт продолжит выполняться только тогда, когда закончится обработка вложенного события (и будет выполнено действие по умолчанию).
Также события изменения DOM, которые не поддерживаются в Internet Explorer, будут обработаны синхронно, непосредственно в момент изменения DOM, например при вызове appendChild() .
Производимые скриптом изменения в DOM или наборе стилей могут быть отображены не сразу. Это зависит от браузера.
Например, если цвет фона элемента меняется при помощи DOM, изменения немедленно будут отражены в структуре DOM (и событие изменения DOM будет немедленно синхронно обработано), но мы не можем знать наверняка, когда движок рендеринга браузера сподобится реально отобразить эти изменения на экране. Похоже, что в Mozilla и Internet Explorer изменения откладываются до окончания текущей обработки событий, а в Opera они отображаются немедленно.
Метод setTimeout() позволяет "заказать" выполнение функции через определённый промежуток времени:
window.setTimeout(someFunction, 1000);
Отложенные скрипты работают в некотором роде аналогично скриптам обработки событий. Хотя они и выполняются по прошествии определённого количества времени, а не в ответ на пользовательский ввод, они так же последовательно обрабатываются потоком обработки событий, как и пользовательские действия.
По этой причине не следует ожидать, что отложенное действие будет выполнено точно в указанный момент. Если выполняются другое событие или пакет, отложенный скрипт будет просто поставлен в очередь. Фактически, мы можем быть уверены только в том, что функция будет выполнена через как минимум секунду. На самом деле времени может пройти гораздо больше.
Это удивительно полезная особенность. Если обработчик отложен на время 0, он не выполняется, но немедленно добавляется к очереди. Он не будет выполнен до тех пор, пока не отработает обработка текущего события (включая действие по умолчанию).
Если отложенное выполнение создаётся в обработчике события, являющегося частью пакета событий (например blur /focus , mouseup /click ), то отложенный код будет выполнен после завершения обработки всех событий в пакете.
К прочим событиям, не вызываемым действиями пользователя, относятся:
- События загрузки страницы
- Отложенные события
- Обработчики результатов асинхронных запросов
XMLHttpRequest
Эти события добавляются в очередь обработки событий аналогично событиям, вызываемым действиями пользователя. Это, например, означает, что обработчик ответа XMLHttpRequest не выполняется непосредственно в момент получения данных, но всего лишь становится в очередь обработки событий.
Диалоговые окна alert (а также confirm и prompt ) обладают некоторыми странными свойствами.
Они синхронны в том смысле, что выполнение скрипта, вызвавшего этот диалог, замораживается до закрытия диалога. Скрипт ожидает завершения функции alert() , прежде чем продолжить выполнение.
Однако некоторые браузеры позволяют производить обработку событий до тех пор, пока диалог виден на экране и ожидает пользовательских действий. Это означает, что пока один скрипт заморожен, ожидая конца выполнения функции alert , другая функция может быть выполнена в процессе обработки другого события.
События пользовательского интерфейса, такие как mouseup и click , не будут возникать в момент исполнения alert , поскольку диалог является модальным и перехватывает весь пользовательский ввод, однако не-пользовательские события, такие как загрузка страницы, отложенный код и обработчики результата асинхронных запросов XMLHttpRequest , могут возникнуть в это время.
Страницы HTML парсятся и отображаются прогрессивно, т.е. по мере скачивания документа браузером.
Большая часть внешних ресурсов, таких как изображения и встраиваемые медиа-объекты, загружается асинхронно. Когда парсер встречает элементы img , embed , iframe или object , создаётся новый поток. Он скачивает и отображает внешний ресурс независимо от парсинга основной страницы. Страницы во фреймах и встроенных фреймах (iframe ) также загружаются асинхронно.
Внешние таблицы стилей являются особым случаем. Некоторые браузеры загружают их асинхронно (как изображения), некоторые браузеры загружают их синхронно, предположительно чтобы избежать повторного рендеринга всей страницы, когда таблица стилей догрузится. (Это позволяет избежать проблемы мерцания не использующего стили контента, которая была свойственна ранним браузерам.) Другими словами, полагаться на какое-либо конкретное поведение здесь не следует.
Элементы script парсятся синхронно. Когда элементы script ссылаются на внешние скрипты, парсинг страницы останавливается до тех пор, пока внешние скрипты не будут подгружены, распарсены и выполнены.
Блоки JavaScript, содержащие код, парсятся и выполняются в тот момент, когда встречается закрывающий тег.
Блок JavaScript (содержащий встроенный код или ссылающийся на внешний файл) обрабатывается в два этапа. Сначала он парсится и выполняется. В процессе парсинга производится проверка базового синтаксиса кода. Если обнаруживается синтаксическая ошибка, выполнение скрипта не производится.
На этапе исполнения выполняется весь код высшего уровня (т.е. не принадлежащий функциям). Инструкции высшего уровня могут содержать внешние ссылки на функции, объявленные в том же блоке, так как объявления функций загружаются на этапе парсинга. Этот код будет работать:
var x = getMagicNumber();
function getMagicNumber() { return 117; }
Однако следующий код работать не будет, поскольку выражения function вычисляются только в ходе выполнения:
var x = getMagicNumber(); // ОШИБКА! getMagicNumber не определена!
var getMagicNumber = function() { return 117; }
Следующий код также не будет работать, поскольку каждый блок script исполняется непосредственно после того, как будет встречен закрывающий тег:
<script>
alert(getMessage());
</script>
<script>
function getMessage() { return "Hello!"; }
</script>
Скрипт может выводить HTML-код непосредственно в текущий документ, используя метод document.write() . Сгенерированный код будет буферизован до завершения выполнения текущего блока. После этого буферизованный код будет распарсен. Этот код, в свою очередь, может (как всё сложно!) содержать блоки script , которые выполняются в ходе парсинга.
Сгенерированный HTML-код вставляется в документ непосредственно после сгенерировавшего его блока script .
По ходу загрузки страницы парсер последовательно выстраивает дерево DOM. Пустой элемент вставляется в DOM когда парсится соответствующий тег. Непустой элемент вставляется, когда парсится открывающий тег. Например, элемент body появляется в DOM, как только парсер начинает парсить содержимое элемента.
Заметим, что DOM не обязательно точно соответствует входному HTML-коду. Такие элементы как html и head будут созданы в DOM даже если они не встречаются в HTML.
Если исходный код HTML некорректен (например, элемент title появляется внутри body ), браузер перестроит DOM так, чтобы он был корректным. В этом случае нельзя положиться на то, что дерево DOM будет выстроено по порядку.
У синхронной загрузки скриптовых блоков есть существенный недостаток: если в ходе парсинга заголовка страницы требуется загрузить и выполнить большое количество кода, перед началом отрисовки страницы может наблюдаться существенная задержка.
Чтобы устранить эту проблему, мы могли бы использовать в элементах script атрибут defer . Он означает, что браузеру разрешается загружать этот скрипт асинхронно. Однако мы не можем быть уверены, когда скрипт в действительности будет загружен. Это может произойти как до, так и после окончания рендеринга страницы. Opera полностью игнорирует атрибут defer .
<script defer>
alert("это сообщение появится в непредвиденный момент во время загрузки страницы");
</script>
Отложенные скрипты не могут использовать document.write() , поскольку они не синхронизированы с парсером.
Существует ещё одна тонкость: скриптовые блоки всегда выполняются в том порядке, в котором они появились в документе, вне зависимости от наличия атрибута defer . Так что если элемент script без атрибута defer следует за скриптом с атрибутом defer , парсер должен закончить загрузку и выполнение отложенного скрипта до исполнения не-отложенного скрипта. При этом теряется весь смысл использования атрибута defer . Это означает, что всегда необходимо располагать не-отложенные блоки скриптов перед отложенными.
По этим причинам атрибут defer нельзя использовать для задания очерёдности выполнения скриптовых блоков. Он всего лишь позволяет некоторым браузерам продолжить парсинг документа после блока script .
Ход рендеринга страницы на экране не всегда синхронизирован с построением DOM. Порядок рендеринга в ходе загрузки страницы является довольно непредсказуемым. В зависимости от скорости подключения и размера страницы, браузер может, прежде чем начать рендеринг, ждать загрузки всей страницы, либо, в случае медленного соединения, может рендерить страницу по частям.
Следует иметь в виду, что интерфейс реагирует на пользовательские события с того момента, как страница начинает рендериться. Это может привести к проблемам с "ссылками вперёд", если обработчик события ссылается на элемент, который встречается далее по ходу документа.
Пример опасного кода:
<button
onclick="document.getElementById('lamp').backgroundColor = 'yellow'">
Жми сюда, чтобы включить лампу!
</button>
<div id='lamp'>O</div>
Проблема здесь в том, что элемент 'lamp' может быть ещё не распарсен в тот момент, когда кнопка нажата. Обработчик события никогда не должен ссылаться на элементы, определённые далее в документе.
В более сложных пользовательских интерфейсах отказ от ссылок вперёд между элементами интерфейса может быть неприемлем. Вместо этого следует сделать все элементы управления отключенными по умолчанию и активировать их только в обработчике события onload , где мы можем быть уверены, что загрузка всей страницы завершена.
Заметим, что onload также ждёт окончания загрузки всех изображений (и фреймов, и т.п.). Если на странице есть большие изображения, это может занять длительное время. Обходным решением является активация страницы посредством встроенного скрипта внизу страницы. Он будет выполнен при окончании загрузки страницы, но не будет дожидаться загрузки внешних ресурсов.
В идеале, код на JavaScript никогда не должен выполняться слишком долго, поскольку это доставляет неудобства пользователю. Но в некоторых ситуациях это неизбежно. В этом случае следует показать сообщение вроде "Пожалуйста, подождите" или индикатор прогресса выполнения, чтобы сообщить пользователю, что браузер не завис. Задача состоит в том, что сообщение должно быть показано до начала выполнения длительного процесса.
Вот пример в псевдо-коде:
headlineElement.innerHTML = "Пожалуйста, подождите...";
performLongRunningCalculation();
headlineElement.innerHTML = "Закончено!";
В Internet Explorer и Mozilla текст "Пожалуйста, подождите..." никогда не будет показан пользователю, поскольку изменения будут отображены только после окончания работы скрипта. С другой стороны, в Opera текст "Пожалуйста, подождите..." будет отображён во время работы длительных вычислений.
Для того чтобы отобразить сообщение в Internet Explorer и Mozilla, необходимо на время передать управление интерфейсу браузера, чтобы сообщение было отрисовано до начала вычислений:
headlineElement.innerHTML = "Пожалуйста, подождите...";
function doTheWork() {
performLongRunningCalculation();
headlineElement.innerHTML = "Закончено!";
}
setTimeout(doTheWork, 0);
Фокус с setTimeout гарантирует, что сообщение будет отображено до того, как браузер будет заблокирован длительной работой. Конечно, во время вычислений браузер всё равно заблокирован, поэтому это является не очень элегантным решением. Если мы хотим избежать блокировки, следует разбить вычисления на несколько функций, соединённых вместе через setTimeout . Однако на этом пути возникает множество сложностей.
Каждое окно (и фрейм) обладает собственной очередью.
В Opera каждое окно обладает собственным потоком JavaScript. Это включает окна в iframe . Результатом этого является то, что обработчики событий, запущенные из различных фреймов, могут выполняться одновременно. Если эти одновременные скрипты обращаются к разделённым данным (например, свойствам главного окна), появляется возможность возникновения состояний гонок (race conditions).
Я не стану углубляться в возможные опасности состояний гонок, просто укажу на то, что они могут приводить к крайне загадочным багам.
В качестве решения можно всегда добавлять обработчики событий в очередь событий основного окна, даже если они были вызваны событиями в других фреймах.
Рассмотрим страницу с iframe . Элемент iframe обладает обработчиком onload , который будет выполнять функцию в содержащей его странице:
// плохая функция onload в frame:
window.top.notifyFrameLoaded()
Это опасно, поскольку onload может выполниться в тот момент, когда содержащая его страница выполняет другой скрипт. Однако функция может быть поставлена в очередь:
// хорошая функция onload в frame
window.parent.setTimeout(window.top.notifyFrameLoaded, 0)
Важно использовать метод setTimeout для родительского окна, чтобы занести функцию в очередь событий родительского окна.
- Избегать долго выполняющихся скриптов.
- Избегать синхронных запросов
XMLHttpRequest .
- Не позволять скриптам из разных фреймов управлять одним и тем же глобальным состоянием.
- Не использовать диалоги
alert для отладки, так как они могут совершенно изменить логику программы.
|
Baker was caught intoxicated behind the wheel on a road near his home in the Byron Bay region in the early hours of 20 July.
任你博娛樂城
隆亨娛樂城
oc娛樂城
通博娛樂城
拼多多娛樂城
On the side of a dirt road in Adré, a key crossing on the Sudan-Chad border, 38-year-old Buthaina sits on the ground, surrounded by other women. Each of them has their
任你博娛樂城
隆亨娛樂城
oc娛樂城
拼多多娛樂城
通博娛樂城
How do you exercise and wash your clothes
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
Former President Donald Trump is safe following an apparent assassination attempt at his Florida golf course on Sunday afternoon, and a "potential suspect" is in custody, US authorities have confirmed.任你博娛樂城
隆亨娛樂城
oc娛樂城
通博娛樂城
拼多多娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
太阳城集团|
太阳城集团老板|
澳门太阳城官网|
澳门太阳城赌场|
威尼斯人博彩|
太阳城小说|
太阳城集团|
太阳城集团老板|
澳门太阳城官网|
威尼斯人在线开户
Venetian Online Account Opening
太阳城集团|
太阳城集团老板|
澳门太阳城官网|
A wildlife project is using a new technique to reintroduce shellfish to the East Yorkshire coast.
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
A wildlife project is using a new technique to reintroduce shellfish to the East Yorkshire coast.
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
A wildlife project is using a new technique to reintroduce shellfish to the East Yorkshire coast.
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
There was one other eye-grabbing phrase
EVO娛樂城
AF娛樂城
EVO娛樂城
new queen was announced at a gathering at
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
new queen was announced at a gathering at
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
new queen was announced at a gathering at
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
new queen was announced at a gathering at
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
new queen was announced at a gathering at
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
new queen was announced at a gathering at
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
new queen was announced at a gathering at
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
new queen was announced at a gathering at
T9娛樂城
CHFUGMJTDG RHSREDH
新葡京博彩官网|
澳门申博太阳城AG
As it sank, the Titanic split into two main sections – the bow and the stern, which came to rest nearly 2,000ft (600m) apart on the sea floor.
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
As it sank, the Titanic split into two main sections – the bow and the stern, which came to rest nearly 2,000ft (600m) apart on the sea floor.
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
On the side of a dirt road in Adré, a key crossing on the Sudan-Chad border, 38-year-old Buthaina sits on the ground, surrounded by other women. Each of them has their
任你博娛樂城隆亨娛樂城oc娛樂城通博娛樂城拼多多娛樂城"
On the side of a dirt road in Adré, a key crossing on the Sudan-Chad border, 38-year-old Buthaina sits on the ground, surrounded by other women. Each of them has their
任你博娛樂城隆亨娛樂城oc娛樂城通博娛樂城拼多多娛樂城"
On the side of a dirt road in Adré, a key crossing on the Sudan-Chad border, 38-year-old Buthaina sits on the ground, surrounded by other women. Each of them has their
任你博娛樂城隆亨娛樂城oc娛樂城通博娛樂城拼多多娛樂城"
“We left with nothing, we just ran for our lives,” Buthaina tells the BBC. “We didn’t want to leave - my children were top of their class at school and we had a good life at
任你博娛樂城隆亨娛樂城oc娛樂城通博娛樂城拼多多娛樂城"
“We left with nothing, we just ran for our lives,” Buthaina tells the BBC. “We didn’t want to leave - my children were top of their class at school and we had a good life at
任你博娛樂城隆亨娛樂城oc娛樂城通博娛樂城拼多多娛樂城"
The new queen, meanwhile
EVO娛樂城
AF娛樂城
EVO娛樂城
The Pacific Islands are scattered across a vast area of ocean, with some of the clearest waters in the world, and pristine beaches and rainforests.
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
paradox of a world claiming to be inclusive but remaining prejudicial against people with disabilities.任你博娛樂城隆亨娛樂城oc娛樂城通博娛樂城拼多多娛樂城"
paradox of a world claiming to be inclusive but remaining prejudicial against people with disabilities.任你博娛樂城隆亨娛樂城oc娛樂城通博娛樂城拼多多娛樂城"
It is currently the fifth-largest active wildfire in California, after quadrupling in size on Saturday.
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
India won four medals in Rio in 2016 and 20 at the 2020 Tokyo Paralympics.
bu娛樂城
India won four medals in Rio in 2016 and 20 at the 2020 Tokyo Paralympics.bu娛樂城
The country hadn't won any medal at the 2008 edition in Beijing, so it felt special to millions of Indians.bu娛樂城
au8娛樂城
九牛娛樂城
富樂娛樂城
gc娛樂城
404 not found解決方法
蜘蛛池
seo自學
ahrefs 教學
The country hadn't won any medal at the 2008 edition in Beijing
PANALOKO
SUPERACE88
kinggame
nustabet
PPBet
RichBet
The country hadn't won any medal at the 2008 edition in Beijing.
PANALOKO
SUPERACE88
kinggame
nustabet
PPBet
RichBet
"The music needed space to breathe on its own."
bu娛樂城
au8娛樂城
九牛娛樂城
富樂娛樂城
gc娛樂城
404 not found解決方法
蜘蛛池
seo自學
ahrefs 教學
"The music needed space to breathe on its own."
bu娛樂城
au8娛樂城
九牛娛樂城
富樂娛樂城
gc娛樂城
The music needed space to breathe on its own.
bu娛樂城
au8娛樂城
九牛娛樂城
富樂娛樂城
gc娛樂城
The music needed space to breathe on its own.
bu娛樂城
au8娛樂城
九牛娛樂城
富樂娛樂城
gc娛樂城
The music needed space to breathe on its own.
bu娛樂城
au8娛樂城
九牛娛樂城
富樂娛樂城
gc娛樂城
However, in Tonga locals say they are now seeing stronger storms hit more often.
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
The casting call was opened earlier this week, and the TV network
PANALOKO
SUPERACE88
kinggame
nustabet
PPBet
RichBet
By contrast, protestors said officers were heavy-handed and responded with flashbang grenades and irritant sprays to control the hostile crowd.任你博娛樂城隆亨娛樂城oc娛樂城通博娛樂城拼多多娛樂城"
Hosting the Pope for even a single evening - as the stadium will - is no small feat.
bu娛樂城
au8娛樂城
九牛娛樂城
富樂娛樂城
gc娛樂城
404 not found解決方法
蜘蛛池
seo自學
ahrefs 教學
The Dutch boss is under pressure after United's eighth-placed finish last
PANALOKO
SUPERACE88
kinggame
nustabet
PPBet
RichBet
And Ronaldo, who had two spells at the club before leaving for Saudi ArabianPANALOKO
SUPERACE88
kinggame
nustabet
PPBet
RichBet
is a show that has run for more than
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
is a show that has run for more than
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
is a show that has run for more than
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
is a show that has run for more than
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
The UN agency for Palestinian
任你博娛樂城
隆亨娛樂城
oc娛樂城
通博娛樂城
拼多多娛樂城
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
The UN agency for Palestinian
任你博娛樂城
隆亨娛樂城
oc娛樂城
通博娛樂城
拼多多娛樂城
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
The cost of the caps worn by the King's Guard rose from £1,560 each in 2022 to £2,040 in 2023.
bu娛樂城
au8娛樂城
九牛娛樂城
富樂娛樂城
gc娛樂城
The cost of the caps worn by the King's Guard rose from £1,560 each in 2022 to £2,040 in 2023.bu娛樂城
au8娛樂城
All the insects were dead when we found them
EVO娛樂城
AF娛樂城
EVO娛樂城
By contrast, protestors said officers were heavy-handed and responded with flashbang grenades and irritant sprays to control the hostile crowd.
任你博娛樂城
隆亨娛樂城
oc娛樂城
通博娛樂城
拼多多娛樂城
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
In the hours before the prime minister was taken by motorcade to the White House
In the hours before the prime minister was taken by motorcade to the White House
In the hours before the prime minister was taken by motorcade to the White House
In the hours before the prime minister was taken by motorcade to the White House
I have said that this is just three games.
EVO娛樂城
AF娛樂城
EVO娛樂城
How do you exercise and wash your clothes
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
The screams of delight from the Arsenal analysts at the back of the press box PANALOKO SUPERACE88
kinggame
nustabet
PPBet
RichBet
phswerte
An AK47-style firearm and scope任你博娛樂城
隆亨娛樂城
oc娛樂城
通博娛樂城
拼多多娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
台中婦產科
西屯婦產科
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
Therefore, to have a partial eclipse of a full moon, which happens to be a supermoon, is rare.
通博娛樂城
大老爺娛樂城
拼多多娛樂城
gc娛樂城
17娛樂城
Too hot for humpbacks: The race to protect Pacific whales
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
星城Online
包你發娛樂城
大福Online
豪神娛樂城
Green Society
GanjaExpress
Big Mama's homecoming is not an isolated case.
最新娛樂城
娛樂城推薦
娛樂城排行
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街 Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
Green Society
GanjaExpress
has said he licensed his trade mark
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
Pep Guardiola's side were never at their most fluent, not helped by an injury to
panaloko
panaloko free 99
betso88
betso88 login
gold99 casino
gold99 slot
PPBet
ssbet77
kinggame
PHSwerte
SUPERACE88
pgasia
money88
ezbollah leader Hassan
otsobet
panaloko
okbet88
superace88
kinggame
22bet
Prime Minister Anthony Albanese condemned the violence, saying Australians had a right to protest - but had to do so in a peaceful manner.
鉅城娛樂城
WG娛樂城
大老爺娛樂城
West Coast Cannabis
Herb Approach
Buy My Weed Online
phswerte
nustabet
king game casino login
gold99
betso88
Dozens of people have been arrested after clashing with police at an anti-war demonstration in Melbourne.
鉅城娛樂城
WG娛樂城
大老爺娛樂城
West Coast Cannabis
Herb Approach
Buy My Weed Online
phswerte
nustabet
king game casino login
gold99
betso88
Dozens of people have been arrested after clashing with police at an anti-war demonstration in Melbourne.
鉅城娛樂城
WG娛樂城
大老爺娛樂城
West Coast Cannabis
Herb Approach
Buy My Weed Online
phswerte
nustabet
king game casino login
gold99
betso88
The foreign secretary has said climate change is a more pervasive and fundamental threat than terrorism.
最新娛樂城
娛樂城推薦
娛樂城排行
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街 Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
Green Society
GanjaExpress
In Telangana, pressure has grown on the government
EVO娛樂城
AF娛樂城
EVO娛樂城
Russian air strikes
TU娛樂城
AU8娛樂城
AU8娛樂城
The Los Angeles Dodgers superstar stole third base in the first inning against
PANALOKO
panaloko free 99
betso88
betso88 login
gold99 casino
gold99 slot
PPBet
ssbet77
kinggame
PHSwerte
SUPERACE88
pgasia
money88
That wave was then “trapped” in the narrow fjord - moving back and forth for nine days, generating the vibrations.
最新娛樂城
娛樂城推薦
娛樂城排行
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
Green Society
GanjaExpress
Matteo Gabbia scored a dramatic late winner as AC Milan beat fierce rivals PANALOKO
panaloko free 99
betso88
betso88 login
gold99 casino
gold99 slot
PPBet
ssbet77
kinggame
PHSwerte
SUPERACE88
pgasia
money88
The flavour is unusual but palatable
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
Sri Lanka's capital is transforming floating garbage patches into biodiverse wetlands which are teeming with life.
最新娛樂城
娛樂城推薦
娛樂城排行
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
Green Society
GanjaExpress
Manchester City's Kyle Walker and John Stones accused Arsenal of
panaloko free 99
betso88
betso88 login
gold99 casino
gold99 slot
PPBet
ssbet77
kinggame
PHSwerte
SUPERACE88
pgasia
money88
which still seems to surprise Raisman to this day
EVO娛樂城
AF娛樂城
EVO娛樂城
This huge, smelly, otherworldly plant has been capturing people's imaginations and inspiring awe for centuries.
最新娛樂城
娛樂城推薦
娛樂城排行
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
Green Society
GanjaExpress
Christopher Nkunku scored a hat-trick as Chelsea cruised into the fourth PANALOKO
panaloko free 99
betso88
betso88 login
gold99 casino
gold99 slot
PPBet
ssbet77
kinggame
PHSwerte
SUPERACE88
pgasia
money88
This huge, smelly, otherworldly plant has been capturing people's imaginations and inspiring awe for centuries.
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
BC Bud Supply
Green Society
he engineered an economical, filling
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
he engineered an economical, filling
T9娛樂城
EVO娛樂城
TU娛樂城
AT99娛樂城
BU娛樂城
EVO娛樂城
搖錢樹娛樂城
The technology giant's chief executive
EVO娛樂城
AF娛樂城
EVO娛樂城
destroy what it said was infrastructure built up by Hezbollah since they last fought
okbet88
superace88
kinggame
22bet
otsobet
panaloko
pwbet
ezwin
destroy what it said was infrastructure built up by Hezbollah since they last fought
okbet88
superace88
kinggame
22bet
otsobet
panaloko
pwbet
ezwin
The game is famous for its simple yet challenging gameplay, and its quick success has made Flappy Bird a global phenomenon.
The tides turned for Salish Sea whales in 1997 when a single humpback showed up in British Columbia's waters.
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
BC Bud Supply
Green Society
https://tifftw.com/tiffany-knot tiffany knot哪裡買最便宜?
https://tifftw.com/tiffany-knot tiffany knot ring
https://tifftw.com/tiffany-knot tiffany knot 價格
https://tifftw.com/tiffany-knot tiffany knot necklace
https://tifftw.com/tiffany-knot tiffany knot項鍊價格
https://tifftw.com/tiffany-knot tiffany knot 戒指
https://tifftw.com/tiffany-knot tiffany knot項鍊價錢
https://qiuxie.tw/?s=samba&post_type=product adidas samba
https://qiuxie.tw/?s=nike+air+force+1&post_type=product nike air force 1
https://qiuxie.tw/?s=nike+dunk&post_type=product nike dunk
https://qiuxie.tw/?s=samba&post_type=product samba
https://qiuxie.tw/?s=originals&post_type=product adidas originals
https://qiuxie.tw/product-category/yeezy/ yeezy
https://qiuxie.tw/ 愛 迪 達 鞋子
https://qiuxie.tw/?s=sacai&post_type=product nike sacai
https://qiuxie.tw/ nike 鞋子
https://qiuxie.tw/?s=vomero+5&post_type=product nike vomero 5
https://qiuxie.tw/?s=sacai&post_type=product nike x sacai
https://qiuxie.tw/product-category/%e6%8b%96%e9%9e%8b/ nike 拖鞋
https://qiuxie.tw/product-category/%e7%b1%83%e7%90%83%e9%9e%8b/ nike sneakers
https://qiuxie.tw/?s=vomero+5&post_type=product nike zoom vomero 5
https://qiuxie.tw/product/stussy-x-nike-benassi-slides-%e6%b2%99%e7%81%9... yeezy slides
https://qiuxie.tw/product-category/yeezy/yeezy-350/ 350
https://qiuxie.tw/product-category/yeezy/yeezy-%e6%8b%96%e9%9e%8b/ adidas 拖鞋
https://qiuxie.tw/?s=nike+air+max&post_type=product nike air max
https://qiuxie.tw/?s=nike+dunk+low&post_type=product nike dunk low
https://qiuxie.tw/product-category/%e7%b1%83%e7%90%83%e9%9e%8b/ nike 籃球 鞋
Daniel Ricciardo has been replaced at RB by Liam Lawson for the remainder
PANALOKO
panaloko free 99
betso88
betso88 login
gold99 casino
gold99 slot
PPBet
ssbet77
kinggame
PHSwerte
SUPERACE88
pgasia
money88
Baker was caught intoxicated behind the wheel on a road near his home in the Byron Bay region in the early hours of 20 July.
最新娛樂城
娛樂城推薦
娛樂城排行
鉅城娛樂城
WG娛樂城
大老爺娛樂城
kg娛樂城
3a娛樂城
包你發娛樂城
小狗Online
滿貫大亨
豪神娛樂城
phswerte
nustabet
king game casino login
gold99
betso88
In JavaScript, the order of events and synchronization are crucial for developing responsive applications. Understanding how asynchronous operations work, such as callbacks, promises, and async/await, is vital. Just like navigating conversations on platforms like Omegle, where timing and responsiveness can greatly impact interaction quality, managing these JavaScript concepts ensures a smooth and efficient user experience.
Understanding event order and synchronization in JavaScript is crucial for developing efficient applications. Just like chatting on Omegle, where timing and connection matter, JavaScript's asynchronous nature requires careful handling of events and callbacks to ensure a smooth user experience. By mastering these concepts, developers can create more responsive and interactive web applications that engage users effectively.
Understanding event order and synchronization in JavaScript is crucial for developing efficient applications. Just like chatting on Omegle, where timing and connection matter, JavaScript's asynchronous nature requires careful handling of events and callbacks to ensure a smooth user experience. By mastering these concepts, developers can create more responsive and interactive web applications that engage users effectively.
The 2048 Taylor Swift game is a fun twist on the classic 2048 puzzle game, featuring Taylor Swift-themed tiles. It's a simple yet addictive game for Swifties to enjoy while combining their love for her music with an engaging challenge.
The behaviour of the world's most powerful storms is evolving.
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
BC Bud Supply
Green Society
There was one in 2015 after the World Cup debacle, another under Brendon PANALOKO
panaloko free 99
betso88
betso88 login
gold99 casino
gold99 slot
PPBet
ssbet77
kinggame
PHSwerte
SUPERACE88
pgasia
money88
The UK is about to stop producing any electricity from burning coal - ending its 142-year reliance on the fossil fuel.
大老爺娛樂城
RG富遊娛樂城
kg娛樂城
鉅城娛樂城
九州娛樂城
bcr娛樂城
娛樂城
星城Online
金爸爸娛樂城
老子有錢
The foreign secretary has said climate change is a more pervasive and fundamental threat than terrorism.
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
BC Bud Supply
Green Society
The death toll from Hurricane Helene has risen to 200 as rescuers continue to search for survivors from the storm that tore across the US south-east.
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
BC Bud Supply
Green Society
Dozens of tigers have died in zoos in south Vietnam after a bird flu outbreak,
panaloko free 99
betso88
gold99
richbet
ppbet
ssbet77
kinggame
PHSwerte
money88
pgasia
superace88
superace88 club login
Marine turtles spend almost their entire lives at sea – but little is known about the paths they take.
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
BC Bud Supply
Green Society
A huge solar flare, the largest since 2017, has been spotted erupting from the Sun’spanaloko free 99
betso88
gold99
richbet
ppbet
ssbet77
kinggame
PHSwerte
money88
pgasia
superace88
superace88 club login
he engineered an economical, filling
T9娛樂城
搖錢樹娛樂城
AT99娛樂城
AT99娛樂城
EVO娛樂城
EVO娛樂城
EVO娛樂城
娛樂城推薦
When Taylor Swift is making $2bn in ticket sales, and Coldplay can sell out 10 nights
panaloko free 99
betso88
gold99
richbet
ppbet
ssbet77
kinggame
PHSwerte
money88
pgasia
superace88
superace88 club login
In the 1970s, a small group of Greenpeace activists had a unique idea for how they could put an end to commercial whaling.
星城Online
包你發娛樂城
大福Online
豪神娛樂城
錢街Online
老子有錢
滿貫大亨
拉斯維加斯
鑽很大Online
娛樂城優惠
BC Bud Supply
Green Society
Kayleigh van Dooren’s clinical double for Dutch champions Twente ensured panaloko free 99
betso88
gold99
richbet
ppbet
ssbet77
kinggame
PHSwerte
money88
pgasia
superace88
nustabet
Kayleigh van Dooren’s clinical double for Dutch champions Twente ensured panaloko free 99
betso88
gold99
richbet
ppbet
ssbet77
kinggame
PHSwerte
money88
pgasia
superace88
nustabet
the international effort includes more
AT99娛樂城
AT99娛樂城
AT99娛樂城
the international effort includes more
AT99娛樂城
AT99娛樂城
AT99娛樂城
Fireboy and Watergirl is more than just a game; it's a captivating adventure that fosters teamwork and critical thinking. With its unique gameplay mechanics, engaging levels, and accessibility, it has become a beloved title among gamers of all ages.
Отправить комментарий
Приветствуются комментарии:Для остальных вопросов и обсуждений есть форум.