【C++】string的底层剖析以及模拟实现

一、字符串类的认识

C语言中,字符串是以'\0'结尾的一些字符的集合,为了操作方便,C标准库中提供了一些str系列的库函数, 但是这些库函数与字符串是分离开的,不太符合OOP的思想,而且底层空间需要用户自己管理,稍不留神可能还会越界访问。在OJ中,有关字符串的题目基本以string类的形式出现,而且在常规工作中,为了简单、方便、快捷,基本 都使用string类,很少有人去使用C库中的字符串操作函数。为了增加自己对于string的理解,自己将模仿库中string类有的方法,设计一个简单的string类。 其中类成员包括以下:

cpp 复制代码
class string
    {

    private:

        char* _str;//字符串首地址

        size_t _capacity;//字符串容量

        size_t _size;//有效数据的个数

    public:

        typedef char* iterator;
    }

二、库中string常用的方法

我主要会实现string中经常会用到的方法,若大家想要了解更多关于string的细节,可以登录这个C++查询网站https://cplusplus.com/reference/自行查询下面是一些常用方法以及代码片段,可能前面出现的方法会用到后面出现的方法的实现,若有疑问可以看最后面的完整代码

正向迭代器

cpp 复制代码
iterator begin()
{
    return _str;
}

iterator end()
{
    return _str + _size;
}

+=

cpp 复制代码
        string& operator+=(char c)
        {
            if (_size == _capacity)
            {
                _capacity = _capacity == 0 ? 4 : 2 * _capacity;
                char* tmp = new char[_capacity +1];
                strcpy(tmp, _str);
                delete[] _str;
                _str = tmp;
            }

            _str[_size] = c;
            _str[_size + 1] = '\0';
            _size++;

            return *this;
        }

        string& operator+=(const char* str)
        {
            append(str);
            return *this;
        }

push_back(尾插)

cpp 复制代码
        void push_back(char c)
        {
            *this += c;
        }

append(在字符串末尾追加)

cpp 复制代码
        void append(const char* str)
        {
            int i = 0;
            while (str[i])
            {
                push_back(str[i]);
                i++;
            }
        }

clear(清除掉字符串的数据)

cpp 复制代码
        void clear()
        {
            _size = 0;
            _str[0] = '\0';
        }

swap(交换两个字符串的内容)

cpp 复制代码
        void swap(string& s)
        {
            std::swap(_str,s._str);
            std::swap(_size, s._size);
            std::swap(_capacity, s._capacity);
        }

c_str(返回字符串的首地址)

cpp 复制代码
        const char* c_str()const
        {
            return _str;
        }

resize(将字符串设定为指定大小,字符串占满所开辟的空间)

cpp 复制代码
        void resize(size_t n, char c = '\0')
        {
            if (n > _capacity)
            {
                reserve(n);
                for (int i = _size; i < _capacity; i++)
                {
                    _str[i] = c;
                }
                _size = _capacity;
            }
            else
            {
                _size = n;
            }
        }

reserve(预开辟出空间,字符串还是原来的大小(一般不缩容))

cpp 复制代码
        void reserve(size_t n)
        {
            if (n > _capacity)
            {
                _capacity = n;
                char* tmp = new char[_capacity + 1];
                strcpy(tmp, _str);
                delete[] _str;
                _str = tmp;
            }
        }

find(返回字符c在string中第一次出现的位置/返回子串s在string中第一次出现的位置**)**

cpp 复制代码
        size_t find(char c, size_t pos = 0) const
        {
            for (size_t i = pos; i < _size; i++)
            {
                if (_str[i] == c)
                    return i;
            }
            return std::string::npos;
        }
        size_t find(const char* s, size_t pos = 0) const
        {
            const char* ptr = std::strstr(_str + pos, s);

            if (ptr == nullptr)
                return std::string::npos;
            else
            {
                return ptr - _str;
            }
        }

insert(在pos位置上插入字符c/字符串str,并返回该字符的位置)

