ThinkPHP 5小知识和常见问题

安装

应用项目:https://github.com/top-think/think 核心框架:https://github.com/top-think/framework 文档:https://www.kancloud.cn/manual/thinkphp5 以上可选择指定版本下载,然后将应用项目解压到网站根目录,进行重命名,如alan,将核心框架拷贝alan目录下,重命名为thinkphp,此时即完成安装。通过访问http://localhost/alan/public/验证安装是否成功。

依赖注入input助手函数 1.自带Server 切换到public目录下执行:

即可在浏览器中执行http://localhost:8080直接进行访问(实际上这是PHP自带的功能) 2.Debug PHP可以通过xdebug来进行断点调试等操作,安装方法: 通过将phpinfo()输出的内容源代码拷贝到https://xdebug.org/wizard.php可获取对应需下载的版本及安装方法,按提示操作即可,在php.ini中需加入的配置

PHP Storm中的配置 Edit Configurations…>左上角加号>PHP Web Application 进入Run/Debug Configurations页面,设置名称 在右侧Configuration下点击Server弹出Servers窗口 点击左上角加号,修改Name,配置Host: localhost 回到Run/Debug Configurations页面在Start URL处输入初始调试页面 点击右上角爬虫图标进行调试 PHP中配置xdebug 3.验证器 独立验证在文件内

验证器封装在单独的文件内

4. 异常Exception 自定义异常可通过修改config.php中的exception_handle来实现 5.日志 日志路径是在框架文件下的base.php中定义的,想改默认路径可在入口文件index.php中进行定义

5. 日志

关闭默认日志将config.php进行如下修改:

手动开启,在方法中加入

6. 数据库

原生SQL: Db::query(‘SELECT * FROM….’)

链式查询:Db::table(‘dbname’)->where(‘id’,’=’, $id)->select();

模型类,继承内置的Model,在controller中执行get方法,如BannerModel::get($id),默认TP会根据模型的类名去数据库中查询同名表,如想要不同的话,在类中定义一个protected $table变量来指定表名即可。

不同于上述方法返回组,采取这种方法返回的是一个对象,因此可以将return json($result)修改成return $result,但默认的返回类型是text/html,可在config.php中

7.模型类

可以通过在模型类中设置protected $hidden = []来指定隐藏的变量,也可以通过protected $visible = []来指定显示的字段

在application目录下创建extra目录并放置配置文件(如setting.php),默认为自动被框架加载,调用可通过config(‘setting.xxx’)来获取

模型类中可通过读取器public function getXxxAttr($value)来对指定数据表中xxx字段进行设置

关联其它模型有一对一有hasOne, belongsTo,区别在于拥有外键方关联另一张表时使用belongsTo, 而关联拥有外键的表时用hasOne

8. 常见配置

9. 路由

10. 事务

 

小技巧

1.PHPStorm中去除未被使用的引用快捷键(Ctrl+Option/Alt+O)

2.PHPStorm自动补全Namespace Preferences > Directories 点击application目录,选中Sources,点击Source Folders下application右侧的P图标,在弹出的窗口中输入app(可通过application/config.php修改默认值app)保存。再次创建类文件时会自动填写Namespace部分的内容

3.命令行创建模型、控制器

扩展:PSR-4与PSR-0

CentOS 6.x 上如何升级php

我们都知道CentOS 6.x上的PHP版本为5.3.3,那么如果想要升级到更高的版本要怎么办呢?

这里使用Magento官方文档中提到的rpm包来进行相关安装

# 不加第一个有可能会报Requires: libmcrypt.so.4的错误
rpm -Uvh https://dl.fedoraproject.org/pub/epel/epel-release-latest-6.noarch.rpm
rpm -Uvh https://mirror.webtatic.com/yum/el6/latest.rpm
yum install -y php56w php56w-opcache php56w-xml php56w-mcrypt php56w-gd php56w-devel php56w-mysql php56w-intl php56w-mbstring php56w-bcmath

但是如果之前已经安装过php的朋友可能会发现无法进行安装,此时请先删除php-common即执行

yum remove php-common -y 

再进行安装可发现会正常进行新版本的升级,我这里演示的是5.6.x的版本,您也可以选择PHP 7.0(php70w...),如果您选择的是其它的rpm包,可通过如下命令来查看可供安装的PHP文件版本

yum list available | grep php

安装完成后可通过php -v或php -version来验证已安装的PHP版本

其它参考链接:https://foreigngods.com/blog/2016/03/14/56/upgrade-to-php-56-on-centos-6/

细说PHP精要版读书笔记

最早接触PHP就是看高洛峰的视频,坦白的说他第一套视频录得并不算好,有些冗长,晚上看会有催眠的效果,但那是在当时能找到关于PHP最好的视频。很遗憾虽然通过这套视频了解了PHP基础知识,但受限于当时对于编程的理解,始终没能做到有所突破,离专业程序员还有相当大的差距。这次决定通过书本来重新梳理一遍,期待能有所突破,因《细说PHP》第三版要到暑假后才会出,所以买了精要版来进行学习。

细说PHP精要版封面

第一个示例文件:

