PHP文件上传详解

简介:

上传文件分为两个部分,HTML显示部分和PHP处理部分,HTML部分主要是让用户来选择所要上传的文件,然后通过PHP中的$_FILES,我们可以把文件上传到服务器的指定目录。

先来看一下HTML部分。

< form  action = "upload.php"  method = "post"  enctype = "multipart/form-data" >
上传:< input  type = "file"  name = "myfile"  />
< input  type = "submit"  name = "submit"  value = "上传"  />
</ form >

说明:
form标答的action="upload.php"是指点击这个form中的submit的时候,这个上传命令会被发送到这个叫 upload.php的页面去处理。method="post"是指以post方式去送,enctype="multipart/form-data"属 性规定了在提交这个表单时要使用哪种内容类型,在表单需要二进制数据时,比如文件内容,请使用"multipart/form-data",如果要上传文 件,这个属性是必要的。input中的type="file"时,规定了应该把输入作为文件来处理,并且在input后面会有一个浏览的按钮。

我们再来看一个PHP处理页面 upload.php

<?php
if ( $_FILES [ 'myfile' ][ 'name' ] != '' ) {
   if ( $_FILES [ 'myfile' ][ 'error' ] > 0) {
     echo  "错误状态:"  . $_FILES [ 'myfile' ][ 'error' ];
  } else  {
    move_uploaded_file( $_FILES [ 'myfile' ][ 'tmp_name' ] , "uploads/"  . $FILES [ 'myfile' ][ 'name' ]);
     echo  "<script>alert(上传成功!);</script>" ;
  }
} else {
   echo  "<script>alert(请上传文件!);</script>" ;
}
?>

说明:
在解释这篇代码前,我们有必要了解以下知识。

$_FILES['myfile']['name']  是指被上传文件的名称
$_FILES['myfile']['type']  是指被上传文件的类型
$_FILES['myfile']['size']  是指被上传文件的大小,单位为字节(B)
$_FILES['myfile']['tmp_name']  是指被上传文件存在服务器中的临时副本文件名称,文件被移动到指定目录后临文件将被自动消毁。
$_FILES['myfile']["error"]  是指由文件上传中有可能出现的错误的状态码,关于各状态含义后在会说明。

了解了这些,我们再来看一下upload.php的代码。
首先,$_FILES['myfile']['name']中的myfile是指在上面HTML页面中上传文件标签的name值,根据这个我们才能知道我 们正在处理的文件是哪一个input提交过来的,然后再来判断一下 $_FILES['myfile']['name'] 不是否为空,根据这个我们可以知道用户有没有上传文件,从而执行不同的操作。如果上传了文件并且状态是0就说明上传成功,我们就可以用 move_uploaded_file方法把上传的文件存放到指定目录,上面这个例子是指把上传的文件移动到同目录下的uploads文件夹下,这个路径 是相对于这个PHP文件(既upload.php)的相对目录。比如,我们想把上传的文件移动到upload.php上一层叫user的文件夹中的话,我 们就可以这样写:move_uploaded_file($_FILES['myfile']['tmp_name'] , "../user/" . $FILES['myfile']['name']),这种方法使用起来很方便、灵活,这样一个文件就被上传到服务器中了,可以打开服务器中的目录查看该 文件。

允许用户上传文件是一个有巨大的安全风险的行为,因此,通常情况下,我们会对用户上传的文件做一些限制,比如常见的限制文件
类型和文件大小,来看一下。

<?php
if ( $_FILES [ 'myfile' ][ 'name' ] != '' ) {
   if ( $_FILES [ 'myfile' ][ 'error' ] > 0) {
     echo  "错误状态:"  . $_FILES [ 'myfile' ][ 'error' ];
  } else  {
     if ( $_FILES [ 'myfile' ][ 'type' ] == 'image/jpeg'  or  $_FILES [ 'myfile' ][ 'type' ] == 'image/pjpeg'  or  $_FILES [ 'myfile' ][ 'type' ] == 'image/gif'  && $_FILES [ 'myfile' ][ 'size' ] < 20480){
      move_uploaded_file( $_FILES [ 'myfile' ][ 'tmp_name' ] , "uploads/"  . $FILES [ 'myfile' ][ 'name' ]);
       echo  "<script>alert(上传成功!);</script>" ;
    } else  {
       echo  "<script>alert(请上传小于2MB的jpeg或Gif类型的附件);<script>" ;
    }
  }
} else  {
   echo  "<script>alert(请上传文件!);</script>" ;
}
?>

