php路径问题及其解决方法

初学PHP的时候,我们经常被PHP的绝对路径和相对路径弄的晕头转向。路径问题经常导致include及require命令不能加载到指定的页面,从而导致Web程序运行错误。本文将跟大家一起探讨一下PHP路径问题的常用解决方案。
下面我们先做一个个简单的示例:

上图是一个简单Web项目的结构图,其中各文件的代码如下:
root/index.php

GetBlogInfo(); ?>

root/app/blog.php

’; ?>

首先,我们从浏览器里直接输入http://localhost/root/app/blog.php,浏览器显示:
———————————————————————-
root/app/blog.php loaded successed!
———————————————————————-
文件加载成功,没发生任何警告或错误。
接下来,我们在浏览器中请求如下网站:http://localhost/root/index.php,浏览器显示出如下错误信息:
———————————————————————————
Warning: require_once(../lib/smarty/functions.php) [function.require-once]: failed to open stream: No such file or directory in T:\Study\PHP_REL\Projects\root\app\blog.php on line 2
Fatal error: require_once() [function.require]: Failed opening required ‘../lib/smarty/functions.php’ (include_path=’.;C:\php5\pear’) in T:\Study\PHP_REL\Projects\root\app\blog.php on line 2
———————————————————————————
为什么’../lib/smarty/functions.php’没能加载? 我们访问http://localhost/root/app/blog.php没有出现任何异常,这说明出现错误的原因很可能在**root/index.php**的require_once ‘app/blog.php’语句。从Google里搜索一下:php****路径问题,你就能找到问题的答案。当root/index.php引用root/app/blog.php后,blog.php中的require_once语句的参照点变成了root/index.php所在的目录(根目录root)。所以,以root/index.php为参照点加载‘../lib/smarty/functions.php’时,自然就出错了。
找到了问题所在,接下来,我们就来寻找解决问题的办法:
**1.**将所有文件放在一个文件夹下
将所有文件放在一个文件夹下,那么就不会存在路径问题了。但是,这绝对是个馊主意!!除非你写一个小的不能再小的项目,不然请不要尝试这种方法,没有结构的系统太可怕了!
**2.**使用绝对路径
**注:**PHP中的include和require使用的是文件系统的绝对路径,如“c:\wwwroot\yourproject\index.php”
刚刚发生异常是因为我们使用了相对路径,如果换成绝对路径就不会出现以上错误。让我们简单修改一下root/app/blog.php:
将前三行由
step1:

修改为:

现在,我们访问:http://localhost/root/index.php ,浏览器显示:
—————————————-
root/app/blog.php loaded successed!
This is a test of blog!
—————————————-
程序执行成功了。
step2:虽然程序不报错了,但很明显,我们并没有真正的解决问题。没有人会在程序里写require_once ‘T:\Study\PHP_REL\Projects…functions.php’这样的东西,这样写将会使程序完全丧失灵活性,使程序难以移植!
让我们再重新回想一下出现错误的原因:1.不同层次文件之间的引用使require_once的参照点发生了变化;2.参照点发生变化后,按相对路径加载文件会出错。如果我们把require_once的参照点固定,问题不就解决了吗.如何固定参照点?当然是使用绝对路径,具体实现方法:一个函数+一个常量:dirname()和__FILE__。让我们重新修改root/app/blog.php如下:
将前三行由

修改为:

再次访问http://localhost/root/index.php程序运行正常,但这种解决方式很明显优于step1中的解决方式。应该说,这是一种**"****绝对路径+相对路径"**的解决方式。
step3:上面的方法已经能够解决路径问题,但感觉代码不够优雅。让代码更优雅,我们可以这样做:在根目录(root)下新建一个settings.php:

root/settings.php :

root/app/blog.php中的代码修改为:

同时root/index.php中的代码修改为:

GetBlogInfo(); ?>

仔细考虑一下,如果直接访问http://localhost/root/app/blog.php又会出现问题:常量ABSPATH没有定义。所以,如果你的程序有直接访问[http://localhost/root/app/blog.php](http://localhost/root/app/blog.php)这种情况类似的情况,那么最好直接使用dirname(__FILE__) .‘/’.‘相对路径’,或者在使用ABSPATH前加一个判断(但这样有点脱裤子放X的感觉)。
:在WordPress中使用了ABSPATH与dirname(FILE).‘/’.‘相对路径’相结合的方法,从网站统一入口(根目录/index.php)加载的文件,使用ABSPATH的解决方法(ABSPATH在_根目录/wp-config-sample.php_ 中定义),而那些不直接通过统一入口访问的php文件,WP使用dirname(FILE).‘/’.‘相对路径’的解决方案。
3.设置Apache的include_path参数
在前面的错误信息中,有一句值得我们注意:
———————————————
Fatal error: require_once() [function.require]: Failed opening required ‘../lib/smarty/functions.php’ (include_path=’.;C:\php5\pear’) in T:\Study\PHP_REL\Projects\root\app\blog.php on line 2
———————————————
Apache的include_path参数保存的是require/include的读取目录,在上面的错误信息里,include_path包含了两个位置:

  1. "."表示从当前文件所在的目录中加载
  2. "C:\php5\pear"表示从C盘的php5/pear目录下加载。
    php函数库为我们提供了set_include_path()函数用于设置include_path参数。通过set_include_path()函数,我们可以自定义加载位置(ZendFramework中就是使用set_include_path()函数来解决路径问题)。
    下面我们来演示一下如何用set_include_path()函数:
    root/index.php :
GetBlogInfo(); ?>

root/app/blog.php ;

’; ?>

测试http://localhost/root/index.php,运行正常。从root/app/blog.php可以看出,require_once的路径写法更简洁了(不需要使用ABSPATH或者dirname(__FILE__))。这种实现方式跟ABSPATH的解决方式一样,需要保证系统有统一的入口点(一般通过[**.htaccess****文件**](http://www.myleoliu.com/archives/5.html)实现)。
当然,解决php路径的方法不止以上几种,网上有很多人提供了针对php路径问题的解决方案,但多数都有一定的适用场景,不能生搬硬套。在下一篇日志中,我会对php路径问题的其他解决方法做一个简单的总结。