Статьи,обзоры,справочники по PHP

       

Ошибка программы простого текстового счетчика


Давайте сделаем такую программу. Итак, у нас есть какая-то страница, на которой хочется повесить счетчик. Обудим алгоритм:

  • считать число из файла
  • записать увеличенное число обратно
  • вывести его на экран

Согласитесь, программа простая, но может привести к ошибке, что и показано ниже.

Если вызывать данную программу очень часто, значение счетчика иногда будет обнуляться. Это произойдет из-за того, что в некоторый момент программа прочитает из файла пустое значение, к которому потом прибавляется единица ("пусто" + число 1 = число 1). Собственно, это и есть сброс счетчика.

Рассмотрим подробно, когда это произойдет. Представьте, что в один момент времени стартовали 2 копии данного скрипта. Одновременно ничего нигде не проиходит, в т.ч. и запуск скриптов, но время между запуском может быть очень маленькое. Процессор выполняет скрипты с разной скоростью, т.е. вы не должны удивляться тому, в каком порядке далее будут рассматриваться команды. Итак, ход программы (на примере "скрипта N1" и "скрипта N2"):

скрипткомандакомментарий (что сделает данная команда)
1 запуск первого скрипта--
1$counter=file("counter.txt");в переменной (массиве $counter) теперь храниться текущее число счетчика. Допустим, там было 1234, тогда это число будет в переменной $counter[0].
2запуск второго скрипта--
1$f=fopen("counter.txt","w+");
  • открывает файл
  • обнуляет его
  • если файл не был создан, создает его (если позволят права). Но файл создан нами заранее, этот вариант исключен.
  • 2$counter=file("counter.txt");читает содержимое пустого файла и записывает в массив $counter пустой массив. Переменная $counter[0] не существует.
    1fputs($f,$counter[0]+1);пишет в файл число 1234 (т.к. в $counter[0] лежит число 1234)
    2$f=fopen("counter.txt","w+");см. комментарий выше
    1fclose($f); и конец работы--
    2fputs($f,$counter[0]+1);записывает в файл число 1, т.к результат сложения несуществующей переменной и числа 1 равен числу 1
    2fclose($f); и конец работы--
    <

    Содержание раздела