memcpy以及string中c_str()、data()、copy(p,n)函数的用法

本文深入探讨了C++中标准库string类的三个关键成员函数:c_str()、data()和copy(p,n)的用法及注意事项。c_str()生成指向以空字符终止的数组的const char*指针,但其内容易失效。data()类似c_str(),但返回的数组不以空字符终止。copy(p,n)用于将string类型对象中的字符复制到由p指向的空间中,用户需确保该空间足够大。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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 = s.c_str(); 不是一个好习惯。既然c指针指向的内容容易失效,我们就应该按照上面的方法,那怎么把数据复制出来呢?这就要用到strcpy等函数(推荐)。

//const char* c; //①
//char* c; //②
//char c[20];
char* c=new char[20];
string s=“1234”;
//c = s.c_str();
strcpy(c,s.c_str());
cout<<c<<endl; //输出:1234
s=“abcd”;
cout<<c<<endl; //输出:1234
注意:不能再像上面一样①所示了,const还怎么向里面写入值啊;也不能②所示,使用了未初始化的局部变量“c”,运行会出错的 。

② c_str()返回一个客户程序可读不可改的指向字符数组的指针,不需要手动释放或删除这个指针。

  1. data():与c_str()类似,但是返回的数组不以空字符终止。

  2. copy(p,n,size_type _Off = 0):从string类型对象中至多复制n个字符到字符指针p指向的空间中。默认从首字符开始,但是也可以指定,开始的位置(记住从0开始)。返回真正从对象中复制的字符。------用户要确保p指向的空间足够保存n个字符。

// basic_string_copy.cpp
// compile with: /EHsc /W3
#include
#include

int main( )
{
using namespace std;
string str1 ( “1234567890” );
basic_string ::iterator str_Iter;
char array1 [ 20 ] = { 0 };
char array2 [ 10 ] = { 0 };
basic_string :: pointer array1Ptr = array1;
basic_string :: value_type *array2Ptr = array2;

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;
 
注意一定要使array3有足够的空间
//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;

}

memcpy

函数原型为void *memcpy(void *destin, void *source, unsigned n);函数的功能是从源内存地址的起始位置开始拷贝若干个字节到目标内存地址中,即从源source中拷贝n个字节到目标destin中。是C 库函数。

当然,我可以帮你创建一个名为`MyString`的类,类似于C++标准库中的`std::string`。以下是这个类的基本实现: ```cpp #include <iostream> class MyString { private: char* data; // 存储字符串的字符数组 int length; // 字符串长度 int capacity; // 容量,即最多能容纳的字符数 public: // 构造函数:默认构造函数,初始容量为16 MyString() : data(new char[16]), length(0), capacity(16) {} // 构造函数:指定初始值 explicit MyString(const char* str) : data(new char[strlen(str) + 1]), length(strlen(str)), capacity(strlen(str) + 1) { strcpy(data, str); } // 拷贝构造函数 MyString(const MyString& other) : data(new char[other.capacity]), length(other.length), capacity(other.capacity) { if (other.data != nullptr) { memcpy(data, other.data, other.length); } } // 解析构造函数:从C风格字符串复制数据 MyString(const string& c_str) : data(new char[c_str.size() + 1]), length(c_str.size()), capacity(c_str.size() + 1) { strcpy(data, c_str.c_str()); } // 析构函数:释放内存 ~MyString() { delete[] data; } // 返回字符串长度 int size() const { return length; } // 检查字符串是否为空 bool empty() const { return length == 0; } // 返回指定索引处的字符 char operator[](int index) const { if (index >= 0 && index < length) { return data[index]; } else { throw out_of_range("Index out of range"); } } // 将字符串连接到自身 MyString& append(const char* str) { if (length + strlen(str) > capacity) { resize(length + strlen(str)); } strcat(data + length, str); length += strlen(str); return *this; } // 调整字符串容量,增加到新的大小 void resize(int newSize) { if (newSize > capacity) { char* newData = new char[newSize]; memcpy(newData, data, length); delete[] data; data = newData; capacity = newSize; } } // 显示字符串 friend std::ostream& operator<<(std::ostream& os, const MyString& str) { for (int i = 0; i < str.length; ++i) { os.put(str.data[i]); } return os; } // 复制操作,浅拷贝 MyString copy() const { MyString copied(*this); return copied; } private: // 禁止赋值操作,防止意外改变原始字符串 MyString& operator=(const MyString&) = delete; }; int main() { MyString s1("Hello, "); MyString s2("World!"); s1.append(s2.copy()); // 连接两个字符串 std::cout << s1 << std::endl; // 输出结果:Hello, World! return 0; } ``` 这个`MyString`类包含了常见的构造函数、析构函数、拷贝构造函数以及一些常用的成员函数,如检查空字符串、获取字符、连接字符串等。注意,为了简洁起见,这里没有实现所有可能的成员函数,例如插入、替换和删除字符的功能。另外,为了避免深拷贝,`copy`函数仅创建了一个浅拷贝。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值