<html>
	<head>
		<meta http-equiv="content-type" content="text/html;charset=utf-8">
		<title>第一个PHP程序(获取服务器信息)</title>
	</head>
	<body>
		<?php
			$sysos = $_SERVER["SERVER_SOFTWARE"];
			$phpversion = PHP_VERSION;
                        //连接数据库并获取MySQL版本
			mysql_connect("localhost","root","");
			$mysqlinfo = mysql_get_server_info();
                        //获取GD库的信息
			if(function_exists("gd_info")){
				$gd = gd_info();
				$gdinfo = $gd["GD Version"];
			}else{
				$gdinfo = "未知";
			}
                        //查看GD库是否支持FreeType字体
			$freetype = $gd["FreeType Support"] ? "支持" : "不支持" ;

			$allowurl = ini_get("allow_url_fopen") ? "支持" : "不支持" ;

			$max_upload = ini_get("file_uploads") ? ini_get("upload_max_filesize") : "Disabled";

			$max_ex_time = ini_get("max_execution_time")."秒";

			date_default_timezone_set("Etc/GMT-8");
			$systemtime = date("Y-m-d H:i:s", time());

			echo "<table align=center cellspacing=0 cellpadding=0>";
			echo "<caption><h2>系统信息</h2></caption>";
			echo "<tr><td>Web服务器:</td><td>$sysos</td></tr>";
			echo "<tr><td>PHP版本:</td><td>$phpversion</td></tr>";
			echo "<tr><td>MySQL版本:</td><td>$mysqlinfo</td></tr>";
			echo "<tr><td>GD库版本:</td><td>$gdinfo</td></tr>";
			echo "<tr><td>FileType:</td><td>$freetype</td></tr>";
			echo "<tr><td>远程文件获取:</td><td>$allowurl</td></tr>";
			echo "<tr><td>最大上传限制:</td><td>$max_upload</td></tr>";
			echo "<tr><td>最大执行时间:</td><td>$max_ex_time</td></tr>";
			echo "<tr><td>服务器时间:</td><td>$systemtime</td></tr>";

			echo "</table>";
		?>
	</body>
</html>

第二天

如同任何一门编程语言一样,首先复习的自然还是基础语法结构,对于PHP而言代码以<?开启,以?>关闭(非强制性要求),变量以$开头。然后自然少不了数据类型,有布尔型、整型、浮点型和字符串四种标量类型,数组和对象两种复合类型以及资源和NULL两种特殊类型。PHP是弱类型语言,因而无需像JAVA等语言那样要求先声明数据类型。

对应于变量的还有常量,通常采用大写字母来进行命名,常量多用于配置文件,如数据库用户名、密码等则可以通过常量写入配置文件,方便程序中各处调用,这样修改后其它代码均无需修改,且由于其是常量,不会担心别人在代码中对其重新赋值。

运行在计算机上的程序自然少不了计算的功能,对于不同运算则需采用不同运算符,像最基础的算术运算符+,-,*,/,%,++,–,字符串运算符点.,对应的赋值运算符有+=,-=,*=,/=,%=,.=,此外还有比较运算符、逻辑运算符、位运算符等。

第三天

程序中另一个基础的内容就是流程控制,if…elseif…elseif条件分支语句,另一种分支语句switch

switch(expr){
	case 1:
		expr 1;
		break;
	case 2:
		expr 2;
		break;
	...
	default:
		expr;
}

另一种则是循环语句,有while(){}循环,与之对应至少执行一次的do{}while();循环,还有应用更为广泛的for(;;){}循环。

此外还有一些流程控制语句如break退出循环,值得一提的是break 1退出一层循环,break 2退出两重循环,所以在break后添加数字可以退出一重或多重循环。cotninue语句则用于跳过某一次循环,接着执行下面的循环,exit()和die()可以退出当前脚本。

第四天

函数是一个被命名的、独立的代码段,它执行特定的任务,并可能给调用 它的程序返回一个值。

function 函数名([参数1, 参数2, ... 参数n])
{
    函数休;
    return 返回值;
}

定义函数时函数名后面括号内的表达式称为形式参数(”形参“),而调用函数时后面括号中的表达式称为实际参数(“实参”)。使用global关键字或超全局变量$GLOBALS[]来调用函数外声明的变量或进行重新赋值。在定义函数时对形参赋初值则形成了带有默认参数的函数,在调用时可不传参数值或仅传入部分值,在实参少于形参时,靠右边的形参不会被传值。

而要定义可变个数参数的函数则需要借助于函数func_get_args(),示例如下

function more_args(){
	$args = func_get_args();
	for($i=0; $i<func_num_args(); $i++){ //或count($args)
		echo "第{$i}个参数是:".$args[$i]."<br />";
	}
}

require()和include()语句性能相似,都是包含并运行指定文件。不同之处在于,对include()语句来说,在执行文件时每次都要进行读取和评估,而对require()语句文件只处理了一次。所以需多次执行代码,require()效率比较高,而每次执行代码时如读取不同的文件,则使用Include()语句。

第五天

数组根据索引下标为数字和字符串划分为索引数组和关联数组。调试数组时可通过print_r()和var_dump()打印出下标和值,声明数组可以通过$arr[“key”] = “value”的方式,也可以通过

$arr = array( key1 => value1, key2 => value2 …);

的方式。

遍历数组:for()循环的方式进行遍历,但这仅限于索引数组且下标为连续数字的数组;另一种更为常用的方式是用foreach()来对数组进行遍历,不仅适用于索引数据还适用于关联数组。

foreach($array as $value){
   循环体;
}
foreach($array as $key => $value){
   循环体;
}

PHP中有一套预定义数组,叫做全局变量或超全局变量,如$_SERVER, $_ENV, $_GET, $_POST, $_REQUEST, $_FILES, $_COOKIE, $_SESSION, $GLOBALS

第六天

面向对象(Object Oriented Programming , OOP)。对象需要对类进行实例化,声明一个类需通过关键字class来进行(成员属性,成员方法),示例如下

class SimpleClass
{
    // property declaration
    public $var = 'a default value';

    // method declaration
    public function displayVar() {
        echo $this->var;
    }
}

通关键字new来实例化一个类,如$obj = new class(),访问类中的成中属性或成员方法则通过->符号,如$obj->name,引用对象内的成员属性或方法使用$this,如$this->name。

