PHP基于迭代实现文件夹复制、删除、查看大小等操作的方法

发布时间 - 2026-01-11 02:45:02    点击率:

本文实例讲述了PHP基于迭代实现文件夹复制、删除、查看大小等操作的方法。分享给大家供大家参考,具体如下:

前面一篇 PHP递归实现文件夹的复制、删除、查看大小操作 分析了递归操作使用技巧,这里再来分析一下迭代的操作技巧。

“既然递归能很好的解决,为什么还要用迭代呢”?主要的原因还是效率问题……

递归的概念是函数调用自身,把一个复杂的问题分解成与其相似的多个子问题来解决,可以极大的减少代码量,使得程序看起来非常优雅。

由于系统要为每次函数调用分配运行空间,并使用压栈予以记录。在函数调用结束后,系统需要释放空间,并弹栈恢复断点。所以递归的消耗还是比较大的。

即使语言设计时已经将函数调用优化的极度完美,达到可以忽略递归造成的资源浪费,但是递归的深度仍然会受到系统栈容量的限制,否则将会抛出 StackOverflowError 错误。

而迭代能很好的利用计算机适合做重复操作的特点,并且从理论上说,所有的递归函数都可以转换为迭代函数,所以尽量能不用递归就不用递归,能用迭代代替就用迭代代替。

查看文件夹大小

迭代的思路是让计算机对一组指令进行重复执行,在每次执行这组指令时,都从变量的原值推出其它的新值……重复这一过程直到达到结束条件或没有新值产生。

由于递归相当于循环加堆栈,所以可以在迭代中使用堆栈来进行递归和迭代的转换。

/**
 * 文件夹大小
 * @param $path
 * @return int
 */
function dirsize($path)
{
  /* 初始条件 */
  $size = 0;
  $stack = array();
  if (file_exists($path)) {
    $path = realpath($path) . '/';
    array_push($stack, '');
  } else {
    return -1;
  }
  /* 迭代条件 */
  while (count($stack) !== 0) {
    $dir = array_pop($stack);
    $handle = opendir($path . $dir);
    /* 执行过程 */
    while (($item = readdir($handle)) !== false) {
      if ($item == '.' || $item == '..') continue;
      $_path = $path . $dir . $item;
      if (is_file($_path)) $size += filesize($_path);
      /* 更新条件 */
      if (is_dir($_path)) array_push($stack, $dir . $item . '/');
    }
    closedir($handle);
  }
  return $size;
}

复制文件夹

迭代和递归都具有初始化变量、判断结束条件、执行实际操作、产生新变量这四个步骤,只不过所在的位置不同罢了。

比如初始化变量这一步骤,在迭代中是位于函数的开始部分,而在递归中是指其他函数传递参数这一过程;

判断结束条件这一步骤,在迭代中用于判断循环是否继续,在递归中用于判断递归的结束位置;

执行实际操作在递归和迭代中都是函数的核心部分,位于产生新变量步骤之前;

产生新变量在迭代中是迭代继续的条件,在递归中是下一次递归的基础,由于产生了新变量才使得递归或迭代继续进行。

/**
 * 复制文件夹
 * @param $source
 * @param $dest
 * @return string
 */
function copydir($source, $dest)
{
  /* 初始条件 */
  $stack = array();
  $target = '';
  if (file_exists($source)) {
    if (!file_exists($dest)) mkdir($dest);
    $source = realpath($source) . '/';
    $dest = realpath($dest) . '/';
    $target = realpath($dest);
    array_push($stack, '');
  }
  /* 迭代条件 */
  while (count($stack) !== 0) {
    $dir = array_pop($stack);
    $handle = opendir($source . $dir);
    if (!file_exists($dest . $dir)) mkdir($dest . $dir);
    /* 执行过程 */
    while (($item = readdir($handle)) !== false) {
      if ($item == '.' || $item == '..') continue;
      $_source = $source . $dir . $item;
      $_dest = $dest . $dir . $item;
      if (is_file($_source)) copy($_source, $_dest);
      /* 更新条件 */
      if (is_dir($_source)) array_push($stack, $dir . $item . '/');
    }
    closedir($handle);
  }
  return $target;
}

删除文件夹

抛开语言特性影响性能最多的就是冗余代码了,冗余代码通常是由于设计不到位而产生的。

多数情况下递归要比迭代冗余代码更多,这也是造成递归效率低的一大因素。

但当递归代码足够简练,冗余度足够低时,迭代的性能未必就比递归高。

比如这个用迭代实现的文件夹删除函数,速度就比递归要慢20%,主要原因是空文件夹的判断,在递归中当文件夹没有子文件夹时,函数会直接删除所有文件和当前文件夹,递归结束。

在迭代中即使文件夹为空也需要将其存入堆栈,下次迭代时再判断是否为空,之后才能删除。这就相比递归多了判断文件为空、存入堆栈、取出迭代等冗余操作,所以处理速度会比递归更慢。

/**
 * 删除文件夹
 * @param $path
 * @return bool
 */
function rmdirs($path)
{
  /* 初始化条件 */
  $stack = array();
  if (!file_exists($path)) return false;
  $path = realpath($path) . '/';
  array_push($stack, '');
  /* 迭代条件 */
  while (count($stack) !== 0) {
    $dir = end($stack);
    $items = scandir($path . $dir);
    /* 执行过程 */
    if (count($items) === 2) {
      rmdir($path . $dir);
      array_pop($stack);
      continue;
    }
    /* 执行过程 */
    foreach ($items as $item) {
      if ($item == '.' || $item == '..') continue;
      $_path = $path . $dir . $item;
      if (is_file($_path)) unlink($_path);
      /* 更新条件 */
      if (is_dir($_path)) array_push($stack, $dir . $item . '/');
    }
  }
  return !(file_exists($path));
}

