分类目录归档:PHP

PHP源码,PHP扩展,PHP程序

一些sina面试题目的解答

一些sina面试题目的解答
在phpchina上看到了这些题目,看完解答后,很纠结!

1. echo count(“abc”); 输出什么?
答案:出1
解释:在PHP的源码中可以看到,仅对IS_NULL,IS_ARRAY,IS_OBJECT有特殊处理,其它所有的类型都返回1(RETURN_LONG(1);)

2. 用PHP写出显示客户端IP与服务器IP的代码
答案:
“SERVER_ADDR” 当前运行脚本所在的服务器的 IP 地址。
“REMOTE_ADDR” 正在浏览当前页面用户的 IP 地址。

3. error_reporting(2047)什么作用?
答案:error_reporting(E_ALL)
显示所有PHP错误和警告

4. echo,print()和print_r()有什么区别?
答案:echo, print是语言结构,并不是一个真正的函数,print_r是函数打印变量信息
解释:print() is not actually a real function (it is a language construct) so you are not required to use parentheses with its argument list.
这个问题看别人的答案后最纠结

5. 打开php.ini中的Safe_mode,会影响哪些函数?至少说出6个。
1:用户输入输出函数(fopen() file()require(),只能用于调用这些函数有相同脚本的拥有者)
2:创建新文件(限制用户只在该用户拥有目录下创建文件)
3:用户调用popen() systen()exec()等脚本,只有脚本处在safe_mode_exec_dir配置指令指定的目 录中才可能
4:加强HTTP认证,认证脚本拥有者的UID的划入认证领域范围内,此外启用安全模式下,不会设置PHP_AUTH
5:mysql服务器所用的用户名必须与调用mysql_connect()的文件的拥有者用户名相同
6:受影响的函数变量以及配置命令达到40个

6. 写个函数来解决多线程同时读写一个文件的问题。
答案:锁

1
2
3
4
5
6
7
8
9
10
11
12
13
/**
 * 文件写入函数
 * @param     string    $data    需要写入文件的数据
 * @param    string    $filename    文件名
 * @param    string    $type    文件访问类型
 */
function write_file($data, $filename, $type='a') {
    $fp = @fopen($filename, $type);
    flock($fp, LOCK_EX) ;
    fwrite($fp, $data);
    flock($fp, LOCK_UN);
    fclose($fp);
}

7. 请写一个函数验证电子邮件的格式是否正确(要求使用正则)

1
2
3
4
5
6
7
8
/**
 * 验证是否为字符串
 * @param string $email 需要验证的字符串
 * @return bool  返回0或1
 */
function isEmail($email) {
    return preg_match("/^\w+([-+.]\w+)*@\w+([-.]\w+)*\.[a-z]{1,4}$/", $email);
}

8. 考SQL语句的题,题太长了,实在不好回忆了。

9. MySQL数据库,一天一万条以上的增量,怎么优化?

10. 写出一种排序算法(要写出代码),并说出优化它的方法。
快速排序(Quicksort)是对冒泡排序的一种改进。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
function qsort(&$array, $low, $high) {
    $i = $low;
    $j = $high;
    $x = $array[$low];
    while ($i < $j) {
 
        while($i < $j && $array[$j] >= $x) {
            $j--;
        }
        $array[$i] = $array[$j];
 
        while ($i < $j && $array[$i] <= $x) {
            $i++;
        }
        $array[$j] = $array[$i];
    }
 
    $array[$i] = $x;
 
    if ($low < $i - 1) {
        qsort($array, $low, $i - 1);
    }
 
    if ($i + 1 < $high) {
        qsort($array, $i + 1, $high);
    }
}
 
$array = array(3, 2, 4, 1, 4, 0, 11, 333, 444, 22, 111, 22, 2);
qsort($array, 0, count($array) - 1);
print_r($array);

11. 写个函数用来对二维数组排序。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
/*
 *  对二维数组进行排序
 *  @param  $array
 *  @param  $keyid  排序的键值
 *  @param  $order  排序方式 'asc':升序 'desc':降序
 *  @param  $type   键值类型 'number':数字 'string':字符串
 */
function sort_array($array, $keyid, $order='asc', $type='number') {
    if(is_array($array)) {
        foreach($array as $val) {
            $order_arr[] = $val[$keyid];
        }
 
        $order = ($order == 'asc') ? SORT_ASC: SORT_DESC;
        $type  = ($type == 'number') ? SORT_NUMERIC: SORT_STRING;
 
        array_multisort($order_arr, $order, $type, $array);
    }
}

12. 写5个不同的自己的函数,来截取一个全路径的文件的扩展名,允许封装php库中已有的函数。
写了5种

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
/**
 * 取文件的后缀,通过字符串截取
 * @param string $filename  文件名
 * @return string   文件后缀
 */