构造方法:对象创建后第一个自动调用的方法__construct(),析构方法:对象销毁前最后调用的方法__destruct(),

封装性

在类中成员属性或方法前添加private关键字可限制外部的访问,默认为public(成员属性前未
添加public, private时应使用var)。

魔术方法:除__construct(),__desctruct()外,还有__set(), __get(), __isset(), __unset()

继承性

PHP中通过关键字extends一个类可以继承另一个类,不同于C++,PHP仅支持单继承。

访问控制修饰符public公有的,private私有的,protected受保护的

PHP对象访问控制修饰符

PHP中不能定义第一名的函数,所以不能进行重载,但子类中的方法可以和父类同名,也便实现了重载。使用”parent::方法名“可在子类中重载的同时使用父类中其要覆盖的方法。

常见关键字和魔术方法:final标识的类不能被继承,final标识的成员方法不能被覆盖;static可将类中的成员属性和成员方法标识为静态的,可通过类名::静态成员属性名类名::静态成员方法名()来进行调用,在类中可使用self::::静态成员属性名self::静态成员方法名()进行调用,此外静态方法中只能访问静态成员;类中的常量不能使用define()函数定义,而要使用关键字const,访问方法和静态成员属性相同;clone用来克隆对象,如$obj2 = clone $obj1,克隆时需对成员属性重新赋值时可在类中声明一个魔术方法__clone();通过在类中声明魔术方法__toString(),则可直接通过echo加对象名输出类中所返回的字符串;在类中声明魔术方法__call()来自定义调用不存在方法时的提示信息,同时不会退出程序而将继续向下执行;__autoload()魔术方法用于自动包含满足指定条件的文件。

第七天

使用大括号的方法可以像访问数组那样访问字符串中指定位置的字符,如访问字符串最后一个字符$str{strlen(str)-1}

字符串输出函数echo(), print(), die(), printf(), sprintf()

PHP字符串格式化函数

第八天

正则表达式通过特定规则配合函数实现对字符串的匹配、查找、替换和分割等操作。

与Perl语言兼容的正则表达式处理函数

 

正则表达式常用通用字符类型

在正则表达式模式或部分模式两边添加括号()可将匹配内容存储到临时缓冲区区,依次通过\\1, \\2…来进行调用,若不想保存相关匹配,可在括号内添加非捕获元字符?:, ?=, ?!忽略保存相关匹配。

正则表达式模式匹配优先级

正则表达式模式修正符

正则表达式相关函数:

int preg_match ( string pattern , string subject [, array matches])

int preg_match_all ( string pattern , string subject , array matches [, int flags])

array preg_grep ( string $pattern , array $input)

strstr(), strpos(), strrpos(), substr(), str_replace(),preg_split(),explode(),implode()

mixed preg_replace ( mixed pattern , mixed replacement , mixed subject [, int limit])

第九天

PHP文件属性处理函数 使用clearstatcache()函数可清除被PHP缓存的文件信息。UNIX系统使用正斜线/作为路径分隔符,Windows中默认使用反斜线\作为分隔符,在程序中需要使用\用进行转义,Windows也支持使用正斜线/作为路径分隔符。PHP中内置了常量DIRECTORY_SEPARATOR来表示不同操作系统中的默认文件路径分隔符。与路径相关的函数有basename(),dirname(),pathinfo()。目录相关的函数有opendir(),readdir(),closedir(),rewinddir()。复制、删除非空目录需要自定义函数来进行操作,以下为复制目录的示例代码:

function copyDir($dirSrc, $dirTo){
	if(is_file($dirTo)){
		echo "目标不是目录不能创建!!";
		return;
	}

	if(!file_exists($dirTo)){
		mkdir($dirTo);
	}

	if($dir_handle = @opendir($dirSrc)){
		while($filename = readdir($dir_handle)){
			if($filename != "." && $filename != ".."){
				$subSrcFile = $dirSrc."/".$filename;
				$subToFile = $dirTo."/".$filename;

				if(is_dir($subSrcFile))
					copyDir($subSrcFile, $subToFile);
				if(is_file($subSrcFile))
					copy($subSrcFile, $subToFile);
			}
		}
		closedir($dir_handle);
	}
}

copyDir("phpMyAdmin", "D:/admin");

操作文件的相关函数

fopen(), fclose(), fwrite()/fputs(),file_put_contents(), feof(), ftell(), fseek(), rewind(), flock(), copy(), unlink(), ftruncate(), rename()

读取文件内容的相关函数

PHP读取文件内容函数

文件上传在form表单中通过method=”post”指明发送数据的方法,enctype=”multipart/form-data”来指定表单编码数据方式 ,另外在form表单中可以添加一个类型为hidden的input,name设置为MAX_FILE_UPLOAD,用value来限制所上传文件的大小,当然这个值不能大于php.ini中的upload_max_filesize。PHP为上传文件专门提供了另外两个函数is_uploaded_file()和move_uploaded_file(),处理单个文件上传的示例脚本代码如下:

