PHP递归创建多级目录(一道面试题的解题过程)

今天看到一道面试题,要写出一个可以创建多级目录的函数:

我的第一个感觉就是用递归创建,具体思路如下:

function Directory($dir){

    if(is_dir($dir) || @mkdir($dir,0777)){ //查看目录是否已经存在或尝试创建,加一个@抑制符号是因为第一次创建失败,会报一个“父目录不存在”的警告。

        echo $dir."创建成功<br>";   //输出创建成功的目录

    }else{

        $dirArr=explode('/',$dir); //当子目录没创建成功时,试图创建父目录,用explode()函数以'/'分隔符切割成一个数组

        array_pop($dirArr); //将数组中的最后一项(即子目录)弹出来,

        $newDir=implode('/',$dirArr); //重新组合成一个文件夹字符串

        Directory($newDir); //试图创建父目录

        if(@mkdir($dir,0777)){

            echo $dir."创建成功<br>";

        } //再次试图创建子目录,成功输出目录名

    }

}

Directory("A/B/C/D/E/F");

输出结果如图:

但是可以看得出来,写得也太麻烦了,在手册里翻看文件函数,看到一个dirname()函数,其原型如下:

string dirname ( string $path ) 给出一个包含有指向一个文件的全路径的字符串,本函数返回去掉文件名后的目录名。  在 Windows 中,斜线(/)和反斜线(\)都可以用作目录分隔符。在其它环境下是斜线(/)。

可以稍稍地优化一下:

function Directory($dir){

    if(is_dir($dir) || @mkdir($dir,0777)){ 

        echo $dir."创建成功<br>";   

    }else{

        Directory(dirname($dir));

        if(@mkdir($dir,0777)){

            echo $dir."创建成功<br>";

        }

    }

}

效果一样。

之后我在在网上搜一下答案,找到一个异常精辟的:

function  Directory( $dir ){    

     return   is_dir ( $dir )  or  Directory(dirname( $dir ))  and   mkdir ( $dir , 0777);

}

现在来解释一下整个函数:

先介绍一下PHP中逻辑运算符的优先级顺序:&& > || > and > or,即符号型>字母型,AND型>OR型,所以函数体可以看成:

is_dir ( $dir )  or  (Directory(dirname( $dir ))  and   mkdir ( $dir , 0777));

先判断目标目录是否存在,若存在,依or的短路特性,后面的整体被短路,跳过执行;若目标目录不存在,则执行后面的函数体:

Directory(dirname( $dir ))  and   mkdir ( $dir , 0777) 

我考虑了一下先进行递归的用意:先执行递归,意在确认其父目录(dirname($dir))都已经创建完毕,使后面的mkdir()函数不会创建子目录时找不到父目录发出警告。

进入递归深处后,确认最深处的根目录存在后,从根目录向下依次创建目录。

最后,建议要找工作的亲们,去网上找些大公司面试题做一下,毕竟他们考得较为综合较深,在学习知识的时候,也刷一下题,另外也一定要做一下,因为很容易眼高手低,一开始的函数,我优化了好几遍才能正常使用。

以后我也会再找些有意思的面试题跟大家分享。

如果您觉得本文对您有帮助,您可以推荐或关注我,如果您有什么问题,可以在下方留言讨论,谢谢。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏比原链

剥开比原看代码12:比原是如何通过/create-account-receiver创建地址的?

Gitee地址:https://gitee.com/BytomBlockchain/bytom

811
来自专栏开发技术

spring-boot-2.0.3不一样系列之源码篇 - 阶段总结

    朋友喜欢去按摩,第一次推门进来的是一个学生美眉,感觉还不错;后来经常去,有时是护士,有时是空姐,有时候是教师。昨天晚上推门进去的是一个女警察,长得贼好看...

632
来自专栏24K纯开源

CMake结合Visual Studio中开发Qt应用程序注意事项

2392
来自专栏恰同学骚年

.NET单元测试的艺术-1.入门

开篇:最近在看Roy Osherove的《单元测试的艺术》一书,颇有收获。因此,将其记录下来,并分为四个部分分享成文,与各位Share。本篇作为入门,介绍了单元...

802
来自专栏Ryan Miao

使用dropwizard(3)-加入DI-dagger2

前言 习惯了Spring全家桶,对spring的容器爱不释手。使用dropwizard,看起来确实很轻,然而,真正使用的时候不得不面临一个问题。我们不可能一个...

2507
来自专栏抠抠空间

Django之admin的使用和源码剖析

1570
来自专栏ThoughtWorks

TW洞见 | 李光磊:JavaScript语言中五种消除分支的方法

最近开始使用JavaScript。回顾了一下这几天的代码,发现圈复杂度为1。30几个函数40多行,超过两行的函数都很少 (当然那种当做对象来用的函数除外,只说实...

3316
来自专栏代码世界

Django-admin管理工具

1222
来自专栏上善若水

026android初级篇之ActivityManager的Proxy模式初识

取代了简单的指针,它在访问对象时执行一些附加操作。对指向实际对象的引用计数, 这样当该对象没有引用时,可以自动释放它,智能指针当第一次引用一个持久对象时,将它...

814
来自专栏Python

wtforms

 简介 WTForms是一个支持多个web框架的form组件,主要用于对用户请求数据进行验证。 安装: ? 1 pip3 install wtf...

1947

扫码关注云+社区