Вот сценарий:
Моим пользователям представлена сетка, в основном упрощенная версия электронной таблицы. В каждой строке сетки есть текстовые поля. Когда они изменяют значение в текстовом поле, я выполняю проверку их ввода, обновляю коллекцию, которая управляет сеткой, и перерисовываю промежуточные итоги на странице. Все это обрабатывается событием OnChange каждого текстового поля.
Когда они нажимают кнопку «Сохранить», я использую событие OnClick кнопки, чтобы выполнить окончательную проверку сумм, а затем отправляю весь свой вклад в веб-службу, сохраняя ее.
По крайней мере, так будет, если они перейдут через форму к кнопке «Отправить».
Проблема в том, что если они вводят значение, а затем сразу же нажимают кнопку сохранения, SaveForm () начинает выполняться до завершения UserInputChanged () - условие гонки. Мой код не использует setTimeout, но я использую его для имитации медленного кода проверки UserInputChanged:
<!-- snip -->
<script>
var amount = null;
var currentControl = null;
function UserInputChanged(control) {
currentControl = control;
// use setTimeout to simulate slow validation code (production code does not use setTimeout)
setTimeout("ValidateAmount()", 100);
}
function SaveForm() {
// call web service to save value
document.getElementById("SavedAmount").innerHTML = amount;
}
function ValidateAmount() {
// various validationey functions here
amount = currentControl.value; // save value to collection
document.getElementById("Subtotal").innerHTML = amount; // update subtotals
}
</script>
<!-- snip -->
Amount: <input type="text" id="UserInputValue" onchange="UserInputChanged(this);" /> <br />
Subtotal: <span id="Subtotal"></span> <br />
<input type="button" onclick="SaveForm();" value="Save" /> <br /><br />
Saved amount: <span id="SavedAmount"></span>
<!-- snip -->
Не думаю, что смогу ускорить код проверки - он довольно легкий, но, по-видимому, достаточно медленный, чтобы код пытался вызвать веб-сервис до завершения проверки.
На моей машине ~ 95 мс - это магическое число между тем, выполняется ли код проверки перед началом сохранения. Это может быть выше или ниже в зависимости от скорости компьютера пользователя.
У кого-нибудь есть идеи, как справиться с этим условием? Коллега предложил использовать семафор во время выполнения кода проверки и цикл занятости в коде сохранения, чтобы дождаться разблокировки семафора, но я бы хотел избежать использования любого типа цикла занятости в моем коде.