PHP实现通过CURL上传文件功能示例

转载https://www.jb51.net/article/141156.htm

本文实例讲述了PHP实现通过CURL上传文件功能。分享给大家供大家参考,具体如下:

PHP使用CURL上传文件只需发送一个POST请求就可以了,在请求中设置某个字段为需要上传的文件全路径,并且以"@"开头,然后使用CURL把该变量以POST方式发送到服务器,在服务端即可以从超级全局变量$_FILES中取到相应的上传文件信息。

下面我们以一个例子来展示这个过程。

假设本地有一个文本文件log.txt,其路径为"/www/test/log.txt",内容如下:

1

2

this is a file for test

hello PythonTab!

为了把这个文件上传给服务端的脚本http://yourwebname.com/upload.php,我们在本地写了一个名为curl_file.php的脚本,内容如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

<?php

$url = "http://yourwebname.com/upload.php";

$post_data = array(

"foo" => "bar",

//要上传的本地文件地址

"upload" = > "@/www/test/log.txt"

);

$ch = curl_init();

curl_setopt($ch , CURLOPT_URL , $url);

curl_setopt($ch , CURLOPT_RETURNTRANSFER, 1);

curl_setopt($ch , CURLOPT_POST, 1);

curl_setopt($ch , CURLOPT_POSTFIELDS, $post_data);

$output = curl_exec($ch);

curl_close($ch);

echo $output;

?>

curl_file.php的逻辑很简单,设置POST变量$post_data,其中upload指向需要发送的文件。这里要注意,我们之前使用POST都是发送一个字符串,然后在服务器端使用file_get_contents("php//input")来获取该字符串,和这里的用法不一样,其实POST也可以像GET一样,发送键值对。在服务端有一个超级全局变量$_POST可以像$_GET一样,获取相应的POST数据的值。需要注意的是,上传文件的变量不是存在着$_POST中,而是在$_FILES中。

为了展示服务端收到上述代码的文件上传请求的逻辑,我们在upload.php中写了以下代码:

1

2

3

4

5

<?php

echo var_export($_FILES,true);

echo file_get_contents($_FILES['upload']['tmp_name']);

copy($_FILES['upload']['tmp_name'], "./log_copy.txt");

?>

upload.php首先使用var_export$_FILES变量输出到标准输出,然后使用file_get_contents读取$_FILES['upload']['tmp_name']所指文件的内容,并输出到标准输出,然后把$_FILES['upload']['tmp_name']所指文件自制到当前目录的log_copy.txt文件中。该脚本的标准输出如下:

array(
'upload' =>
array(
'name' => 'log.txt',
'type' => 'application/octet-stream',
'tmp_name' => '/tmp/phpLUB59F',
'error' => 0,
'size' => 36,
)
)
this is a file for test
hello PythonTab!

可以看到$_FILES变量中有一个upload数组,对应到上传文件描述信息,其中name和type分别表示名称和类型。tmp_name比较关键,服务端在接收到上传文件之后,会把文件写在一个临时文件中,这个临时文件的名字就是tmp_name的值,这也是为什么我们读取该文件可以获取一log.txt的文件内容。一般在服务端接收到上传文件后都需要立即读取该文件或者把文件复制到别外一个文件中,因为tmp_name所指的临时文件在服务端脚本执行完毕后会被删除掉,upload.php脚本的最后一行就是把临时文件复制到我们的目标文件中。

案例
$url = "https://api.weixin.qq.com/cgi-bin/media/uploadimg?access_token=".getAccessToken();
        $post_data = array(
            'media' => "@".$result['url'],
        );
        $response = curl_http($url, 'POST', $post_data);

/**
 * http请求方式: 默认GET
 */
function curl_http($url, $method="GET", $postfields){
    $ch = curl_init();
    curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE);
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
    curl_setopt($ch, CURLOPT_URL, $url);
    switch ($method) {
        case "POST":
            curl_setopt($ch, CURLOPT_POST, true);
            if (!empty($postfields)) {
                $hadFile = false;
                if (is_array($postfields) && isset($postfields['media'])) {
                    /* 支持文件上传 */
                    if (class_exists('\CURLFile')) {
                        curl_setopt($ch, CURLOPT_SAFE_UPLOAD, true);
                        foreach ($postfields as $key => $value) {
                            if (isPostHasFile($value)) {
                                $postfields[$key] = new \CURLFile(realpath(ltrim($value, '@')));
                                $hadFile = true;
                            }
                        }
                    } elseif (defined('CURLOPT_SAFE_UPLOAD')) {
                        if (isPostHasFile($value)) {
                            curl_setopt($ch, CURLOPT_SAFE_UPLOAD, false);
                            $hadFile = true;
                        }
                    }
                }
                $tmpdatastr = (!$hadFile && is_array($postfields)) ? http_build_query($postfields) : $postfields;
                curl_setopt($ch, CURLOPT_POSTFIELDS, $tmpdatastr);
            }
            break;
        default:
            curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $method); /* //设置请求方式 */
            break;
    }
    $ssl = preg_match('/^https:\/\//i',$url) ? TRUE : FALSE;
    curl_setopt($ch, CURLOPT_URL, $url);
    if($ssl){
        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE); // https请求 不验证证书和hosts
        curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE); // 不从证书中检查SSL加密算法是否存在
    }
    $response = curl_exec($ch);
    curl_close($ch);
    if(empty($response)){
        exit("错误请求");
    }
    return $response;
}
//判断文件是否存在
function isPostHasFile($value)
{
    if (is_string($value) && strpos($value, '@') === 0 && is_file(realpath(ltrim($value, '@')))) {
        return true;
    }
    return false;
}

 

©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页