如何从时间戳数组中获取最早和最晚时钟时间的原始时间戳

聖光之護
发布: 2025-09-27 22:22:01
原创
720人浏览过

如何从时间戳数组中获取最早和最晚时钟时间的原始时间戳

本文详细介绍了如何从一个包含大量时间戳的数组中,高效地找出对应最早和最晚“时钟时间”(即一天中的时间,不考虑日期)的原始完整时间戳。通过利用 PHP 的 array_reduce 函数,结合对时间字符串的比较,我们能够精准定位并返回这些特定时间点的原始日期和时间信息,避免了常见方法中日期被重置的问题。

场景与挑战

在处理包含大量时间戳的数据时,一个常见的需求是识别出一天中最早和最晚的“时钟时间”所对应的原始完整时间戳。例如,我们可能有一个包含数千个不同日期和时间的时间戳数组,需要找出其中时间部分(如“00:00:01 am”或“11:59:59 pm”)最早和最晚的那些记录,并保留其原始的日期信息。

一个直观但存在问题的尝试是先将所有时间戳转换为仅包含时钟时间的部分,然后找出最小值和最大值,例如:

$timestamps = array();
for ($i = 0; $i < 5000; $i++) {
    $timestamps[] = mt_rand(strtotime('1900-01-01 00:00:00 am'), strtotime('2100-12-31 11:59:59 pm'));
}

function callback($timestamp) {
    return strtotime(date('h:i:s a', $timestamp));
}

// 这种方法会返回最早和最晚的时钟时间,但日期会被重置为当前日期
echo date('Y-m-d h:i:s a', min(array_map('callback', $timestamps)));
echo "\n";
echo date('Y-m-d h:i:s a', max(array_map('callback', $timestamps)));
登录后复制

上述方法虽然能找到最早和最晚的时钟时间,但由于 strtotime(date('h:i:s a', $timestamp)) 默认会将日期部分设置为当前日期,导致我们无法获取到原始的时间戳信息。我们需要一种方法,在比较时只考虑时钟时间,但在返回结果时保留原始的完整时间戳。

解决方案:使用 array_reduce 进行高效查找

为了解决上述问题,我们可以利用 PHP 的 array_reduce 函数。array_reduce 能够迭代数组中的每个元素,并使用回调函数将它们归结为一个单一的结果。这使得它非常适合在一次遍历中同时找出最早和最晚的时钟时间对应的原始时间戳。

以下是实现此功能的代码示例:

<?php

// 模拟生成一个包含大量随机时间戳的数组
$timestamps = array();
for ($i = 0; $i < 5000; $i++) {
    $timestamps[] = mt_rand(strtotime('1900-01-01 00:00:00 am'), strtotime('2100-12-31 11:59:59 pm'));
}

/**
 * 从时间戳数组中找出最早和最晚时钟时间对应的原始时间戳。
 *
 * @param array $timestamps 包含Unix时间戳的数组。
 * @return array 包含 'min' 和 'max' 键的数组,每个键对应一个数组,
 *               其中包含 [0] 时钟时间字符串 (HH:ii:ss) 和 [1] 格式化的原始完整时间戳。
 */
$res = array_reduce(
    $timestamps, // 要遍历的时间戳数组
    function($carry, $currentTimestamp) {
        // 1. 从当前时间戳中提取时钟时间字符串 (24小时制,方便比较)
        $currentTimeString = date('H:i:s', $currentTimestamp);
        // 2. 格式化原始完整时间戳,用于最终结果输出
        $formattedOriginalTimestamp = date('Y-m-d h:i:s a', $currentTimestamp);

        // 3. 检查并更新最早时钟时间
        // 如果 $carry['min'][0] 为 null (初始状态) 或当前时间字符串更早
        if (is_null($carry['min'][0]) || $currentTimeString < $carry['min'][0]) {
            $carry['min'] = [$currentTimeString, $formattedOriginalTimestamp];
        }

        // 4. 检查并更新最晚时钟时间
        // 如果 $carry['max'][0] 为 null (初始状态) 或当前时间字符串更晚
        if (is_null($carry['max'][0]) || $currentTimeString > $carry['max'][0]) {
            $carry['max'] = [$currentTimeString, $formattedOriginalTimestamp];
        }

        // 5. 返回更新后的累加器
        return $carry;
    },
    // 初始累加器值:将 'min' 和 'max' 都初始化为包含两个 null 的数组
    // [0] 用于存储时钟时间字符串进行比较,[1] 用于存储对应的原始格式化时间戳
    ['min' => [null, null], 'max' => [null, null]]
);

