php常见问题及解决方案汇总(中)

1.isset()和empty()的区别
两者都是测试变量用的。但是isset()是测试变量是否被赋值,而empty()是测试一个已经被赋值的变量是否为空。如果一个变量没被赋值就引用在php里是被允许的,但会有notice提示。如果一个变量被赋空值,$foo=””或者$foo=0或者$foo=false,那么empty($foo)返回真,isset($foo)也返回真,就是说赋空值不会注销一个变量。要注销一个变量,可以用 unset($foo)或者$foo=NULL。
2.如何显示错误信息
当php.ini的display_errors = On并且error_reporting = E_ALL时,将显示所有的错误和提示,调试的时候最好打开以便纠错,如果你用以前php写法错误信息多半是关于未定义变量的。变量在赋值以前调用会有提示,解决办法是探测或者屏蔽
3.单引号 双引号什么区别?分别什么时侯用
单引号中,任何变量($var)、特殊转义字符(如”t r n”等)不会被解析,因此PHP的解析速度更快,转义字符仅仅支持”’”和””这样对单引号和反斜杠本身的转义;双引号中,变量($var)值会代入字符串中,特殊转义字符也会被解析成特定的单个字符,还有一些专门针对上述两项特性的特殊功能性转义,例如”$”和”。这样虽然程序编写更加方便,但同时PHP的解析也很慢;数组中,如果下标不是整型,而是字符串类型,请务必用单引号将下标括起,正确的写法为$array[‘key’],而不是$array[key],因为不正确的写法会使PHP解析器认为key是一个常量,进而先判断常量是否存在,不存在时才以”key”作为下标带入表达式中,同时出发错误事件,产生一条Notice级错误。因此,在绝大多数可以使用单引号的场合,不要使用双引号。
4.print,echo,print_r什么区别?分别什么时侯用?
echo和print都可以做输出,不同的是,echo不是函数,没有返回值,而print是一个函数有返回值,所以相对而言如果只是输出 echo 会更快,而print_r通常用于打印变量的相关信息,通常在调试中使用
5.在PHP中有些时候需要打开远程文件
打开远程文件函数为:fopen(http://XXX.com/a.php),fsockopen(http://XXX.com/a.php) , file_get_contents(http://XXX.com/a.php)等)
在php5,apache2.2.X环境下,会提示你无法打开文件流,http请求失败(failed to open stream: HTTP request failed!)
在php.ini中,有这样两个选项:
allow_url_fopen =on(表示可以通过url打开远程文件),
user_agent=”PHP”(表示通过哪种脚本访问网络,默认前面有个 ” ; ” 去掉即可。)
重启Apache服务即可。
6.如何预先获取auto_increment的值?
mysql_connect(’localhost’,’root’,’root’) or die(’不能连接到服务器‘);
mysql_select_db(’test’); //连接数据库
$sql = “show create table id_user”; //id_userd 为表名
$query = mysql_query($sql);
$arr = mysql_fetch_array($query);
$b = strstr($arr[1],’AUTO_INCREMENT=’); //获取子字符串,包含AUTO_INCREMENT=5 DEFAULT CHARSET=utf8 )
$result = intval(substr($b,15)); //substr()从字符串第16个位置获取字符串,再对获取后的字符串进行int类型转换
echo ‘AUTO_INCREMENT的值’ .$result; //输出结果:即为下一次插入数据时的ID值
7.利用获得客户端信息
getenv(”REMOTE_ADDR”)可以得到浏览者的IP
getenv(”HTTP_USER_AGENT”)可以得到浏览者的操作系统类型,浏览器类型
getenv能得到所有$_ENV这个全局变量的所有东西
print_r($_ENV)可以看到很多东西
比如
echo getenv(ALLUSERSPROFILE); 可以看到ALLUSERSPROFILE的值
echo getenv(PATHEXT); 可以看到PATHEXT的值
8.mysql_fetch_row()和mysql_fetch_array()的区别
mysql_fetch_array() 是 mysql_fetch_row() 的扩展版本。除了将数据以数字索引方式储存在数组中之外,还可以将数据作为关联索引储存,用字段名作为键名。用 mysql_fetch_array()并不明显比用 mysql_fetch_row()慢,而且还提供了明显更多的值。
mysql_fetch_array() 中可选的第二个参数 result_type 是一个常量,可以接受以下值:MYSQL_ASSOC,MYSQL_NUM 和MYSQL_BOTH。本特性是PHP 3.0.7 起新加的。本参数的默认值是 MYSQL_BOTH。
9.EOD的用法和功效
就是比单引号和双引号好些,能包含换行。EOD可以改成其他字符。
10.用gdate()获得一长串数字怎么转换成正常的时间?
gmdate 返回 GMT 时间,除此以外和 date() 函数完全一样。如果你需要本地时间,就用 date() 来格式化时间戳。
11.正则里面的()//区别何在?
/ / 是模式定界符,代表里面是一个正则规则。
( ) 代表的是子模式,一个/ /中可以有很多的( )组成,可以用1 2 或 $1 $2来匹配前面子模式的值。
12.require和require_once有啥区别?
require重复调用会多次加载你饮用的文件;require_once只加载一次,而不管你实际上调用了多少次,主要用于复杂的文件包含关系
例如b包含a,c包含a,但同时c又包含了b,那么如果用require的话可能会导致两次加载a
13.三个IP地址的获取有什么区别?
$_SERVER[‘REMOTE_ADDR’];
$_SERVER[‘HTTP_CLIENT_IP’];
$_SERVER[‘HTTP_X_FORWARDED_FOR’];
$_SERVER[‘REMOTE_ADDR’]; //访问端(有可能是用户,有可能是代理的)IP
$_SERVER[‘HTTP_CLIENT_IP’]; //代理端的(有可能存在,可伪造)
$_SERVER[‘HTTP_X_FORWARDED_FOR’]; //用户是在哪个IP使用的代理(有可能存在,也可以伪造)
14.$_GET和$_POST的区别?
一:get 方法是限制传值的大小,不能超过2K. 而post不限制传值的大小.
二:Get是用来从服务器上获得数据,而Post是用来向服务器上传递数据。
三:Get将表单中数据的按照variable=value的形式,添加到action所指向的URL后面,并且两者使用”?”连接,而各个变量之间使用”&”连接;Post是将表单中的数据放在form的数据体中,按照变量和值相对应的方式,传递到action所指向URL。
四:Get是不安全的,因为在传输过程,数据被放在请求的URL中,而如今现有的很多服务器、代理服务器或者用户代理都会将请求URL记录到日志文件中,然后放在某个地方,这样就可能会有一些隐私的信息被第三方看到。另外,用户也可以在浏览器上直接看到提交的数据,一些系统内部消息将会一同显示在用户面前。Post的所有操作对用户来说都是不可见的。
15.在类里面的函数前面可以加public,private,但在函数库中的函数前面不能加呢?
function自定的随便你,public是公开使用的类属性对象,privat是独立使用的类属性对象;
类中 public 或 private 是指这个方法(要注意叫”方法”)是对外公开还是是属于类私有而你指的函数库,它是”函数”的库,不属于类中的方法,所以不必也不能有前置修饰。
16.页面执行时间怎么算的?
代码如下 复制代码
$mtime = explode(’ ‘, microtime());
$starttime = $mtime[1] + $mtime[0];
……
……
……
$mtime = explode(’ ‘, microtime());
$endtime = $mtime[1] + $mtime[0];
$usedtime = $endtime – $starttime;
printf(”
%0.4f s”, $usedtime);
17. UTF8编码脚本session_start(),header(),settcookie()等函数出错,提示”headers already sent “。
UTF8编码脚本通常的编辑器都会在文件头部加上三字节的BOM编码来识别UTF8编码格式,这三个字节是普通文件编辑器看不到,而输出时去先行以HTML输出了。执行以上函数时就会提示以上错误。解决办法:用editplus等可以清除BOM的编辑器,清除BOM(设置为utf8清除BOM)保存一下即可。
18. PHP中单引号,双引号,反引号的区别
PHP单引号(’),双引号(””)反引号(`)都能引用字串。单引号中变量不被转义,双引号转义变量,而反引号中变量转义作为shell命令被执行。
下成介绍了一些常开发中的问题
下面我们从较轻微的问题开始讨论,直至一些致命的错误。共分三部分。
第一部分、较轻微的错误
一、Printf(),
  该函数主要用来格式化显示数据。当你要改变某个数据的显示格式时才使用。
例如以不同的精度来显示PI(3.1415926)的值。
代码如下 复制代码
n”, M_PI);
 printf (“Pi is also: %.3fn
n”, M_PI);
 printf (“Pi is also: %.4fn
n”, M_PI);
?>
  但许多程序员仅仅为显示一些变量值和函数返回值使用该函数。因为Printf()在显示数据前要先格式化该数据以速度较慢,因此,仅为了显示数据时应用print和echo,以提高速度。
二、语意检查
  PHP是一种弱类型语言,也就是说在使用一个变量前不用定义,这样给编程带来了很大的方便和灵活,但你自己必须知道该变量到底应该是哪种类型,因为该变量在运行时仍实际对应着某一种类型(各种类型
之间可以自由互相转换),没有类型的变量是不存在的。有可能PHP并不能检查出你的语意错误,但由于变量类型的变化,会导致一些潜在的问题的发生。另外一个值得注意的问题是变量的范围,它也可能会导致
一些潜在的问题的发生。
在PHP中有以下几种基本变量:
Boolean, resource, integer, double, string, array and object。
三、临时变量的使用
  临时变量的滥用会导致程序运行效率的降低。何时使用临时变量可基于以下两点考虑:
1、该变量是否至少使用两次。
2、该变量的使用是否会显著提高程序的可读性。
如果一条也不满足,则省略该变量的使用。例如:
代码如下 复制代码

就应该改成:

又如:

的可读性不强,可改成:

代码如下 复制代码

网络找到的一些问题

php中常见问题集锦2009-09-10 11:07【1】页面之间无法传递变量
get,post,session在最新的php版本中自动全局变量是关闭的,所以要从上一页面取得提交过来得变量要使用$_GET[‘foo’],$_POST[‘foo’],$_SESSION[‘foo’]来得到
当然也可以修改自动全局变量为开(php.ini改为register_globals = On);考虑到兼容性,还是强迫自己熟悉新的写法比较好。
【2】Win32下apache2 用get方法传递中文参数会出错
test.php?a=你好&b=你也好
传递参数是会导致一个内部错误
解决办法:”test.php?a=”.urlencode(你好).”&b=”.urlencode(你也好)
【3】win32下的session不能正常工作
php.ini默认的session.save_path = /tmp
这显然是linux下的配置,win32下php无法读写session文件导致session无法使用
把它改成一个绝对路径就可以了,例如session.save_path = c:windowstemp
【4】显示错误信息
当php.ini的display_errors = On并且error_reporting = E_ALL时,将显示所有的错误和提示,调试的时候最好打开以便纠错,如果你用以前php写法错误信息多半是关于未定义变量的。变量在赋值以前调用会有提示,解决办法是探测或者屏蔽
例如显示$foo,可以if(isset($foo)) echo $foo 或者echo @$foo
【5】Win32下mail()不能发送电子邮件
在linux下配置好的sendmail可以发送,在win32下需要调用smtp服务器来发送电子邮件
修改php.ini的SMTP = ip //ip是不带验证功能的smtp服务器(网上很难找到)
php发送邮件的最好解决方法是用socket直接发送到对方email服务器而不用转发服务器
有个很好的class,不过需要修改一下发信会暴快,修改后版本将在近日推出
【6】header already sent这个错误通常会在你使用HEADER的时候出现,他可能是几种原因:
1,你在使用HEADER前PRING或者ECHO了
2.你当前文件前面有空行
3.你可能INCLUDE了一个文件,该文件尾部有空行或者输出也会出现这种错误。!
还有使用session_register()
【7】初装的mysql如果没有设置密码,应该使用
update mysql.user set password=password(“yourpassword”) where user=”root”
【8】更改php.ini后没有变化
重新启动web server,比如IIS,Apache等等,然后才会应用最新的设置
【9】php在2003上面安装(ISAPI的安装方法)
PHP4的php4isapi.dll好像和2003有些冲突,只能用CGI模式安装
步骤一,先去在一个安装程序,我是装的是:php-4.2.3-installer.exe,你也可以去找最新的版本,在安装 php-4.2.3-installer.exe之前保证你的IIS6.0启动了,并能够访问。 安装好以后,在默认网站–>应用程序配置
步骤二:点击 web服务扩展 –>新建web服务扩展.
步骤三: 扩展名–>php,然后添加
步骤四:找到php.exe的路径添加上去。
步骤五: 确定就可以了!
步骤六: 选择php的服务扩展,然后点击允许。
【10】有时候sql语句不起作用,对数据库操作失败
最简便的调试方法,echo那句sql,看看变量的值能不能得到
【11】
代码如下 复制代码

使用 GET 表单也类似,只不过要用适当的 GET 预定义变量。GET 也适用于 QUERY_STRING(URL 中在“?”之后的信息)。因此,举例说,http://www.example.com/test.php?id=3 包含有可用 $_GET[‘id’] 访问的 GET 数据。参见 $_REQUEST 和 import_request_variables()。
在 PHP 4.2.0 之前 register_globals 的默认值是 on。在 PHP 3 中其值总是 on。鼓励大家不要依赖此指令,建议在编码时假定其为 off。

【12】暂存你的变量到临时文件:
代码如下 复制代码
if(file_exists(‘temp.php’))
{
$x=r(‘temp.php’);
@eval(“$a = $x;”);
}
if(!is_array($a)){
#重新构造数组A
$a=array(array(“af”=>”fsdf”),”f”=>”df”);
}
w(‘temp.php’,var_export($a,true));
var_dump($a);
读取,写入文档数据
function r($file_name)
{
$filenum=@fopen($file_name,”r”);
@flock($filenum,LOCK_SH);
$file_data=@fread($filenum,filesize($file_name));
@fclose($filenum);
return $file_data;
}
function w($file_name,$data,$method=”w”)
{
$filenum=@fopen($file_name,$method);
flock($filenum,LOCK_EX);
$file_data=fwrite($filenum,$data);
fclose($filenum);
return $file_data;
}
【13】include和require的区别
两者没有太大的区别,如果要包含的文件不存在,include提示notice,然后继续执行下面的语句,require提示致命错误并且退出;据我测试,win32平台下它们都是先包含后执行,所以被包含文件里最好不要再有include或require语句,这样会造成目录混乱。或许linux下情况不同,暂时还没测试;如果一个文件不想被包含多次可以使用include_once或require_once
【14】session在函数和方法中的应用:你打算注册进session的变量都必须是全局的。
原因是这样的:
php的session_register函数只是记住变量的名字,而不会去记变量的值。
真正要在服务器端记住这个变量的值是在整个脚本运行结束之后,也就是说变量的值是在脚本运行结束的时候才会被读取并保存入服务器端的临时目录,这样,在所有函数或方法外的、或在函数或方法内被定义为全局变量的变量才会成功register,而其他的则都会在脚本运行结束时被unset掉。
【15】PHP中的session由于基于cookie实现,所以在所有会话窗口关闭后并不会马上消失。这一点和其他脚本语言不同,同时让我感到郁闷。
答:不不不,楼上的朋友没有把php的session研究透,如果非要说php的session跟cookie有关系,那也就是一个session_id被记录在了客户端
而在服务器端的临时目录下,会生成一个跟session_id大致同名的文件,这个文件才是真正记录你成功register过的变量和它们的值。同样,如果客户端禁止使用cookie,php会自动以get的方式传递session_id的值,使其不会丢失。所以,cookie跟session之间的关系,并不是那么密不可分。而且,php的session也不能说是基于cookie实现的。
【16】isset()和empty()的区别
如果一个变量没被赋值就引用在php里是被允许的,但会有notice提示;
两者都是测试变量用的,但是isset()是测试变量是否被赋值,而empty()是测试一个已经被赋值的变量是否为空
如果一个变量被赋空值,$foo=””或者$foo=0或者 $foo=false,那么empty($foo)返回真,isset($foo)也返回真,就是说赋空值不会注销一个变量。要注销一个变量,可以用 unset($foo)或者$foo=NULL
【17】通过HTTP协议一次上传多个文件的方法
有两个思路,是同一个方法的两种实现。
1,在form中设置多个文件输入框,用数组命名他们的名字,如下:
代码如下 复制代码




这样,在服务器端做以下测试
echo “

"; 
print_r($_FILES); 
echo "

“;
2,在form中设置多个文件输入框,但名字不同,如下:




在服务器端做同样测试:
echo “

"; 
print_r($_FILES); 
echo "

“;
【18】可以创建一个临时的PHP格式的文件,在需要的地方include进来,这样临时的PHP文件中定义的变量就有了,可以取代session。
【19】正如上面说得,现在get,post方式提交的都无法直接使用,有时甚是麻烦,自己写了段代码,把他们转化为全局变量,以便使用,session变量同理可得。
把通过GET或POST方式提交的变量转化为全局变量:
代码如下 复制代码
foreach($_GET as $key=>$value){
$$key=$value;
}
foreach($_POST as $key=>$value){
$$key=$value;
}

Content-disposition 是 MIME 协议的扩展,MIME 协议指示 MIME 用户代理如何显示附加的文件。Content-disposition其实可以控制用户请求所得的内容存为一个文件的时候提供一个默认的文件名,文件直接在浏览器上显示或者在访问时弹出文件下载对话框。

格式说明:
content-disposition = “Content-Disposition” “:” disposition-type *( “;” disposition-parm )  

字段说明:
Content-Disposition为属性名
disposition-type是以什么方式下载,如attachment为以附件方式下载
disposition-parm为默认保存时的文件名
服务端向客户端游览器发送文件时,如果是浏览器支持的文件类型,一般会默认使用浏览器打开,比如txt、jpg等,会直接在浏览器中显示,如果需要提示用户保存,就要利用Content-Disposition进行一下处理,关键在于一定要加上attachment:
代码如下 复制代码
Response.AppendHeader(“Content-Disposition”,”attachment;filename=FileName.txt”);
备注:这样浏览器会提示保存还是打开,即使选择打开,也会使用相关联的程序比如记事本打开,而不是IE直接打开了。
Content-Disposition就是当用户想把请求所得的内容存为一个文件的时候提供一个默认的文件名。具体的定义如下:
代码如下 复制代码
content-disposition = “Content-Disposition” “:”
disposition-type *( “;” disposition-parm )
disposition-type = “attachment” | disp-extension-token
disposition-parm = filename-parm | disp-extension-parm
filename-parm = “filename” “=” quoted-string
disp-extension-token = token
disp-extension-parm = token “=” ( token | quoted-string )
那么由上可知具体的例子:
Content-Disposition: attachment; filename=“filename.xls”
当然filename参数可以包含路径信息,但User-Agnet会忽略掉这些信息,只会把路径信息的最后一部分做为文件名。当你在响应类型为application/octet- stream情况下使用了这个头信息的话,那就意味着你不想直接显示内容,而是弹出一个”文件下载”的对话框,接下来就是由你来决定“打开”还是“保存” 了。
注意事项:
1.当代码里面使用Content-Disposition来确保浏览器弹出下载对话框的时候。 response.addHeader(“Content-Disposition”,”attachment”);一定要确保没有做过关于禁止浏览器缓存的操作。如下:
代码如下 复制代码
response.setHeader(“Pragma”, “No-cache”);
response.setHeader(“Cache-Control”, “No-cache”);
response.setDateHeader(“Expires”, 0);

我们先来讲述一下利用session来登录
登录页面05.php的代码如下:
代码如下 复制代码

管理系统登录
管理员:
密码:
&nbsp ;

该程序是一个管理员登录界面,首先初始化session,然后通过检测session变量是否已经注册来检测用户是否已经登录过,如果登录过,则无需再次登录,直接定向到其他页面。这里还使用了javascript来判断是否输入了用户名和密码。
验证登录的页面
06.php的代码如下:
代码如下 复制代码
登录成功!” ;
header (“Location:shili.php”) ; //登录成功重定向到管理页面
}else{
echo “

” ;
echo “账号或密码错误,或者不是管理员账号
” ;
echo “登录失败!
请重新输入“;
echo “

” ; }
?>

该程序同样是先通过session变量检测用户是否已经登录,如果没有,则要对输入的用户名和密码进行验证,正确则注册session变量,输出登录成功!。如不正确,则提示登录失败。
这是一个简单的用户登录验证的程序,通过session变量阻止了那些没有登录而直接浏览的用户。如果想要对某个页面限制浏览的用户,可以使用同样的方式,只要把如下的代码放在该页面的开头即可:
代码如下 复制代码
” ;
echo “” ;
echo “你没有登录,请登录!” ;
echo “

” ;
exit () ; }
?>

利用session_id实现共享登录
先,多服务器共享session问题,这个大家应该都能够理解的,当一个网站的用户量过大,就会使用服务器集群,例如专门有一个登录用的服务器。用户通过登录服务器登录之后,登录服务器保存了用户的登录信息session,而其他受访问的服务器,例如电影服务器没有这个session,那么我们就要通过一个session的唯一标识来共享这个session了——具体session的共享超出了本文的范围,请自行查阅资料。
第二个用途就是,验证同一用户的不同会话,这个比较难理解。这样说吧,一个用户并非通过浏览器来请求连接,而是通过socket或者其它方式来请求数据的时候,我们首先要对他进行用户登录验证,验证成功之后,就下发一个sessionid给他,然后他每次请求的时候就携带这个sessionid,我们通过这个sessionid来判断session是否已经存在,如果存在我们就认定用户已经登录……
对于第一个问题,我们可以把sessionid保存在数据库中得以实现,这个方法比较安全而且应用广泛,但是不是我们讨论的范围哦
首先验证的时候产生一个sessionid;
代码如下 复制代码

客户端携带sessionid这个变量来请求数据
代码如下 复制代码