我需要将范围中的日期范围数组拆分为其他日期范围。
例如,我的start数组可以是:
$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;
}
}
}
}
但结果是一个包含更多我需要的结果的数组。谢谢你的帮助。
答案 0 :(得分:0)
老实说,我会让这个过程分为多步......虽然我确信它可以一步完成,但代码可能 - 而且可能已经 - 令人困惑如地狱(因为我不喜欢#39;在几秒钟内理解你的代码):
我将从一个传递开始,我将所有to+1
日期和所有from
日期存储到一个数组中并对其进行排序。
现在你有间隔。第一个条目是from
,必须与最小的from
日期匹配。下一个日期可以是to+1
或from
日期。无论哪种方式,间隔都是从from
到下一个间隔减去1.下一个间隔从该值开始,然后上升到下一个-1,依此类推。
下一步,将原始间隔排序到新的间隔中。 (并过滤掉那些不包含任何内容的间隔。)
除非性能/存储是一个非常大的问题,否则这可能已经足够好了,并且可能会被您的代码继承的任何人理解。