My_string类__.h

本文详细介绍了一个自定义字符串类My_string的实现过程,包括构造与析构函数、字符串操作函数及重载运算符等。作者从零开始,逐步构建了一个功能较为完善的字符串类,并分享了开发过程中的心得。

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

#ifndef Mystring
#define Mystring
#include <ostream>
class My_string
{
public:
//构造函数
My_string();
My_string(const My_string& s);
My_string(const char* a);
My_string(const char* a,const int n);
My_string(const int n,const char c);
My_string(const My_string& s,const int pos,const int n);
My_string(char* begin,char* end);
//析构函数
~My_string();

//一般成员函数
My_string& append(const My_string& s);
My_string& append(const My_string& s,const int n);
My_string& append(const char *a,const int n);
My_string& append(const My_string& s,const int m,const int n);
My_string& append(int n, char c);
My_string& assign(const My_string& s);
My_string& assign(const My_string& s,const int n);
My_string& assign(const My_string& s,const int m,const int n);
My_string& assign(const int n,const char c);

char& at(const int n)const;
char& at(const int n);
const char* c_str()const;

int compare(const My_string& s)const;
int compare(int pos,int n,const My_string& s);
int compare(int pos,int n,const My_string& s,int spos,int m);

int copy(My_string& s,const int m,const int n);
char* data();
My_string& erase(const int p0,const int n);

unsigned int find(const My_string& s,int pos=0)const;
unsigned int find(const My_string& s,int pos,int n)const;
unsigned int find(const char c,const int pos=0)const;
unsigned int rfind(const My_string& s,int pos=0)const;
unsigned int rfind(const My_string& s,int pos,int n)const;
unsigned int rfind(const char c,const int pos=0)const;
unsigned int find_first_of(const My_string& s,int pos=0)const;
unsigned int find_first_of(const My_string& s,int pos,int n)const;
unsigned int find_first_of(const char c,const int pos=0)const;
unsigned int find_first_not_of(const My_string& s,int pos=0)const;
unsigned int find_first_not_of(const My_string& s,int pos,int n)const;
unsigned int find_first_not_of(const char c,const int pos=0)const;
unsigned int find_last_of(const My_string& s,int pos=0)const;
unsigned int find_last_of(const My_string& s,int pos,int n)const;
unsigned int find_last_of(const char c,const int pos=0)const;
unsigned int find_last_not_of(const My_string& s,int pos=0)const;
unsigned int find_last_not_of(const My_string& s,int pos,int n)const;
unsigned int find_last_not_of(const char c,const int pos=0)const;

My_string& insert(int pos,const My_string& s);
My_string& insert(int pos,const My_string& s,int n);
My_string& insert(int pos,const My_string& s,int spos,int n);
My_string& insert(int pos,char c,int n);

const bool isEmpty();
const int length();

My_string& replace(int pos,int n,My_string& s);
My_string& replace(int pos,int n,My_string& s,int m);
My_string& replace(int pos,int n,My_string& s,int spos,int m);
My_string& replace(int pos,int n,int m,char c);

void swap(My_string& s);
const int size()const;

//重载函数
const char& operator [](int pos)const;
char& operator [](int pos);
My_string& operator =(const My_string& s);
My_string operator +(const My_string& s);
My_string& operator +=(const My_string& s);
bool operator ==(const My_string& s);
bool operator <(const My_string& s);
bool operator >(const My_string& s);
bool operator <=(const My_string& s);
bool operator >=(const My_string& s);
friend std::ostream& operator <<(std::ostream& output,My_string& s);

private:
const static unsigned int npos;
char* str;
int len;
static int Num_strings;
};
#endif
//此篇只是头文件,包含了几乎所有string类函数的声明,其定义在下篇见,请多多支持啦^-^
/*怎么说,我已经努力了,有时候觉得写代码也是一种美,写了差不多一个来月,即使这个偶尔会觉得很枯燥很幸苦,我一直相信,只要有付出,自己就可换回我想要的回报。模拟系统的string类,虽然写得很烂,但还是很开心,毕竟这是自己努力而来的成果,(*^__^*) 嘻嘻……真的很开心。O(∩_∩)O哈哈~另一方面,我还是一个十足的新手,很多方面都不懂,所以里面可能有许多的Debug,很多不足的的地方,而且还有些应用于容器的函数没有写上去,望某些牛人多多指教啦,多多支持(*^__^*) 嘻嘻……*/

