本文转自:
如何从string 转换为const char *和char * (附带string中c_str()、data()、copy(p,n)函数的用法)
内容做了部分删改和排版。
basic_string是一个模版类,string是模版形参为char的basci_string模版类的类型定义,即typedef
typedef basic_string<char, char_traits<char>, allocator<char> > string;
basic_string是类模版,并且是容器类模版。basic_string类模版的对象管理的序列是标准的C++ 字符串,basic_string包括string、wstring、u16string和u32string)。
所以标准C++ 字符串类是一个容器,因此可以像操作其他容器一样,对C++ string类比较,迭代,STL算法等。与char*不同的是,string不一定以NULL(‘\0’)结束。string长度可以根据length()得到,string可以根据下标访问。所以,不能将string直接赋值给char*。
当我们有一个string类型对象后,,用cout输出字符串的内容没有问题的,但是用printf(“%s”,s1)输出是会出问题的。这是因为
“%s”
要求后面的对象的首地址,但是string不是这样的一个类型,所以肯定出错。
若一定要printf输出,就要根据printf函数的要求进行适当的转换。
string 转换成 char *
如果要将string直接转换成const char *类型。string有2个函数可以运用。
一个是.c_str(),一个是data成员函数。
例子如下:
string s1 = "abcdeg";
const char *k = s1.c_str();
const char *t = s1.data();
printf("%s%s",k,t);
cout<<k<<t<<endl;
如上,都可以输出。data()与c_str()不同之处在于,c_str函数返回的数组以NULL(‘\0’)结束,data函数但是返回的数组不以空字符终止。
无论是data函数还是c_str函数只能转换成const char*,如果去掉const编译不能通过。
如果要
转换成char*
,可以用string的
成员函数copy实现
string的copy(p,n,size_type _off = 0)函数,是从string类型对象中至多复制n个字符到字符指针p指向的空间中。默认从首字符开始(从0开始),但是也可以指定开始的位置。函数的返回值是从string对象中复制的字符内容。使用时要要确保p指向的空间足够保存n个字符,否则会产生错误。
string s1 = "abcdefg";
char *data;
int len = s1.length();
data = (char *)malloc((len+1)*sizeof(char));//C语言的动态内存分配,C++可以使用new
s1.copy(data,len,0);
printf("%s",data);
cout<<data;
char *转换成string
可以直接赋值。
string s;
char *p = "adghrtyh";
s = p;
char[] 转换成string
这个也可以直接赋值。
string转换成char[]
这样的转换不可直接赋值。但是根据length()函数可以得到string的长度,而且又可以根据下标直接访问string对象的内容(字符),所以可以利用循环进行操作:
string pp = "dagah";
char p[8];
int i;
for( i=0;i<pp.length();i++)
p[i] = pp[i];
p[i] = '\0';
printf("%s\n",p);
cout<<p;
最后在强调一下string中
c_str()、data()、copy(p,n)函数
的用法
标准库的string类提供了3个成员函数来从一个string得到c类型的字符数组:c_str()、data()、copy(p,n)。
1. c_str():
生成一个const char*指针,指向以空字符终止的数组。
需要注意的是,这个数组的数据是临时的。当有一个改变这些数据的函数被调用后,其中的数据就会失效。因此要么现用先转换,要么把它的数据复制到用户自己可以管理的内存中。
const char * c;
string s= "1234" ;
c = s.c_str();
cout<<c<<endl; //输出:1234
s= "abcd" ;
cout<<c<<endl; //输出:abcd
正如上例所示,继续用c指针的话,存放的内容从1234变为abcd,这将导致不可想象的错误。
既然c指针指向的内容容易失效,我们就应该即时把数据复制出来,这就要用到strcpy等函数了。
char * c= new char [20];//这是C++申请动态内存的方式
string s= "1234" ;
//c = s.c_str();
strcpy (c,s.c_str());
cout<<c<<endl; //输出:1234
s= "abcd" ;
cout<<c<<endl; //输出:1234
c_str()返回一个客户程序可读不可改的指向字符数组的指针(const char*),不需要手动释放或删除这个指针。
2. data():与c_str()类似,但是返回的数组不以空字符终止。
3. copy(p,n,size_type _off = 0):从string类型对象中至多复制n个字符到字符指针p指向的空间中。默认从首字符开始,但是也可以指定,开始的位置(记住从0开始)。返回真正从对象中复制的字符。使用时要确保p指向的空间足够保存n个字符。
#include <string>
#include <iostream>
using namespace std;
int main( )
{
string str1 ( "1234567890" );
basic_string < char >::iterator str_Iter;//basic_string的迭代器
basic_string < char >:: pointer array1Ptr = array1;
basic_string < char >:: value_type *array2Ptr = array2;
char array1 [ 20 ] = { 0 };
char array2 [ 10 ] = { 0 };
cout << "The original string str1 is: " ;
for ( str_Iter = str1.begin( ); str_Iter != str1.end( ); str_Iter++ )
cout << *str_Iter;
cout << endl;
basic_string < char >:: size_type nArray1;
// Note: string::copy is potentially unsafe, consider
// using string::_Copy_s instead.
nArray1 = str1.copy ( array1Ptr , 12 ); // C4996
cout << "The number of copied characters in array1 is: "
<< nArray1 << endl;
cout << "The copied characters array1 is: " << array1Ptr << endl;
basic_string < char >:: size_type nArray2;
// Note: string::copy is potentially unsafe, consider
// using string::_Copy_s instead.
nArray2 = str1.copy ( array2Ptr , 5 , 6 ); // C4996
cout << "The number of copied characters in array2 is: "
<< nArray2 << endl;
cout << "The copied characters array2 is: " << array2Ptr << endl;
//char array3[5]={0};
//basic_string<char>::pointer array3Ptr=array3;
//basic_string<char>::size_type nArray3;
//nArray3 = str1.copy(array3,9); //错误!!!!空间不够
//cout<<"The number of copied characters in array3 is: "
// <<nArray3<<endl;
//cout<<"The copied characters array3 is: "<<array3Ptr<<endl;
}
上面最后注释掉的部分,虽然编译没有错误,但是运行时会产生错误:Stack around the variable ‘array3’ was corrupted.