<?php
	header("content-type:text/html; charset:utf-8");
	error_reporting(E_ALL & ~(E_STRICT | E_NOTICE));
	$allowtype = array("gif", "png", "jpg");
	$size = 1000000;
	$path = "./uploads";

	if($_FILES['myfile']['error']>0){
		echo "上传错误";
		switch($_FILES['myfile']['error']){
			case 1: die('上传文件大小超出了PHP配置文件中的约定值:upload_max_filesize');
			case 2: die('上传文件大小超出了表单中的约定值:MAX_FILE_SIZE');
			case 3: die('文件只被部分上传');
			case 4: die('没有上传任何文件');
			default: die('未知错误');
		}
	}

	$hz = array_pop(explode(".",$_FILES['myfile']['name']));

	if(!in_array($hz,$allowtype)){
		die("这个后缀名{$hz}不是允许上传的文件类型!");
	}

	if($_FILES['myfile']['size']>$size){
		die("超过允许的<b>{$size}</b>字节大小!");
	}

	$filename = date("YmdHis").rand(100,999).".".$hz;

	if(is_uploaded_file($_FILES['myfile']['tmp_name'])){
		if(!move_uploaded_file($_FILES['myfile']['tmp_name'], $path.'/'.$filename)){
			die("问题:不能将文件移动到指定目录。");
		}
	}else{
		die("问题:上传文件{$_FILES['myfile']['name']}不是一个合法文件!");
	}

	echo "文件{$_FILES['myfile']['name']}上传成功,保存在目录{$path}中,大小为{$_FILES['myfile']['size']}字节";

如果上传多个文件,可在表单中设置name=”myfile[]”,$_FILES[‘myfile’][‘name’]二维数组将会变为三维数组$_FILES[‘myfile’][‘name’][n]

第十天 – PHP动态图像处理

PHP中通过使用GD扩展库实现对图像的处理,创建图像主要有4个步骤:创建画布、绘制图像、输出图像和释放资源。

创建画布:imagecreate(), imagecreatetruecolor()前者支持256色,后者支持真彩色,但不能用于GIF文件格式,可通过imagesx()和imagesy()函数来获取图像的大小,如果画布引用句柄不再使用,则需通过imagedestroy()来进行销毁。

设置颜色:通过imagecolorallocate()函数即可设置图像中的颜色,其后三个参数分别代表RGB,可以是0到255的整数,也可以是十六进制的0x00到0xFF,如果使用Imagecreate()创建画布,第一次调用此函数将会作为背景色进行填充。

生成图像:通过imagegif(), imagejpeg(), imagepng(), imagewbmp()函数可分别生成对应格式的图像,在输出前还要通过header()头信息来通知浏览器使用用正确的MIME类型来接收,如header(“Content-Type: image/gif”)。

