gpt4 book ai didi

c# - 我们可以解码 Guid 以找出它是在何时何地生成的吗?

转载 作者:可可西里 更新时间:2023-11-01 08:43:09 27 4
gpt4 key购买 nike

This article说明如何生成 Guid。

我的问题是,有什么方法可以找出我的网络场中哪台机器生成了这个 Guid 以及何时生成的?

最佳答案

Neil Fenwick 是正确的。然而,我们可以利用这种结构来发挥我们的优势。

版本 4 (.Net)

版本 4 UUID 使用仅依赖于随机数的方案。该算法设置版本号以及两个保留位。所有其他位均使用随机或伪随机数据源设置。版本 4 UUID 的格式为 xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx,其中 x 是任何十六进制数字,y 是 8、9、A 或 B 之一。例如f47ac10b-58cc-4372-a567-0e02b2c3d479。

使用版本字段

我们可以随意更改字节 8 的第一个半字节;因此,如果您的机器少于 17 台,您可以通过更改在每台机器上创建的 GUID 来识别它们。

static Guid NewSystemGuid(int machine)
{
if (machine < 0 | machine > 0xF)
throw new ArgumentOutOfRangeException("machine");
var g = Guid.NewGuid();
var arr = g.ToByteArray();
arr[7] = (byte)((machine << 4) | (arr[7] & 0xF));
return new Guid(arr);
}

static int ExtractMachine(Guid guid)
{
var arr = guid.ToByteArray();
return (arr[7] >> 4) & 0xF;
}

使用版本字段和“y”

我不确定更改 Y 是否会改变 GUID 的唯一性,因此您的里程可能会有所不同。如果您的机器少于 17 台,请坚持使用第一种解决方案。

static Guid NewSystemGuid(int machine)
{
if (machine < 0 | machine > 0xFF)
throw new ArgumentOutOfRangeException("machine");

var m1 = machine & 0xF;
var m2 = (machine >> 4) & 0xF;

var g = Guid.NewGuid();
var arr = g.ToByteArray();
arr[7] = (byte)((m1 << 4) | (arr[7] & 0xF));
arr[8] = (byte)((m2 << 4) | (arr[8] & 0xF));
return new Guid(arr);
}

static int ExtractMachine(Guid guid)
{
var arr = guid.ToByteArray();
return
((arr[7] >> 4) & 0xF) |
(((arr[8] >> 4) & 0xF) << 4);
}

使用版本和“y”(Redux)

您仍然可以通过将机器数量限制为 63 台(使用最后 2 位表示“y”的 4 个可能值)来保留“y”中的值:

static Guid NewSystemGuid(int machine)
{
if (machine < 0 | machine > 0x3F)
throw new ArgumentOutOfRangeException("machine");

var m1 = machine & 0xF;
var m2 = (machine >> 4) & 0xF;

var g = Guid.NewGuid();
var arr = g.ToByteArray();
arr[7] = (byte)((m1 << 4) | (arr[7] & 0xF));

var y = (arr[8] >> 4) & 0xF;
switch (y)
{
case 0x8:
arr[8] = (byte)((m2 << 4) | (arr[8] & 0xF));
break;
case 0x9:
arr[8] = (byte)(((m2 | 0x8) << 4) | (arr[8] & 0xF));
break;
case 0xA:
arr[8] = (byte)(((m2 | 0x4) << 4) | (arr[8] & 0xF));
break;
case 0xB:
arr[8] = (byte)(((m2 | 0xC) << 4) | (arr[8] & 0xF));
break;
default:
throw new Exception();
}
return new Guid(arr);
}

static int ExtractMachine(Guid guid)
{
var arr = guid.ToByteArray();
return
((arr[7] >> 4) & 0xF) |
(((arr[8] >> 4) & 0x3) << 4);
}

使用版本 1 GUID

您还可以使用版本 1 GUID,因为仍然可以生成它们:

class SequentialGuid
{
[DllImport("rpcrt4.dll", SetLastError = true)]
static extern int UuidCreateSequential(out Guid guid);

public static Guid NewGuid()
{
Guid guid;
UuidCreateSequential(out guid);
return guid;
}

public static byte[] ExtractMacAddress(Guid guid)
{
var arr = guid.ToByteArray();
// Require version 1.
if (((arr[7] >> 4) & 0xF) != 1)
throw new ArgumentOutOfRangeException("guid", "GUID is required to be a sequential (version 1) GUID.");

var macLong = BitConverter.ToInt64(arr, arr.Length - 8);
macLong = IPAddress.NetworkToHostOrder(macLong);
arr = BitConverter.GetBytes(macLong);
Array.Resize(ref arr, 6);

return arr;
}
}

关于c# - 我们可以解码 Guid 以找出它是在何时何地生成的吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7159359/

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