C ++ / CLI从System :: String ^转换为std :: string

时间:2009-06-03 19:51:10

标签: string c++-cli managed-c++

有人可以发布一个可以转换的简单代码,

System::String^

要,

C ++ std::string

即,我只想分配值

String^ originalString;

要,

std::string newString;

10 个答案:

答案 0 :(得分:137)

不要自己滚动,使用Microsoft提供的these方便(和可扩展)包装。

例如:

#include <msclr\marshal_cppstd.h>

System::String^ managed = "test";
std::string unmanaged = msclr::interop::marshal_as<std::string>(managed);

答案 1 :(得分:38)

查看System::Runtime::InteropServices::Marshal::StringToCoTaskMemUni()及其朋友。

抱歉,现在无法发布代码;我在这台机器上没有VS来检查它是否在发布之前编译。

答案 2 :(得分:37)

您可以按照以下方式轻松完成此操作

#include <msclr/marshal_cppstd.h>

System::String^ xyz="Hi boys"; 

std::string converted_xyz=msclr::interop::marshal_as< std::string >( xyz);

答案 3 :(得分:19)

这对我有用:

#include <stdlib.h>
#include <string.h>
#include <msclr\marshal_cppstd.h>
//..
using namespace msclr::interop;
//..
System::String^ clrString = (TextoDeBoton);
std::string stdString = marshal_as<std::string>(clrString); //String^ to std
//System::String^ myString = marshal_as<System::String^>(MyBasicStirng); //std to String^
prueba.CopyInfo(stdString); //MyMethod
//..
//Where: String^ = TextoDeBoton;
//and stdString is a "normal" string;

答案 4 :(得分:9)

以下是我多年前为c ++ / cli项目编写的一些转换例程,它们仍然有用。

void StringToStlWString ( System::String const^ s, std::wstring& os)
    {
        String^ string = const_cast<String^>(s);
        const wchar_t* chars = reinterpret_cast<const wchar_t*>((Marshal::StringToHGlobalUni(string)).ToPointer());
        os = chars;
        Marshal::FreeHGlobal(IntPtr((void*)chars));

    }
    System::String^ StlWStringToString (std::wstring const& os) {
        String^ str = gcnew String(os.c_str());
        //String^ str = gcnew String("");
        return str;
    }

    System::String^ WPtrToString(wchar_t const* pData, int length) {
        if (length == 0) {
            //use null termination
            length = wcslen(pData);
            if (length == 0) {
                System::String^ ret = "";
                return ret;
            }
        }

        System::IntPtr bfr = System::IntPtr(const_cast<wchar_t*>(pData));
        System::String^ ret = System::Runtime::InteropServices::Marshal::PtrToStringUni(bfr, length);
        return ret;
    }

    void Utf8ToStlWString(char const* pUtfString, std::wstring& stlString) {
        //wchar_t* pString;
        MAKE_WIDEPTR_FROMUTF8(pString, pUtfString);
        stlString = pString;
    }

    void Utf8ToStlWStringN(char const* pUtfString, std::wstring& stlString, ULONG length) {
        //wchar_t* pString;
        MAKE_WIDEPTR_FROMUTF8N(pString, pUtfString, length);
        stlString = pString;
    }

答案 5 :(得分:5)

我花了好几个小时尝试将Windows窗体列表框ToString值转换为标准字符串,以便我可以将它与fstream一起用于输出到txt文件。我的Visual Studio没有附带编组头文件,我找到了几个使用的答案。经过这么多试验和错误,我终于找到了一个解决方案,解决了刚刚使用System :: Runtime :: InteropServices的问题:

void MarshalString ( String ^ s, string& os ) {
   using namespace Runtime::InteropServices;
   const char* chars = 
      (const char*)(Marshal::StringToHGlobalAnsi(s)).ToPointer();
   os = chars;
   Marshal::FreeHGlobal(IntPtr((void*)chars));
}

//this is the code to use the function:
scheduleBox->SetSelected(0,true);
string a = "test";
String ^ c = gcnew String(scheduleBox->SelectedItem->ToString());
MarshalString(c, a);
filestream << a;

以下是带有示例的MSDN页面: http://msdn.microsoft.com/en-us/library/1b4az623(v=vs.80).aspx

我知道这是一个非常简单的解决方案,但这需要我进行故障排除和访问几个论坛,最终找到有效的方法。

答案 6 :(得分:4)

我发现从String获取std :: string的简单方法是使用sprintf()。

char cStr[50] = { 0 };
String^ clrString = "Hello";
if (clrString->Length < sizeof(cStr))
  sprintf(cStr, "%s", clrString);
std::string stlString(cStr);

无需召唤元帅功能!

UPDATE 感谢Eric,我修改了示例代码以检查输入字符串的大小,以防止缓冲区溢出。

答案 7 :(得分:3)

C#使用UTF16格式作为字符串 因此,除了转换类型之外,您还应该注意字符串的实际格式。

编译多字节字符集时,Visual Studio和Win API采用UTF8(实际上是Windows-28591的Windows编码)。
编译 Unicode字符集时,Visual Studio和Win API采用UTF16。

因此,您必须将字符串从UTF16转换为UTF8格式,而不仅仅是转换为std :: string。 在使用多种字符格式(如某些非拉丁语言)时,这将是必要的。

我们的想法是确定std::wstring 始终代表 UTF16
std::string 始终代表 UTF8

这不是由编译器强制执行的,它是一个更好的策略。

#include "stdafx.h"
#include <string>
#include <codecvt>
#include <msclr\marshal_cppstd.h>

using namespace System;

int main(array<System::String ^> ^args)
{
    System::String^ managedString = "test";

    msclr::interop::marshal_context context;

    //Actual format is UTF16, so represent as wstring
    std::wstring utf16NativeString = context.marshal_as<std::wstring>(managedString); 

    //C++11 format converter
    std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>> convert;

    //convert to UTF8 and std::string
    std::string utf8NativeString = convert.to_bytes(utf16NativeString);

    return 0;
}

或者使用更紧凑的语法:

int main(array<System::String ^> ^args)
{
    System::String^ managedString = "test";

    msclr::interop::marshal_context context;
    std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>> convert;

    std::string utf8NativeString = convert.to_bytes(context.marshal_as<std::wstring>(managedString));

    return 0;
}

答案 8 :(得分:0)

我喜欢远离编队。

Using CString newString(originalString);

对我来说似乎更清洁,更快捷。无需担心创建和删除上下文。

答案 9 :(得分:0)

//我使用VS2012编写下面的代码 - convert_system_string到Standard_Sting

        #include "stdafx.h"
        #include <iostream>
        #include <string> 

        using namespace System;
        using namespace Runtime::InteropServices; 


        void MarshalString ( String^ s, std::string& outputstring )
        {  
           const char* kPtoC =  (const char*) (Marshal::StringToHGlobalAnsi(s)).ToPointer();                                                        
           outputstring = kPtoC;  
           Marshal::FreeHGlobal(IntPtr((void*)kPtoC));  
        }   

        int _tmain(int argc, _TCHAR* argv[])
        {
             std::string strNativeString;  
             String ^ strManagedString = "Temp";

             MarshalString(strManagedString, strNativeString);  
             std::cout << strNativeString << std::endl; 

             return 0;
        }