从上面的代码可以看出,我们规定了上传的文件类型必须是jpeg或者Gif并且必须小于2MB的文件($_FILES['myfile']['size']的默认单位是字节)
这里必须提到的是,对于IE浏览器,它识别jpg文件的类型必须是 pjpeg,而对于 FireFox,则必须是 jpeg,因此,我们必须对jpeg和pjpeg都作判断。
这样一来,我们可以限制用户上传的一些危险的比如木马或者病毒脚本,来保证了服务器的安全运行。
现在,一个上传文件程序就基本成形了。但时在有些时候,考虑到用户体验,我们还可以对用户上传过程中发生的错误作出一些提醒,
让用户明白是哪里出了问题,我们会对$_FILES['myfile']['error']作出一些说明,先来看一下在PHP中对$_FILES['myfile']['error']常见6种状态的定义。

$_FILES [ 'teacher_pic' ][ 'error' ] = 1        文件大小超过了PHP.ini中的文件限制
$_FILES [ 'teacher_pic' ][ 'error' ] = 2        文件大小超过了浏览器限制
$_FILES [ 'teacher_pic' ][ 'error' ] = 3        文件部分被上传
$_FILES [ 'teacher_pic' ][ 'error' ] = 4        没有找到要上传的文件
$_FILES [ 'teacher_pic' ][ 'error' ] = 5        服务器临时文件夹丢失
$_FILES [ 'teacher_pic' ][ 'error' ] = 5        文件写入到临时文件夹出错

错误信息状态为1时说明上传的文件超过了php.ini中的文件大小限制,我们可以打开php.ini这个文件。

来找一下; Maximum allowed size for uploaded files.upload_max_filesize = 2M
我这里是在第516行,这一句说定义了PHP中上传文件的最大字节数,默认情况下是2MB,这个设置是PHP全局上传限 制,权限最高,即使$_FILES['myfile']['size']设为10MB,也只能上传2MB以下的文件。比如,在默认情况下,如果规 定$_FILES['myfile']['size'] < 10MB,在用户上传文件大于2MB的情况下,就会现在$_FILES['teacher_pic']['error'] = 1的情况,一般来说,我们须要把$_FILES['myfile']['size']的值设定在upload_max_filesize值之下(设大了也 没用,呵呵)。当然,你完全可以把php.ini中的upload_max_filesize值调的更大,但实际应用中,我们考虑到服务器的负载能力,不 建议upload_max_filesize的值超过20MB,这样会造成网站附件增大,这在论坛社区上可以很明显的看出来。
了解了这些,我们就可以对错误状态作出定义,我们再来完善一下代码,来看一下。

<?php
if ( $_FILES [ 'myfile' ][ 'name' ] != '' ){
   if ( $_FILES [ 'myfile' ][ 'error' ] > 0){
     switch ( $_FILES [ 'myfile' ][ 'error' ]){
       case  1:
         echo  "文件大小超过了PHP.ini中的文件限制!" ;
         break ;
       case  2:
         echo  "文件大小超过了浏览器限制!" ;
         break ;
       case  3:
         echo  "文件部分被上传!" ;
         break ;
       case  4:
         echo  "没有找到要上传的文件!" ;
         break ;
       case  5:
         echo  "服务器临时文件夹丢失,请重新上传!" ;
         break ;
       case  6:
         echo  "文件写入到临时文件夹出错!" ;
         break ;
    }
  } else  {
     if ( $_FILES [ 'myfile' ][ 'type' ] == 'image/jpeg'  or  $_FILES [ 'myfile' ][ 'type' ] == 'image/pjpeg'  or  $_FILES [ 'myfile' ][ 'type' ] == 'image/gif'  && $_FILES [ 'myfile' ][ 'size' ] < 20480) {
      move_uploaded_file( $_FILES [ 'myfile' ][ 'tmp_name' ] , "uploads/"  . $FILES [ 'myfile' ][ 'name' ]);
       echo  "<script>alert(上传成功!);</script>" ;
    } else  {
       echo  "<script>alert(请上传小于2MB的jpeg或Gif类型的附件);<script>" ;
    }
  }
} else  {
   echo  "<script>alert(请上传文件!);</script>" ;
}
?>

可以看出,我们使用了switch语句来对6种错状态作出定义,这样来下,在发生错误的时间,用户就会明白,是哪里出了问题。
但是还有一种情况就是,用户上传的文件在指定的目录中已经存在,这里我们可以使用file_exists方法来判断一下:

