string类的实现
一般面试,已知类String的原型为:
class String { public: String(const char *str = NULL);// 普通构造函数 String(const String &str);// 拷贝构造函数 ~String(void);// 析构函数 String & operator = (const String &str);// 赋值函数 private: char *m_data;// 用于保存字符串 };
类的实现:
String::String(const char *str) { if (str == NULL) { m_data = new char[1];// 得分点:对空字符串自动申请存放结束标志'\0'的,加分点:对m_data加NULL判断 *m_data = '\0'; } else { int length = strlen(str); m_data = new char[length + 1];// 若能加 NULL 判断则更好 strcpy(m_data, str); } } // String的析构函数 String::~String(void) { delete[] m_data; // 或delete m_data; } //拷贝构造函数 String::String(const String &str)// 得分点:输入参数为const型 { int length = strlen(str.m_data); m_data = new char[length + 1];// 若能加 NULL 判断则更好 strcpy(m_data, str.m_data); } //赋值函数 String & String::operator = (const String &str) // 得分点:输入参数为const型 { if (this == &str)//得分点:检查自赋值 return *this; if (m_data) delete[] m_data;//得分点:释放原有的内存资源 int length = strlen(str.m_data); m_data = new char[length + 1];//加分点:对m_data加NULL判断 strcpy(m_data, str.m_data); return *this;//得分点:返回本对象的引用 }
完整版:
一 整体框架
除了以上四个必须的函数,这里还实现了一些附加的内容。
- 若干个运算符重载,这里的几个是常见的运算符,可以加深对String的认识和运算符重载的理解。
- 两个常用的函数,包括取字符串长度和取C类型的字符串。
- 两个处理输入输出的运算符重载,为了使用的方便,这里把这两个运算符定义为友元函数。
整体的类的框架如下所示。
class String { public: String(const char *str = NULL); //通用构造函数 String(const String &str); //拷贝构造函数 ~String(); //析构函数 String operator+(const String &str) const; //重载+ String& operator=(const String &str); //重载= String& operator+=(const String &str); //重载+= bool operator==(const String &str) const; //重载== char& operator[](int n) const; //重载[] size_t size() const; //获取长度 const char* c_str() const; //获取C字符串 friend istream& operator>>(istream &is, String &str);//输入 friend ostream& operator<<(ostream &os, String &str);//输出 private: char *data; //字符串 size_t length; //长度 }; //注意,类的成员函数中,有一些是加了const修饰的,表示这个函数不会对类的成员进行任何修改。一些函数的输入参数也加了const修饰,表示该函数不会对改变这个参数的值。
二 具体实现
/*下面逐个进行成员函数的实现。 同样构造函数适用一个字符串数组进行String的初始化,默认的字符串数组为空。这里的函数定义中不需要再定义参数的默认值,因为在类中已经声明过了。 另外,适用C函数strlen的时候需要注意字符串参数是否为空,对空指针调用strlen会引发内存错误。*/ String::String(const char *str)//通用构造函数 { if (!str) { length = 0; data = new char[1]; *data = '\0'; } else { length = strlen(str); data = new char[length + 1]; strcpy(data, str); } } //拷贝构造函数需要进行深复制。 String::String(const String &str)//拷贝构造函数 { length = str.size(); data = new char[length + 1]; strcpy(data, str.c_str()); } //析构函数需要进行内存的释放及长度的归零。 String::~String()//析构函数 { delete []data; length = 0; } //重载字符串连接运算,这个运算会返回一个新的字符串。 String String::operator+(const String &str) const//重载+ { String newString; newString.length = length + str.size(); newString.data = new char[newString.length + 1]; strcpy(newString.data, data); strcat(newString.data, str.data); return newString; } //重载字符串赋值运算,这个运算会改变原有字符串的值,为了避免内存泄露,这里释放了原先申请的内存再重新申请一块适当大小的内存存放新的字符串。 String& String::operator=(const String &str)//重载= { if (this == &str) return *this; delete []data; length = str.length; data = new char[length + 1]; strcpy(data, str.c_str()); return *this; } //重载字符串+=操作,总体上是以上两个操作的结合。 String& String::operator+=(const String &str)//重载+= { length += str.length; char *newData = new char[length + 1]; strcpy(newData, data); strcat(newData, str.data); delete []data; data = newData; return *this; } //重载相等关系运算,这里定义为内联函数加快运行速度。 inline bool String::operator==(const String &str) const//重载== { if (length != str.length) return false; return strcmp(data, str.data) ? false : true; } //重载字符串索引运算符,进行了一个简单的错误处理,当长度太大时自动读取最后一个字符。 inline char& String::operator[](int n) const//重载[] { if (n >= length) return data[length-1]; //错误处理 else return data[n]; } //重载两个读取私有成员的函数,分别读取长度和C字符串。 inline size_t String::size() const//获取长度 { return length; } //重载输入运算符,先申请一块足够大的内存用来存放输入字符串,再进行新字符串的生成。这是一个比较简单朴素的实现,网上很多直接is>>str.data的方法是错误的,因为不能确定str.data的大小和即将输入的字符串的大小关系。 istream& operator>>(istream &is, String &str)//输入 { char tem[1000]; //简单的申请一块内存 is >> tem; str.length = strlen(tem); str.data = new char[str.length + 1]; strcpy(str.data, tem); return is; } //重载输出运算符,只需简单地输出字符串的内容即可。注意为了实现形如cout<<a<<b的连续输出,这里需要返回输出流。上面的输入也是类似。 ostream& operator<<(ostream &os, String &str)//输出 { os << str.data; return os; } inline const char* String::c_str() const//获取C字符串 { return data; }
三 功能测试
编码完成后需要对代码进行测试,以下是一个简单但不够严谨的测试。
int main() { String s; cin >> s; cout << s << ": " << s.size() << endl; char a[] = "Hello", b[] = "World!"; String s1(a), s2(b); cout << s1 << " + " << s2 << " = " << s1 + s2 << endl; String s3 = s1 + s2; if (s1 == s3) cout << "First: s1 == s3" << endl; s1 += s2; if (s1 == s3) cout << "Second: s1 == s3" << endl; /*程序输入输出为: 123456789 123456789: 9 Hello + World! = HelloWorld! Second: s1 == s3 Press any key to continue . . . */ }
原文: