gpt4 book ai didi

algorithm - 寻找 George Marsaglia 的 XorShift RNG 的逆运算

转载 作者:塔克拉玛干 更新时间:2023-11-03 02:52:13 26 4
gpt4 key购买 nike

摘要

您好,假设您有 128 位自动机(由四个 32 位字表示 XYZW )根据以下规则更改其状态:

X = ...
Y = ...
Z = ...
W = ...

void next()
{
var t = X ^ (X << 11);

X = Y;
Y = Z;
Z = W;

W = W ^ (W >> 19) ^ (t ^ (t >> 8));
}

^ - 表示二进制 XOR操作

<< - 表示二进制左移操作

>> - 表示二进制右移操作

保证上述自动机不会产生冲突,即每个状态都是一个(且只有一个)先前状态的结果。还保证上述状态机产生 2^128 个唯一状态。

问题

对于任何给定状态 (X,Y,Z,W)产生与next相反的结果, (即 prev )操作会将状态恢复到之前的状态。

换句话说,如果你有以下状态(X=1, Y=2, Z=3, W=4)并将调用next , 状态将变为 (X=2, Y=3, Z=4, W=2061) ,假设在调用 prev 之后状态应该再次等于 (X=1, Y=2, Z=3, W=4) .

附言

next操作是 George Marsaglia 发现的 XorShift 伪随机数生成器的实现之一

https://en.wikipedia.org/wiki/Xorshift

此操作的逆运算通常非常有用,请考虑 Guid.Next(...)、Guid.Prev(...) 可用性的含义


编辑

我稍微改进了 Niklas B. 的原始答案并将结果移植到 C#,所以这是最后一段代码,希望有人能从 Random.Next() 和 Random.Prev() 操作中受益:

public class Xor128
{
public UInt32 X { get; set; }
public UInt32 Y { get; set; }
public UInt32 Z { get; set; }
public UInt32 W { get; set; }

public Xor128()
{

}

public Xor128(UInt32 x, UInt32 y, UInt32 z, UInt32 w)
{
X = x;
Y = y;
Z = z;
W = w;
}

//private UInt32 UnXorShl(UInt32 x, Int32 shift)
//{
// for (var i = shift; i < 32; i <<= 1) {
// x ^= x << i;
// }

// return x;
//}

//private UInt32 UnXorShr(UInt32 x, Int32 shift)
//{
// for (var i = shift; i < 32; i <<= 1) {
// x ^= x >> i;
// }

// return x;
//}

//public UInt32 Prev()
//{
// var t = UnXorShr(W ^ Z ^ (Z >> 19), 8);

// W = Z;
// Z = Y;
// Y = X;
// X = UnXorShl(t, 11);

// return W;
//}

public UInt32 Prev()
{
var t = W ^ Z ^ (Z >> 19);

t ^= t >> 8;
t ^= t >> 16;

W = Z;
Z = Y;
Y = X;

t ^= t << 11;
t ^= t << 22;

X = t;

return W;
}


public UInt32 Curr()
{
return W;
}

public UInt32 Next()
{
UInt32 t = X ^ (X << 11);

X = Y;
Y = Z;
Z = W;

return W = W ^ (W >> 19) ^ (t ^ (t >> 8));
}
}

顺便说一句。这是一个快速版本:

public class Xor128 {
public var X: UInt32
public var Y: UInt32
public var Z: UInt32
public var W: UInt32

public convenience init(uuid: uuid_t) {
let xa = (UInt32(uuid.0 ) << 24)
let xb = (UInt32(uuid.1 ) << 16)
let xc = (UInt32(uuid.2 ) << 8 )
let xd = (UInt32(uuid.3 ) << 0 )

let ya = (UInt32(uuid.4 ) << 24)
let yb = (UInt32(uuid.5 ) << 16)
let yc = (UInt32(uuid.6 ) << 8 )
let yd = (UInt32(uuid.7 ) << 0 )

let za = (UInt32(uuid.8 ) << 24)
let zb = (UInt32(uuid.9 ) << 16)
let zc = (UInt32(uuid.10) << 8 )
let zd = (UInt32(uuid.11) << 0 )

let wa = (UInt32(uuid.12) << 24)
let wb = (UInt32(uuid.13) << 16)
let wc = (UInt32(uuid.14) << 8 )
let wd = (UInt32(uuid.15) << 0)

self.init(
x: xa + xb + xc + xd,
y: ya + yb + yc + yd,
z: za + zb + zc + zd,
w: wa + wb + wc + wd
)
}

public convenience init(uuid: UUID) {
self.init(uuid: uuid.uuid)
}

public init(x: UInt32, y: UInt32, z: uint32, w: UInt32) {
X = x
Y = y
Z = z
W = w
}

@discardableResult
public func next() -> UInt32 {
let t = X ^ (X << 11);

X = Y;
Y = Z;
Z = W;

W = W ^ (W >> 19) ^ (t ^ (t >> 8))

return W;
}

public var curr: UInt32 {
return W
}

@discardableResult
public func prev() -> UInt32 {
var t = W ^ Z ^ (Z >> 19);

t ^= t >> 8;
t ^= t >> 16;

W = Z;
Z = Y;
Y = X;

t ^= t << 11;
t ^= t << 22;

X = t;

return W;
}
}

最佳答案

您需要的基本构建 block 是一种通过左移运算反转 XOR 的算法 f(x) = x ^ (x << s)对于某些 s > 0。给定 f(x),您已经直接知道 x 的低 s 位。

您可以从低位到高位迭代地重建其余位,因为您已经知道在每个点上已经异或得到 f(x) 位的两个位。这是 Python 中的示例:

def reverse_xor_lshift(y, shift, w=32):
x = y & ((1<<shift) - 1)
for i in range(w - shift):
x |= (1 if bool(x & (1<<i)) ^ bool(y & (1<<(shift+i))) else 0)<<(shift+i)
return x

现在剩下的就变得相当容易了。请注意,我正在为右移模拟重用左移反转:

def reverse_bin(x, w=32):
return int(bin(x)[2:].rjust(w, '0')[::-1], 2)

def reverse_xor_rshift(y, shift, w=32):
# for simplicity, we just reuse reverse_xor_lshift here
return reverse_bin(reverse_xor_lshift(reverse_bin(y), shift))

def forward(X, Y, Z, W):
t = (X ^ (X << 11)) & 0xffffffff
X = Y
Y = Z
Z = W
W = W ^ (W >> 19) ^ (t ^ (t >> 8))
return (X, Y, Z, W)

def backward(X, Y, Z, W):
t = reverse_xor_rshift(W ^ Z ^ (Z >> 19), 8)
return (reverse_xor_lshift(t, 11), X, Y, Z)

backward是反转状态转换的函数。一些随机测试:

import random
for _ in range(1000):
X, Y, Z, W = [random.randint(0,2**32-1) for _ in range(4)]
assert backward(*forward(X,Y,Z,W)) == (X, Y, Z, W)

似乎有效。

关于algorithm - 寻找 George Marsaglia 的 XorShift RNG 的逆运算,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31513168/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com