技术头条 - 一个快速在微博传播文章的方式     搜索本站
您现在的位置首页 --> PHP --> PHP上传进度条深度解析

PHP上传进度条深度解析

浏览:4876次  出处信息

随着互联网的发展,越来越多的技术开始注重用户体验,以人为本才是长久之道,于是在上传的时候,大家都不再满足一个单一的“浏览”按钮,纷纷推出了带上传进度条的功能。而作为解释型语言的PHP,如何做到对上传文件的检测,如何实现上传进度条以其背后的原理,54chen将在本文中一步步展开。

一. 实现篇

一般情况,用PHP实现上传进度条就下面两种方法:

1.APC扩展(作者是PHP的创始人,5.2后PHP已经加入APC扩展)

2.PECL扩展模块 uploadprogress

不论是APC还是uploadprogress,都需要编译源码,因为原有的PHP函数根本不可能读取到临时文件夹里的东西。下面来看如何使用以及关键的代码:

APC实现方法:

1.安装APC
2.配置php.ini,设置参数 apc.rfc1867=1
3.关键代码:
1
2
3
4
5
6
7
8
9
10
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;
}
uploadprogress实现方法:
1.使用PECL 安装uploadprogress
2.php.ini里面设置 uploadprogress.file.filename_template = “/tmp/upd_%s.txt”
3.关键代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
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进行源码分析,在代码中作了注释。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
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) { *e = 0; break; } 				if (v != NULL) { //当文件有内容时 					for (index = strlen(v); index > 0; index--) {
						if (v[index] > 32) break; //累计
						v[index] = 0;
					}
				}
				add_assoc_string( return_value, k, v, 1 );
			}
 
			fclose(F);
		}
	}
 
	if (filename) efree(filename);
	return;
}

在源码中还能发现:

1
2
3
4
5
6
PHP_MINIT_FUNCTION(uploadprogress)
{
	REGISTER_INI_ENTRIES();
	php_rfc1867_callback = uploadprogress_php_rfc1867_file;
	return SUCCESS;
}

在MINIT中修改了php_rfc1867_callback,抽取uploadprogress_php_rfc1867_file的关键代码:

1
2
3
4
5
6
7
8
9
10
11
12
upload_id = emalloc(strlen(*e_data->value) + 1);
strcpy(upload_id, *e_data->value);
progress->upload_id = upload_id;
progress->time_last  = time(NULL);
progress->speed_average  = 0;
progress->speed_last     = 0;
progress->bytes_uploaded = read_bytes;
progress->files_uploaded = 0;
progress->est_sec        = 0;
progress->identifier = uploadprogress_mk_filename(upload_id, template);//在指定的模板位置放下了临时文件
progress->identifier_tmp = emalloc(strlen( progress->identifier) + 4);
sprintf( progress->identifier_tmp, "%s.wr", progress->identifier );

关于php_rfc1867_callback是什么,可以看哥学社另一哥鸟哥的分析 http://www.laruence.com/2008/11/07/586.html

三.总结

在探测临时文件大小的时候,APC和 uploadprogress其实是大同小异的方法,先记录,再取大小百分比。

本文提及的文章:

http://www.php.net/releases/5_2_0.php

http://www.ibm.com/developerworks/cn/opensource/os-php-v525/

http://www.laruence.com/2008/11/07/586.html

建议继续学习:

  1. QQ上传大文件为什么这么快    (阅读:12360)
  2. 10个强大的Ajax jQuery文件上传程序    (阅读:7751)
  3. 解决securecrt rz 上传rar,gif文件不正确问题    (阅读:7078)
  4. Codeigniter里的无刷新上传    (阅读:4590)
  5. PHP上传文件类型彻底判断方案及PHP+nginx上传大小彻底控制方案    (阅读:3825)
  6. PHP JAVA C上传文件如何准确判断文件类型-mime知识普及    (阅读:3541)
  7. 渐进增强的无刷新多图片上传控件(iFrame+HTML5)    (阅读:3426)
  8. PHP文件上传源码分析(RFC1867)    (阅读:3333)
  9. php_admin_value open_basedir 引起的上传文件失败解决方法    (阅读:3308)
  10. PHP上传进度条深度解析    (阅读:2582)
QQ技术交流群:445447336,欢迎加入!
扫一扫订阅我的微信号:IT技术博客大学习
© 2009 - 2024 by blogread.cn 微博:@IT技术博客大学习

京ICP备15002552号-1