计算房间在选定日期范围内的动态总价:支持按日期覆盖价格与默认价回退

计算房间在选定日期范围内的动态总价:支持按日期覆盖价格与默认价回退

本文介绍如何为多个房间批量计算指定日期列表的总价格,逻辑为:对每个房间遍历所有选中日期,若该日期在 `options` 中存在则使用对应价格,否则回退至 `default_price`,最终汇总各房间总价。

在酒店预订、共享住宿等场景中,常需根据用户选择的连续日期(如入住/退房区间),为不同房型动态计算总价。由于部分日期可能设置了促销价或旺季加价(即 options 中的自定义价格),而其余日期沿用基础价(default_price),因此不能简单套用固定单价,必须逐日匹配并累加。

以下是一个简洁、健壮且无副作用的实现方案(兼容 PHP 7.4+,无需额外依赖):

$selectedDates = ['10-04-2022', '11-04-2022', '12-04-2022'];

$setRooms = [
    [
        'id'            => 1,
        'title'         => 'Room1',
        'default_price' => 50,
        'options'       => [
            ['date' => '12-04-2022', 'price' => 100],
            ['date' => '13-04-2022', 'price' => 200],
            ['date' => '14-04-2022', 'price' => 200],
        ]
    ],
    [
        'id'            => 2,
        'title'         => 'Room2',
        'default_price' => 120,
        'options'       => [
            ['date' => '11-04-2022', 'price' => 200],
            ['date' => '12-04-2022', 'price' => 300],
            ['date' => '13-04-2022', 'price' => 400],
        ]
    ],
];

$result = [];
foreach ($setRooms as $room) {
    $total = 0;
    foreach ($selectedDates as $date) {
        // 精确匹配日期(字符串直比,避免 Carbon 解析开销;确保输入格式统一)
        $matchedOption = null;
        foreach ($room['options'] as $option) {
            if ($option['date'] === $date) {
                $matchedOption = $option;
                break;
            }
        }

        $total += $matchedOption ? $matchedOption['price'] : $room['default_price'];
    }
    $result[] = [
        'id'         => $room['id'],
        'title'      => $room['title'],
        'total_price' => $total
    ];
}

print_r($result);

输出结果

ArrowMancer

ArrowMancer

手机上的宇宙动作RPG,游戏角色和元素均为AI生成

下载

Array
(
    [0] => Array
        (
            [id] => 1
            [title] => Room1
            [total_price] => 200   // 50 + 50 + 100
        )
    [1] => Array
        (
            [id] => 2
            [title] => Room2
            [total_price] => 620   // 120 + 200 + 300
        )
)

关键说明与最佳实践

  • 避免变量污染:原代码中 $price 在外层循环中未重置,导致后一个房间累加了前一个房间的结果。本方案在每个房间内独立声明 $total = 0,彻底规避该错误。
  • 性能优化:使用原生 foreach 查找匹配项,比 array_filter() + array_key_first() 更高效(尤其在 options 数量较多时),且语义清晰、易于调试。
  • 日期格式一致性:示例中所有日期均为 ‘d-m-Y’ 格式(如 ’12-04-2022’)。务必确保 $selectedDates 与 options.date 格式完全一致,否则字符串比较会失败。如需兼容多种输入格式,建议在预处理阶段统一标准化(例如用 Carbon::parse($date)->format(‘d-m-Y’))。
  • 扩展性提示:若后续需支持时间范围(如“4月10日–4月12日”)、节假日规则或多币种,可将价格查找逻辑封装为独立方法(如 getPriceForDate($room, $date)),便于单元测试和复用。

该方案结构清晰、零外部依赖、易于维护,适用于中小型预订系统的价格聚合场景。

https://www.php.cn/faq/2014820.html

发表回复

Your email address will not be published. Required fields are marked *