PHP中的下载文件

2023-05-24 08:05:45 php 文件 下载

开发网站或者应用程序时,我们经常需要实现文件下载功能,通过PHP可以方便地实现该功能。本文将介绍如何使用php来实现文件下载功能。

一、文件下载流程

在下载文件前,我们需要了解一下下载的流程:

  1. 用户在网页或者应用程序上点击下载按钮;
  2. 服务器接受下载请求;
  3. 服务器检查用户权限和文件存在性;
  4. 服务器将文件发送给用户;
  5. 用户接收文件并保存。

二、准备下载文件

在PHP中,我们可以使用readfile函数来读取文件内容并输出到浏览器。但是,我们需要确保文件存在,否则将会抛出异常。下面是一个简单的代码示例:

$file_path = '/path/to/your/file/yourfile.extension';
if (file_exists($file_path))
{
    header('Content-Type: application/octet-stream');
    header('Content-Disposition: attachment; filename=' . basename($file_path));
    header('Content-Transfer-Encoding: binary');
    header('Content-Length: ' . filesize($file_path));
    readfile($file_path);
    exit;
}
else
{
    die('File not found.');
}

以上代码中,我们首先使用file_exists函数检查文件是否存在,如果文件存在,则设置Http头文件来告诉浏览器将要下载一个文件,并在Content-Disposition中指定文件名,Content-Type指定文件类型,并使用readfile函数来输出文件内容。

三、防止文件下载被盗链

由于浏览器的本质,可能会导致某些不良站点盗用您的文件,因此我们需要防止文件下载被盗链。

我们可以添加以下代码来防止文件下载被盗链:

$referer = $_SERVER['HTTP_REFERER'];
if ($referer && !preg_match('/^https?://' . $_SERVER['SERVER_NAME'] . '/', $referer))
{
    header("HTTP/1.1 403 Forbidden");
    die("Access denied.");
}

该代码将检查HTTP_REFERER头信息并确保其与当前服务器的域名匹配,如果不匹配,则返回403状态码,禁止访问。

四、实现分段下载

在下载较大的文件时,可能需要实现分段下载功能,以加快下载速度并减少网络带宽的占用。借助于HTTP 1.1的Range头信息,我们可以轻松地实现分段下载。

以下是一个示例代码:

$file_path = '/path/to/your/file/yourfile.extension';
if (file_exists($file_path))
{
    $size = filesize($file_path);
    $start = 0;
    $end = $size - 1;
    
    if (isset($_SERVER['HTTP_RANGE']))
    {
        if (preg_match('/bytes=h*(d+)-(d*)[D.*]?/i', $_SERVER['HTTP_RANGE'], $matches))
        {
            $start = intval($matches[1]);
            if (!empty($matches[2]))
            {
                $end = intval($matches[2]);
            }
        }
    }
    if ($start > $end || $end > $size - 1 || $start < 0)
    {
        header('HTTP/1.1 416 Requested Range Not Satisfiable');
        header("Content-Range: bytes $start-$end/$size");
        exit;
    }
    header('Content-Type: application/octet-stream');
    header('Content-Disposition: attachment; filename=' . basename($file_path));
    header('Content-Transfer-Encoding: binary');
    header("Content-Range: bytes $start-$end/$size");
    header('Accept-Ranges: bytes');
    header('Content-Length: ' . ($end - $start + 1));
    header("Cache-control: private");
    header('Pragma: private');
    header('Expires: ' . gmdate('D, d M Y H:i:s T', time() + 3600));
    
    $fp = fopen($file_path, 'rb');
    fseek($fp, $start);
    $buffer_size = 1024 * 8; //每次读取8 KB
    $bytes_send = 0;
    while (!feof($fp) && ($bytes_send < $end - $start + 1))
    {
        $buffer = fread($fp, $buffer_size);
        echo $buffer;
        flush();
        $bytes_send += strlen($buffer);
    }
    fclose($fp);
    exit;
}
else
{
    die('File not found.');
}

以上代码中,我们首先检查HTTP_RANGE头信息,如果该头信息不存在,则输出整个文件。如果该头信息存在,则解析出起始偏移量和结束偏移量,然后使用fseek函数定位文件指针,使用fread函数读取文件内容并输出到浏览器。

五、结论

在PHP中实现文件下载功能不难,我们只需要检查文件是否存在并设置正确的HTTP头信息即可。使用HTTP 1.1 Range头信息可以轻松地实现分段下载,防止文件下载被盗链可以保护文件的安全。因此,我们可以根据需求来选择实现所需的功能,以达到更好的用户体验。

以上就是PHP中的下载文件的详细内容,更多请关注其它相关文章!

相关文章