【C++/STL】:string类的基本使用

简介: 【C++/STL】:string类的基本使用

引言

string 就是我们常说的"串",它是一种字符数组,只不过这个数组具备扩容,增删查改等功能。string类在我们日常生活中是十分常用的,并且在笔试,面试中也经常出现,它是学习C++的不可缺少的一部分。

string类大概有120个函数接口,注意下面只讲解最常用的接口,想要了解更多,前往https://legacy.cplusplus.com/reference/string/string/网站里浏览。

注意:使用string类时要包含 < string >

一,string类对象的常见构造

代码演示:

void test_string1()
{
  string s1;//无参默认构造
  string s2("hello world");//用字符串构造
  //从str中的pos下标位置,拷贝len个字符
  string s4(s2, 3, 5);
  
  //不传第三个参数,默认拷贝到结尾,缺省参数npos是整型最大值
  string s5(s2, 3);
  string s3(s2);//拷贝构造
}

二,string类对象的容量操作

代码演示1:

// size/clear/resize
void Test_string2()
{
  //注意:string类对象支持直接用cin和cout进行输入和输出
  string s("hello, world!!!");
  
  //计算字符个数,不算\0
  cout << s.size() << endl;
  cout << s.length() << endl;
  
  //计算容量,不算\0
  //capacity比实际空间少一个,有一个多的是预留给\0的
  cout << s.capacity() << endl;
  cout << s << endl;
  //将s中的字符串清空,注意清空时只是将size清0,不改变底层空间的大小
  s.clear();
  cout << s.size() << endl;
  cout << s.capacity() << endl;
  // 将s中有效字符个数增加到10个,多出位置用'a'进行填充
  //当传的值大于size时,此时会影响capacity+size,两个都会增加
  // “aaaaaaaaaa”
  s.resize(10, 'a');
  cout << s.size() << endl;
  cout << s.capacity() << endl;
  // 将s中有效字符个数增加到15个,多出位置用缺省值'\0'进行填充
  // "aaaaaaaaaa\0\0\0\0\0"
  // 注意此时s中有效字符个数已经增加到15个
  s.resize(15);
  cout << s.size() << endl;
  cout << s.capacity() << endl;
  cout << s << endl;
  // 将s中有效字符个数缩小到5个
  s.resize(5);
  cout << s.size() << endl;
  cout << s.capacity() << endl;
  cout << s << endl;
}

代码演示1:

//测试reserve
void test_string3()
{
  string s;
  
  //一般用在知道需要多少空间,提前开好
  //这样可以避免每次都要开空间,提升效率
  s.reserve(100);
  string s1("111111111");
  cout << s1.capacity() << endl;//15
  //扩容
  //reserve只影响capacity,不影响size,即不改变里面的数据
  s1.reserve(100);
  cout << s1.capacity() << endl;//111
  //缩容
  //传的值比当前的capacity小时,
  // vs一般不缩容,g++会缩
  s1.reserve(10);
  cout << s1.capacity() << endl;//15
}

注意:

  1. size()与length()方法底层实现原理完全相同,引入size()的原因是为了与其他容器的接口保持一致,一般情况下基本都是用size()。
  2. clear()只是将string中有效字符清空,使size = 0,不改变底层空间大小。
  3. resize(size_t n) 与 resize(size_t n, char c)都是将字符串中有效字符个数改变到n个,不同的是当字符个数增多时:resize(size_t n)用0来填充多出的元素空间,resize(size_t n, char c)用字符c来填充多出的元素空间。注意:resize在改变元素个数时,如果是将元素个数增多,可能会改变底层容量的大小,如果是将元素个数减少,底层空间总大小不变。
  4. reserve(size_t res_arg=0):为string预留空间,不改变有效元素个数,当reserve的参数小于string的底层空间总大小时,reserver不会改变容量大小。

三,string类对象的访问及遍历操作

string类对象的访问及遍历有三种方式:

1. 迭代器:begin()+end()

现在我们暂时把迭代器理解成像指针一样的东西,但是不一定是指针,后面会细讲:

begin返回第一个数据位置的iterator;

end返回最后一个数据的下一个位置(\0)的iterator;

2. for+[]

这是一个[ ]运算符重载。利用下面的模拟代码来说明几个问题:

(1) 引用返回的作用:一是减少拷贝,二是修改返回对象。

(2) 为什么可以用引用返回:_str[i]出了作用域还在,因为_str开辟在堆上,它返回的是堆上的一个字符的引用别名。