// 输出结果
print_r($res);

?>
登录后复制

代码解析与工作原理

  1. array_reduce($timestamps, function($carry, $currentTimestamp) { ... }, ['min' =youjiankuohaophpcn [null, null], 'max' => [null, null]]):

    • $timestamps:我们要处理的原始时间戳数组。
    • 匿名函数:这是 array_reduce 的回调函数,它在每次迭代中被调用。
      • $carry:累加器,它在每次迭代中保存上一次回调函数返回的值。初始值是第三个参数 ['min' => [null, null], 'max' => [null, null]]。
      • $currentTimestamp:当前正在处理的时间戳。
    • ['min' => [null, null], 'max' => [null, null]]:累加器的初始值。我们用一个数组来存储 min 和 max,每个又是一个包含两个元素的数组:第一个元素用来存储时钟时间字符串(用于比较),第二个元素用来存储对应的原始格式化时间戳(用于最终输出)。
  2. $currentTimeString = date('H:i:s', $currentTimestamp);:

    • 在每次迭代中,我们首先使用 date('H:i:s', $currentTimestamp) 将当前时间戳格式化为 HH:ii:ss(24小时制)字符串。这种格式的字符串可以直接进行字典序比较,从而判断时间的早晚。
  3. $formattedOriginalTimestamp = date('Y-m-d h:i:s a', $currentTimestamp);:

    美间AI
    美间AI

    美间AI:让设计更简单

    美间AI 45
    查看详情 美间AI
    • 同时,我们将原始的 $currentTimestamp 格式化为我们希望在最终结果中显示的完整日期和时间字符串(例如 1997-05-03 12:00:30 am)。
  4. 比较逻辑 (if (is_null($carry['min'][0]) || $currentTimeString < $carry['min'][0])):

    • 初始化检查: is_null($carry['min'][0]) 用于处理第一次迭代。在第一次迭代中,$carry['min'][0] 和 $carry['max'][0] 都是 null,所以它们会被当前时间戳的值初始化。
    • 最小值更新: 如果当前时钟时间字符串 $currentTimeString 比当前累加器中记录的最小时间 $carry['min'][0] 更早,我们就更新 $carry['min'] 为当前的时间字符串和格式化的原始时间戳。
    • 最大值更新: 同样地,如果当前时钟时间字符串 $currentTimeString 比当前累加器中记录的最大时间 $carry['max'][0] 更晚,我们就更新 $carry['max']。
  5. return $carry;:

    • 每次迭代结束时,回调函数返回更新后的 $carry 数组,这个数组将作为下一次迭代的 $carry 参数。

示例输出

运行上述代码,你将得到类似以下的结果(具体日期和时间取决于随机生成的时间戳):

Array
(
    [min] => Array
        (
            [0] => 00:00:30
            [1] => 1997-05-03 12:00:30 am
        )

    [max] => Array
        (
            [0] => 23:59:36
            [1] => 1983-07-21 11:59:36 pm
        )

)
登录后复制

结果清晰地展示了最早的时钟时间(00:00:30)及其对应的原始完整时间戳 (1997-05-03 12:00:30 am),以及最晚的时钟时间(23:59:36)及其对应的原始完整时间戳 (1983-07-21 11:59:36 pm)。

注意事项与总结

  • 效率: array_reduce 只需要对数组进行一次遍历,这对于处理大型时间戳数组非常高效。
  • 字符串比较: 关键在于将时间戳转换为 HH:ii:ss 格式的字符串进行比较。这种格式确保了字符串的字典序比较结果与时间的早晚顺序一致。
  • 数据结构: 累加器 $carry 的设计至关重要,它同时存储了用于比较的时间字符串和用于输出的原始格式化时间戳,确保了信息的完整性。
  • 时区: 在处理时间戳时,始终要注意服务器的时区设置 (date_default_timezone_set()),以确保 date() 函数返回的时间是预期的。

通过这种 array_reduce 的方法,我们不仅能够精确地找出最早和最晚的时钟时间,还能保留它们原始的日期信息,完美解决了在不丢失上下文的情况下进行时间比较的需求。

以上就是如何从时间戳数组中获取最早和最晚时钟时间的原始时间戳的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号