file_get_contents () с контекстом для использования http / 1.1 значительно медленные скорости загрузки
Вопрос
Используя код под каждым загрузкой изображения) File_Get_Contents ()) в среднем 8-15 секунд .....
Если я не использую контекст на file_get_contents (), то загрузка изображения меньше секунды.
Если я изменю $ OPTS, ниже, то я получаю то же производительность, что и file_get_contents () без контекста, который принимает Appox 13 секунд для обработки 2500 изображений.
$opts = array(
'http'=>array(
'protocol_version'=>'1.1',
'method'=>'GET',
'header'=>array(
'Connection: close'
),
'user_agent'=>'Image Resizer'
)
);
Воспроизводить:
$start_time = mktime();
$products = array(
array( 'code'=>'A123', 'image_url'=>'http://www.google.com/intl/en_ALL/images/srpr/logo1w.png' ),
array( 'code'=>'A124', 'image_url'=>'http://www.google.com/intl/en_ALL/images/srpr/logo1w.png' ),
array( 'code'=>'A125', 'image_url'=>'http://www.google.com/intl/en_ALL/images/srpr/logo1w.png' ),
array( 'code'=>'A126', 'image_url'=>'http://www.google.com/intl/en_ALL/images/srpr/logo1w.png' ),
array( 'code'=>'A127', 'image_url'=>'http://www.google.com/intl/en_ALL/images/srpr/logo1w.png' ),
array( 'code'=>'A128', 'image_url'=>'http://www.google.com/intl/en_ALL/images/srpr/logo1w.png' ),
array( 'code'=>'A134', 'image_url'=>'http://www.google.com/intl/en_ALL/images/srpr/logo1w.png' ),
array( 'code'=>'A135', 'image_url'=>'http://www.google.com/intl/en_ALL/images/srpr/logo1w.png' ),
array( 'code'=>'A146', 'image_url'=>'http://www.google.com/intl/en_ALL/images/srpr/logo1w.png' ),
array( 'code'=>'A165', 'image_url'=>'http://www.google.com/intl/en_ALL/images/srpr/logo1w.png' )
);
if ( count( $products ) > 0 ) {
$opts = array(
'http'=>array(
'protocol_version'=>'1.1',
'method'=>'GET',
'user_agent'=>'Image Resizer'
)
);
$context = stream_context_create($opts);
$def_width = 100;
$max_width = $def_width;
foreach( $products as $product ) {
$code = $product['code'];
$folder = substr( $code, 0, 3 );
echo( 'Looking at: ' .$product['code'] ."<br />" );
$file = '/tmp/' .$folder .'/' .$code .'_' .$def_width .'.jpg';
$filemtime = @filemtime($file);
$gen_file = true;
if ( $filemtime !== false ) {
$file_age = (time() - $filemtime);
if ( $file_age <= 300 ) {
$gen_file = false;
}
}
echo( ' File not cached or cached file has expired<br />' );
if ( $gen_file ) {
echo( ' Getting File...');
$imgStr = file_get_contents( $product['image_url'], false, $context );
$img = @imagecreatefromstring($imgStr);
if ( is_resource( $img ) ) {
echo( 'DONE' .'<br />' );
$image_x = imagesx($img);
$image_y = imagesy($img);
if ( $def_width >= $image_x ) {
$def_width = $image_x;
}
echo( ' Calculating Scale<br />' );
$ts = min( $max_width/$image_x,$max_width/$image_y);
$thumbhght = $ts * $image_y;
$thumbwth = $ts * $image_x;
$thumb_image_resized = imagecreatetruecolor( $thumbwth, $thumbhght);
imagecopyresampled($thumb_image_resized, $img, 0, 0, 0, 0, $thumbwth, $thumbhght, $image_x, $image_y );
echo( ' Checking For Directory<br />' );
if ( !is_dir( '/tmp/' .$folder ) ) {
mkdir( '/tmp/' .$folder );
}
echo( ' Writing File<br />' );
$new_file = '/tmp/' .$folder .'/' .$code .'_' .$def_width .'.jpg';
imagejpeg( $thumb_image_resized, $new_file, 100);
echo( ' DONE<br />' );
imagedestroy($img);
imagedestroy($thumb_image_resized);
} else {
echo( 'Problem Getting Image<br />' );
die();
}
} else {
echo( ' Already Exists<br />' );
}
}
}
$end_time = mktime();
echo( 'Completed In...' .($end_time - $start_time ) .' seconds(s)<br />' );
Решение
Запросы HTTP 1.1 по умолчанию являются трубопроводными. Если вы не «подключаетесь: закрыть», он предполагает «соединение: сохранить живую», а затем вы должны ждать подключения к времени (так как вы никогда не его явно закрыли) до начала следующего цикла.
Другие советы
Ваш контекст рассказывает File_Get_Contents (), чтобы закрыть HTTP-соединение каждый раз. Возможно, именно поэтому код занимает так много времени, так как он закрывается и переосветвляет соединения много раз? Я не знаком с внутренними пользователями File_Get_Contents (), но вы сможете настроить контекст для использования «Подключение: сохранить» для всех, кроме последнего подключения и «подключение: закрытие» для последнего.