function fileext($filename) {
    return strtolower(trim(substr(strrchr($filename, '.'), 1, 10)));
}
 
/**
 *正则
 */
function fileext2($filename) {
    preg_match_all("/^.*\.([^.]+)$/", $filename, $matches);
    return strtolower(trim(substr($matches[1][0], 0, 10)));
}
 
function fileext3($filename) {
    $filename = strtolower(trim(basename($filename)));
    return substr($filename, strrpos($filename, '.') + 1);
}
/**
 * 内置函数
 */
function fileext4($filename) {
    $pathinfo = pathinfo($filename);
    return strtolower($pathinfo['extension']);
}
/**
 * 数组
 */
function fileext5($filename) {
    $arr = explode('.', $filename);
    return strtolower(array_pop($arr));
}
$filename = "/usr/bin/file.txt";
echo fileext($filename);
echo fileext2($filename);
echo fileext3($filename);
echo fileext4($filename);
echo fileext5($filename);
die();

13. 一群猴子排成一圈,按1,2,…,n依次编号。然后从第1只开始数,数到第m只,把它踢出圈,从它后面再开始数,再数到第m只,在把它踢出去…,如此不停的进行下去,直到最后只剩下一只猴子为止,那只猴子就叫做大王。要求编程模拟此过程,输入m、n, 输出最后那个大王的编号。
入门的题目,学C语言的时候都有做过的,貌似计算机三级考试中也有此题目

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
 
/**
 * 约瑟夫出圈问题
 * 模拟双向链表实现,没有考虑时间复杂度
 * @param int $m
 * @param int $n
 * @return 
 */
function josegh($m, $n) {
    if ($m < 1 || $n < 1) {
        return FALSE;
    }
 
    $link = array();
    /* 初始化数组值 */
    for ($i = 1; $i <= $n; $i++) {
        $link[$i]['value'] = $i;
    }
 
    /* 初始化下一元素 */
    $link[$n]['next'] = 1;
    for ($i = 1; $i < $n; $i++) {
        $link[$i]['next'] = $i + 1;
    }
 
    /* 初始化上一元素 */
    $link[1]['pre'] = $n;
    for ($i = 2; $i <= $n; $i++) {
        $link[$i]['pre'] = $i - 1;
    }
 
    $rest = $n;
    $index = 1;
    $count = 0;
    while ($rest > 1) {
        $count++;
 
        if ($count % $m == 0) {
            $pop_index = $index;
            $link[$link[$index]['pre']]['next'] = $link[$index]['next'];
            $link[$link[$index]['next']]['pre'] = $link[$index]['pre'];
            $index = $link[$index]['next'];
            $rest--;
            unset($link[$pop_index]);
 
            $count = 0;
        }else{
            $index = $link[$index]['next'];
        }
    }
 
    $rs = array_pop($link);
    return $rs['value'];
}
 
echo josegh(2, 3);

PHP 源码阅读笔记二十三 :urlencode函数

PHP 源码阅读笔记二十三 :urlencode函数
有一段时间没有看PHP的源码了,最近一直在看以前买的书,有一些书已经看过一遍了,但是事隔一年又有不同的感受

urlencode函数在开发的过程中经常有遇到,它作用于字符串编码并将其用于 URL 的请求部分
urlencode函数的作用是编码 URL 字符串
string urlencode ( string str )
返回字符串,此字符串中除了 -_. 之外的所有非字母数字字符都将被替换成百分号(%)后跟两位十六进制数,空格则编码为加号(+)。此编码与 WWW 表单 POST 数据的编码方式是一样的,同时与 application/x-www-form-urlencoded 的媒体类型编码方式一样。由于历史原因,此编码在将空格编码为加号(+)方面与 RFC1738 编码(参见 rawurlencode())不同。
在standard/url.c文件的493行,可以看到此函数的实现

1
2
out_str = php_url_encode(in_str, in_str_len, &out_str_len);
RETURN_STRINGL(out_str, out_str_len, 0);

在查看php_url_encode函数时,我纠结了好一段时间,因为它有一个#ifndef CHARSET_EBCDIC的编译判断,我一直将#ifndef看成了#ifdef,导致理解起来怪怪的
php_url_encode函数是一个遍历整个字符串,并针对每个字符进行替换的过程,
对于除了 -_. 之外的所有非字母数字字符都将被替换成百分号(%)后跟两位十六进制数,
此处在编译时有针对EBCDIC编码的处理