查看执行时间

这是一个查看代码执行时间(毫秒数)的函数,通过回调方式执行目标代码(或函数),最终计算出执行的时间(毫秒)。通过这个工具可以对比函数之间的性能差距,非常简单实用的一个小工具。

/**
 * 函数执行毫秒数
 * @param $func
 * @return int
 */
function exec_time($func)
{
  $start = explode(' ', microtime());
  $func();// 执行耗时操作
  $end = explode(' ', microtime());
  $sec_time = floatval($end[0]) - floatval($start[0]);
  $mic_time = floatval($end[1]) - floatval($start[1]);
  return intval(($sec_time + $mic_time) * 1000);
}
echo exec_time(function () {
  /* 执行的耗时操作 */
});

更多关于PHP相关内容感兴趣的读者可查看本站专题:《PHP目录操作技巧汇总》、《php文件操作总结》、《PHP常用遍历算法与技巧总结》、《PHP数据结构与算法教程》、《php程序设计算法总结》、《PHP数组(Array)操作技巧大全》及《php字符串(string)用法总结》

希望本文所述对大家PHP程序设计有所帮助。


# PHP  # 迭代  # 文件夹  # 复制  # 删除  # 查看大小  # PHP递归实现文件夹的复制、删除、查看大小操作示例  # PHP文件及文件夹操作之创建、删除、移动、复制  # PHP创建/删除/复制文件夹、文件  # PHP实现递归复制整个文件夹的类实例  # php删除与复制文件夹及其文件夹下所有文件的实现代码  # PHP操作文件类的函数代码(文件和文件夹创建  # 移动和删除)  # 探讨PHP删除文件夹的三种方法  # PHP 删除文件与文件夹操作 unlink()与rmdir()这两个函数的使用  # php定时删除文件夹下文件(清理缓存文件)  # php删除文件夹及其文件夹下所有文件的函数代码  # php使用递归与迭代实现快速排序示例  # 递归  # 这一  # 很好  # 为空  # 执行时间  # 操作技巧  # 程序设计  # 就比  # 实际操作  # 都是  # 相关内容  # 多个  # 将会  # 最多  # 是指  # 遍历  # 而在  # 将其  # 感兴趣 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Windows驱动无法加载错误解决方法_驱动签名验证失败处理步骤  如何快速搭建安全的FTP站点?  如何在局域网内绑定自建网站域名?  如何正确下载安装西数主机建站助手?  VIVO手机上del键无效OnKeyListener不响应的原因及解决方法  夸克浏览器网页跳转延迟怎么办 夸克浏览器跳转优化  谷歌浏览器如何更改浏览器主题 Google Chrome主题设置教程  文字头像制作网站推荐软件,醒图能自动配文字吗?  简历在线制作网站免费版,如何创建个人简历?  android nfc常用标签读取总结  Laravel如何实现多级无限分类_Laravel递归模型关联与树状数据输出【方法】  Laravel如何实现本地化和多语言支持?(i18n教程)  Laravel如何使用Collections进行数据处理?(实用方法示例)  如何用JavaScript实现文本编辑器_光标和选区怎么处理  Laravel怎么使用Session存储数据_Laravel会话管理与自定义驱动配置【详解】  JS去除重复并统计数量的实现方法  如何快速上传建站程序避免常见错误?  如何在Windows虚拟主机上快速搭建网站?  Laravel如何使用Socialite实现第三方登录?(微信/GitHub示例)  东莞市网站制作公司有哪些,东莞找工作用什么网站好?  Google浏览器为什么这么卡 Google浏览器提速优化设置步骤【方法】  Laravel怎么实现搜索功能_Laravel使用Eloquent实现模糊查询与多条件搜索【实例】  微博html5版本怎么弄发语音微博_语音录制入口及时长限制操作【教程】  python中快速进行多个字符替换的方法小结  Laravel如何安装使用Debugbar工具栏_Laravel性能调试与SQL监控插件【步骤】  微信h5制作网站有哪些,免费微信H5页面制作工具?  详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)  在线教育网站制作平台,山西立德教育官网?  高端建站三要素:定制模板、企业官网与响应式设计优化  零基础网站服务器架设实战:轻量应用与域名解析配置指南  HTML5段落标签p和br怎么选_文本排版常用标签对比【解答】  如何在Windows 2008云服务器安全搭建网站?  Laravel Octane如何提升性能_使用Laravel Octane加速你的应用  百度浏览器网页无法复制文字怎么办 百度浏览器复制修复  php后缀怎么变mp4格式错误_修改扩展名提示格式不对怎么办【技巧】  Laravel怎么清理缓存_Laravel optimize clear命令详解  西安专业网站制作公司有哪些,陕西省建行官方网站?  Laravel如何编写单元测试和功能测试?(PHPUnit示例)  Laravel Admin后台管理框架推荐_Laravel快速开发后台工具  网页设计与网站制作内容,怎样注册网站?  如何在万网开始建站?分步指南解析  Android okhttputils现在进度显示实例代码  EditPlus中的正则表达式实战(6)  Laravel表单请求验证类怎么用_Laravel Form Request分离验证逻辑教程  如何在宝塔面板中修改默认建站目录?  如何在 Python 中将列表项按字母顺序编号(a.、b.、c. …)  IOS倒计时设置UIButton标题title的抖动问题  nodejs redis 发布订阅机制封装实现方法及实例代码  laravel怎么为应用开启和关闭维护模式_laravel应用维护模式开启与关闭方法  如何利用DOS批处理实现定时关机操作详解