php require_once用法与相对目录要谨慎

wwwroot   //网站根目录 绝对路径为: F:/wwwroot

-- folder_a // 文件夹A

   file_a_a.php

   file_a_b.php

   file_a_c.php

-- folder_b // 文件夹B

   file_b_a.php

   file_b_b.php

   file_b_c.php

-- index.php

*************************************************************

这个目录层次已经很清楚了:

wwwroot为根目录,下面有index.php文件和folder_a,folder_b两个文件夹

这两个文件夹分别有3个php文件

先来看index.php文件的内容:

 代码如下 复制代码

<?php

     require_once("folder_a/file_a_a.php");

     echo "文件folder_a_a.php被包含成功";

?>

再来看folder_a/folder_a_a.php文件的内容:

 代码如下 复制代码

<?php

     require_once("../folder_b/file_b_a.php");

     $x = new X();

     $x.printInfo();

?>

最后再来看看folder_b/folder_b_a.php文件的内容:

 代码如下 复制代码

<?php

     class X{

          function printInfo(){

               echo 'success;

          }

     }

?>

ok 如果我现在直接运行 floder_a/file_a_a.php

那么会输出: success

如果我运行 wwwroot下的index.php

那么会报错,原因是找不到包含文件:file_b_a.php

但是如果我在所有的require_once()中加入 dirname(__FILE__).'/'

那么无论运行 file_a_a.php 还是 index.php 都可以正常输出

*********************************************************

问题:

我第一次使用的是相对路径,所以重复包含的时候就出错了

而我第二次使用的是绝对路径,所以不出错.可是我还是有点疑惑:

我先分析了以下使用相对路径会出错的原因:

我运行index.php,它能找到 folder_a目录,也能找到该目录下的file_a_a.php,于是它就把 folder_a/file_a_a.php 里的内容复制到了index.php 的第一行(包含语句的那行),然后再继续运行(也就是运行包含过来的内容),因此此时等于是在index.php中运行 file_a_a.php 中的require_once('../folder_b/file_b_a.php'); 它根据当前index.php的位置去找这个路径文件(file_b_a.php),当然找不到啦,所以它出错了.

可是我使用绝对路径的时候不是一样的道理吗?可为什么它就不出错?可能这句话大家有点不解了,我来详细说明(根据程序的运行顺序来说明).

程序先运行 index.php(注意此时我加上了 dirname(__FILE__),所以当前是绝对路径),

index.php先运行第一句代码:require_once(dirname(__FILE__).'/'.'folder_a/file_a_a.php');

dirname(__FILE__) 就是 f:/wwwroot/ 所以这句代码包含的路径也就是:

f:/wwwroot/folder_a/file_a_a.php

这个路径是正确的,因此没问题是吧

ok 第一步正确完成

然后它就把 file_a_a.php 中的代码复制到的 index.php 的这个地方:

然后仍然继续运行: 这是也就是在index.php 中运行file_a_a.php 中的所有代码,那么我们来看看它运行那些代码?

 代码如下 复制代码

<?php

     require_once(dirname(__FILE__).'/'."../folder_b/file_b_a.php");

     $x = new X();

     $x.printInfo();

?>

对就是这些,需要注意的是,这些代码已经被复制到了index.php,也就是说,现在index.php的内容实际上就变成了:

 代码如下 复制代码

<?php

     require_once(dirname(__FILE__).'/'."../folder_b/file_b_a.php");

     $x = new X();

     $x.printInfo();

     echo "文件folder_a_a.php被包含成功";

?>

我们来看个注意事项

假设有如下三个文件, c.php a.php b.php 对应的存放目录为:localhost/ localhost/ localhost/demo

 代码如下 复制代码

c.php

require_once("a.php");
require_once("demo/b.php");
B::demo();a.php
class A
{
}

b.php的内容比较有意思,因为它自己要继承 CLASS A 所以自己把a.php也引入进去了

 代码如下 复制代码

require_once("../a.php");
class B extends A
{
    public static function demo()
    {
    echo "xx";
    }
}

执行localhost/c.php 系统报错,报错信息如下
Warning: require_once(../a.php) [function.require-once]: failed to open stream: No such file or directory in F:wwwdemob.php on line 2
Fatal error: require_once() [function.require]: Failed opening required '../a.php' (include_path='.;C:php5pear') in F:wwwdemob.php on line 2但是,惊奇的发现,如果去掉b.php里面的require_once语句,执行正常,那么一定是require_once语句定义多了吗?原因就是Class A重定义了两次?可是不会啊。如果我只在c.php里面加require_once(‘a.php’);这条语句,哪怕我写两遍也是没错的,那到底是咋回事呢?
原因就是,b.php定义的目录和c.php执行文件的目录层级不一致,导致在c.php里面require_once语句有两条。使其相当于

 代码如下 复制代码

require_once("a.php");
require_once("../a.php");
class B extends A
{
    public static function demo()
    {
    echo "xx";
    }
}
B::demo();

