腾讯PHP面试题
2013-02-28 11:51
302 查看
1. 请对POSIX风格和兼容Perl风格两种正则表达式的主要函数进行类比说明
ereg preg_match
ereg_replace preg_replace
ereg/preg_match 为执行正则匹配,ereg里面是不需要也不能使用分隔符和修饰符的,所以ereg的功能比preg要弱上不少。
关于”.”:点在正则里面一般是除了换行符以外的全部字符,但是在ereg里面的”.”是任意字符,即包括换行符!如果在preg里面希望”.” 能够包括换行符,可以在修饰符中加上”s”
ereg默认使用贪婪模式,并且不能修改,这个给很多替换和匹配带来麻烦。
速度:这个或许是很多人关心的问题,会不会preg功能强大是以速度来换取的?不用担心,preg的速度要远远比ereg快3倍。
3. PHP5中魔术方法函数有哪几个,请举例说明各自的用法
魔术函数
__construct()
实例化对象时被调用,当__construct和以类名为函数名的函数同时存在时,__construct将被调用,另一个不被调用。
__destruct()
当删除一个对象或对象操作终止时被调用。
__call()
对象调用某个方法,若方法存在,则直接调用;若不存在,则会去调用__call函数。
[php]
view plaincopyprint?
class Test { public function __call($Key, $Args) { echo “您要调用的 {$Key} 方法不存在。你传入的参数是:” . print_r($Args, true); } } $t = new Test(); $t->getName(aning,go);
您要调用的 getName 方法不存在。参数是:Array
(
[0] => aning
[1] => go
)
__get()
读取一个对象的属性时,若属性存在,则直接返回属性值;若不存在,则会调用__get函数。如果试图读取一个对象并不存在的属性的时候,PHP就会给出错误信息。如果在类里添加__get方法,并且我们可以用这个函数实现类似java中反射的各种操作。
[php]
view plaincopyprint?
class Test { public function __get($key) { echo $key . ” 不存在”; } } $t = new Test(); echo $t->name;
name 不存在
__set()
设置一个对象的属性时,若属性存在,则直接赋值;若不存在,则会调用__set函数。
[php]
view plaincopyprint?
class Test
{
public function __set($key,$value)
{
echo ‘对’.$key . “附值”.$value;
}
}
$t = new Test();
$t->name = “aninggo”;
就会输出:
对 name 附值 aninggo
__toString()
打印一个对象的时被调用。如echo $obj;或print $obj;
[php]
view plaincopyprint?
class Test { public function __toString() { return “打印 Test”; } } $t = new Test(); echo $t;
运行echo $t;的时候,就会调用$t->__toString();从而输出
打印 Test
__clone()
克隆对象时被调用。如:$t=new Test();$t1=clone $t;
[php]
view plaincopyprint?
class Test { public function __clone() { echo “我被复制了!”; } } $t = new Test(); $t1 = clone $t;
程序输出:
我被复制了
__sleep()
serialize之前被调用。若对象比较大,想删减一点东东再序列化,可考虑一下此函数。
__wakeup()
unserialize时被调用,做些对象的初始化工作。
__isset()
检测一个对象的属性是否存在时被调用。如:isset($c->name)。
__unset()
unset一个对象的属性时被调用。如:unset($c->name)。
__set_state()
调用var_export时,被调用。用__set_state的返回值做为var_export的返回值。
__autoload()
实例化一个对象时,如果对应的类不存在,则该方法被调用。
魔术常量
__LINE__
返回文件中的当前行号。
__FILE__
返回文件的完整路径和文件名。如果用在包含文件中,则返回包含文件名。自 PHP 4.0.2 起,__FILE__ 总是包含一个绝对路径,而在此之前的版本有时会包含一个相对路径。
__FUNCTION__
返回函数名称(PHP 4.3.0 新加)。自 PHP 5 起本常量返回该函数被定义时的名字(区分大小写)。在 PHP 4 中该值总是小写字母的。
__CLASS__
返回类的名称(PHP 4.3.0 新加)。自 PHP 5 起本常量返回该类被定义时的名字(区分大小写)。在 PHP 4 中该值总是小写字母的。
__METHOD__
返回类的方法名(PHP 5.0.0 新加)。返回该方法被定义时的名字(区分大小写)。
4. 请写出让,并说明如何在命令行下运行PHP脚本(写出两种方式)同时向PHP脚本传递参数?
这个比较简单:第一种方法是配置环境变量,windows操作下步骤:我的电脑—属性–高级系统设置–高级–环境变量—找到名为path的变量添加(“;您的PHP文件夹路径”例如我的是:”;D:\xampp\php”)运行命令为
D:\>php d:/1.php
hello
第二种:为
D:\xampp\php>php.exe d:/1.php
hello;
5. PHP的垃圾收集机制是怎样的
PHP作为脚本语言是页面结束即释放变量所占内存的。 当一个 PHP线程结束时,当前占用的所有内存空间都会被销毁,当前程序中所有对象同时被销毁。GC进程一般都跟着每起一个SESSION而开始运行的.gc目的是为了在session文件过期以后自动销毁删除这些文件. 在PHP中,没有任何变量指向这个对象时,这个对象就成为垃圾。PHP会将其在内存中销毁;这是PHP 的GC垃圾处理机制,防止内存溢出。 执行这些函数也可以起到回收作用 __destruct /unset/mysql_close /fclose. php对session有明确的gc处理时间设定
session.gc_maxlifetime 如果说有垃圾,那就是整体的程序在框架使用中,会多次调用同一文件等等造成的非单件模式等。所以在出来的时候,必要的用_once 引用,在声明类的时候使用单件模式。还有简化逻辑等等。而如果妄想让PHP自己本身管理内存,进行垃圾管理。呵呵。好像PHP还办不到,对于析构函数,ANDI在他的书里写的很明白。可有可无,不可置否。而内存管理的东西一般都是桌面程序更多去考虑的。
6.使对象可以像数组一样进行foreach循环,要求属性必须是私有。
(Iterator模式的PHP5实现,写一类实现Iterator接口)
[php]
view plaincopyprint?
<?php class MyClass implements Iterator{ private $vars = array(); private $var1; private $var2; private $var3; function __set($name , $value){ $this->vars[$name] = $value; } function __get($name){ return $this->vars[$name]; } public function rewind() { reset($this->vars); } public function current() { return current($this->vars); } public function key() { return key($this->vars); } public function next() { return next($this->vars); } public function valid() { return $this->current() !== false; } } $it = new MyClass($values); $it->var1 = 1; $it->var2 = 2; $it->var3 = 3; foreach ($it as $a => $b) { print “$a: $b “; } ?>
7.请写一段PHP代码,确保多个进程同时写入同一个文件成功
[php]
view plaincopyprint?
//WRITE
$fp = fopen("test.txt", 'ab'); //From the end
flock($fp, LOCK_EX); //lock the file for waiting...
fwrite($fp, 'Just A Test String.......'); //Start writing...
flock($fp, LOCK_UN); //Release write lock
fclose($fp); //Close the file读操作:
//READ
$fp = fopen("test.txt", 'r');
flock($fp, LOCK_SH);
//Read from the file.......
flock($fp, LOCK_UN);
fclose($fp);
其函数原型为:bool flock ( int handle, int operation [, int &wouldblock] )
PHP 支持以咨询方式(也就是说所有访问程序必须使用同一方式锁定, 否则它不会工作)锁定全部文件的一种轻便方法,
operation 可以是以下值之一:
要取得共享锁定(读取的程序),将 operation 设为 LOCK_SH(PHP 4.0.1 以前的版本设置为 1)。
要取得独占锁定(写入的程序),将 operation 设为 LOCK_EX(PHP 4.0.1 以前的版本中设置为 2)。
要释放锁定(无论共享或独占),将 operation 设为 LOCK_UN(PHP 4.0.1 以前的版本中设置为 3)。
如果不希望 flock() 在锁定时堵塞,则给 operation 加上 LOCK_NB(PHP 4.0.1 以前的版本中设置为 4)。
flock() 允许执行一个简单的可以在任何平台中使用的读取/写入模型(包括大部分的 Unix 派生版和甚至是 Windows)。如果锁定会堵塞的话(EWOULDBLOCK 错误码情况下),可选的第三个参数会被设置为 TRUE。锁定操作也可以被 fclose() 释放(代码执行完毕时也会自动调用)。如果成功则返回 TRUE,失败则返回 FALSE。
来看看 PHP manual 上面经典的实例:
[php]
view plaincopyprint?
<?php
$fp = fopen("/tmp/lock.txt", "w+");
if (flock($fp, LOCK_EX)) { // 进行排它型锁定
fwrite($fp, "Write something here\n");
flock($fp, LOCK_UN); // 释放锁定
} else {
echo "Couldn't lock the file !";
}
fclose($fp);
?>
注意:
在 Windows 下 flock() 将会强制执行。flock() 操作的 handle 必须是一个已经打开的文件指针。
由于 flock() 需要一个文件指针, 因此可能不得不用一个特殊的锁定文件来保护打算通过写模式打开的文件的访问(在 fopen() 函数中加入 "w" 或 "w+")。
flock() 不能用于 NFS 以及其它一些网络文件系统。flock() 不支持旧的文件系统,如 FAT 以及它的派生系统。因此,此环境下总是返回 FALSE(尤其是对 Windows)详细资料查看自己操作系统的文档。
在部分操作系统中 flock() 以进程级实现。当用一个多线程服务器 API(比如 ISAPI)时,可能不可以依靠 flock() 来保护文件,因为运行于同一服务器实例中其它并行线程的 PHP 脚本可以对该文件进行处理。
8.用PHP实现一个双向队列
[php]
view plaincopyprint?
class DoubleEndedQueue{ protected $elements; public function __construct(){ $this->elements = array(); } public function push($element){ array_unshift($this->elements , $element); } public function pop(){ return array_shift($this->element); } public function inject($element){ $this->elements[] = $element; } public function eject(){ array_pop($this->elements); } } ?>
9. 使用正则表达式提取一段标识语言(html或xml)代码段中指定标签的指定属性值(需考虑属性值对不规则的情况,如大小写不敏感,属性名值与等号间有空格等)。此处假设需提取test标签的attr属性值,请自行构建包含该标签的串
$str = '‘;
if(preg_match(‘//i’ , $str , $matches)){
echo $matches[1];
}
?>
10.请使用socket相关函数(非curl)实现如下功能:构造一个post请求,发送到指定http server的指定端口的指定请求路径(如http://www.example.com:8080/test)。请求中包含以下变量:
用户名(username):温柔一刀
密码(pwd):&123=321&321=123&
个人简介(intro):Hello world!
且该http server需要以下cookie来进行简单的用户动作跟踪:
cur_query:you&me
last_tm:…(上次请求的unix时间戳,定为当前请求时间前10分钟)
cur_tm:…(当前请求的unix时间戳)
设置超时为10秒,发出请求后,将http server的响应内容输出。
[php]
view plaincopyprint?
<?php
function encode($data, $sep = '&')
{
while (list($k,$v) = each($data))
{
$encoded .= ($encoded ? "$sep" : "");
$encoded .= rawurlencode($k)."=".rawurlencode($v);
}
Return $encoded;
}
function post($url, $data, $cookie){
$url = parse_url($url);
$post = encode($data, '&');
$cookie = encode($cookie, ';');
$fp = fsockopen($url['host'], $url['port'] ? $url['port'] : 80, $errno, $errstr, 10);
if (!$fp)
return "Failed to open socket to $url[host]";
fputs($fp, sprintf("POST %s%s%s HTTP/1.0n", $url['path'], $url['query'] ? "?" : "", $url['query']));
fputs($fp, "Host: $url[host]n");
fputs($fp, "Content-type: application/x-www-form-urlencodedn");
fputs($fp, "Content-length: " . strlen($post) . "n");
fputs($fp, "Cookie: $cookie");
fputs($fp, "Connection: closenn");
fputs($fp, "$post n");
while (!feof($fp)) {
echo fgets($fp, 128);
}
fclose($fp);
}
$url = 'http://www.example.com:8080/test';
$post = array(
'username'=> '温柔一刀',
'pwd' => '&123=321&321=123&',
'intro' => 'Hello world!'
);
$cookie = array(
'cur_query' => 'you&me',
'last_tm' => time() – 600,
'cur_tm '=> time()
);
Post($url, $post, $cookie);
$host = 'www.test.com';
$port = 80;
$path = '/mytest.php';
$timeout = 10;
$fp = fsockopen($host , $port , $errno , $errstr , $timeout);
if(!$fp){
die('scoket open failed');
}
$cookie_header = 'Cookie: cur_query='.urlencode('you&me').'; last_tm='.urlencode(time() - 10 * 60).'; cur_tm='.urlencode(time())."\r\n";
$body = "username=".urlencode('温柔一刀').'&pwd='.urlencode('&123=321&321=123&').'&intro='.urlencode('Hello world!');
$header = "POST $path HTTP/1.0\r\n";
$header .= "User-Agent: User-Agent: Mozilla/4.0 (+http://labs.chinamobile.com; compatible; MSIE 7.0; Windows NT 5.1; No cache)\r\n";
$header .= "Host: $host\r\n";
$header .= "Accept: image/gif, image/x-xbitmap, image/jpeg, */*\r\n";
$header .= $cookie_header;
$header .= "Content-Type: application/x-www-form-urlencoded\r\n";
$header .= "Content-Length: ".strlen($body)."\r\n\r\n";
$str = $header.$body;
fwrite($fp , $str , strlen($str));
$result = '';
$i = 0;
while($currentHeader = fgets($fp)){
if($currentHeader == "\r\n"){
break;
}
}
$data = '';
do{
$block = fread($fp , 5000000);
if(strlen($block) == 0){
break;
}
$data .= $block;
}while(true);
fclose($fp);
echo $data;
?>
11.你用什么方法检查PHP脚本的执行效率(通常是脚本执行时间)和数据库SQL的效率(通常是数据库Query时间),并定位和分析脚本执行和数据库查询的瓶颈所在?
脚本执行效率一般用zend profile,用declare来定位分析脚本的执行效率。
SQL一般用Mysql的slow_query_log和explain来定位和分析SQL的效率
1.脚本执行时间,启用xdebug,使用WinCacheGrind分析。
2.数据库查询,mysql使用EXPLAIN分析查询,启用slow query log记录慢查询。
PHP LAMP Engineer Test Paper
Question 1
What does print out?
A) 3
B) False
C) Null
D) 1
E) 0
Question 2
Which of the following snippets prints a representation of 42 with two decimal places?
A) printf(“%.2dn”, 42);
B) printf(“%1.2fn”, 42);
C) printf(“%1.2un”, 42);
ereg preg_match
ereg_replace preg_replace
ereg/preg_match 为执行正则匹配,ereg里面是不需要也不能使用分隔符和修饰符的,所以ereg的功能比preg要弱上不少。
关于”.”:点在正则里面一般是除了换行符以外的全部字符,但是在ereg里面的”.”是任意字符,即包括换行符!如果在preg里面希望”.” 能够包括换行符,可以在修饰符中加上”s”
ereg默认使用贪婪模式,并且不能修改,这个给很多替换和匹配带来麻烦。
速度:这个或许是很多人关心的问题,会不会preg功能强大是以速度来换取的?不用担心,preg的速度要远远比ereg快3倍。
3. PHP5中魔术方法函数有哪几个,请举例说明各自的用法
魔术函数
__construct()
实例化对象时被调用,当__construct和以类名为函数名的函数同时存在时,__construct将被调用,另一个不被调用。
__destruct()
当删除一个对象或对象操作终止时被调用。
__call()
对象调用某个方法,若方法存在,则直接调用;若不存在,则会去调用__call函数。
[php]
view plaincopyprint?
class Test { public function __call($Key, $Args) { echo “您要调用的 {$Key} 方法不存在。你传入的参数是:” . print_r($Args, true); } } $t = new Test(); $t->getName(aning,go);
class Test { public function __call($Key, $Args) { echo “您要调用的 {$Key} 方法不存在。你传入的参数是:” . print_r($Args, true); } } $t = new Test(); $t->getName(aning,go);程序将会输出:
您要调用的 getName 方法不存在。参数是:Array
(
[0] => aning
[1] => go
)
__get()
读取一个对象的属性时,若属性存在,则直接返回属性值;若不存在,则会调用__get函数。如果试图读取一个对象并不存在的属性的时候,PHP就会给出错误信息。如果在类里添加__get方法,并且我们可以用这个函数实现类似java中反射的各种操作。
[php]
view plaincopyprint?
class Test { public function __get($key) { echo $key . ” 不存在”; } } $t = new Test(); echo $t->name;
class Test { public function __get($key) { echo $key . ” 不存在”; } } $t = new Test(); echo $t->name;就会输出:
name 不存在
__set()
设置一个对象的属性时,若属性存在,则直接赋值;若不存在,则会调用__set函数。
[php]
view plaincopyprint?
class Test
{
public function __set($key,$value)
{
echo ‘对’.$key . “附值”.$value;
}
}
$t = new Test();
$t->name = “aninggo”;
class Test { public function __set($key,$value) { echo ‘对’.$key . “附值”.$value; } } $t = new Test(); $t->name = “aninggo”;
就会输出:
对 name 附值 aninggo
__toString()
打印一个对象的时被调用。如echo $obj;或print $obj;
[php]
view plaincopyprint?
class Test { public function __toString() { return “打印 Test”; } } $t = new Test(); echo $t;
class Test { public function __toString() { return “打印 Test”; } } $t = new Test(); echo $t;
运行echo $t;的时候,就会调用$t->__toString();从而输出
打印 Test
__clone()
克隆对象时被调用。如:$t=new Test();$t1=clone $t;
[php]
view plaincopyprint?
class Test { public function __clone() { echo “我被复制了!”; } } $t = new Test(); $t1 = clone $t;
class Test { public function __clone() { echo “我被复制了!”; } } $t = new Test(); $t1 = clone $t;
程序输出:
我被复制了
__sleep()
serialize之前被调用。若对象比较大,想删减一点东东再序列化,可考虑一下此函数。
__wakeup()
unserialize时被调用,做些对象的初始化工作。
__isset()
检测一个对象的属性是否存在时被调用。如:isset($c->name)。
__unset()
unset一个对象的属性时被调用。如:unset($c->name)。
__set_state()
调用var_export时,被调用。用__set_state的返回值做为var_export的返回值。
__autoload()
实例化一个对象时,如果对应的类不存在,则该方法被调用。
魔术常量
__LINE__
返回文件中的当前行号。
__FILE__
返回文件的完整路径和文件名。如果用在包含文件中,则返回包含文件名。自 PHP 4.0.2 起,__FILE__ 总是包含一个绝对路径,而在此之前的版本有时会包含一个相对路径。
__FUNCTION__
返回函数名称(PHP 4.3.0 新加)。自 PHP 5 起本常量返回该函数被定义时的名字(区分大小写)。在 PHP 4 中该值总是小写字母的。
__CLASS__
返回类的名称(PHP 4.3.0 新加)。自 PHP 5 起本常量返回该类被定义时的名字(区分大小写)。在 PHP 4 中该值总是小写字母的。
__METHOD__
返回类的方法名(PHP 5.0.0 新加)。返回该方法被定义时的名字(区分大小写)。
4. 请写出让,并说明如何在命令行下运行PHP脚本(写出两种方式)同时向PHP脚本传递参数?
这个比较简单:第一种方法是配置环境变量,windows操作下步骤:我的电脑—属性–高级系统设置–高级–环境变量—找到名为path的变量添加(“;您的PHP文件夹路径”例如我的是:”;D:\xampp\php”)运行命令为
D:\>php d:/1.php
hello
第二种:为
D:\xampp\php>php.exe d:/1.php
hello;
5. PHP的垃圾收集机制是怎样的
PHP作为脚本语言是页面结束即释放变量所占内存的。 当一个 PHP线程结束时,当前占用的所有内存空间都会被销毁,当前程序中所有对象同时被销毁。GC进程一般都跟着每起一个SESSION而开始运行的.gc目的是为了在session文件过期以后自动销毁删除这些文件. 在PHP中,没有任何变量指向这个对象时,这个对象就成为垃圾。PHP会将其在内存中销毁;这是PHP 的GC垃圾处理机制,防止内存溢出。 执行这些函数也可以起到回收作用 __destruct /unset/mysql_close /fclose. php对session有明确的gc处理时间设定
session.gc_maxlifetime 如果说有垃圾,那就是整体的程序在框架使用中,会多次调用同一文件等等造成的非单件模式等。所以在出来的时候,必要的用_once 引用,在声明类的时候使用单件模式。还有简化逻辑等等。而如果妄想让PHP自己本身管理内存,进行垃圾管理。呵呵。好像PHP还办不到,对于析构函数,ANDI在他的书里写的很明白。可有可无,不可置否。而内存管理的东西一般都是桌面程序更多去考虑的。
6.使对象可以像数组一样进行foreach循环,要求属性必须是私有。
(Iterator模式的PHP5实现,写一类实现Iterator接口)
[php]
view plaincopyprint?
<?php class MyClass implements Iterator{ private $vars = array(); private $var1; private $var2; private $var3; function __set($name , $value){ $this->vars[$name] = $value; } function __get($name){ return $this->vars[$name]; } public function rewind() { reset($this->vars); } public function current() { return current($this->vars); } public function key() { return key($this->vars); } public function next() { return next($this->vars); } public function valid() { return $this->current() !== false; } } $it = new MyClass($values); $it->var1 = 1; $it->var2 = 2; $it->var3 = 3; foreach ($it as $a => $b) { print “$a: $b “; } ?>
<?php class MyClass implements Iterator{ private $vars = array(); private $var1; private $var2; private $var3; function __set($name , $value){ $this->vars[$name] = $value; } function __get($name){ return $this->vars[$name]; } public function rewind() { reset($this->vars); } public function current() { return current($this->vars); } public function key() { return key($this->vars); } public function next() { return next($this->vars); } public function valid() { return $this->current() !== false; } } $it = new MyClass($values); $it->var1 = 1; $it->var2 = 2; $it->var3 = 3; foreach ($it as $a => $b) { print “$a: $b “; } ?>
7.请写一段PHP代码,确保多个进程同时写入同一个文件成功
[php]
view plaincopyprint?
//WRITE
$fp = fopen("test.txt", 'ab'); //From the end
flock($fp, LOCK_EX); //lock the file for waiting...
fwrite($fp, 'Just A Test String.......'); //Start writing...
flock($fp, LOCK_UN); //Release write lock
fclose($fp); //Close the file读操作:
//READ
$fp = fopen("test.txt", 'r');
flock($fp, LOCK_SH);
//Read from the file.......
flock($fp, LOCK_UN);
fclose($fp);
//WRITE $fp = fopen("test.txt", 'ab'); //From the end flock($fp, LOCK_EX); //lock the file for waiting... fwrite($fp, 'Just A Test String.......'); //Start writing... flock($fp, LOCK_UN); //Release write lock fclose($fp); //Close the file读操作: //READ $fp = fopen("test.txt", 'r'); flock($fp, LOCK_SH); //Read from the file....... flock($fp, LOCK_UN); fclose($fp);flock -- 轻便的咨询文件锁定
其函数原型为:bool flock ( int handle, int operation [, int &wouldblock] )
PHP 支持以咨询方式(也就是说所有访问程序必须使用同一方式锁定, 否则它不会工作)锁定全部文件的一种轻便方法,
operation 可以是以下值之一:
要取得共享锁定(读取的程序),将 operation 设为 LOCK_SH(PHP 4.0.1 以前的版本设置为 1)。
要取得独占锁定(写入的程序),将 operation 设为 LOCK_EX(PHP 4.0.1 以前的版本中设置为 2)。
要释放锁定(无论共享或独占),将 operation 设为 LOCK_UN(PHP 4.0.1 以前的版本中设置为 3)。
如果不希望 flock() 在锁定时堵塞,则给 operation 加上 LOCK_NB(PHP 4.0.1 以前的版本中设置为 4)。
flock() 允许执行一个简单的可以在任何平台中使用的读取/写入模型(包括大部分的 Unix 派生版和甚至是 Windows)。如果锁定会堵塞的话(EWOULDBLOCK 错误码情况下),可选的第三个参数会被设置为 TRUE。锁定操作也可以被 fclose() 释放(代码执行完毕时也会自动调用)。如果成功则返回 TRUE,失败则返回 FALSE。
来看看 PHP manual 上面经典的实例:
[php]
view plaincopyprint?
<?php
$fp = fopen("/tmp/lock.txt", "w+");
if (flock($fp, LOCK_EX)) { // 进行排它型锁定
fwrite($fp, "Write something here\n");
flock($fp, LOCK_UN); // 释放锁定
} else {
echo "Couldn't lock the file !";
}
fclose($fp);
?>
<?php $fp = fopen("/tmp/lock.txt", "w+"); if (flock($fp, LOCK_EX)) { // 进行排它型锁定 fwrite($fp, "Write something here\n"); flock($fp, LOCK_UN); // 释放锁定 } else { echo "Couldn't lock the file !"; } fclose($fp); ?>
注意:
在 Windows 下 flock() 将会强制执行。flock() 操作的 handle 必须是一个已经打开的文件指针。
由于 flock() 需要一个文件指针, 因此可能不得不用一个特殊的锁定文件来保护打算通过写模式打开的文件的访问(在 fopen() 函数中加入 "w" 或 "w+")。
flock() 不能用于 NFS 以及其它一些网络文件系统。flock() 不支持旧的文件系统,如 FAT 以及它的派生系统。因此,此环境下总是返回 FALSE(尤其是对 Windows)详细资料查看自己操作系统的文档。
在部分操作系统中 flock() 以进程级实现。当用一个多线程服务器 API(比如 ISAPI)时,可能不可以依靠 flock() 来保护文件,因为运行于同一服务器实例中其它并行线程的 PHP 脚本可以对该文件进行处理。
8.用PHP实现一个双向队列
[php]
view plaincopyprint?
class DoubleEndedQueue{ protected $elements; public function __construct(){ $this->elements = array(); } public function push($element){ array_unshift($this->elements , $element); } public function pop(){ return array_shift($this->element); } public function inject($element){ $this->elements[] = $element; } public function eject(){ array_pop($this->elements); } } ?>
class DoubleEndedQueue{ protected $elements; public function __construct(){ $this->elements = array(); } public function push($element){ array_unshift($this->elements , $element); } public function pop(){ return array_shift($this->element); } public function inject($element){ $this->elements[] = $element; } public function eject(){ array_pop($this->elements); } } ?>
9. 使用正则表达式提取一段标识语言(html或xml)代码段中指定标签的指定属性值(需考虑属性值对不规则的情况,如大小写不敏感,属性名值与等号间有空格等)。此处假设需提取test标签的attr属性值,请自行构建包含该标签的串
$str = '‘;
if(preg_match(‘//i’ , $str , $matches)){
echo $matches[1];
}
?>
10.请使用socket相关函数(非curl)实现如下功能:构造一个post请求,发送到指定http server的指定端口的指定请求路径(如http://www.example.com:8080/test)。请求中包含以下变量:
用户名(username):温柔一刀
密码(pwd):&123=321&321=123&
个人简介(intro):Hello world!
且该http server需要以下cookie来进行简单的用户动作跟踪:
cur_query:you&me
last_tm:…(上次请求的unix时间戳,定为当前请求时间前10分钟)
cur_tm:…(当前请求的unix时间戳)
设置超时为10秒,发出请求后,将http server的响应内容输出。
[php]
view plaincopyprint?
<?php
function encode($data, $sep = '&')
{
while (list($k,$v) = each($data))
{
$encoded .= ($encoded ? "$sep" : "");
$encoded .= rawurlencode($k)."=".rawurlencode($v);
}
Return $encoded;
}
function post($url, $data, $cookie){
$url = parse_url($url);
$post = encode($data, '&');
$cookie = encode($cookie, ';');
$fp = fsockopen($url['host'], $url['port'] ? $url['port'] : 80, $errno, $errstr, 10);
if (!$fp)
return "Failed to open socket to $url[host]";
fputs($fp, sprintf("POST %s%s%s HTTP/1.0n", $url['path'], $url['query'] ? "?" : "", $url['query']));
fputs($fp, "Host: $url[host]n");
fputs($fp, "Content-type: application/x-www-form-urlencodedn");
fputs($fp, "Content-length: " . strlen($post) . "n");
fputs($fp, "Cookie: $cookie");
fputs($fp, "Connection: closenn");
fputs($fp, "$post n");
while (!feof($fp)) {
echo fgets($fp, 128);
}
fclose($fp);
}
$url = 'http://www.example.com:8080/test';
$post = array(
'username'=> '温柔一刀',
'pwd' => '&123=321&321=123&',
'intro' => 'Hello world!'
);
$cookie = array(
'cur_query' => 'you&me',
'last_tm' => time() – 600,
'cur_tm '=> time()
);
Post($url, $post, $cookie);
$host = 'www.test.com';
$port = 80;
$path = '/mytest.php';
$timeout = 10;
$fp = fsockopen($host , $port , $errno , $errstr , $timeout);
if(!$fp){
die('scoket open failed');
}
$cookie_header = 'Cookie: cur_query='.urlencode('you&me').'; last_tm='.urlencode(time() - 10 * 60).'; cur_tm='.urlencode(time())."\r\n";
$body = "username=".urlencode('温柔一刀').'&pwd='.urlencode('&123=321&321=123&').'&intro='.urlencode('Hello world!');
$header = "POST $path HTTP/1.0\r\n";
$header .= "User-Agent: User-Agent: Mozilla/4.0 (+http://labs.chinamobile.com; compatible; MSIE 7.0; Windows NT 5.1; No cache)\r\n";
$header .= "Host: $host\r\n";
$header .= "Accept: image/gif, image/x-xbitmap, image/jpeg, */*\r\n";
$header .= $cookie_header;
$header .= "Content-Type: application/x-www-form-urlencoded\r\n";
$header .= "Content-Length: ".strlen($body)."\r\n\r\n";
$str = $header.$body;
fwrite($fp , $str , strlen($str));
$result = '';
$i = 0;
while($currentHeader = fgets($fp)){
if($currentHeader == "\r\n"){
break;
}
}
$data = '';
do{
$block = fread($fp , 5000000);
if(strlen($block) == 0){
break;
}
$data .= $block;
}while(true);
fclose($fp);
echo $data;
?>
<?php function encode($data, $sep = '&') { while (list($k,$v) = each($data)) { $encoded .= ($encoded ? "$sep" : ""); $encoded .= rawurlencode($k)."=".rawurlencode($v); } Return $encoded; } function post($url, $data, $cookie){ $url = parse_url($url); $post = encode($data, '&'); $cookie = encode($cookie, ';'); $fp = fsockopen($url['host'], $url['port'] ? $url['port'] : 80, $errno, $errstr, 10); if (!$fp) return "Failed to open socket to $url[host]"; fputs($fp, sprintf("POST %s%s%s HTTP/1.0n", $url['path'], $url['query'] ? "?" : "", $url['query'])); fputs($fp, "Host: $url[host]n"); fputs($fp, "Content-type: application/x-www-form-urlencodedn"); fputs($fp, "Content-length: " . strlen($post) . "n"); fputs($fp, "Cookie: $cookie"); fputs($fp, "Connection: closenn"); fputs($fp, "$post n"); while (!feof($fp)) { echo fgets($fp, 128); } fclose($fp); } $url = 'http://www.example.com:8080/test'; $post = array( 'username'=> '温柔一刀', 'pwd' => '&123=321&321=123&', 'intro' => 'Hello world!' ); $cookie = array( 'cur_query' => 'you&me', 'last_tm' => time() – 600, 'cur_tm '=> time() ); Post($url, $post, $cookie); $host = 'www.test.com'; $port = 80; $path = '/mytest.php'; $timeout = 10; $fp = fsockopen($host , $port , $errno , $errstr , $timeout); if(!$fp){ die('scoket open failed'); } $cookie_header = 'Cookie: cur_query='.urlencode('you&me').'; last_tm='.urlencode(time() - 10 * 60).'; cur_tm='.urlencode(time())."\r\n"; $body = "username=".urlencode('温柔一刀').'&pwd='.urlencode('&123=321&321=123&').'&intro='.urlencode('Hello world!'); $header = "POST $path HTTP/1.0\r\n"; $header .= "User-Agent: User-Agent: Mozilla/4.0 (+http://labs.chinamobile.com; compatible; MSIE 7.0; Windows NT 5.1; No cache)\r\n"; $header .= "Host: $host\r\n"; $header .= "Accept: image/gif, image/x-xbitmap, image/jpeg, */*\r\n"; $header .= $cookie_header; $header .= "Content-Type: application/x-www-form-urlencoded\r\n"; $header .= "Content-Length: ".strlen($body)."\r\n\r\n"; $str = $header.$body; fwrite($fp , $str , strlen($str)); $result = ''; $i = 0; while($currentHeader = fgets($fp)){ if($currentHeader == "\r\n"){ break; } } $data = ''; do{ $block = fread($fp , 5000000); if(strlen($block) == 0){ break; } $data .= $block; }while(true); fclose($fp); echo $data; ?>
11.你用什么方法检查PHP脚本的执行效率(通常是脚本执行时间)和数据库SQL的效率(通常是数据库Query时间),并定位和分析脚本执行和数据库查询的瓶颈所在?
脚本执行效率一般用zend profile,用declare来定位分析脚本的执行效率。
SQL一般用Mysql的slow_query_log和explain来定位和分析SQL的效率
1.脚本执行时间,启用xdebug,使用WinCacheGrind分析。
2.数据库查询,mysql使用EXPLAIN分析查询,启用slow query log记录慢查询。
PHP LAMP Engineer Test Paper
Question 1
What does print out?
A) 3
B) False
C) Null
D) 1
E) 0
Question 2
Which of the following snippets prints a representation of 42 with two decimal places?
A) printf(“%.2dn”, 42);
B) printf(“%1.2fn”, 42);
C) printf(“%1.2un”, 42);