绘制图像:绘制图像的函数有图形区域填充函数imagefill(), 绘制点和线的函数imagesetpixel()和imageline(), 绘制矩形函数imagerectangle()和imagefilledrectangle(), 绘制多边形函数imagepolygon()和imagefilledpolygon(), 绘制椭圆函数imageellipse(()和imagefilledellipse(), 绘制弧线函数imagearc()和imagefilledarc()。

绘制文字:通过函数imagestring(), imagestringup(), imagechar(), imagecharup(), imagettftext()来实现图像中文字的绘制,后者按坐标从左下角开始绘制,而其它的则是从左上角开始绘制 ,关于中文字符可通过iconv()或mb_convert_encoding()进行转码,但在新版本中似乎无需进行转码即可直接输出。

从现有图片创建画布资源:imagecreatefromjpeg(), imagecreatefrompng(), imagecreatefrompng(), 获取图像宽高和类型使用函数 getimagesize()

图片缩放图片裁剪函数:imagecopyresized(), imagecopyresampled(), 后者处理后质量会更好些。为图片添加水印的函数imagecopy(), 图片旋转函数imagerotate(), 水平翻转代码示例如下,进行垂直翻转时只需对代码进行微调

 function turn_y($filename){
		$back = imagecreatefromjpeg($filename);

		$width = imagesx($back);
		$height = imagesy($back);

		$new = imagecreatetruecolor($width, $height);

		for($x=0; $x<$width; $x++){
			imagecopy($new, $back, $width-$x-1, 0, $x, 0, 1, $height);
		}

		imagejpeg($new, $filename);

		imagedestroy($back);
		imagedestroy($new);
	}

 

第十一天 – MySQL数据库

SQL语言包含四个部分:数据定义语言(DDL)如CREATE, DROP, ALTER, 数据操作语言(DML)如INSERT, UPDATE, DELETE, 数据查询语言(DQL)如SELECT, 数据控制语言(DCL)如GRANT, REVOKE, COMMIT, ROLLBACK。

添加列:ALTER TABLE 表名 ADD 字段名 <建表语句> [FIRST|AFTER 列名]

修改列:ALTER TABLE 表名 CHANGE(MODIFY) 列名 <建表语句>

数据表重命名:ALTER TABLE 旧表名 RENAME AS 新表名

字符集相关:配置文件中character_set_server, collation_server等,客户端查看SHOW VARIABLES LIKE ‘character%’; 创建数据表时可在最后添加类似DEFAULT CHARACTER SET utf8 COLLATE utf8_general_ci的语句来进行设置。

索引相关:主键索引(PRIMARY KEY), 唯一索引(UNIQUE), 常规索引(INDEX|KEY)和全文索引(FULLTEXT)

PDO(PHP Data Object) Window下通过php.ini中extension=php_pdo_mysql.dll进行支持。连接数据库创建PDO对象:

try{
		$dbh = new PDO('mysql:dbname=testdb;host=localhost','user','pwd');
	}catch(PDOException $e){
		echo '数据库连接失败:'.$e->getMessage();
		exit;
	}

常用的一些方法:PDO::exec(), PDo::query(),预处理PDO::prepare(),绑定参数bindParam(),绑定后执行execute(),获取数据fetch()和fetchAll()。

第十二天 – 会话控制

bool setcookie ( string $name [, string $value [, int $expire = 0 [, string $path [, string $domain [, bool $secure = false [, bool $httponly = false ]]]]]] )

PHP setCookie()函数参数说明

会话的声明通过session_start()开启会话和$_SESSION全局数组赋值,这样就会将对应的值保存到php,ini中通过session.save_path指定的路径下,通过session_destroy()可以销毁当前会话,要删除SESSION,可通过$_SESSION=array()将其赋值为空数组,或unset()来删除特定变量,此外要彻底注销一个session,还应将cookie中对应的session id删除,所以删除一个session应有以下几步

        session_start();
	//$_SESSION['username']="Alan";
	$_SESSION = array();
	if(isset($_COOKIE[session_name()])){
		setcookie(session_name(),'',time()-3600,'/');
	}
	session_destroy();

第十三天 – PHP模板引擎Smarty

在http://www.smarty.net/download选择Smarty的对应版本(这里选用3.*.*)然后到Github去进行下载,解压出libs文件夹即可,使用时通过require()语句加载该文件夹中的Smarty.class.php类文件即可以进行使用了。通过$smarty = new Smarty()即可对该类进行实例化。

Smarty类中的成员属性

根据书的建议在libs所在的同一级目录中新建一个Init.inc.php文件来设置上述的成员属性以供其它php脚本文件包含

	define("ROOT",str_replace("\\","/",dirname(__FILE__)).'/'); //指定项目的根目录
	require ROOT.'libs/Smarty.class.php';						//加载Smarty类文件
	$smarty = new Smarty();										//实例化Smarty类

	$smarty->setTemplateDir(ROOT.'templates/')					//设置模板文件存放路径
		   //->addTemplateDir(ROOT.'templatess/')				//添加多个模板目录
		   ->setCompileDir(ROOT.'templates_c')					//设置编译后文件存放路径
		   ->setPluginsDir(ROOT.'plugins/')						//设置插件存放路径
		   ->setCacheDir(ROOT.'cache/')							//设置缓存存放路径
		   ->setConfigDir(ROOT.'configs/');						//设置模板配置文件存放路径

	$smarty->caching = false;									//Smarty缓存开关
	$smarty->cache_lifetime = 60*60*24;							//缓存有效期
	$smarty->left_delimiter = '<{';								//模板语言左起始符
	$smarty->right_delimiter = '}>';							//模板语言右结束符

通过assign()方法赋值,display()方法输出,Smarty中可通过{*comment*}的方式来添加评论并且不会在前端页面输出

PHP与数据库

PHP支持哪些数据库

PHP通过安装相应的扩展来实现数据库操作,现代应用程序的设计离不开数据库的应用,当前主流的数据库有MsSQL,MySQL,Sybase,DB2,Oracle,PostgreSQL,Access等,这些数据库PHP都能够安装扩展来支持,一般情况下常说的LAMP架构指的是:Linux、Apache、MySql、PHP,因此Mysql数据库在PHP中的应用非常广泛,我们会进一步简单地了解Mysql的操作方法。

数据库扩展

PHP中一个数据库可能有一个或者多个扩展,其中既有官方的,也有第三方提供的。像Mysql常用的扩展有原生的mysql库,也可以使用增强版的mysqli扩展,还可以使用PDO进行连接与操作。

不同的扩展提供基本相近的操作方法,不同的是可能具备一些新特性,以及操作性能可能会有所不同。

mysql扩展进行数据库连接的方法:

mysqli扩展:

PDO扩展

 

连接MySQL数据库

PHP要对数据库进行操作,首先要做的是与数据库建立连接,通常我们使用mysql_connect函数进行数据库连接,该函数需要指定数据库的地址,用户名及密码。

PHP连接数据库的方式类似于直接在命令行下通过进行连接,类似:mysql -hlocalhost -ucode1 -p,当连接成功以后,我们需要选择一个操作的数据库,通过mysql_select_db函数来选择数据库。

通常我们会先设置一下当前连接使用的字符编码,一般的我们会使用utf8编码。

通过上面的步骤,我们就与数据库建立了连接,可以进行数据操作了。

执行MySQL查询

在数据库建立连接以后就可以进行查询,采用mysql_query加sql语句的形式向数据库发送查询指令。

对于查询类的语句会返回一个资源句柄(resource),可以通过该资源获取查询结果集中的数据。

默认的,PHP使用最近的数据库连接执行查询,但如果存在多个连接的情况,则可以通过参数指令从那个连接中进行查询。

 

插入新数据到MySQL中

当我们了解了如何使用mysql_query进行数据查询以后,那么类似的,插入数据其实也是通过执行一个sql语句来实现,例如:

通常数据都是存储在变量或者数组中,因此sql语句需要先进行字符串拼接得到。

在mysql中,执行插入语句以后,可以得到自增的主键id,通过PHP的mysql_insert_id函数可以获取该id。

这个id的作用非常大,通常可以用来判断是否插入成功,或者作为关联ID进行其他的数据操作。

取得数据查询结果

通过前面的学习,我们发现PHP操作数据库跟MySql客户端上操作极为相似,先进行连接,然后执行sql语句,再然后获取我们想要的结果集。

PHP有多个函数可以获取数据集中的一行数据,最常用的是mysql_fetch_array,可以通过设定参数来更改行数据的下标,默认的会包含数字索引的下标以及字段名的关联索引下标。

可以通过设定参数MYSQL_NUM只获取数字索引数组,等同于mysql_fetch_row函数,如果设定参数为MYSQL_ASSOC则只获取关联索引数组,等同于mysql_fetch_assoc函数。

如果要获取数据集中的所有数据,我们通过循环来遍历整个结果集。

 

查询分页数据

上面,我们了解到通过循环可以获取一个查询的所有数据,在实际应用中,我们并不希望一次性获取数据表中的所有数据,那样性能会非常的低,因此会使用翻页功能,每页仅显示10条或者20条数据。

通过mysql的limit可以很容易的实现分页,limit m,n表示从m行后取n行数据,在PHP中我们需要构造m与n来实现获取某一页的所有数据。

假定当前页为$page,每页显示$n条数据,那么m为当前页前面所有的数据,既$m = ($page-1) * $n,在知道了翻页原理以后,那么我们很容易通过构造SQL语句在PHP中实现数据翻页。

在上面的例子中,我们使用了$m与$n变量来表示偏移量与每页数据条数,但我们推荐使用更有意义的变量名来表示,比如$pagesize, $start, $offset等,这样更容易理解,有助于团队协作开发。

更新与删除数据

数据的更新与删除相对比较简单,只需要构建好相应的sql语句,然后调用mysql_query执行就能完成相应的更新与删除操作。

同样的删除可以使用类似以下的代码:

对于删除与更新操作,可以通过mysql_affected_rows函数来获取更新过的数据行数,如果数据没有变化,则结果为0。

 

关闭MySQL连接

当数据库操作完成以后,可以使用mysql_close关闭数据库连接,默认的,当PHP执行完毕以后,会自动的关闭数据库连接。

虽然PHP会自动关闭数据库连接,一般情况下已经满足需求,但是在对性能要求比较高的情况下,可以在进行完数据库操作之后尽快关闭数据库连接,以节省资源,提高性能。

在存在多个数据库连接的情况下,可以设定连接资源参数来关闭指定的数据库连接。

PHP异常处理

抛出一个异常

从PHP5开始,PHP支持异常处理,异常处理是面向对象一个重要特性,PHP代码中的异常通过throw抛出,异常抛出之后,后面的代码将不会再被执行。

既然抛出异常会中断程序执行,那么为什么还需要使用异常处理?

异常抛出被用于在遇到未知错误,或者不符合预先设定的条件时,通知客户程序,以便进行其他相关处理,不至于使程序直接报错中断。

当代码中使用了try catch的时候,抛出的异常会在catch中捕获,否则会直接中断。

 

1、基本语法
try{
//可能出现错误或异常的代码
//catch表示捕获,Exception是php已定义好的异常类
} catch(Exception $e){
//对异常处理,方法:
//1、自己处理
//2、不处理,将其再次抛出
}
2、处理处理程序应当包括:
Try – 使用异常的函数应该位于 “try”  代码块内。如果没有触发异常,则代码将照常继续执行。但是如果异常被触发,会抛出一个异常。
Throw – 这里规定如何触发异常。注意:每一个 “throw” 必须对应至少一个 “catch”,当然可以对应多个”catch”
Catch – “catch” 代码块会捕获异常,并创建一个包含异常信息的对象。

上面代码将获得类似这样一个错误:

捕获异常:: 异常提示-数字必须小于等于1

例子解释:

上面的代码抛出了一个异常,并捕获了它:

创建 checkNum() 函数。它检测数字是否大于 1。如果是,则抛出一个异常。
在 “try” 代码块中调用 checkNum() 函数。
checkNum() 函数中的异常被抛出
“catch” 代码块接收到该异常,并创建一个包含异常信息的对象 ($e)。
通过从这个 exception 对象调用 $e->getMessage(),输出来自该异常的错误消息

异常处理类

PHP具有很多异常处理类,其中Exception是所有异常处理的基类。

Exception具有几个基本属性与方法,其中包括了:

message 异常消息内容
code 异常代码
file 抛出异常的文件名
line 抛出异常在该文件的行数

其中常用的方法有:

getTrace 获取异常追踪信息
getTraceAsString 获取异常追踪信息的字符串
getMessage 获取出错信息

如果必要的话,可以通过继承Exception类来建立自定义的异常处理类。

捕获异常信息

在了解了异常处理的基本原理之后,我们可以通过try catch来捕获异常,我们将执行的代码放在try代码块中,一旦其中的代码抛出异常,就能在catch中捕获。

这里我们只是通过案例来了解try catch的机制以及异常捕获的方法,在实际应用中,不会轻易的抛出异常,只有在极端情况或者非常重要的情况下,才会抛出异常,抛出异常,可以保障程序的正确性与安全,避免导致不可预知的bug。

一般的异常处理流程代码为:

 

获取错误发生的所在行

在异常被捕获之后,我们可以通过异常处理对象获取其中的异常信息,前面我们已经了解捕获方式,以及获取基本的错误信息。

在实际应用中,我们通常会获取足够多的异常信息,然后写入到错误日志中。

通过我们需要将报错的文件名、行号、错误信息、异常追踪信息等记录到日志中,以便调试与修复问题。

PHP图像处理

GD库简介

GD指的是Graphic Device,PHP的GD库是用来处理图形的扩展库,通过GD库提供的一系列API,可以对图像进行处理或者直接生成新的图片。

PHP除了能进行文本处理以外,通过GD库,可以对JPG、PNG、GIF、SWF等图片进行处理。GD库常用在图片加水印,验证码生成等方面。

PHP默认已经集成了GD库,只需要在安装的时候开启就行。

绘制线条

要对图形进行操作,首先要新建一个画布,通过imagecreatetruecolor函数可以创建一个真彩色的空白图片:

GD库中对于画笔所用的颜色,需要通过imagecolorallocate函数进行分配,通过参数设定RGB的颜色值来确定画笔的颜色:

然后我们通过调用绘制线段函数imageline进行线条的绘制,通过指定起点跟终点来最终得到线条。

线条绘制好以后,通过header与imagepng进行图像的输出。

最后可以调用imagedestroy释放该图片占用的内存。

通过上面的步骤,可以发现PHP绘制图形非常的简单,但很多时候我们不只是需要输出图片,可能我们还需要得到一个图片文件,可以通过imagepng函数指定文件名将绘制后的图像保存到文件中。

在图像中绘制文字

GD库可以进行多种图形的基本操作,常用的有绘制线条,背景填充,画矩形,绘制文字等。

跟绘制线条类似,首先需要新建一个图片与初始化颜色。

然后使用imagestring函数来进行文字的绘制,这个函数的参数很多:imagestring ( resource $image , int $font , int $x , int $y , string $s , int $col ),可以通过$font来设置字体的大小,x,y设置文字显示的位置,$s是要绘制的文字,$col是文字的颜色。

 

输出图像文件

前面我们已经了解到,通过imagepng可以直接输出图像到浏览器,但是很多时候,我们希望将处理好的图像保存到文件,以便可以多次使用。通过指定路径参数将图像保存到文件中。

使用imagepng可以将图像保存成png格式,如果要保存成其他格式需要使用不同的函数,使用imagejpeg将图片保存成jpeg格式,imagegif将图片保存成gif格式,需要说明的是,imagejpeg会对图片进行压缩,因此还可以设置一个质量参数。

 

生成图像验证码

简单的验证码其实就是在图片中输出了几个字符,通过我们前面章节讲到的imagestring函数就能实现。

但是在处理上,为了使验证码更加的安全,防止其他程序自动识别,因此常常需要对验证码进行一些干扰处理,通常会采用绘制一些噪点,干扰线段,对输出的字符进行倾斜、扭曲等操作。

可以使用imagesetpixel绘制点来实现噪点干扰,但是只绘制一个点的作用不大,因此这里常常会使用循环进行随机绘制。

 

给图片添加水印

给图片添加水印的方法一般有两种,一种是在图片上面加上一个字符串,另一种是在图片上加上一个logo或者其他的图片。

因为这里处理的是已经存在的图片,所以可以直接从已存在的图片建立画布,通过imagecreatefromjpeg可以直接从图片文件创建图像。

创建图像对象以后,我们就可以通过前面的GD函数,绘制字符串到图像上。如果要加的水印是一个logo图片,那么就需要再建立一个图像对象,然后通过GD函数imagecopy将logo的图像复制到源图像中。

当将logo图片复制到原图片上以后,将加水印后的图片输出保存就完成了加水印处理。

PHP日期和时间

取得当前的Unix时间戳

UNIX 时间戳(英文叫做:timestamp)是 PHP 中关于时间与日期的一个很重要的概念,它表示从 1970年1月1日 00:00:00 到当前时间的秒数之和。

PHP提供了内置函数 time() 来取得服务器当前时间的时间戳。那么获取当前的UNIX时间戳就很简单了。

 

取得当前的日期

php内置了date()函数,来取得当前的日期。

函数说明:date(时间戳的格式, 规定时间戳【默认是当前的日期和时间,可选】)

返回值:函数日期和时间

例子:

取得日期的Unix时间戳

UNIX 时间戳(英文叫做:timestamp)是 PHP 中关于时间与日期的一个很重要的概念,它表示从 1970年1月1日 00:00:00 到当前时间的秒数之和。

PHP提供了内置函数strtotime实现功能:获取某个日期的时间戳,或获取某个时间的时间戳。例如:

 

将格式化的日期字符串转换为Unix时间戳

strtotime函数预期接受一个包含美国英语日期格式的字符串并尝试将其解析为 Unix 时间戳。

函数说明:strtotime(要解析的时间字符串, 计算返回值的时间戳【默认是当前的时间,可选】)
返回值:成功则返回时间戳,否则返回 FALSE

比如

格式化格林威治(GMT)标准时间

gmdate 函数能格式化一个GMT的日期和时间,返回的是格林威治标准时(GMT)。

PHP文件处理

读取文件内容

PHP具有丰富的文件操作函数,最简单的读取文件的函数为file_get_contents,可以将整个文件全部读取到一个字符串中。

file_get_contents也可以通过参数控制读取内容的开始点以及长度。

PHP也提供类似于C语言操作文件的方法,使用fopen,fgets,fread等方法,fgets可以从文件指针中读取一行,freads可以读取指定长度的字符串。

 

使用fopen打开的文件,最好使用fclose关闭文件指针,以避免文件句柄被占用。

判断文件是否存在

一般情况下在对文件进行操作的时候需要先判断文件是否存在,PHP中常用来判断文件存在的函数有两个:is_filefile_exists.

如果只是判断文件存在,使用file_exists就行,file_exists不仅可以判断文件是否存在,同时也可以判断目录是否存在,从函数名可以看出,is_file是确切的判断给定的路径是否是一个文件。

更加精确的可以使用is_readableis_writeable在文件是否存在的基础上,判断文件是否可读与可写。

写入内容到文件

与读取文件对应,PHP写文件也具有两种方式,最简单的方式是采用file_put_contents。

上例中,$data参数可以是一个一维数组,当$data是数组的时候,会自动的将数组连接起来,相当于$data=implode('', $data);

同样的,PHP也支持类似C语言风格的操作方式,采用fwrite进行文件写入。

 

取得文件的修改时间

文件有很多元属性,包括:文件的所有者、创建时间、修改时间、最后的访问时间等。

其中最常用的是文件的修改时间,通过文件的修改时间,可以判断文件的时效性,经常用在静态文件或者缓存数据的更新。

 

取得文件的大小

通过filesize函数可以取得文件的大小,文件大小是以字节数表示的。

如果要转换文件大小的单位,可以自己定义函数来实现。

值得注意的是,没法通过简单的函数来取得目录的大小,目录的大小是该目录下所有子目录以及文件大小的总和,因此需要通过递归的方法来循环计算目录的大小。

删除文件

跟Unix系统命令类似,PHP使用unlink函数进行文件删除。


删除文件夹使用rmdir函数,文件夹必须为空,如果不为空或者没有权限则会提示失败。

如果文件夹中存在文件,可以先循环删除目录中的所有文件,然后再删除该目录,循环删除可以使用glob函数遍历所有文件。

PHP会话控制

Cookie简介

Cookie是存储在客户端浏览器中的数据,我们通过Cookie来跟踪与存储用户数据。一般情况下,Cookie通过HTTP headers从服务端返回到客户端。多数web程序都支持Cookie的操作,因为Cookie是存在于HTTP的标头之中,所以必须在其他信息输出以前进行设置,类似于header函数的使用限制。

PHP通过setcookie函数进行Cookie的设置,任何从浏览器发回的Cookie,PHP都会自动的将他存储在$_COOKIE的全局变量之中,因此我们可以通过$_COOKIE[‘key’]的形式来读取某个Cookie值。

PHP中的Cookie具有非常广泛的使用,经常用来存储用户的登录信息,购物车等,且在使用会话Session时通常使用Cookie来存储会话id以识别用户,Cookie具备有效期,当有效期结束之后,Cookie会自动的从客户端删除。同时为了进行安全控制,Cookie还可以设置域跟路径,我们会在稍后详细地进行讲解。

设置cookie

PHP设置Cookie最常用的方法就是使用setcookie函数,setcookie具有7个可选参数,我们常用到的为前5个:

name( Cookie名)可以通过$_COOKIE[‘name’] 进行访问
value(Cookie的值)
expire(过期时间)Unix时间戳格式,默认为0,表示浏览器关闭即失效
path(有效路径)如果路径设置为’/’,则整个网站都有效
domain(有效域)默认整个域名都有效,如果设置了’www.alanhou.org’,则只在www子域中有效

PHP中还有一个设置Cookie的函数setrawcookie,setrawcookie跟setcookie基本一样,唯一的不同就是value值不会自动的进行urlencode,因此在需要的时候要手动的进行urlencode。

因为Cookie是通过HTTP标头进行设置的,所以也可以直接使用header方法进行设置。

 

cookie的删除与过期时间

通过前面的学习,我们了解了设置cookie的函数,但是我们却发现php中没有删除Cookie的函数,在PHP中删除cookie也是采用setcookie函数来实现。

可以看到将cookie的过期时间设置到当前时间之前,则该cookie会自动失效,也就达到了删除cookie的目的。之所以这么设计是因为cookie是通过HTTP的标头来传递的,客户端根据服务端返回的Set-Cookie段来进行cookie的设置,如果删除cookie需要使用新的Del-Cookie来实现,则HTTP头就会变得复杂,实际上仅通过Set-Cookie就可以简单明了的实现Cookie的设置、更新与删除。

了解原理以后,我们也可以直接通过header来删除cookie。

这里用到了gmdate,用来生成格林威治标准时间,以便排除时差的影响。

cookie的有效路径

cookie中的路径用来控制设置的cookie在哪个路径下有效,默认为’/’,在所有路径下都有,当设定了其他路径之后,则只在设定的路径以及子路径下有效,例如:

上面的设置会使test在/path以及子路径/path/abc下都有效,但是在根目录下就读取不到test的cookie值。

一般情况下,大多是使用所有路径的,只有在极少数有特殊需求的时候,会设置路径,这种情况下只在指定的路径中才会传递cookie值,可以节省数据的传输,增强安全性以及提高性能。

当我们设置了有效路径的时候,不在当前路径的时候则看不到当前cookie。

 

session与cookie的异同

cookie将数据存储在客户端,建立起用户与服务器之间的联系,通常可以解决很多问题,但是cookie仍然具有一些局限:

cookie相对不是太安全,容易被盗用导致cookie欺骗
单个cookie的值最大只能存储4k
每次请求都要进行网络传输,占用带宽

session是将用户的会话数据存储在服务端,没有大小限制,通过一个session_id进行用户识别,PHP默认情况下session id是通过cookie来保存的,因此从某种程度上来说,seesion依赖于cookie。但这不是绝对的,session id也可以通过参数来实现,只要能将session id传递到服务端进行识别的机制都可以使用session。

使用session

在PHP中使用session非常简单,先执行session_start方法开启session,然后通过全局变量$_SESSION进行session的读写。

session会自动的对要设置的值进行encode与decode,因此session可以支持任意数据类型,包括数据与对象等。

默认情况下,session是以文件形式存储在服务器上的,因此当一个页面开启了session之后,会独占这个session文件,这样会导致当前用户的其他并发访问无法执行而等待。可以采用缓存或者数据库的形式存储来解决这个问题。

删除与销毁session

删除某个session值可以使用PHP的unset函数,删除后就会从全局变量$_SESSION中去除,无法访问。

如果要删除所有的session,可以使用session_destroy函数销毁当前session,session_destroy会删除所有数据,但是session_id仍然存在。

值得注意的是,session_destroy并不会立即的销毁全局变量$_SESSION中的值,只有当下次再访问的时候,$_SESSION才为空,因此如果需要立即销毁$_SESSION,可以使用unset函数。

如果需要同时销毁cookie中的session_id,通常在用户退出的时候可能会用到,则还需要显式的调用setcookie方法删除session_id的cookie值。

使用session来存储用户的登录信息

session可以用来存储多种类型的数据,因此具有很多的用途,常用来存储用户的登录信息,购物车数据,或者一些临时使用的暂存数据等。

用户在登录成功以后,通常可以将用户的信息存储在session中,一般的会单独的将一些重要的字段单独存储,然后所有的用户信息独立存储。

一般来说,登录信息既可以存储在session中,也可以存储在cookie中,他们之间的差别在于session可以方便的存取多种数据类型,而cookie只支持字符串类型,同时对于一些安全性比较高的数据,cookie需要进行格式化与加密存储,而session存储在服务端则安全性较高。