回顾
数组: 数据的组合
数组定义: 三种方式(常用array())
数组访问: 下标
数组分类: 关联数组, 索引数组和混合数组
数组循环遍历: foreach遍历(foreach原理)
二维数组: 不是真正的二维数组(遍历一维: 第二维下标访问)
Each+list结构
数组函数: 指针函数, 数据结构模拟(栈和队列), 数组字符串函数(explode,implode), 验证码
数组运算: + 和array_merge
数组比较: 数组长度,元素出现顺序,下标元素位置, 数据值,数据类型
算法: 冒泡算法
表单数据提交
表单数据提交: 用户通过浏览器将数据提交服务器.
浏览器提交数据的方式是什么?
浏览器提交数据的本质: 是基于HTTP协议.
浏览器提交数据的方式有两种:
URL传值: 将数据以键值对的形式,绑定到URL之后
表单传值: 通过form表单
提交的本质形式有两种:
GET传值: URL传值和form表单的get方式传值
POST传值: POST表单传值
数据提交方式
GET传值
方案1: 直接绑定到URL之后: 键值对的形式
如: http://www.itcast.cn/gz/php/index.php?name=张三&age=30
代表传递两组数据: name和age,其值分别为张三和30
方案2: 可以通过form表单,使用get方式实现
POST传值
只有一种方案: form表单使用post方式提交数据
GET与POST区别
理论上讲: GET与POST有本质区别
GET: get传递的数据不应该影响服务器已经存在的数据(查询)
POST:post传递的数据就是用来修改数据的(更改)
INPUT: input传递的数据是用来增加数据(新增)
DELETE: delete传递的数据应该用来删除数据(删除)
GET和POST的区别
- 传递数据给服务器的方式不一样: GET是在URL后显示提交; POST是隐式提交(HTTP协议)
- 安全性能不一样: GET不安全,POST安全
- GET和POST提交的数据大小不一样: 事实是HTTP协议从未规定GET与POST大小不一样
a)
Get是查询: 有浏览器规定GET只能最大2K(只有IE)
b)
Post是数据提交: 浏览器规定POST最大只能传递8M(服务器也有限制)
数据接收
服务器接收: 浏览器将数据提交给服务器: 但是apache是无法接收数据.PHP可以接收数据.
PHP针对数据提价的方式不同,有不同的接收方式: 都是PHP自动接收
GET数据接收
在PHP中: 使用$_GET超全局预定义变量(系统定义)来接收get方式提交的数据. 所有的超全局预定义变量都是数组类型: 所有的数据提交后都是以键值对的形式在$_GET中存在.
POST数据接收
在PHP中: 使用$_POST超全局预定义变量来接收post方式提交的数据
PHP中,考虑到有可能数据的来源不确定(POST或者GET不清楚):可以通过判断如果不在GET中就在POST中: 但是判断就会降低效率, 可以使用PHP的另外一种接收方式:$_REQUEST.
混合数据接收
$_REQUEST是PHP中提供了默认合并$_GET和$_POST的一种数据保存.
$_REQUEST虽然可以不用区分get和post数据: 但是正是因为其不区分,所以导致不安全:
$_REQUEST并不是接收数据: 而是合并数据($_GET和$_POST): 合并原理是使用array_merge,会有覆盖的危险.
$_REQUEST的合并规则在配置文件中可以配置: request_order
如果是低版本中的PHP,可能没有该选项: 默认是走variables_order
复选框传值
复选框: checkbox
复选数据库提交
- 复选框: 只有选中的复选框才会提交,没有选中不会提交
- 因为一类复选框: 名字都相同: 名字提交到服务器中之后,作为数组的下标: 下标具有唯一性: 多个相同的会覆盖,只会保留最后一个
解决被覆盖的问题
解决方案1: 修改名字,让复选框的名字不一样(不用): 复选框往往是作为一类数据,名字应该都是一样的
解决方案2: 服务器因为同名问题覆盖原始数据,而浏览器不区分是否同名: PHP有一个特点: PHP只要见到[]就会当做数组处理: 如果中括号内部没有内容,系统自动添加索引下标使其变得不一样: 给checkbox的name属性增加[]
服务器接收效果
复选框入库
- 如何设计表?
方案1: 单独一张表: 爱好表, 每个爱好有一个对应的字段
篮球 |
足球 |
羽毛球 |
排球 |
台球 |
网球 |
手球 |
地球 |
篮球 |
台球 |
||||||
足球 |
羽毛球 |
排球 |
方案1可以解决复选框数据存储的问题, 但是却让设计变得非常的不灵活
方案2: 在表中增加一个字段: 存放所有的爱好: 爱好通过PHP数组转换成带有分隔符的字符串
用户名 |
年龄 |
爱好 |
张三 |
30 |
篮球,足球,羽毛球 |
李四 |
25 |
地球,台球 |
- 程序代码如何实现?
存的时候将数组转换成字符串;
使用的时候将字符串分解成数组.
存储
提取
- 将数据如何回显给用户查看?
用户查看: 依然用复选框查看
①如何让复选框默认被选中? 给checkbox增加属性: checked = ‘checked’
- 增加一个用户显示checkbox数据的显示界面: html文件
- 使用PHP来判断用户的哪些被选中: 选中的数据增加checked=checked属性
- 修改逻辑: 有被选中的才有属性,没有选中的就没有属性: 当前html肯定是用于被PHP脚本包含, 所以肯定也是使用PHP中准备的爱好信息: PHP中的爱好是一个数组: 判断当前的爱好选项是否在数组中存在; in_array()
- 在需要显示数据的PHP脚本中引入对应的HTML显示文件.
文件上传
文件上传: 将文件从本机电脑上传到服务器电脑.
文件上传条件
- 服务器允许用户上传文件: apache不能处理文件: PHP才能处理
a)
文件是否允许上传
b)
文件上传到服务器之后保存到什么位置
- 浏览器需要上传条件
a)
浏览器需要表单: form, 只能POST提交
b)
表单中必须使用文件上传表单域: <input type=”file”>
c)
文件是二进制数据: 但是表单默认只能提交字符流数据: 如果要实现提交,需要告诉浏览器: 当前内容中有二进制数据, 需要在表单中增加一个文件上传的属性: enctype=”multipart/form-data”
PHP接收文件
如果文件上传的表单域存在: 但是没有给form增加enctype属性: 文件不可能上传, 但是表单可以上传: 上传的数据很明显会被$_POST接收.
PHP中接收文件上传信息: 由$_FILES超全局预定义变量实现
之所以看不到文件: PHP没有处理,就会被系统回收: 系统在PHP脚本执行结束,php没有处理,操作系统自动回收: 如何让脚本执行周期变长?
Sleep(10): 休息10秒
移动文件
文件上传本质是系统接收: 跟PHP没有关系, PHP只是在脚本周期内有权利去访问存在的临时文件: 如果不处理,操作系统会自动清理: 在PHP可以操作的周期内,将临时文件移动到目标文件夹内,
然后给其新的名字(临时文件不能使用)
PHP提供了一个系统函数能够移动文件(专门移动上传的文件)
Move_uploaded_file(临时文件路径, 目标路径); //两个路径都带文件名(copy())
文件上传错误
文件在上传的过程中,很有可能出现多种错误: 用户根本没有选中要上传的文件,或者网络不稳定等,都有可能出现错误: 而这种错误不是由PHP来抓取: PHP只负责处理最终由系统接收好的临时文件.
系统在接收文件的过程中,会将已经出现的错误信息变成对应的错误代码error提示PHP: PHP根据错误提示进行相应的处理.
封装文件上传函数
- 创建一个单独的文件: 里面只存放上传文件的函数
- 确定需要实现上传函数的参数: 要上传的文件数组信息, 要上传的文件路径, 允许上传的类型数组, 当前允许的大小(size)有默认值
- 确定返回值: 成功返回文件的新名字, 失败返回false
- 因为文件上传会出现错误:要将错误信息反馈出来: 引用传递错误信息
- 判断文件文件信息
a)
判断文件信息是否有效: 必须是数组,而且必须是5个元素数组
b)
判断系统错误(文件中的error信息)
c)
判断文件类型是否合法: 如果类型不再允许的类型数组中
d)
判断文件大小是否符合当前要求
- 需要对文件进行重命名: 时间 + 随机字符串: 新建一个函数专门产生随机名字
- 移动文件到指定目录
- 验证函数是否正确: 调用
总结:一般的数据提交方式是:post表单方式因为post是隐藏式提交相对于get方式比较安全,一般链接都是get方式提交,重要的数据传输都是post,在php中的接受数据方式分别为$_POST,$_GET,还有一种方式是$_REQUEST表示两者的混合
复选框:例如一个人的爱好有很多,但是type的类型只能是一个(爱好);我们可以这样做将踢足球,打篮球等这些放到爱好了,爱好是数组,踢足球和打篮球是他的子元素;
文件上传:要考虑符合文件上传的规则,例如明白文件上传的五要素(文件的信息,文件的路径,文件的类型,文件的大小,
<?php //接收上传文件
header('Content-type:text/html;charset=utf-8'); //查看文件上传信息
//var_dump($_POST); //文件上传信息: $_FILES
echo '<pre>';
var_dump($_FILES); //休息一下
//sleep(10); //接收文件信息
$file = $_FILES['myfile']; //$file是一个数组: 五个元素 //移动临时文件
//$res = move_uploaded_file($file['tmp_name'],'uploads/' . //$file['name']);
//var_dump($res); //调用文件上传函数实现文件上传
include_once 'demo11_fileupload_function.php'; //自定义允许的类型: image/pjpeg是IE浏览器下特有的一种MIME类型
$allow = array('image/png','image/jpg','image/jpeg','image/pjpeg','image/gif'); //使用函数
$filename = uploadfile($file,'uploads',$allow,$error);
//$error虽然没有定义,但是不会出错: 函数内部不是使用$error,而是赋值 //判断结果
if($filename){
//成功
echo $filename;
}else{
//失败
echo $error;
}
文件上传代码
<?php //文件上传函数 /*
* 实现文件上传
* @param1 array $file,要上传的文件信息(文件五要素数组)
* @param2 string $path,要上传的文件路径
* @param3 array $allow,允许上传是文件类型(MIME)
* @示例 $allow = array('image/jpg','image/jpeg'...)
* @param4 string &$error,用来记录错误信息的变量
* @param5 int $maxsize = 1000000, 默认为1M
* @return 成功返回新的文件名,失败返回false
*/
function uploadFile($file,$path,$allow,&$error,$maxsize = 1000000){
//判断文件是否有效
if(!is_array($file) || count($file) != 5){
//文件无效
$error = '无效文件,不能上传!';
return false;
} //判断系统错误(系统在接收数据的时候发现的错误)
switch($file['error']){
case 1: //文件超过服务器允许的大小
$error = '文件超过服务器允许的大小!'; //ini_get('upload_max_filesize')获取配置文件中指定配置项的值
return false;
case 2: //超过浏览器表单允许大小
$error = '文件超过表单允许大小!';
return false;
case 3: //上传部分
$error = '文件只有部分上传成功!';
return false;
case 4: //没有选中文件
$error = '没有选中要上传的文件!';
return false;
case 6:
case 7:
$error = '服务器错误!';
return false;
} //判断文件类型是否合法
if(!in_array($file['type'],$allow)){
//类型不存在
$error = '当前文件的类型不合法!'; //增加允许的类型
return false;
} //判断文件的大小是否满足当前需求
if($file['size'] > $maxsize){
//超出大小
$error = '文件超出当前允许的范围!'; //告诉用户大小
return false;
} //获取文件的新名字
$filename = getRandomName($file['name']); //移动文件
if(move_uploaded_file($file['tmp_name'],$path . '/' . $filename)){
//D:/server//20150923abcdef.jpg
//移动成功: 返回名字
return $filename;
}else{
//移动失败
$error = '文件移动失败!';
return false;
}
} /*
* 生成随机名字: YYYYMMDDHHIISS + 随机6为字符串(小写字母)
* @param1 string $filename,原始文件的名字
* @return string $newname,新的文件名字
*/
function getRandomName($filename){
//生成时间部分名字
$newname = date('YmdHis'); //获取随机部分
$str = 'abcdefghijklmnopqrstuvwxyz';
for($i = 0;$i < 6;$i++){
//每次从字符串中取一个
$newname .= $str[mt_rand(0,strlen($str)-1)];
} //补全后缀名
return $newname . strrchr($filename,'.');
}
封装好的文件上传代码