在 C++ 中处理汉字字符
Dealing with kanji characters in C++
我有一个用 C++ 编写的 windows 桌面应用程序(名为:Timestamp),它使用名为 CLR 的 .NET。
我还有一个用本机 c++ 编写的 DLL 项目(名为:Amscpprest)并使用 CPPREST SDK 从服务器获取 json 数据并将数据传递到我的 Timestamp 应用程序。
场景如下:
这是来自我的服务器的 return json 数据,它是一个员工姓名列表,其中大部分是用汉字字符书写的日文姓名。
[
{
"staff": {
"id": 121,
"name": "福士 達哉",
"department": [
{
"_id": 3,
"name": "事業推進本部"
}
]
}
},
{
"staff": {
"id": 12,
"name": "北島 美奈",
"department": [
{
"_id": 4,
"name": "事業開発本部"
}
]
}
},
{
"staff": {
"id": 151,
"name": "大河原 紗希",
"department": [
{
"_id": 3,
"name": "事業推進本部"
}
]
}
}
]
这是我的 DLL 项目 (Amscpprest) 中的代码。这是获取数据并传递到我的 CLR 项目的方式:
std::map<int, std::string> staffMap;
auto GetStaffMap() -> std::map<int, std::string> {
return staffMap;
}
void display_json(json::value const & jvalue, utility::string_t const & prefix)
{
try {
//==== Iterate through json data and make an associative array ====/
auto DataArray = jvalue.as_array();
// loop through 'data' object
for (int i = 0; i < DataArray.size(); i++)
{
try
{
auto data = DataArray[i];
auto dataObj = data.at(U("staff")).as_object();
int key;
std::string value;
// loop through each object of 'data'
for (auto iterInner = dataObj.cbegin(); iterInner != dataObj.cend(); ++iterInner)
{
auto &propertyName = iterInner->first;
auto &propertyValue = iterInner->second;
if (propertyName == L"_id")
{
key = propertyValue.as_integer();
}
else if (propertyName == L"name")
{
value = conversions::to_utf8string(propertyValue.as_string());
}
}
staffMap.insert(std::make_pair(key, value));
}
catch (const std::exception& e)
{
std::wcout << e.what() << std::endl;
}
}
}
catch (const std::exception& e) {
std::wcout << e.what() << std::endl;
}
}
pplx::task<http_response> task_request(http_client & client, method mtd, json::value const & jvalue, std::string searchText)
{
//std::string url = "/api/authenticate/searchStaffs/";
std::string url = "/api/authenticate/oldgms/staffs_id_name/";
return client.request(mtd, utility::conversions::to_string_t(url));
}
void make_request(http_client & client, method mtd, json::value const & jvalue, std::string searchText)
{
task_request(client, mtd, jvalue, searchText)
.then([](http_response response)
{
if (response.status_code() == status_codes::OK)
{
return response.extract_json();
}
return pplx::task_from_result(json::value());
})
.then([](pplx::task<json::value> previousTask)
{
try
{
display_json(previousTask.get(), L"R: ");
}
catch (http_exception const & e)
{
std::wcout << e.what() << std::endl;
}
})
.wait();
}
int SearchStaff(std::string searchText)
{
//clear staffMap every call
staffMap.clear();
http_client client(U("http://52.68.13.154:3000"));
auto nullValue = json::value::null();
//std::string search_text = conversions::to_utf8string(L"北島 美奈");
make_request(client, methods::GET, nullValue, searchText);
return staff_id;
}
这是我在 CLR 项目(时间戳)中的代码。这就是我从我的 dll 项目中接受数据并显示到用户界面的方式。
String^ input = searchBox->Text;
std::string searchText = msclr::interop::marshal_as<std::string>(input);
// Clear listView item every type in searchbox
listView1->Items->Clear();
Staffs::SearchStaff(searchText);
std::map<int, std::string> staffMap = Staffs::GetStaffMap();
std::map<int, std::string>::iterator iter;
for (iter = staffMap.begin(); iter != staffMap.end(); iter++)
{
String^ value = msclr::interop::marshal_as<System::String^>(iter->second);
int key = iter->first;
listViewItem = gcnew Windows::Forms::ListViewItem(value);
listViewItem->SubItems->Add(System::Convert::ToString(key));
this->listView1->Items->Add(this->listViewItem);
}
我希望它应该在列表视图中正确显示名称和 ID,但这是结果:
我希望有人能帮我解决这个问题。
我认为您有两个截然不同的问题。
首先,在迭代中,您尝试读取一个名为 _id
的键,它不存在(应该是 id
),因此您的 int key
永远不会被分配一个值(并且它没有被初始化,这就是为什么你在列表视图中得到一个奇怪的数字)。
其次,您必须将 utf8(存储在 std::string
中)转换为 ucs2,这是 .NET 字符串的组成部分。您可以使用 UTF8Encoding
class (docs here) 来实现。所以,而不是这个:
String^ value = msclr::interop::marshal_as<System::String^>(iter->second);
你需要这样的东西:
//make a byte array to hold the string chars
array<Byte>^ bytes = gcnew array<Byte>(iter->second.size());
//copy the string chars into the byte array
System::Runtime::InteropServices::Marshal::Copy(IntPtr(&iter->second[0]), bytes, 0, iter->second.size());
//get a string from the bytes, using UTF8Encoding
String^ value = System::Text::UTF8Encoding::UTF8->GetString(bytes);
我有一个用 C++ 编写的 windows 桌面应用程序(名为:Timestamp),它使用名为 CLR 的 .NET。
我还有一个用本机 c++ 编写的 DLL 项目(名为:Amscpprest)并使用 CPPREST SDK 从服务器获取 json 数据并将数据传递到我的 Timestamp 应用程序。
场景如下: 这是来自我的服务器的 return json 数据,它是一个员工姓名列表,其中大部分是用汉字字符书写的日文姓名。
[
{
"staff": {
"id": 121,
"name": "福士 達哉",
"department": [
{
"_id": 3,
"name": "事業推進本部"
}
]
}
},
{
"staff": {
"id": 12,
"name": "北島 美奈",
"department": [
{
"_id": 4,
"name": "事業開発本部"
}
]
}
},
{
"staff": {
"id": 151,
"name": "大河原 紗希",
"department": [
{
"_id": 3,
"name": "事業推進本部"
}
]
}
}
]
这是我的 DLL 项目 (Amscpprest) 中的代码。这是获取数据并传递到我的 CLR 项目的方式:
std::map<int, std::string> staffMap;
auto GetStaffMap() -> std::map<int, std::string> {
return staffMap;
}
void display_json(json::value const & jvalue, utility::string_t const & prefix)
{
try {
//==== Iterate through json data and make an associative array ====/
auto DataArray = jvalue.as_array();
// loop through 'data' object
for (int i = 0; i < DataArray.size(); i++)
{
try
{
auto data = DataArray[i];
auto dataObj = data.at(U("staff")).as_object();
int key;
std::string value;
// loop through each object of 'data'
for (auto iterInner = dataObj.cbegin(); iterInner != dataObj.cend(); ++iterInner)
{
auto &propertyName = iterInner->first;
auto &propertyValue = iterInner->second;
if (propertyName == L"_id")
{
key = propertyValue.as_integer();
}
else if (propertyName == L"name")
{
value = conversions::to_utf8string(propertyValue.as_string());
}
}
staffMap.insert(std::make_pair(key, value));
}
catch (const std::exception& e)
{
std::wcout << e.what() << std::endl;
}
}
}
catch (const std::exception& e) {
std::wcout << e.what() << std::endl;
}
}
pplx::task<http_response> task_request(http_client & client, method mtd, json::value const & jvalue, std::string searchText)
{
//std::string url = "/api/authenticate/searchStaffs/";
std::string url = "/api/authenticate/oldgms/staffs_id_name/";
return client.request(mtd, utility::conversions::to_string_t(url));
}
void make_request(http_client & client, method mtd, json::value const & jvalue, std::string searchText)
{
task_request(client, mtd, jvalue, searchText)
.then([](http_response response)
{
if (response.status_code() == status_codes::OK)
{
return response.extract_json();
}
return pplx::task_from_result(json::value());
})
.then([](pplx::task<json::value> previousTask)
{
try
{
display_json(previousTask.get(), L"R: ");
}
catch (http_exception const & e)
{
std::wcout << e.what() << std::endl;
}
})
.wait();
}
int SearchStaff(std::string searchText)
{
//clear staffMap every call
staffMap.clear();
http_client client(U("http://52.68.13.154:3000"));
auto nullValue = json::value::null();
//std::string search_text = conversions::to_utf8string(L"北島 美奈");
make_request(client, methods::GET, nullValue, searchText);
return staff_id;
}
这是我在 CLR 项目(时间戳)中的代码。这就是我从我的 dll 项目中接受数据并显示到用户界面的方式。
String^ input = searchBox->Text;
std::string searchText = msclr::interop::marshal_as<std::string>(input);
// Clear listView item every type in searchbox
listView1->Items->Clear();
Staffs::SearchStaff(searchText);
std::map<int, std::string> staffMap = Staffs::GetStaffMap();
std::map<int, std::string>::iterator iter;
for (iter = staffMap.begin(); iter != staffMap.end(); iter++)
{
String^ value = msclr::interop::marshal_as<System::String^>(iter->second);
int key = iter->first;
listViewItem = gcnew Windows::Forms::ListViewItem(value);
listViewItem->SubItems->Add(System::Convert::ToString(key));
this->listView1->Items->Add(this->listViewItem);
}
我希望它应该在列表视图中正确显示名称和 ID,但这是结果:
我希望有人能帮我解决这个问题。
我认为您有两个截然不同的问题。
首先,在迭代中,您尝试读取一个名为 _id
的键,它不存在(应该是 id
),因此您的 int key
永远不会被分配一个值(并且它没有被初始化,这就是为什么你在列表视图中得到一个奇怪的数字)。
其次,您必须将 utf8(存储在 std::string
中)转换为 ucs2,这是 .NET 字符串的组成部分。您可以使用 UTF8Encoding
class (docs here) 来实现。所以,而不是这个:
String^ value = msclr::interop::marshal_as<System::String^>(iter->second);
你需要这样的东西:
//make a byte array to hold the string chars
array<Byte>^ bytes = gcnew array<Byte>(iter->second.size());
//copy the string chars into the byte array
System::Runtime::InteropServices::Marshal::Copy(IntPtr(&iter->second[0]), bytes, 0, iter->second.size());
//get a string from the bytes, using UTF8Encoding
String^ value = System::Text::UTF8Encoding::UTF8->GetString(bytes);