<?php
if ( $_FILES [ 'myfile' ][ 'name' ] != '' ){
   if ( $_FILES [ 'myfile' ][ 'error' ] > 0){
     switch ( $_FILES [ 'myfile' ][ 'error' ]){
       case  1:
         echo  "文件大小超过了PHP.ini中的文件限制!" ;
         break ;
       case  2:
         echo  "文件大小超过了浏览器限制!" ;
         break ;
       case  3:
         echo  "文件部分被上传!" ;
         break ;
       case  4:
         echo  "没有找到要上传的文件!" ;
         break ;
       case  5:
         echo  "服务器临时文件夹丢失,请重新上传!" ;
         break ;
       case  6:
         echo  "文件写入到临时文件夹出错!" ;
         break ;
    }
  } else  {
     if ( $_FILES [ 'myfile' ][ 'type' ] == 'image/jpeg'  or  $_FILES [ 'myfile' ][ 'type' ] == 'image/pjpeg'  or  $_FILES [ 'myfile' ][ 'type' ] == 'image/gif'  && $_FILES [ 'myfile' ][ 'size' ] < 20480) {
       if  (! file_exists ( "uploads/"  . $_FILES [ "myfile" ][ "name" ]))
        move_uploaded_file( $_FILES [ 'myfile' ][ 'tmp_name' ] , "uploads/"  . $FILES [ 'myfile' ][ 'name' ]);
         echo  "<script>alert(上传成功!);</script>" ;
      } else {
         echo  "<script>alert(您上传的文件已经存在!);</script>" ;
      }
    } else  {
       echo  "<script>alert(请上传小于2MB的jpeg或Gif类型的附件);<script>" ;
    }
  }
} else  {
   echo  "<script>alert(请上传文件!);</script>" ;
}
?>

我的个娘呀,终于写完了,一个完整的上传程序终于完成了,这只是上传文件最原始的方法,这样更容易自己理解,使用时大家可以考虑把它写成类
现在我们再来总结一下上传中的逻辑判断顺吧。


1. 先判断是否上传文件
2. 如果有再来判断上传中是否出错
3. 如果出错,则提示出错信息
4. 如查没出错,再判断文件类型
5. 如果类型符合条件,再判断指定目录中有没有存在该文件
6. 如果没有就把该文件移至指定目录

本文转自博客园知识天地的博客,原文链接:PHP文件上传详解,如需转载请自行联系原博主。

相关文章
|
6月前
thinkphp5.1隐藏index.php入口文件
thinkphp5.1隐藏index.php入口文件
61 0
thinkphp5.1隐藏index.php入口文件
|
2月前
|
PHP
php常见问题,php.ini文件不存在或者找不到,mb_strlen()函数未定义系列问题,dll模块找不到的解决
本文介绍了解决PHP常见问题的步骤,包括定位和创建`php.ini`文件,以及解决`mb_strlen()`函数未定义和DLL模块加载错误的具体方法。
php常见问题,php.ini文件不存在或者找不到,mb_strlen()函数未定义系列问题,dll模块找不到的解决
|
5月前
|
存储 运维 Serverless
函数计算产品使用问题之在YAML文件中配置了环境变量,但在PHP代码中无法读取到这些环境变量,是什么原因
函数计算产品作为一种事件驱动的全托管计算服务,让用户能够专注于业务逻辑的编写,而无需关心底层服务器的管理与运维。你可以有效地利用函数计算产品来支撑各类应用场景,从简单的数据处理到复杂的业务逻辑,实现快速、高效、低成本的云上部署与运维。以下是一些关于使用函数计算产品的合集和要点,帮助你更好地理解和应用这一服务。
|
2月前
|
前端开发 PHP
php学习笔记-php文件表单上传-day06
本文介绍了PHP文件上传处理流程、预定义变量`$_FILES`的使用、文件上传状态代码以及文件上传实现函数。同时,通过一个文件上传的小例子,演示了文件上传表单的创建、文件上传表单处理的PHP页面编写以及运行测试输出。
php学习笔记-php文件表单上传-day06
|
6月前
|
关系型数据库 MySQL
web简易开发(二){html5+php实现文件上传及通过关键字搜索已上传图片)}
web简易开发(二){html5+php实现文件上传及通过关键字搜索已上传图片)}
|
2月前
|
缓存 监控 算法
分析慢日志文件来优化 PHP 脚本的性能
分析慢日志文件来优化 PHP 脚本的性能
|
2月前
进入靶场,出现一张照片,右击查看源代码,发现有一个注释的source.php文件
这段代码实现了一个网站上弹出的促销海报动画效果,包含一个关闭按钮。当促销海报弹出时,会在三秒后开始抖动一两下。海报使用固定定位居中显示,带有阴影和圆角,关闭按钮位于右上角。可以通过修改时间参数调整弹出时间。
19 0
|
2月前
|
PHP
深入浅出PHP之文件上传功能
【9月更文挑战第26天】本文将带你了解PHP中的文件上传功能,从基本概念到实战操作,一步步教你如何实现文件上传。我们将通过代码示例和详细的解释,让你轻松掌握这一技能。
|
3月前
|
存储 安全 数据库连接
php.ini 文件的用途是什么?
【8月更文挑战第29天】
64 1
|
3月前
|
PHP
PHP遍历文件并同步上传到服务器
在进行网站迁移时,由于原网站的图片文件过多,采用打包下载再上传的方式耗时过长,且尝试使用FTP工具从旧服务器传输至新服务器时失败。为解决此问题,特使用PHP编写了一款工具,该工具能扫描指定目录下的所有`.webp`图像文件,并将其上传至新的服务器,极大地提高了迁移效率。
103 16