Работа с символами кандзи в C++

У меня есть настольное приложение для Windows (с именем Timestamp), написанное на C++, которое использует .NET под названием CLR.

У меня также есть проект DLL (с ​​именем: Amscpprest), написанный на собственном С++ и использующий CPPREST SDK для получения данных json с сервера и передачи данных в мое приложение Timestamp.

Вот сценарий: это возвращаемые данные 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 (Timestamp). Вот как я принимаю данные из моего проекта 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);
}

Я ожидаю, что он должен правильно отображать имена и идентификатор в списке, но это было результатом: введите здесь описание изображения

Я надеюсь, что кто-то может помочь мне решить эту проблему.


person noyruto88    schedule 06.07.2018    source источник
comment
У вас есть несоответствия в обработке вашей кодировки символов. Ваш графический интерфейс, скорее всего, ожидает, что это будет UCS-16. В чем кодируется JSON?   -  person Jodocus    schedule 06.07.2018
comment
Возможен ли кандзи в std::string = std::basic_string<char> или вам нужен std::wstring = std::basic_string<wchar>?   -  person JHBonarius    schedule 06.07.2018
comment
Ваш map также может быть map<int,std::wstring>или даже map<int, String^>, так что вам не нужно снова и снова кодировать и декодировать значения вашего имени. См. docs.microsoft.com/en-us/cpp/dotnet. /карта-stl-clr.   -  person Hello Everyone    schedule 06.07.2018
comment
@OstrichGroomer, проект DLL является родным C ++, поэтому String^ там нет, не так ли?   -  person JHBonarius    schedule 06.07.2018
comment
Нет, это не так. Используйте 1_ .   -  person Hello Everyone    schedule 06.07.2018
comment
Я пытался заставить ваш код работать в течение некоторого времени. Не могли бы вы включить минимально воспроизводимый пример?   -  person JHBonarius    schedule 06.07.2018


Ответы (1)


Я думаю, что у вас есть две разные проблемы здесь.

Во-первых, в итерации вы пытаетесь прочитать ключ с именем _id, которого там нет (он должен быть id), поэтому вашему int key никогда не присваивается значение (и он не инициализируется, поэтому вы получаете странное число в вашем Посмотреть список).

Во-вторых, вам нужно преобразовать utf8 (хранящийся в std::string) в ucs2, из которого состоит строка .NET. Этого можно добиться с помощью класса UTF8Encoding (документы здесь) . Итак, вместо этого:

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);
person p-a-o-l-o    schedule 06.07.2018
comment
Да, сэр, по поводу проблемы с _id, я это уже заметил и изменил. - person noyruto88; 06.07.2018
comment
Произошла ошибка, когда я попытался запустить этот код. '^': нельзя использовать эту косвенность для типа 'std::array' и; использование неопределенного типа 'std::array' - person noyruto88; 09.07.2018
comment
Вы добавили #include <array> сверху? - person p-a-o-l-o; 09.07.2018
comment
Да сэр. я добавил это - person noyruto88; 09.07.2018
comment
Теперь я нашел решение этой ошибки. отсюда stackoverflow.com/questions/23062179/ Я просто добавляю cli:: перед array‹Byte› вот так: cli::array‹Byte›^ bytes и это сработало. - person noyruto88; 09.07.2018