<think>好的,用户想了解C或C++中的#ifndef指令在包含守卫中的用法,以及相关的编译问题。首先,我需要回忆#ifndef的基本作用。它是预处理器指令,用于条件编译,防止头文件被重复包含。 接下来,我应该解释包含守卫的基本结构。通常,头文件开始于#ifndef一个唯一的标识符,接着#define这个标识符,最后在结尾#endif。这样,如果同一个头文件被多次包含,预处理器会跳过重复的内容,避免重定义错误。 然后,需要考虑用户可能遇到的问题。比如,忘记写#define导致守卫失效,或者标识符冲突。比如两个不同的头文件使用相同的标识符,可能导致其中一个被错误地排除。这时候应该建议使用唯一的命名约定,比如包含项目名、文件名等。 还要提到现代编译器中的替代方法,比如#pragma once,但需要注意它的可移植性问题,因为不是所有编译器都支持。这时候需要结合引用中的内容,比如引用[1]提到包含头文件可以使用其中的和函数,所以正确使用包含守卫很重要。 用户可能还会想知道如何选择标识符。通常用全大写字母,以下划线和项目名称为前缀,例如MYPROJECT_MYCLASS_H。这样可以减少冲突的可能性。 另外,可能需要提到在嵌套包含的情况下,包含守卫如何工作,以及可能的陷阱,比如在头文件内部有其他#include语句时,守卫的位置是否正确放置。 还要考虑示例代码,比如展示一个典型的头文件结构,使用#ifndef,然后如何防止重复包含。同时,可以引用用户提供的代码示例中的命名空间用法,比如引用[3]中的my::String和your::String,说明如果头文件没有正确守卫,可能会导致命名冲突。 最后,总结使用时需要注意的事项,比如唯一标识符、现代编译器的替代方案、常见错误等。这样用户能全面了解#ifndef的用法和解决相关问题的方法。</think>在C/C++中,$#ifndef$指令主要用于头文件的包含守卫(Include Guards),其核心作用是防止头文件内容被重复包含。以下是详细说明和常见问题解决方案: ### 一、基本用法 头文件的标准结构示例: ```cpp // myheader.h #ifndef MYHEADER_H // 条件检查 #define MYHEADER_H // 定义标识符 // 头文件内容(声明、函数原型等) namespace my { class String {...}; } #endif // 结束条件编译 ``` 当预处理器首次遇到该头文件时,$MYHEADER_H$未定义,代码块被包含;后续包含时因标识符已定义,自动跳过该文件内容[^1]。 ### 二、关键作用 1. **防止重复定义错误** 当多个源文件包含同一头文件时,避免/函数重复定义: ```cpp // 错误示例:未使用包含守卫时 class MyClass {...}; // 第一次包含时定义 class MyClass {...}; // 第二次包含时报重定义错误 ``` 2. **解决循环包含问题** 当$A.h$包含$B.h$,同时$B.h$又包含$A.h$时,包含守卫能中断递归包含。 ### 三、常见问题与解决方案 | 问题现象 | 原因分析 | 解决方案 | |---------|---------|---------| | 链接错误LNK2005 | 头文件中存在变量/函数的定义而非声明 | 将定义移至源文件,或使用$inline$关键字 | | 守卫标识符冲突 | 不同头文件使用相同宏名称(如$COMMON_H$) | 采用`<项目>_<路径>_<文件名>_H`格式命名 | | 条件编译失效 | 忘记在头文件结尾写$#endif$ | 使用IDE代码模板自动生成守卫结构 | ### 四、现代替代方案 多数现代编译器支持更简洁的指令: ```cpp #pragma once // 非标准但广泛支持 ``` 其优势在于自动基于文件路径生成唯一标识符,但需注意: - 在GCC 4.2+、MSVC 19.0+、Clang 3.0+中可用 - 无法处理通过符号链接指向同一文件的场景 ### 五、最佳实践建议 1. 在开源项目中优先使用传统的$#ifndef$守卫以保证最大兼容性[^2] 2. 守卫标识符命名建议包含命名空间,如: ```cpp #ifndef MYLIB_UTILS_STRING_H ``` 3. 使用Clang-Format等工具自动对齐$#endif$注释 : 包含守卫确保了头文件内容的唯一包含性,这与头文件提供可用性的机制直接相关 [^2]: 遵循C++11兼容性原则时,需注意预处理器指令的标准化特性 [^3]: 命名空间的合理使用能增强包含守卫的冲突规避能力
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值