是否有可能创建一个比默认guid格式短的真正唯一的目录名(即基于uuid)?
到目前为止,我已经想出了这个:
function Get-UglyButShortUniqueDirname {
[CmdletBinding()]
param ()
$t = "$([System.Guid]::NewGuid())".Replace("-", "")
Write-Verbose "base guid: $t"
$t = "$(0..$t.Length | % { if (($_ -lt $t.Length) -and !($_%2)) { [char][byte]"0x$($t[$_])$($t[$_+1])" } })".replace(" ", "").Trim()
Write-Verbose "guid as ascii: $t"
([System.IO.Path]::GetInvalidFileNameChars() | % { $t = $t.replace($_, '.') })
Write-Verbose "dirname: $t"
$t
}
有了这个,我可以生成看起来很奇怪的目录名,但只需要大约16个字符,这比普通guid的默认32个字符(没有破折号)要好得多。
我有点担心的是:由于‘无效文件名字符’被删除并替换为点,这些标识符不符合guid所做的相同的“唯一性承诺”。
(在基于Win的自动化环境中与遗留的260 char路径名限制作斗争:-/)
发布于 2022-01-26 15:42:32
将您的quid转换为Base64,这为您提供了一个24个字符的字符串,并且(正如zett42所提到的),需要替换可能的斜杠(/
)。此外,您还可以通过删除不必要的填充来保存另外两个字符:
[System.Convert]::ToBase64String((NewGuid).ToByteArray()).SubString(0,22).Replace('/', '-')
zp92wiHcdU+0Eb9Cw2z0VA
但是,这个想法实际上存在一个缺陷:文件夹名不区分大小写,这意味着文件夹命名可能不像原始guid
那样唯一。
因此,您可能希望回到Base32
(它需要26个字符)上,这有点复杂,因为没有标准的.Net方法:
$Chars = ('A'..'Z') + ('2'..'7')
$Bytes = (New-Guid).ToByteArray()
$Bits = -join $Bytes.ForEach{ [Convert]::ToString($_, 2).PadLeft(8, '0') }
-Join ($Bits -Split '(?<=\G.{5})').foreach{ $Chars[[Convert]::ToInt32($_, 2)] }
DZ77OUQNDRQUTGP5ATAM7KCWCB
您可能会做一些类似的事情来包含特殊字符,但是我会非常小心,因为不是每个文件系统都可能支持这一点。
发布于 2022-01-26 17:09:11
我要Base32,-encode,GUID。使用26个字符,结果将略长于Base64,但在不区分大小写的文件系统中不会丢失一些随机性。此外,它只使用基本字母数字字符,而IMO看起来更好;-)。
不幸的是,在Base32中没有内置的.NET编码器。首先,我采用了这个C#的答案的编码部分,但后来我对它进行了修改,使其使用Z-基座32变体,这在人眼上更容易,而且不用填充就节省了几个字符。
Add-Type -TypeDefinition @'
public class ZBase32Encoder {
private const string Charset = "ybndrfg8ejkmcpqxot1uwisza345h769";
public static string ToString(byte[] input) {
if (input == null || input.Length == 0) {
throw new System.ArgumentNullException("input");
}
long returnLength = (input.Length * 8L - 1) / 5 + 1;
if( returnLength > (long) System.Int32.MaxValue ) {
throw new System.ArgumentException("Argument length is too large. (Parameter 'input')");
}
char[] returnArray = new char[returnLength];
byte nextChar = 0, bitsRemaining = 5;
int arrayIndex = 0;
foreach (byte b in input) {
nextChar = (byte)(nextChar | (b >> (8 - bitsRemaining)));
returnArray[arrayIndex++] = Charset[nextChar];
if (bitsRemaining < 4) {
nextChar = (byte)((b >> (3 - bitsRemaining)) & 31);
returnArray[arrayIndex++] = Charset[nextChar];
bitsRemaining += 5;
}
bitsRemaining -= 3;
nextChar = (byte)((b << bitsRemaining) & 31);
}
//if we didn't end with a full char
if (arrayIndex != returnLength) {
returnArray[arrayIndex++] = Charset[nextChar];
}
return new string(returnArray);
}
}
'@
foreach( $i in 1..5 ) {
$s = [ZBase32Encoder]::ToString($(New-Guid).ToByteArray())
"$s (Length: $($s.Length))"
}
输出:
81u68ug6txxwpbqz4znzgq3hfa (Length: 26)
sseik38xykrr5n99zedj96nsoy (Length: 26)
a353cgcyhefwdc8euk34zbytxa (Length: 26)
e3x8zd576zcrzn3nyxwxncenho (Length: 26)
7idr4xencm9rmp8wkzidk1fyhe (Length: 26)
https://stackoverflow.com/questions/70865502
复制相似问题