Самый эффективный способ отображения 1x1 GIF (пиксель отслеживания, веб-маяк)

Я создаю базовый аналитический сервис, теоретически основанный на том, как работает Google Analytics, но вместо запроса фактического изображения я направляю запрос изображения на script, который принимает данные, а затем выводит образ. Поскольку браузеры будут запрашивать это изображение при каждой загрузке, каждая миллисекунда считается.

Я ищу наиболее эффективный способ для файла выводить gif файл из PHP script. До сих пор я создал 3 основных метода.

Есть ли более эффективный способ для вывода 1x1 GIF файла из PHP script? Если нет, то какой из них является наиболее эффективным и масштабируемым?

Три идентифицированных метода

Библиотеки для создания образов PHP

$im = imagecreatetruecolor(1, 1);
imagefilledrectangle($im, 0, 0, 0, 0, 0xFb6b6F);
header('Content-Type: image/gif');
imagegif($im);
imagedestroy($im);

file_get_contents изображение с сервера и вывести его

$im = file_get_contents('raw.gif'); 
header('Content-Type: image/gif'); 
echo $im; 

base64_decode изображение

header('Content-Type: image/gif');
echo base64_decode("R0lGODdhAQABAIAAAPxqbAAAACwAAAAAAQABAAACAkQBADs=");

(У меня было то, что base64 был бы самым быстрым, но я понятия не имею, насколько ресурсоемкой является эта функция, и что file_get_contents, вероятно, будут масштабироваться слабее, поскольку он добавляет еще одно действие файловой системы.)

Для справки, GIF, который я использую, приведен здесь: http://i.stack.imgur.com/LQ1CR.gif

ИЗМЕНИТЬ

Итак, причина, по которой я обслуживаю этот образ, заключается в том, что моя библиотека аналитики строит строку запроса и присоединяет ее к этому запросу изображения. Вместо того, чтобы анализировать журналы, я направляю запрос на PHP script, который обрабатывает данные и отвечает изображением, так что браузер конечного пользователя не зависает или не выдает ошибку. Мой вопрос: как лучше всего служить этому изображению в пределах script?

Ответ 1

возможно

header('Content-Type: image/gif');
//equivalent to readfile('pixel.gif')
echo "\x47\x49\x46\x38\x37\x61\x1\x0\x1\x0\x80\x0\x0\xfc\x6a\x6c\x0\x0\x0\x2c\x0\x0\x0\x0\x1\x0\x1\x0\x0\x2\x2\x44\x1\x0\x3b";

Это выведет двоичную строку, идентичную двоичному файлу, содержащему прозрачный gif 1x1. Я утверждаю, что это эффективно, основываясь на том, что он не делает медленного ввода-вывода, такого как чтение файла, и я не вызываю никаких функций.

Если вы хотите создать свою собственную версию вышеупомянутой шестнадцатеричной строки, возможно, чтобы вы могли изменить цвет, вы можете использовать ее для генерации php-кода для оператора echo.

printf('echo "%s";', preg_replace_callback('/./s', function ($matches) {
    return '\x' . dechex(ord($matches[0]));
}, file_get_contents('https://upload.wikimedia.org/wikipedia/en/d/d0/Clear.gif')));

Ответ 2

   header('Content-Type: image/gif'); 
   header("Content-Length: " . filesize("image.gif"));
   $f = fopen('image.gif', 'rb');
   fpassthru($f);
   fclose($f);

Вероятно, это было бы самым быстрым для изображения с диска, но (особенно, если вы используете кэширование байтовых кодов) для небольших изображений, известных заранее, путь base64 будет самым быстрым, я думаю. Отправка Content-Length может быть хорошей идеей, так как для небольшого изображения браузер в большинстве случаев не будет ждать ничего после получения байтов, так что, пока ваш сервер займет столько же времени, опыт использования будет заметно лучше.

Еще один способ - позволить Apache/lighttpd/nginx обслуживать изображение, регистрировать доступ и анализировать его в автономном режиме.

Ответ 3

Вместо динамического создания/вывода изображения, почему бы просто не перенаправить на статическое изображение?

<?php
// process query param stuff

header('Location: pixel.gif');
exit();
?>

Ответ 4

С Laravel:

$pixel = "\x47\x49\x46\x38\x39\x61\x1\x0\x1\x0\x80\x0\x0\xff\xff\xff\x0\x0\x0\x21\xf9\x4\x1\x0\x0\x0\x0\x2c\x0\x0\x0\x0\x1\x0\x1\x0\x0\x2\x2\x44\x1\x0\x3b";
return response($pixel,200,[
    'Content-Type' => 'image/gif',
    'Content-Length' => strlen($pixel),
]);

Если кто-то хочет это по какой-то причине.

В качестве альтернативы, если вам не нравятся длинные (ish) шестнадцатеричные строки в вашем коде:

base64_decode('R0lGODlhAQABAIAAAP///wAAACH5BAEAAAAALAAAAAABAAEAAAICRAEAOw')