Hackerrank圆形阵列旋转在php中无法进行100,000次旋转测试

时间:2016-09-07 09:04:21

标签: php arrays rotation

我的第一个解决方案,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";
}

?>

1 个答案:

答案 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 可以实现同样的效果。