- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试仅使用单个一维数组将数组左移 n 个位置。我可以在两个数组中完成它,但我还没有想出如何使用一个数组来完成它。请提出您的建议
最佳答案
实际上有一个聪明的算法。我们将使用 A
表示数组,N
表示数组大小,和 n
表示要移动的位置数。换类后您想要 i-th
要移动到 ((i + n) mod N)-th
的元素位置,因此我们可以通过以下映射定义新位置:
f(j) := (j + n) mod N (j = 0,...,N - 1)
i
处的元素开始.我们要做的是将元素移动到位置
i
到位置
f(i)
,但随后我们将覆盖该位置的元素,因此我们需要先保存位置
f(i)
处的元素然后执行移位。一旦我们移动了第一个元素,我们需要选择另一个元素来移动。由于我们想要节省空间,显而易见的候选对象是我们刚刚保存的元素(位于
f(i)
中的元素)。和以前一样,我们将元素保存在位置
f(f(i))
然后将我们保存的元素复制到那个位置。我们不断重复这个过程(经过位置
i, f(i), f(f(i)), f(f(f(i))), ...
),直到我们到达一个我们已经移动的元素(我们保证这样做,因为有有限多个位置)。如果我们通过了所有元素,那么我们就完成了,如果没有,那么我们选择另一个元素(还没有被移动),比如位置
j
,并重复该过程(通过
j, f(j), f(f(j)), f(f(f(j))), ...
)。就是这样。但是在我们可以实现这样的算法之前,或者甚至在我们确定这是否确实是一个好的算法之前,我们需要回答几个问题:
i, f(i), f(f(i)), ...
.我们怎么知道我们到达了一个已经转移的位置?我们是否需要保存我们经过的每个位置?如果我们这样做,那么这意味着我们需要保存一个大小为 N 的数组(以覆盖所有位置),并且我们还需要在每次移动元素时执行查找。这将使算法非常低效。幸运的是,这不是必需的,因为序列 i, f(i), f(f(i)), ...
必须在位置 i
处环绕自身,所以我们只需要等到我们到达那个位置。我们可以如下证明这个断言:假设我们遇到的第一个重复元素不是 i
.那么我们必须有 2 个不同的元素,当移动时会到达相同的位置 - 一个矛盾。 i, f(i), f(f(i)), ...
,但仍有元素未移动(我们可以通过计算移动了多少个元素来判断)。我们现在如何找到位置 j
包含这样的元素?而且,一旦我们完成第二次迭代(通过 j, f(j), f(f(j)), ...
)我们如何找到第三个位置 k
带有未移位的元素?等等.. 这也可能表明我们需要保存一个数组来说明使用过的\未使用的元素,并在每次需要查找未使用的元素时执行查找。然而,我们可以再次放松,因为我们很快就会展示,所有的起始位置(我们用 i
、 j
和 k
表示)都是相邻的。这意味着,如果我们从位置 i
开始,我们接下来选择 i + 1
,然后 i + 2
,等等…… i, f(i), f(f(i)), ...
和 j, f(j), f(f(j)), ...
(其中 i
和 j
不同)包含共同元素?如果他们这样做将意味着该算法是无用的,因为它可以将同一个元素移动两次 - 导致它最终处于错误的位置。那么答案(当然)是它们不能包含公共(public)元素。我们将说明原因。 d := gcd(N, n)
.对于每一个整数:
i = 0,...,d - 1
我们定义了以下集合:
S(i) := { kd + i | k = 0,...,N/d - 1}
S(0),...,S(d - 1)
合盖套
{0,...,N - 1}
.我们还观察到,在对集合中的元素进行划分时
S(i)
来自
d
,我们只剩下余数
i
,并将元素从不同的集合中划分
S(j)
来自
d
会给我们留下不同的余数(
j
)。因此,没有两个集合包含一个公共(public)元素。有了这个,我们已经建立了集合
S(0),...,S(d - 1)
形成
{0,...,N - 1}
的分区
i = 0,...,d - 1
,我们将定义集合
T(i)
如
i, f(i), f(f(i)), ...
.根据
f
的定义我们可以写
T(i)
如下:
T(i) = {(kn + i) mod N | k is an integer}
x
是
T(i)
中的一个元素,那么我们可以写一些
k
:
x = (kn + i) mod N = (k(n/d)d + i) mod N
z := k(n/d) mod N/d
,然后乘以
d
, 我们有:
kn mod N = zd
x = (kn + i) mod N = zd + i
x
也在
S(i)
.同样,如果我们取一些
y
来自
S(i)
我们观察到,对于一些
k
:
y = kd + i
gcd(n/d, N/d) = 1
存在一个
q
使得
q(n/d) mod N/d = 1
(模逆),因此我们可以写(乘以
kd
):
kd = kqn mod N
y = kd + i = ((kq)n + i) mod N
y
也在
T(i)
.我们得出的结论是
T(i) = S(i)
.从这个事实我们可以很容易地展示我们之前的断言。首先,由于集合形成了
{0,...,N - 1}
的分区。满足第三个断言(没有两个序列包含公共(public)元素)。二、由集合的定义
S(i)
我们可以带任何组
d
{0,...N - 1}
中的相邻元素并且它们中的每一个都将被放置在不同的集合中。这满足第二个断言。
0, d, 2d, ..., (N/d - 1)d
中的所有元素通过简单地替换位置
n mod N
处的元素元素位于
0
,位置
2n mod N
处的元素元素位于
n mod N
,依此类推……直到我们返回位置
0
中的元素(我们确信会发生)。下面是一个伪代码示例:
temp <- A[0]
j <- N - (n mod N)
while j != 0 do
A[(j + n) mod N] <- A[j];
j <- (j - n) mod N
A[n mod N] <- temp;
S(0)
.覆盖其余的集合,即
S(1), … ,S(d-1)
,我们将简单地以与第一个相同的方式迭代每个集合:
for i <- 0 to d - 1
temp <- A[i]
j <- N - ((n - i) mod N)
while j != i do
A[(j + n) mod N] <- A[j];
j <- (j - n) mod N
A[(i + n) mod N] <- temp;
O(1)
空间。 Java 中的实现示例:
public static int gcd(int a, int b) {
while(b != 0) {
int c = a;
a = b;
b = c % a;
}
return a;
}
public static void shift_array(int[] A, int n) {
int N = A.length;
n %= N;
if(n < 0)
n = N + n;
int d = gcd(N, n);
for(int i = 0; i < d; i++) {
int temp = A[i];
for(int j = i - n + N; j != i; j = (j - n + N) % N)
A[(j + n) % N] = A[j];
A[i + n] = temp;
}
}
关于java - 在java中将数组循环左移n个位置,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11893053/
我是 PHP 新手。我一直在脚本中使用 for 循环、while 循环、foreach 循环。我想知道 哪个性能更好? 选择循环的标准是什么? 当我们在另一个循环中循环时应该使用哪个? 我一直想知道要
我在高中的编程课上,我的作业是制作一个基本的小计和顶级计算器,但我在一家餐馆工作,所以制作一个只能让你在一种食物中读到。因此,我尝试让它能够接收多种食品并将它们添加到一个价格变量中。抱歉,如果某些代码
这是我正在学习的一本教科书。 var ingredients = ["eggs", "milk", "flour", "sugar", "baking soda", "baking powder",
我正在从字符串中提取数字并将其传递给函数。我想给它加 1,然后返回字符串,同时保留前导零。我可以使用 while 循环来完成此操作,但不能使用 for 循环。 for 循环只是跳过零。 var add
编辑:我已经在程序的输出中进行了编辑。 该程序要求估计给定值 mu。用户给出一个值 mu,同时还提供了四个不等于 1 的不同数字(称为 w、x、y、z)。然后,程序尝试使用 de Jaeger 公式找
我正在编写一个算法,该算法对一个整数数组从末尾到开头执行一个大循环,其中包含一个 if 条件。第一次条件为假时,循环可以终止。 因此,对于 for 循环,如果条件为假,它会继续迭代并进行简单的变量更改
现在我已经习惯了在内存非常有限的情况下进行编程,但我没有答案的一个问题是:哪个内存效率更高;- for(;;) 或 while() ?还是它们可以平等互换?如果有的话,还要对效率问题发表评论! 最佳答
这个问题已经有答案了: How do I compare strings in Java? (23 个回答) 已关闭 8 年前。 我正在尝试创建一个小程序,我可以在其中读取该程序的单词。如果单词有 6
这个问题在这里已经有了答案: python : list index out of range error while iteratively popping elements (12 个答案) 关
我正在尝试向用户请求 4 到 10 之间的整数。如果他们回答超出该范围,它将进入循环。当用户第一次正确输入数字时,它不会中断并继续执行 else 语句。如果用户在 else 语句中正确输入数字,它将正
我尝试创建一个带有嵌套 foreach 循环的列表。第一个循环是循环一些数字,第二个循环是循环日期。我想给一个日期写一个数字。所以还有另一个功能来检查它。但结果是数字多次写入日期。 Out 是这样的:
我想要做的事情是使用循环创建一个数组,然后在另一个类中调用该数组,这不会做,也可能永远不会做。解决这个问题最好的方法是什么?我已经寻找了所有解决方案,但它们无法编译。感谢您的帮助。 import ja
我尝试创建一个带有嵌套 foreach 循环的列表。第一个循环是循环一些数字,第二个循环是循环日期。我想给一个日期写一个数字。所以还有另一个功能来检查它。但结果是数字多次写入日期。 Out 是这样的:
我正在模拟一家快餐店三个多小时。这三个小时分为 18 个间隔,每个间隔 600 秒。每个间隔都会输出有关这 600 秒内发生的情况的统计信息。 我原来的结构是这样的: int i; for (i=0;
这个问题已经有答案了: IE8 for...in enumerator (3 个回答) How do I check if an object has a specific property in J
哪个对性能更好?这可能与其他编程语言不一致,所以如果它们不同,或者如果你能用你对特定语言的知识回答我的问题,请解释。 我将使用 c++ 作为示例,但我想知道它在 java、c 或任何其他主流语言中的工
这个问题不太可能帮助任何 future 的访问者;它只与一个小的地理区域、一个特定的时间点或一个非常狭窄的情况有关,这些情况并不普遍适用于互联网的全局受众。为了帮助使这个问题更广泛地适用,visit
我是 C 编程和编写代码的新手,以确定 M 测试用例的质因数分解。如果我一次只扫描一次,该功能本身就可以工作,但是当我尝试执行 M 次时却惨遭失败。 我不知道为什么 scanf() 循环有问题。 in
这个问题已经有答案了: JavaScript by reference vs. by value [duplicate] (4 个回答) 已关闭 3 年前。 我在使用 TSlint 时遇到问题,并且理
我尝试在下面的代码中添加 foreach 或 for 循环,以便为 Charts.js 创建多个数据集。这将允许我在此折线图上创建多条线。 我有一个 PHP 对象,我可以对其进行编码以稍后填充变量,但
我是一名优秀的程序员,十分优秀!