(3) 重载的底层也是用assert断言的,只要下标越界直接终止报错。

//大概模拟string类说明几个问题
class string
{
public:
  //引用返回
    //1.减少拷贝
    //2.修改返回对象  //s1[0] = 'x';
  char& operator[](int i)
  {
    assert(i < _size);
    return _str[i];//返回的是第i个字符的别名
     //用引用返回:_str[i]出了作用域还在,因为_str开辟在堆上
     //它返回的是堆上的一个字符的引用别名
  }
private:
  char* _str;
  int _size;
  int _capacity;
};

3. 范围for

注意:string遍历时使用最多的还是for+下标 或者 范围for(C++11后才支持)

begin()+end()大多数使用在需要使用STL提供的算法操作string和后面的list容器,set容器,比如:采用reverse逆置string,使用sort按字典序排序(按ASCII码值排序)。

代码演示:

void Teststring4()
{
  string s("hello Bit");
  
  // 3种遍历方式:
  // 需要注意的以下三种方式除了遍历string对象,还可以遍历是修改string中的字符,
  // 另外以下三种方式对于string而言,第一种使用最多
  
  // 1. for+operator[]
  for (size_t i = 0; i < s.size(); ++i)
    cout << s[i] << endl;
  // 2.迭代器
  //string::iterator it = s.begin();
  //iterator 是typedef在类域中的,类似域内部类,所以在域外使用时要指定类域
  
  auto it1 = s1.begin();
  while (it != s.end())
  {
    cout << *it << endl;
    ++it;
  }
  // string::reverse_iterator rit = s.rbegin();
  // C++11之后,直接使用auto定义迭代器,让编译器推到迭代器的类型
  auto rit = s.rbegin();
  while (rit != s.rend())
  {
    cout << *rit << endl;
  }
  // 3.范围for
  //自动取出s中的数据赋给ch,自动判断结束,自动++。
  //其实底层就是迭代器。
  for (auto ch : s)
  {
    cout << ch << endl;
  }
}

使用sort按字典序排序(按ASCII码值排序):

void test_string5()
{
  string s1("hello world");
  cout << s1 << endl;
  //按字典序排序(按ASCII码值排序)
  //用排序函数sort [first last)左闭右开,last传的不是有效数据
  //sort(s1.begin(), s1.end());
  //第一个和最后一个不参与排序
  //sort(++s1.begin(), --s1.end());
  //前5个排序  [0,5)
  sort(s1.begin(), s1.begin() + 5);
  cout << s1 << endl;
}

四,string类对象的修改操作

  1. 插入(拼接)方式:push_back append operator+=
  2. 正向和反向查找:find() + rfind()
  3. 截取子串:substr()
  4. 删除:erase

代码演示1:

void test_string6()
{
  string s1("hello world");
  cout << s1 << endl;
  s1.push_back('x');//一个字符一个字符尾插
  cout << s1 << endl;
  s1.append(" yyyyyyy!!");//尾插一个字符串
  cout << s1 << endl;
  string s2("22222");
  //直接尾插
  s1 += 'aaa';
  s1 += 'd';
  s1 += s2;
  cout << s1 << endl;
}

代码演示2:

void test_string7()
{
    // 获取file的后缀
  string file("string.cpp");
  
  //从后往前找'.'
  size_t pos = file.rfind('.');
  string suffix(file.substr(pos, file.size() - pos));
  cout << suffix << endl;
  // npos是string里面的一个静态成员变量
  // static const size_t npos = -1;
  // 取出url中的域名
  string url("http://www.cplusplus.com/reference/string/string/find/");
  cout << url << endl;
  size_t start = url.find("://");
  
  if (start == string::npos)
  {
    cout << "invalid url" << endl;
    return;
  }
  
  start += 3;
  size_t finish = url.find('/', start);
  string address = url.substr(start, finish - start);
  cout << address << endl;
  // 删除url的协议前缀
  pos = url.find("://");
  url.erase(0, pos + 3);
  cout << url << endl;
}

注意:

  1. 在string尾部追加字符时,s.push_back( c ) / s.append(1, c) / s += 'c’三种的实现方式差不多,一般情况下string类的+=操作用的比较多,+=操作不仅可以连接单个字符,还可以连接字符串。
  2. 对string操作时,如果能够大概预估到放多少字符,可以先通过reserve把空间预留好。

五,string类非成员函数