空格则编码为加号(+)(
if (c == ‘ ‘) {
*to++ = ‘+’;
}

另外:将urlencode从代码中分离出来的独立程序请猛击:www.wqsl.net/blogs/web/php/10/200804/03-60.html
不过没有测试过此段代码

PHP中的魔术方法总结 :__construct, __destruct , __call, __callStatic,__get, __set, __isset, __unset , __sleep, __wakeup, __toString, __set_state, __clone and __autoload

PHP中的魔术方法总结 :__construct, __destruct , __call, __callStatic,__get, __set, __isset, __unset , __sleep, __wakeup, __toString, __set_state, __clone and __autoload

1、__get、__set
这两个方法是为在类和他们的父类中没有声明的属性而设计的
__get( $property ) 当调用一个未定义的属性时访问此方法
__set( $property, $value ) 给一个未定义的属性赋值时调用
这里的没有声明包括当使用对象调用时,访问控制为proteced,private的属性(即没有权限访问的属性)

2、__isset、__unset
__isset( $property ) 当在一个未定义的属性上调用isset()函数时调用此方法
__unset( $property ) 当在一个未定义的属性上调用unset()函数时调用此方法
与__get方法和__set方法相同,这里的没有声明包括当使用对象调用时,访问控制为proteced,private的属性(即没有权限访问的属性)
3、__call
__call( $method, $arg_array ) 当调用一个未定义的方法是调用此访求
这里的未定义的方法包括没有权限访问的方法

4、__autoload
__autoload 函数,它会在试图使用尚未被定义的类时自动调用。通过调用此函数,脚本引擎在 PHP 出错失败前有了最后一个机会加载所需的类。
注意: 在 __autoload 函数中抛出的异常不能被 catch 语句块捕获并导致致命错误。

5、__construct、__destruct
__construct 构造方法,当一个对象创建时调用此方法,使用此方法的好处是:可以使构造方法有一个独一无二的名称,无论它所在的类的名称是什么.这样你在改变类的名称时,就不需要改变构造方法的名称
__destruct 析构方法,PHP将在对象被销毁前(即从内存中清除前)调用这个方法
默认情况下,PHP仅仅释放对象属性所占用的内存并销毁对象相关的资源.
析构函数允许你在使用一个对象之后执行任意代码来清除内存.
当PHP决定你的脚本不再与对象相关时,析构函数将被调用.
在一个函数的命名空间内,这会发生在函数return的时候.
对于全局变量,这发生于脚本结束的时候.如果你想明确地销毁一个对象,你可以给指向该对象的变量分配任何其它值.通常将变量赋值勤为NULL或者调用unset.

6、__clone
PHP5中的对象赋值是使用的引用赋值,如果想复制一个对象则需要使用clone方法,在调用此方法是对象会自动调用__clone魔术方法
如果在对象复制需要执行某些初始化操作,可以在__clone方法实现

7、__toString
__toString方法在将一个对象转化成字符串时自动调用,比如使用echo打印对象时
如果类没有实现此方法,则无法通过echo打印对象,否则会显示:Catchable fatal error: Object of class test could not be converted to string in
此方法必须返回一个字符串

在PHP 5.2.0之前,__toString方法只有结合使用echo() 或 print()时 才能生效。PHP 5.2.0之后,则可以在任何字符串环境生效(例如通过printf(),使用%s修饰符),但 不能用于非字符串环境(如使用%d修饰符)。从PHP 5.2.0,如果将一个未定义__toString方法的对象 转换为字符串,会报出一个E_RECOVERABLE_ERROR错误。

8、__sleep、__wakeup
__sleep 串行化的时候用
__wakeup 反串行化的时候调用
serialize() 检查类中是否有魔术名称 __sleep 的函数。如果这样,该函数将在任何序列化之前运行。它可以清除对象并应该返回一个包含有该对象中应被序列化的所有变量名的数组。
使用 __sleep 的目的是关闭对象可能具有的任何数据库连接,提交等待中的数据或进行类似的清除任务。此外,如果有非常大的对象而并不需要完全储存下来时此函数也很有用。
相反地,unserialize() 检查具有魔术名称 __wakeup 的函数的存在。如果存在,此函数可以重建对象可能具有的任何资源。
使用 __wakeup 的目的是重建在序列化中可能丢失的任何数据库连接以及处理其它重新初始化的任务。

9、__set_state
当调用var_export()时,这个静态 方法会被调用(自PHP 5.1.0起有效)。
本方法的唯一参数是一个数组,其中包含按array(‘property’ => value, …)格式排列的类属性。

10、__invoke
当尝试以调用函数的方式调用一个对象时,__invoke 方法会被自动调用。
PHP5.3.0以上版本有效
11、__callStatic
它的工作方式类似于 __call() 魔术方法,__callStatic() 是为了处理静态方法调用,
PHP5.3.0以上版本有效
PHP 确实加强了对 __callStatic() 方法的定义;它必须是公共的,并且必须被声明为静态的。同样,__call() 魔术方法必须被定义为公共的,所有其他魔术方法都必须如此。