.NET System::String to UTF8-bytes stored in char*(.NET System::String 到存储在 char* 中的 UTF8 字节)
问题描述
我在 .NET 项目中封装了一些非托管 C++ 代码.为此,我需要将 System::String
转换为存储在 char*
中的 UTF8 字节.
I am wrapping some unmanaged C++ code inside a .NET project. For this I need to convert System::String
to UTF8-bytes stored in char*
.
我不确定这是否是最好的甚至是正确的方法,如果有人可以查看并提供反馈,我将不胜感激.
I am unsure if this is the best or even a correct way to do this and I'd appreciate if someone could take a look and provide feedback.
谢谢,
/大卫
// Copy into blank VisualStudio C++/CLR command line solution.
#include "stdafx.h"
#include <stdio.h>
using namespace System;
using namespace System::Text;
using namespace System::Runtime::InteropServices;
// Test for calling with char* argument.
void MyTest(const char* buffer)
{
printf_s("%s
", buffer);
return;
}
int main()
{
// Create a UTF-8 encoding.
UTF8Encoding^ utf8 = gcnew UTF8Encoding;
// A Unicode string with two characters outside an 8-bit code range.
String^ unicodeString = L"This unicode string contains two characters with codes outside an 8-bit code range, Pi (u03a0) and Sigma (u03a3).";
Console::WriteLine(unicodeString);
// Encode the string.
array<Byte>^encodedBytes = utf8->GetBytes(unicodeString);
// Get pointer to unmanaged char array
int size = Marshal::SizeOf(encodedBytes[0]) * encodedBytes->Length;
IntPtr pnt = Marshal::AllocHGlobal(size);
Marshal::Copy(encodedBytes, 0, pnt, encodedBytes->Length);
// Ugly, but necessary?
char *charPnt= (char *)pnt.ToPointer();
MyTest(charPnt);
Marshal::FreeHGlobal(pnt);
}
推荐答案
无需创建编码器实例,使用静态实例即可.
You don't need to create an encoder instance, you can use the static instances.
如果被调用的函数不期望指向 HGlobal 堆的指针,您可以对缓冲区使用普通的 C/C++ 内存分配(new 或 malloc).
If the called function doesn't expect a pointer to the HGlobal heap you can just use plain C/C++ memory allocation (new or malloc) for the buffer.
在您的示例中,该函数不具有所有权,因此您根本不需要副本,只需固定缓冲区即可.
In your example the function doesn't take ownership so you don't need a copy at all, just pin the buffer.
类似:
// Encode the text as UTF8
array<Byte>^ encodedBytes = Encoding::UTF8->GetBytes(unicodeString);
// prevent GC moving the bytes around while this variable is on the stack
pin_ptr<Byte> pinnedBytes = &encodedBytes[0];
// Call the function, typecast from byte* -> char* is required
MyTest(reinterpret_cast<char*>(pinnedBytes), encodedBytes->Length);
或者,如果您需要像大多数 C 函数(包括 OP 中的示例)一样以零结尾的字符串,那么您可能应该添加一个零字节.
Or if you need the string zero-terminated like most C functions (including the example in the OP) then you should probably add a zero byte.
// Encode the text as UTF8, making sure the array is zero terminated
array<Byte>^ encodedBytes = Encoding::UTF8->GetBytes(unicodeString + " ");
// prevent GC moving the bytes around while this variable is on the stack
pin_ptr<Byte> pinnedBytes = &encodedBytes[0];
// Call the function, typecast from byte* -> char* is required
MyTest(reinterpret_cast<char*>(pinnedBytes));
这篇关于.NET System::String 到存储在 char* 中的 UTF8 字节的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:.NET System::String 到存储在 char* 中的 UTF8 字节


- 为什么 C# 中的堆栈大小正好是 1 MB? 2022-01-01
- CanBeNull和ReSharper-将其用于异步任务? 2022-01-01
- 在 LINQ to SQL 中使用 contains() 2022-01-01
- Windows 喜欢在 LINUX 中使用 MONO 进行服务开发? 2022-01-01
- 使用 rss + c# 2022-01-01
- 在 C# 中异步处理项目队列 2022-01-01
- Azure Active Directory 与 MVC,客户端和资源标识同一 2022-01-01
- 带问号的 nvarchar 列结果 2022-01-01
- 是否可以在 .Net 3.5 中进行通用控件? 2022-01-01
- C# 通过连接字符串检索正确的 DbConnection 对象 2022-01-01