这是我的代码:
$language = $_GET['soundtype'];
$word = $_GET['sound'];
$word = urlencode($word);
if ($language == 'english') {
$url = "<the first url>";
} else if ($language == 'chinese') {
$url = "<the second url>";
}
$opts = array(
'http'=>array(
'method'=>"GET",
'header'=>"User-Agent: <my user agent>"
)
);
$context = stream_context_create($opts);
$page = file_get_contents($url, false, $context);
header('Content-Type: audio/mpeg');
echo $page;
但我发现这种速度非常慢。
有没有可能的优化方法?
注意: $url
是一个远程网址。
答案 0 :(得分:9)
它很慢,因为file_get_contents()
将整个文件读入$page
,PHP在输出内容之前等待接收文件。所以你要做的是:在服务器端下载整个文件,然后将其作为一个巨大的字符串输出。
file_get_contents()
不支持流式传输或抓取远程文件的偏移量。一个选项是创建一个带有fsockopen()
的原始套接字,执行HTTP请求,并在循环中读取响应,当您读取每个块时,将其输出到浏览器。这将更快,因为文件将被流式传输。
手册中的示例:
$fp = fsockopen("www.example.com", 80, $errno, $errstr, 30);
if (!$fp) {
echo "$errstr ($errno)<br />\n";
} else {
header('Content-Type: audio/mpeg');
$out = "GET / HTTP/1.1\r\n";
$out .= "Host: www.example.com\r\n";
$out .= "Connection: Close\r\n\r\n";
fwrite($fp, $out);
while (!feof($fp)) {
echo fgets($fp, 128);
}
fclose($fp);
}
上面是循环,而仍有内容可用,在每次迭代时它读取128个字节,然后将其输出到浏览器。同样的原则也适用于您正在做的事情。您需要确保不输出响应HTTP标头,这将是前几行,因为您正在执行原始请求,您将获得包含标头的原始响应。如果输出响应头,则最终会出现损坏的文件。
答案 1 :(得分:2)
正如@MrCode所解释的那样,首先将文件下载到服务器,然后将其传递给客户端,当然会导致下载时间加倍。如果要将文件直接传递到客户端,请使用readfile
。
或者,考虑一下您是否不能使用header("Location: $url")
将客户端重定向到文件URL,以便客户端可以直接从源文件获取文件。
答案 2 :(得分:2)
不要在输出之前下载整个文件,而应考虑将其直接输出:
$in = fopen($url, 'rb', false, $context);
$out = fopen('php://output', 'wb');
header('Content-Type: video/mpeg');
stream_copy_to_stream($in, $out);
如果你大胆,你甚至可以尝试(但这绝对是实验性的):
header('Content-Type: video/mpeg');
copy($url, 'php://output');
另一种选择是使用内部重定向并让您的Web服务器代理您的请求。这将释放PHP以做其他事情。另请参阅my post regarding X-Sendfile and friends。