我的第一个解决方案,POP - >依次取消每个元素的移位,工作正常,但由于超时而失败。
所以我重构了它,现在它没有通过100,000次旋转(500个元素)测试。
我不知道如何解决这个问题,所以它可以很快地运作。
有什么想法吗?
<?php
$handle = fopen("php://stdin", "r");
// n = array length
// k = rotations
// q = # of queries in array
list($n, $k, $q,) = explode(' ', trim(fgets($handle)));
// what's our data
$arr = explode(' ', trim(fgets($handle)));
// pull queries
for($i = 0; $i < $q; $i++) {
$_pos[] = trim(fgets($handle));
}
// NOTE: This is where the test case of 100k rotations on a
// 500 element array fails
// rotate array
$slice = array_slice($arr, -$k, $k);
$arr = array_slice($arr, 0, (count($arr) - $k));
$arr = array_merge($slice, $arr);
// ask the questions
for($i = 0; $i < $q; $i++) {
echo $arr[$_pos[$i]] . "\n";
}
?>
答案 0 :(得分:1)
这是我的解决方案:
//Fetch the input.
$handle = fopen("php://stdin", "r");
fscanf($handle, "%d %d %d", $n, $k, $q);
$inputArray = explode(" ", trim(fgets($handle)));
array_walk($inputArray, 'intval');
//Rotate the array
$totalTimesToRotate = ($k < $n) ? $k : ($k % $n);
if ($totalTimesToRotate > 0) {
$slicedArray = array_splice($inputArray, -$totalTimesToRotate);
$inputArray = array_merge($slicedArray, $inputArray);
}
//For all the queries, return the o/p
for ($i = 0; $i < $q; $i++) {
fscanf($handle, "%d", $m);
echo $inputArray[$m], "\n";
}
说明: 对于hackerrank上提到的示例用例: https://www.hackerrank.com/challenges/circular-array-rotation
3 2 3
1 2 3
0
1
2
所以,n (total array items)
是3
k (total rotations to be performed)
2
q ( total queries)
这一行的目的
$totalTimesToRotate = ($k < $n) ? $k : ($k % $n);
是减少总转数。
假设n
为3,k
为5
First rotation: [3, 1, 2]
Second rotation: [2, 3, 1]
Third rotation: [1, 2, 3]
Fourth rotation: [3, 1, 2]
Fifth rotation: [2, 3, 1]
注意在第五轮和第二轮的结果是如何相同的。
当对长度为n的数组执行n次旋转时,您会获得相同的输出。 因此,使用mod %
操作,我们可以避免不必要地迭代。
array_splice将删除与您要执行操作的次数相等的项目数。
所以你得到[2, 3]
作为输出,inputArray是[1]
在下一步中,[array_merge]将导致[2, 3, 1]
,这是所需的输出。
您的解决方案导致超时的 原因 是因为您使用 array_slice
两次而使用 array_splice
可以实现同样的效果。