URL의 경우 base64에 대한 guid
질문: 그렇게 하는 더 좋은 방법이 있습니까?
VB.Net
Function GuidToBase64(ByVal guid As Guid) As String
Return Convert.ToBase64String(guid.ToByteArray).Replace("/", "-").Replace("+", "_").Replace("=", "")
End Function
Function Base64ToGuid(ByVal base64 As String) As Guid
Dim guid As Guid
base64 = base64.Replace("-", "/").Replace("_", "+") & "=="
Try
guid = New Guid(Convert.FromBase64String(base64))
Catch ex As Exception
Throw New Exception("Bad Base64 conversion to GUID", ex)
End Try
Return guid
End Function
C#
public string GuidToBase64(Guid guid)
{
return Convert.ToBase64String(guid.ToByteArray()).Replace("/", "-").Replace("+", "_").Replace("=", "");
}
public Guid Base64ToGuid(string base64)
{
Guid guid = default(Guid);
base64 = base64.Replace("-", "/").Replace("_", "+") + "==";
try {
guid = new Guid(Convert.FromBase64String(base64));
}
catch (Exception ex) {
throw new Exception("Bad Base64 conversion to GUID", ex);
}
return guid;
}
다음 사이트를 확인해 보십시오. http://prettycode.org/2009/11/12/short-guid/
당신이 하고 있는 일과 매우 가까워 보입니다.
public class ShortGuid
{
private readonly Guid guid;
private readonly string value;
/// <summary>Create a 22-character case-sensitive short GUID.</summary>
public ShortGuid(Guid guid)
{
if (guid == null)
{
throw new ArgumentNullException("guid");
}
this.guid = guid;
this.value = Convert.ToBase64String(guid.ToByteArray())
.Substring(0, 22)
.Replace("/", "_")
.Replace("+", "-");
}
/// <summary>Get the short GUID as a string.</summary>
public override string ToString()
{
return this.value;
}
/// <summary>Get the Guid object from which the short GUID was created.</summary>
public Guid ToGuid()
{
return this.guid;
}
/// <summary>Get a short GUID as a Guid object.</summary>
/// <exception cref="System.ArgumentNullException"></exception>
/// <exception cref="System.FormatException"></exception>
public static ShortGuid Parse(string shortGuid)
{
if (shortGuid == null)
{
throw new ArgumentNullException("shortGuid");
}
else if (shortGuid.Length != 22)
{
throw new FormatException("Input string was not in a correct format.");
}
return new ShortGuid(new Guid(Convert.FromBase64String
(shortGuid.Replace("_", "/").Replace("-", "+") + "==")));
}
public static implicit operator String(ShortGuid guid)
{
return guid.ToString();
}
public static implicit operator Guid(ShortGuid shortGuid)
{
return shortGuid.guid;
}
}
이 기술을 사용하여 URL 또는 파일 이름에 사용할 GUID를 포맷할 때 한 가지 문제는 두 개의 고유한 GUID가 다음과 같은 경우에만 다른 두 개의 값을 생성할 수 있다는 것입니다.
var b1 = GuidToBase64(new Guid("c9d045f3-e21c-46d0-971d-b92ebc2ab83c"));
var b2 = GuidToBase64(new Guid("c9d045f3-e21c-46d0-971d-b92ebc2ab8a4"));
Console.WriteLine(b1); // 80XQyRzi0EaXHbkuvCq4PA
Console.WriteLine(b2); // 80XQyRzi0EaXHbkuvCq4pA
URL은 대소문자를 구분하지 않는 것으로 해석되기 때문에 윈도우즈 파일 경로 및 파일 이름은 대소문자를 구분하지 않습니다.이것은 충돌로 이어질 수 있습니다.
결국 ==를 클리핑하는 이유는 GUID(16바이트)의 경우 인코딩된 문자열이 항상 ==로 끝날 것이라는 확신을 가질 수 있기 때문인 것으로 알고 있습니다.그래서 변환할 때마다 2개의 문자를 저장할 수 있습니다.
이미 @Skurmedal이 언급한 요점 외에도(잘못된 문자열을 입력으로 사용할 경우 예외를 던져야 함), 저는 당신이 게시한 코드가 충분하다고 생각합니다.
만약 당신의 방법으로 전달된 Base64를 GUID로 변환할 수 없다면, 당신은 예외를 던져야 하지 않나요?메소드에 전달된 데이터는 분명히 오류가 있습니다.
인.NET Core
스팬을 사용하면 메모리 할당 없이 성능을 향상시킬 수 있습니다.
using System.Buffers.Text;
using System.Runtime.InteropServices;
namespace Extensions;
public static class GuidExtensions
{
private const char Dash = '-';
private const char EqualsChar = '=';
private const byte ForwardSlashByte = (byte)Slash;
private const char Plus = '+';
private const byte PlusByte = (byte)Plus;
private const char Slash = '/';
private const char Underscore = '_';
private const int Base64LengthWithoutEquals = 22;
public static string EncodeBase64String(this Guid guid)
{
Span<byte> guidBytes = stackalloc byte[16];
Span<byte> encodedBytes = stackalloc byte[24];
MemoryMarshal.TryWrite(guidBytes, ref guid);
Base64.EncodeToUtf8(guidBytes, encodedBytes, out _, out _);
Span<char> chars = stackalloc char[Base64LengthWithoutEquals];
// Replace any characters which are not URL safe.
// And skip the final two bytes as these will be '==' padding we don't need.
for (int i = 0; i < Base64LengthWithoutEquals; i++)
{
chars[i] = encodedBytes[i] switch
{
ForwardSlashByte => Dash,
PlusByte => Underscore,
_ => (char)encodedBytes[i],
};
}
return new(chars);
}
public static Guid DecodeBase64String(this ReadOnlySpan<char> id)
{
Span<char> base64Chars = stackalloc char[24];
for (var i = 0; i < Base64LengthWithoutEquals; i++)
{
base64Chars[i] = id[i] switch
{
Dash => Slash,
Underscore => Plus,
_ => id[i],
};
}
base64Chars[22] = EqualsChar;
base64Chars[23] = EqualsChar;
Span<byte> idBytes = stackalloc byte[16];
Convert.TryFromBase64Chars(base64Chars, idBytes, out _);
return new(idBytes);
}
}
using AutoFixture.Xunit2;
using FluentAssertions;
using Extensions;
using Xunit;
namespace ExtensionTests;
public class GuidExtensionsTests
{
private const int Base64LengthWithoutEquals = 22;
private const string EmptyBase64 = "AAAAAAAAAAAAAAAAAAAAAA";
[Theory]
[AutoData]
public void EncodeBase64String_DecodeBase64String_Should_ReturnInitialGuid(Guid guid)
{
string actualBase64 = guid.EncodeBase64String();
actualBase64.Should().NotBe(string.Empty)
.And.HaveLength(Base64LengthWithoutEquals);
Guid actualGuid = ((ReadOnlySpan<char>)actualBase64).DecodeBase64String();
actualGuid.Should().Be(guid);
}
[Theory]
[InlineData(EmptyBase64)]
public void EncodeBase64String_Should_ReturnEmptyBase64_When_GuidIsEmpty(string expected)
{
string actualBase64 = Guid.Empty.EncodeBase64String();
actualBase64.Should().Be(expected);
}
[Theory]
[InlineData(EmptyBase64)]
public void DecodeBase64String_Should_ReturnEmptyGuid_When_StringIsEmptyBase64(string base64)
{
Guid actual = ((ReadOnlySpan<char>)base64).DecodeBase64String();
actual.Should().Be(Guid.Empty);
}
}
고성능 기술을 사용하여 Base64를 인코딩하는 방법에 대한 자세한 내용은 GUID와 매우 멋진 비디오 설명을 참조하십시오.
URL(Guid)을 단축하고 인코딩하는 데 사용하는 방법이 있습니다. https://dotnetfiddle.net/iQ7nGv
public static void Main()
{
Guid gg = Guid.NewGuid();
string ss = Encode(gg);
Console.WriteLine(gg);
Console.WriteLine(ss);
Console.WriteLine(Decode(ss));
}
public static string Encode(Guid guid)
{
string encoded = Convert.ToBase64String(guid.ToByteArray());
encoded = encoded.Replace("/", "_").Replace("+", "-");
return encoded.Substring(0, 22);
}
public static Guid Decode(string value)
{
value = value.Replace("_", "/").Replace("-", "+");
byte[] buffer = Convert.FromBase64String(value + "==");
return new Guid(buffer);
}
언급URL : https://stackoverflow.com/questions/1032376/guid-to-base64-for-url
'programing' 카테고리의 다른 글
OSX의 Xcode를 어떻게 최신 버전으로 업데이트합니까? (0) | 2023.05.09 |
---|---|
비동기는 C#에서 어떻게 작동합니까? (0) | 2023.05.09 |
Postgre에서 여러 테이블을 삭제하는 방법와일드카드를 사용한 SQL (0) | 2023.05.09 |
VSCode에서 모듈 '@angular/core' 또는 다른 모듈을 찾을 수 없습니다. (0) | 2023.05.09 |
이미 대상 분기에 있는 커밋을 표시하는 GitHub 풀 요청 (0) | 2023.05.09 |