日期:2014-05-17 浏览次数:20629 次
随着互联网的发展,越来越多的技术开始注重用户体验,以人为本才是长久之道,于是在上传的时候,大家都不再满足一个单一的“浏览”按钮,纷纷推出了带上传进度条的功能。而作为解释型语言的PHP,如何做到对上传文件的检测,如何实现上传进度条以其背后的原理,54chen将在本文中一步步展开。
一. 实现篇
一般情况,用PHP实现上传进度条就下面两种方法:
1.APC扩展(作者是PHP的创始人,5.2后PHP已经加入APC扩展)
2.PECL扩展模块 uploadprogress
不论是APC还是uploadprogress,都需要编译源码,因为原有的PHP函数根本不可能读取到临时文件夹里的东西。下面来看如何使用以及关键的代码:
if ($_SERVER['REQUEST_METHOD'] == ‘POST’) { ?//上传请求$status = apc_fetch(‘upload_’ . $_POST['APC_UPLOAD_PROGRESS']);$status['done'] = 1;echo json_encode($status); ?//输出给用户端页面里的ajax调用,相关文档请自己寻找exit;} elseif (isset($_GET['progress_key'])) { ? //读取上传进度$status = apc_fetch(‘upload_’.$_GET['progress_key']);echo json_encode($status);exit;}
if($_SERVER['REQUEST_METHOD']==’POST’) {if (is_uploaded_file($_FILES['upfile']['tmp_name'])) {$upload_dir = ‘your_path/’;$ext ? ? ? ?= strrchr($_FILES['video']['name'], ‘.’);$sessid ? ? = $_POST['UPLOAD_IDENTIFIER'] ;$tmpfile ? ?= $upload_dir . $sessid;$sessfile ? = $upload_dir . $sessid .$ext;if (move_uploaded_file($_FILES['upfile']['tmp_name'],$tmpfile)) {//上传成功}}} elseif (!empty($_GET['sessid'])) {header(“Expires: Mon, 26 Jul 1997 05:00:00 GMT”);header(“Last-Modified: ” . gmdate(“D, d M Y H:i:s”) . ” GMT”);header(“Cache-Control: no-store, no-cache, must-revalidate”);header(“Cache-Control: post-check=0, pre-check=0″, false);header(“Pragma: no-cache”);header(“Content-Type:text/html;charset=UTF-8″);$unique_id = $_GET['sessid'];$uploadvalues = uploadprogress_get_info($unique_id);if (is_array($uploadvalues)) {echo json_encode($uploadvalues);} else {//读取进度失败,另外处理逻辑}}
二. 原理篇
注意上一篇中的红色函数。
下载到uploadprogress1.0.1进行源码分析,在代码中作了注释。
static void uploadprogress_file_php_get_info(char * id, zval * return_value)
{
char s[1024];
char * filename;
char * template;
FILE *F;
TSRMLS_FETCH();
template = INI_STR(“uploadprogress.file.filename_template”); <<这里读取设置好的模板
if (strcmp(template, “”) == 0) ?{
return;
} else {
filename = uploadprogress_mk_filename( id, template );<<<存在的话,会创建
if (!filename) return;
F = VCWD_FOPEN(filename, “rb”);
if (F) {
array_init(return_value);
while ( fgets(s, 1000, F) ) {<<<从流中读取一字符串?*s结果数据的首地址;1000-1:一次读入数据块的长度,其默认值为1k,即1024;F文件指针
char *k, *v, *e;
int index = 0;
e = strchr(s,’='); <<<查找字符串s中首次出现字符=的位置
if (!e) continue;
*e = 0; /* break the line into 2 parts */
v = e+1;
k = s;
/* trim spaces in front of the name/value */
while (*k && *k <= 32) k++;
while (*v && *v <= 32) v++;
/* trim spaces everywhere in the name */
for (e=k; *e; e++) if (*e <= 32) { *e = 0; break; }
/* trim spaces only at the end of the value */
/* http://pecl.php.net/bugs/bug.php?id=14525 */
//for (e=v; *e; e++) if (*e <= 32)