cpp 复制代码
        string& insert(size_t pos, char c)
        {
            if (_size == _capacity)
            {
                reserve(_capacity == 0 ? 4 : 2 * _capacity);
            }
            size_t end = _size - 1;
            while (end >= pos)
            {
                _str[end + 1] = _str[end];
                end--;
            }
            _str[pos] = c;
            return *this;
        }

        string& insert(size_t pos, const char* str)
        {
            int len = 0;
            while (str[len++]);
            if (_size + len > _capacity)
            {
                reserve(_size + len);
            }

            memmove(_str + pos + len, _str + pos, len * sizeof(char));

            for (int i = pos; i < pos + len; i++)
            {
                _str[i] = str[i - pos];
            }
            _size += len;
            return *this;
        }

erase(删除pos位置上的元素,并返回该string)

cpp 复制代码
        string& erase(size_t pos, size_t len)
        {
            memmove(_str + pos, _str + pos + len, (_size - pos-len) * sizeof(char));
            _size -= len;
            return *this;
        }

三、完整代码

cpp 复制代码
//string.h
#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include <iostream>
#include <string>
//using namespace std;

namespace sxb
{

    class string
    {

        friend std::ostream& operator<<(std::ostream& _cout, const string& s);

        friend std::istream& operator>>(std::istream& _cin, string& s);
    private:

        char* _str;

        size_t _capacity;

        size_t _size;

    public:

        typedef char* iterator;

    public:

        string(const char* str = "")
        {
            //_str = str;
            int len = 0;
            while(str[len] != ' ' && str[len] != '\0')
            {
                len++;
            }
            _str = new char[len + 1];
            for (int i = 0; i < len; i++)
            {
                _str[i] = str[i];
            }
            _str[len] = '\0';
            _capacity = len;
            _size = len;
        }

        string(const string& s)
        {
            _str = new char[s.size() + 1];
            strcpy(_str, s.c_str());
            _str[s.size()] = '\0';
            _capacity = s.size();
            _size = s.size();
        }

        string& operator=(const string& s)
        {
            for (int i = 0; i < size(); i++)
            {
                _str += s[i];
            }
            return *this;
        }

        ~string()
        {
            delete[] _str;
            _size = 0;
            _capacity = 0;
        }



            //

            // iterator

        iterator begin()
        {
            return _str;
        }

        iterator end()
        {
            return _str + _size;
        }



        //    /

        //    // modify

        void push_back(char c)
        {
            *this += c;
        }

        string& operator+=(char c)
        {
            if (_size == _capacity)
            {
                _capacity = _capacity == 0 ? 4 : 2 * _capacity;
                char* tmp = new char[_capacity +1];
                strcpy(tmp, _str);
                delete[] _str;
                _str = tmp;
            }

            _str[_size] = c;
            _str[_size + 1] = '\0';
            _size++;

            return *this;
        }

        void append(const char* str)
        {
            int i = 0;
            while (str[i])
            {
                push_back(str[i]);
                i++;
            }
        }

        string& operator+=(const char* str)
        {
            append(str);
            return *this;
        }

        void clear()
        {
            _size = 0;
            _str[0] = '\0';
        }

        void swap(string& s)
        {
            std::swap(_str,s._str);
            std::swap(_size, s._size);
            std::swap(_capacity, s._capacity);
        }

        const char* c_str()const
        {
            return _str;
        }



        ///

         capacity

        size_t size()const
        {
            return _size;
        }

        size_t capacity()const
        {
            return _capacity;
        }

        bool empty()const
        {
            return _str[0] == '\0';
        }

        void resize(size_t n, char c = '\0')
        {
            if (n > _capacity)
            {
                reserve(n);
                for (int i = _size; i < _capacity; i++)
                {
                    _str[i] = c;
                }
                _size = _capacity;
            }
            else
            {
                _size = n;
            }
        }