原因找到了,因为在c.php里面,其相对目录 “..”就是 c.php的上一层了,导致文件找不到报错。
所以,我们的结论是,在 PHP 里面,使用require_once的时候,存在不同层级关系,且有相对目录的使用那么一定要谨慎,小心。

require_once很简单用但在使用时大家尽量使用绝对路径了。

时间: 2024-10-02 09:43:06

php require_once用法与相对目录要谨慎的相关文章

php中require_once()函数相对目录要谨慎

假设有如下三个文件, c.php a.php b.php 对应的存放目录为:localhost/ localhost/ localhost/demo  代码如下 复制代码 c.php require_once("a.php"); require_once("demo/b.php"); B::demo();a.php class A { } b.php的内容比较有意思,因为它自己要继承 CLASS A 所以自己把a.php也引入进去了  代码如下 复制代码 requi

php递归用法与递归目录实例

看一个简单的递归实例 例1  代码如下 复制代码 function demo($a) {      static $sum=1;      if($a > 1){          $sum*=$a;          demo(--$a);      }else{          $a=$sum;          }          return $sum;  }    echo demo(10); 例2 遍历目录  代码如下 复制代码 <?php class listdir{ var

php中stream(流)的用法

 Stream是PHP开发里最容易被忽视的函数系列(SPL系列,Stream系列,pack函数,封装协议)之一,但其是个很有用也很重要的函数.Stream可以翻译为"流",下面是使用方法 在Java里,流是一个很重要的概念.   流(stream)的概念源于UNIX中管道(pipe)的概念.在UNIX中,管道是一条不间断的字节流,用来实现程序或进程间的通信,或读写外围设备.外部文件等.根据流的方向又可以分为输入流和输出流,同时可以在其外围再套上其它流,比如缓冲流,这样就可以得到更多流处

php中stream(流)的用法_php实例

在Java里,流是一个很重要的概念. 流(stream)的概念源于UNIX中管道(pipe)的概念.在UNIX中,管道是一条不间断的字节流,用来实现程序或进程间的通信,或读写外围设备.外部文件等.根据流的方向又可以分为输入流和输出流,同时可以在其外围再套上其它流,比如缓冲流,这样就可以得到更多流处理方法. PHP里的流和Java里的流实际上是同一个概念,只是简单了一点.由于PHP主要用于Web开发,所以"流"这块的概念被提到的较少.如果有Java基础,对于PHP里的流就更容易理解了.其

linux下面某些常用命令的用法【转】

转自:http://blog.csdn.net/luo3532869/article/details/7584290 ls 命令用于常看目录,用法:ls [选项][目录或文件]例:使用ls命令显示/home目录下的文件与目录(不包含隐藏文件) ls /home  显示/home目录下的文件和目录 ls -a  /home显示/home目录下的所有的文件和目录,包括隐藏文件 ls -l  /home 显示/home目录下的文件和目录的详细信息,比如权限.所属用户.创建时间等等 ls -c  /ho

apache中伪静态设置.htaccess详解

.htaccess是什么 .htaccess文件(或者"分布式配置文件")提供了针对目录改变配置的方法, 即,在一个特定的文档目录中放置一个包含一个或多个指令的文件, 以作用于此目录及其所有子目录.作为用户,所能使用的命令受到限制.管理员可以通过Apache的AllowOverride指令来设置. 概述来说,htaccess文件是Apache服务器中的一个配置文件,它负责相关目录下的网页配置.通过htaccess文件,可以帮我们实现:网页301重定向.自定义404错误页面.改变文件扩展

htaccess语法教程_相关技巧

虽然以前的文章没有了,但我至少得保证原来到域名能够继续访问到新的博客,浏览到新的文章,于是想起用htaccess来做重定向.这次对htaccess的写法做个记录,免得今后还要到处搜罗. 这是我这次用到的几条重定向规则 复制代码 代码如下: RewriteEngine on #silentash全局切换 Rewritecond %{HTTP_HOST} ^(www\.)?silentash.com$ [nc] Rewriterule ^(.*)$ http://jb51.net/$1 [r=301

MySQL远程连接

MySQL默认情况下用户只允许在本地登录,如果需要远程登录该如何操作呢? 1.创建新用户 格式: grant 权限 on 数据库名.表名 用户@登录主机 identified by "用户密码"; grant select,update,insert,delete on *.* to name@IP identified by "password";     创建一个用户名为name,密码是password的用户,且只允许IP地址为"IP"的主机

浅谈PHP中Stream(流)_php技巧

流(stream)的概念源于UNIX中管道(pipe)的概念.在UNIX中,管道是一条不间断的字节流,用来实现程序或进程间的通信,或读写外围设备.外部文件等.根据流的方向又可以分为输入流和输出流,同时可以在其外围再套上其它流,比如缓冲流,这样就可以得到更多流处理方法. PHP里的流和Java里的流实际上是同一个概念,只是简单了一点.由于PHP主要用于Web开发,所以"流"这块的概念被提到的较少.如果有Java基础,对于PHP里的流就更容易理解了.其实PHP里的许多高级特性,比如SPL,