代码演示1:

//+运算符
void test_string8()
{
  string s1 = "hello";
  string s2 = "world";
  string ret1 = s1 + s2;
  cout << ret1 << endl;
  string ret2 = s1 + "xxxxx";
  cout << ret2 << endl;
  string ret3 = "xxxxx" + s1;
  cout << ret3 << endl;
  //按字典序比较
  cout << (s1 < s2) << endl;
}

代码演示2:

//getline
int main()
{
  string str;
  string str2;
  //如何停止输入?
  //ctrl+c
  //ctrl+z+空格
  //  while (cin >> str2)
  //  {
  //    cout << str2 << endl;
  //  }
  //cin 遇到空格或者换行会停止提取
  //cin >> str;
  //获取一行包含空格的字符串
  getline(cin, str);
  int pos = str.rfind(' ');
  cout << str.size() - (pos + 1) << endl;
  return 0;
}

六,整形与字符串的转换

int main()
{
  int x = 0, y = 0;
  cin >> x >> y;
  
  //to_string:整形转字符串
  string str = to_string(x + y);
  cout << str << endl;
  
  //stoi:字符串转整形
  int aa = stoi(str);
  cout << aa << endl;
  return 0;
}

七,string类中的insert,assign,erase,replace

注意:

insert,erase,replace要慎用,因为string的底层其实就是数组顺序表,在pos位置插入,删除和替换时,需要挪动数据,时间复杂度为O(N),效率不高。

代码演示1:

// insert/assign
void test_string9()
{
  string s1("hello world");
  cout << s1 << endl;
  //变相赋值,会把原来的字符串全部替换掉
  s1.assign("111111");
  cout << s1 << endl;
  //效率不高 O(N)
  //头插一个字符串
  string s2("hello world");
  
  //下标位置  字符串
  s2.insert(0, "aaaaaa");
  cout << s2 << endl;
  //头插一个字符
  //下标位置  插入的个数 字符
  s2.insert(0, 1, 'f');
  cout << s2 << endl;
}

代码演示2:

// erase/replace
void test_string10()
{
  string s1("hello world");
  cout << s1 << endl;
  //效率不高 O(N)
  //头删字符
  s1.erase(0, 2);//下标位置  字符个数
  cout << s1 << endl;
  //效率不高 O(N)
  //替换部分字符串
  string s2("hello world");
  
  //下标位置 字符个数 替换内容 
  s2.replace(5, 1, "%20");
  cout << s2 << endl;
}
目录
打赏
0
0
0
0
7
分享
相关文章
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
39 16
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
18天前
|
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
63 6
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
14天前
|
《从头开始学java,一天一个知识点》之:字符串处理:String类的核心API
🌱 **《字符串处理:String类的核心API》一分钟速通!** 本文快速介绍Java中String类的3个高频API:`substring`、`indexOf`和`split`,并通过代码示例展示其用法。重点提示:`substring`的结束索引不包含该位置,`split`支持正则表达式。进一步探讨了String不可变性的高效设计原理及企业级编码规范,如避免使用`new String()`、拼接时使用`StringBuilder`等。最后通过互动解密游戏帮助读者巩固知识。 (上一篇:《多维数组与常见操作》 | 下一篇预告:《输入与输出:Scanner与System类》)
44 11
|
20天前
|
课时14:Java数据类型划分(初见String类)
课时14介绍Java数据类型,重点初见String类。通过三个范例讲解:观察String型变量、&quot;+&quot;操作符的使用问题及转义字符的应用。String不是基本数据类型而是引用类型,但使用方式类似基本类型。课程涵盖字符串连接、数学运算与字符串混合使用时的注意事项以及常用转义字符的用法。
课时44:String类对象两种实例化方式比较
本次课程的主要讨论了两种处理模式在Java程序中的应用,直接赋值和构造方法实例化。此外,还讨论了字符串池的概念,指出在Java程序的底层,DOM提供了专门的字符串池,用于存储和查找字符串。 1.直接赋值的对象化模式 2.字符串池的概念 3.构造方法实例化
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
144 2
|
5月前
|
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
本文深入探讨了Java中方法参数的传递机制,包括值传递和引用传递的区别,以及String类对象的不可变性。通过详细讲解和示例代码,帮助读者理解参数传递的内部原理,并掌握在实际编程中正确处理参数传递的方法。关键词:Java, 方法参数传递, 值传递, 引用传递, String不可变性。
116 1
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性