php - 将日期范围拆分为多个部分

时间:2016-07-19 06:40:40

标签: php

我需要将范围中的日期范围数组拆分为其他日期范围。

例如,我的s​​tart数组可以是:

$items = array(
    array(
        'ID' => 2,
        'from' => new DateTime('2016-07-01'),
        'to' => new DateTime('2016-07-19')
    ),
    array(
        'ID' => 3,
        'from' => new DateTime('2016-07-15'),
        'to' => new DateTime('2016-07-29')
    ),
    array(
        'ID' => 1,
        'from' => new DateTime('2016-01-01'),
        'to' => new DateTime('2016-12-31')
    ),
    array(
        'ID' => 4,
        'from' => new DateTime('2016-06-15'),
        'to' => new DateTime('2016-07-07')
    ),
    array(
        'ID' => 5,
        'from' => new DateTime('2016-07-05'),
        'to' => new DateTime('2016-07-17')
    ),
);

我期望的结果是一个数组,其中所有句点都参考了项目的ID,如下所示:

$result = array(
    array(
        'from' => new DateTime('2016-01-01'),
        'to' => new DateTime('2016-06-14'),
        'ids' => array(
            1
        )
    ),
    array(
        'from' => new DateTime('2016-06-15'),
        'to' => new DateTime('2016-06-30'),
        'ids' => array(
            1, 4
        )
    ),
    array(
        'from' => new DateTime('2016-07-01'),
        'to' => new DateTime('2016-07-04'),
        'ids' => array(
            1, 2, 4
        )
    ),
    array(
        'from' => new DateTime('2016-07-05'),
        'to' => new DateTime('2016-07-07'),
        'ids' => array(
            1, 2, 4, 5
        )
    ),
    array(
        'from' => new DateTime('2016-07-08'),
        'to' => new DateTime('2016-07-14'),
        'ids' => array(
            1, 2, 5
        )
    ),
    array(
        'from' => new DateTime('2016-07-15'),
        'to' => new DateTime('2016-07-17'),
        'ids' => array(
            1, 2, 3, 5
        )
    ),
    array(
        'from' => new DateTime('2016-07-18'),
        'to' => new DateTime('2016-07-19'),
        'ids' => array(
            1, 2, 3
        )
    ),
    array(
        'from' => new DateTime('2016-07-20'),
        'to' => new DateTime('2016-07-29'),
        'ids' => array(
            1, 3
        )
    ),
    array(
        'from' => new DateTime('2016-07-30'),
        'to' => new DateTime('2016-12-31'),
        'ids' => array(
            1
        )
    ),
);
你能帮帮我吗?感谢

更新 现在我试图通过'from'项目和循环项目以及范围数组来排序数组。这是我的代码:

usort($items, function($a, $b) {
    if ($a['from'] == $b['from']) {
        return 0;
    }
    if ($a['from'] < $b['from']) {
        return -1;
    }
    return 1;
});

$ranges = array(
    array(
        'from' => clone $items[0]['from'],
        'to' => clone $items[0]['to'],
        'ids' => array($items[0]['ID'])
    )
);
unset($items[0]);
foreach($items as $item) {
    foreach($ranges as $k => $range) {
        if (($range['from'] <= $item['from']) || ($range['to'] > $item['to'])) {
            if ($range['from'] <= $item['from']) {
                $clone = $range;
                $clone['from'] = $item['from'];
                $clone['to'] = clone $item['to'];
                $clone['ids'][] = $item['ID'];

                $ranges[$k]['to'] = clone $item['from'];
                $ranges[$k]['to']->modify('-1 day');
                $ranges[] = $clone;
            }

            if ($range['to'] > $item['to']) {
                $clone = $range;
                $clone['from'] = clone $item['to'];
                $clone['from']->modify('+1 day');
                $clone['to'] = clone $range['to'];
                $ranges[] = $clone;
            }
        }
    }
}

但结果是一个包含更多我需要的结果的数组。谢谢你的帮助。

1 个答案:

答案 0 :(得分:0)

老实说,我会让这个过程分为多步......虽然我确信它可以一步完成,但代码可能 - 而且可能已经 - 令人困惑如地狱(因为我不喜欢#39;在几秒钟内理解你的代码):

我将从一个传递开始,我将所有to+1日期和所有from日期存储到一个数组中并对其进行排序。

现在你有间隔。第一个条目是from,必须与最小的from日期匹配。下一个日期可以是to+1from日期。无论哪种方式,间隔都是从from到下一个间隔减去1.下一个间隔从该值开始,然后上升到下一个-1,依此类推。

下一步,将原始间隔排序到新的间隔中。 (并过滤掉那些不包含任何内容的间隔。)

除非性能/存储是一个非常大的问题,否则这可能已经足够好了,并且可能会被您的代码继承的任何人理解。