如何在 PHP 中仅用 for 循环获取当前项的前一项与后两项(带边界限制)

本文讲解如何在不使用内置函数(如 range())的前提下,仅通过 for 循环和纯数学公式,动态计算并输出以当前索引为中心、包含前 1 项与后 2 项(共 4 项)的序列,同时严格满足页码范围 [1, $pager_max] 的边界约束。

要实现「显示当前项的前一项 + 当前项 + 后两项」共 4 个连续整数,并确保所有值始终落在 [1, $pager_max] 范围内,关键在于动态确定循环起始点(start)和终止点(end),而非对每个 $i 做条件过滤(如原代码中 if ($i > ... && $i

正确思路是:先通过纯算术公式推导出合法的最小起始值和最大结束值,再用 for 循环遍历该区间。

✅ 推导逻辑(无函数,仅公式)

设:

  • $pager_max:总页数上限(正整数)
  • $current:当前选中页码(1 ≤ $current ≤ $pager_max)

我们需要输出 4 个数:[prev, current, next1, next2] → 即 [c−1, c, c+1, c+2]
但需强制截断至 [1, $pager_max]:

  • 左边界:不能小于 1 → start = max(1, $current - 1)
  • 右边界:不能大于 $pager_max → end = min($pager_max, $current + 2)

但注意:我们必须保证恰好输出 4 个数(除非边界严重受限,如 $pager_max

$current 期望输出 实际跨度 起始 start 结束 end
1 1,2,3,4 1→4 1 4
2 1,2,3,4 1→4 1 4
3 2,3,4,5 2→5 2 5
7 5,6,7,8 5→8 5 8
8 5,6,7,8 5→8 5 8

可见:

  • 当 $current ≤ 2:start = 1,end = 4
  • 当 $current ≥ $pager_max - 1(即 $current ≥ 7 当 $pager_max=8):start = $pager_max - 3,end = $pager_max
  • 其他情况:start = $current - 1,end = $current + 2

统一为两个公式(不依赖 if/else,仅用 min/max 运算):

$start = max(1, min($current - 1, $pager_max - 3));
$end   = min($pager_max, max($current + 2, 4));

但题干要求“只用 for 循环,不用函数”,而 max()/min() 是语言内置函数。因此,我们必须用纯比较运算模拟:

✅ 正确且符合题意的 for 循环解法(零函数,纯公式+条件判断):

= $pager_max - 1) { // 即 current == pager_max 或 pager_max-1
    $start = $pager_max - 3;
    $end   = $pager_max;
} else {
    $start = $current - 1;
    $end   = $current + 2;
}

// 用 for 循环输出
for ($i = $start; $i <= $end; $i++) {
    // 高亮当前项
    $output = ($i == $current) ? "$i" : $i;
    echo $output . '
'; } ?>
? 为什么原代码失效? 原条件 if ($i > ($current - $pager_max ) + 6 && $i 1,漏掉 i=1;且未处理 $current 接近边界的场景,缺乏兜底机制。

✅ 通用性验证(更换 $pager_max)

将 $pager_max = 12,$current = 10:

  • $current >= $pager_max - 1 → 10 >= 11?否 → 进入 else → start=9, end=12 → 输出 9,10,11,12 ✔
    若 $current = 11:11 >= 11 → true → start = 12-3 = 9, end = 12 → 9,10,11,12 ✔
    若 $current = 1:start=1, end=4 → 1,2,3,4 ✔

完全符合题设行为模式。

⚠️ 注意事项

  • 本方案假设 $pager_max ≥ 4(否则无法容纳 4 个数),若需兼容更小值,应前置校验:$pager_max = max(4, $pager_max);
  • 所有计算均基于整数运算,无需浮点或额外库;
  • 高亮当前项使用 HTML ,如需纯文本,可改为 [$i] 或其他标记;
  • 该解法时间复杂度 O(1),空间 O(1),高效且可读性强。

通过精准的边界分类与算术推导,我们摆脱了模糊的循环内判断,用清晰、可维护的结构实现了鲁棒的分页上下文渲染。