        void reserve(size_t n)
        {
            if (n > _capacity)
            {
                _capacity = n;
                char* tmp = new char[_capacity + 1];
                strcpy(tmp, _str);
                delete[] _str;
                _str = tmp;
            }
        }



        ///

         access

        char& operator[](size_t index)
        {
            return _str[index];
        }

        const char& operator[](size_t index)const
        {
            return _str[index];
        }



        ///

        relational operators

        bool operator==(const string& s)
        {
            if (_size != s.size())
                return false;
            for (int i = 0; i < _size; i++)
            {
                if (_str[i] != s[i])
                    return false;
            }
            return true;
        }

        bool operator!=(const string& s)
        {
            return !operator==(s);
        }



         返回c在string中第一次出现的位置

        size_t find(char c, size_t pos = 0) const
        {
            for (size_t i = pos; i < _size; i++)
            {
                if (_str[i] == c)
                    return i;
            }
            return std::string::npos;
        }

         返回子串s在string中第一次出现的位置

        size_t find(const char* s, size_t pos = 0) const
        {
            const char* ptr = std::strstr(_str + pos, s);

            if (ptr == nullptr)
                return std::string::npos;
            else
            {
                return ptr - _str;
            }
        }

         在pos位置上插入字符c/字符串str,并返回该字符的位置

        string& insert(size_t pos, char c)
        {
            if (_size == _capacity)
            {
                reserve(_capacity == 0 ? 4 : 2 * _capacity);
            }
            size_t end = _size - 1;
            while (end >= pos)
            {
                _str[end + 1] = _str[end];
                end--;
            }
            _str[pos] = c;
            return *this;
        }

        string& insert(size_t pos, const char* str)
        {
            int len = 0;
            while (str[len++]);
            if (_size + len > _capacity)
            {
                reserve(_size + len);
            }

            memmove(_str + pos + len, _str + pos, len * sizeof(char));

            for (int i = pos; i < pos + len; i++)
            {
                _str[i] = str[i - pos];
            }
            _size += len;
            return *this;
        }



         删除pos位置上的元素,并返回该元素的下一个位置

        string& erase(size_t pos, size_t len)
        {
            memmove(_str + pos, _str + pos + len, (_size - pos-len) * sizeof(char));
            _size -= len;
            return *this;
        }



    };

    std::ostream& operator<<(std::ostream& _cout, const string& s)
    {
        for (int i = 0; i < s.size(); i++)
        {
            _cout << s[i];
        }
        return _cout;
    }

    std::istream& operator>>(std::istream& _cin, string& s)
    {
        char buffer[128];
        int len = 0;
        char bu = _cin.get();
        while (bu != ' ' && bu != '\n')
        {
            buffer[len] = bu;
            len++;
            bu = _cin.get();
        }

        for (int i = 0; i < len; i++)
        {
            s += buffer[i];
        }

        return _cin;
    }

}
相关推荐
奋斗的小花生13 分钟前
c++ 多态性
开发语言·c++
魔道不误砍柴功16 分钟前
Java 中如何巧妙应用 Function 让方法复用性更强
java·开发语言·python
闲晨19 分钟前
C++ 继承:代码传承的魔法棒,开启奇幻编程之旅
java·c语言·开发语言·c++·经验分享
老猿讲编程1 小时前
一个例子来说明Ada语言的实时性支持
开发语言·ada
UestcXiye2 小时前
《TCP/IP网络编程》学习笔记 | Chapter 3:地址族与数据序列
c++·计算机网络·ip·tcp
Chrikk2 小时前
Go-性能调优实战案例
开发语言·后端·golang
幼儿园老大*2 小时前
Go的环境搭建以及GoLand安装教程
开发语言·经验分享·后端·golang·go
canyuemanyue2 小时前
go语言连续监控事件并回调处理
开发语言·后端·golang
杜杜的man2 小时前
【go从零单排】go语言中的指针
开发语言·后端·golang
霁月风3 小时前
设计模式——适配器模式
c++·适配器模式