string【基础篇】

本文详细介绍了C++中的string类,包括构造方法、容量管理、遍历方式、修改操作以及非成员函数,特别关注了reserve和resize的区别,以及如何在C++11中使用。还提供了三个实例,帮助读者理解和应用string类的各种功能。
摘要由CSDN通过智能技术生成

1.1string字符串类

注意:这个类独立于所使用的编码来处理字节:如果用来处理多字节或变长字符(如UTF-8)的序列,这个

类的所有成员(如长度或大小)以及它的迭代器,将仍然按照字节(而不是实际编码的字符)来操作。

总结:

  1. string是表示字符串的字符串类
  2. 该类的接口与常规容器的接口基本相同,再添加了一些专门用来操作string的常规操作。
  3. string在底层实际是:basic_string模板类的别名,typedef basic_string<typedef basic_string<char, char_traits, allocator> string;
  4.  不能操作多字节或者变长字符的序列。

在使用string类时,必须包含#include头文件以及using namespace std;

1.2string接口(函数)

1.string类对象的常见构造

string()

构造空的string类对象,即空字符串

string(const char* s)

用C-string来构造string类对象

string(size_t n, char c)

string类对象中包含n个字符c

string(const string&s)

拷贝构造函数

void Teststring()
{
 string s1; // 构造空的string类对象s1
 string s2("hello bit"); // 用C格式字符串构造string类对象s2
 string s3(s2); // 拷贝构造s3
}

2.容量操作

函数名称

功能说明

size(重点)

返回字符串有效字符长度

length

返回字符串有效字符长度

capacity

返回空间总大小

empty

检测字符串释放为空串,是返回true,否则返回false

clear

清空有效字符(不清理空间【缩容】)

reserve

为字符串预留空间【扩容】(比capacity大扩容)

resize

将有效字符的个数改成n个,多出的空间用字符空白符填充【可扩可缩】

reserse

将容器范围内的元素颠倒顺序放置

shrink_to_fit

缩容(不会为0)

STL扩容机制(在不同编译器有所不同)

int asz=capacity();

注意:下图表示的是reserve与resize的不同

容量原本是10个空间大小,随后我们要求reserve(100)预留了100空间大小,然后使用resize(5),使得元素个数只有5个,其空间容量并不改变。

其可以说明resize不能改变容量的大小。

注意:

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

3.遍历法门

字符串的遍历主要分为三种:引用( [ ] ),迭代器(最实用),访问for

operator[ ] 

返回pos位置的字符,const string类对象调用

begin+ end (从前向后)

begin获取前一个字符的迭代器 + end获取最后一个字符下一个位置的迭代器(可看作'\0'的位置)

rbegin +rend(从后向前)

rbegin获取一个字符的迭代器 + end获取最后一个字符下一个位置的迭代器(反省迭代器)

范围for

C++11支持更简洁的范围for的新遍历方式

迭代器遍历: 

4.修改操作

push_back 

在字符串后尾插字符c

append

在字符串后追加一个字符串

operator+= (重点)

在字符串后追加字符串str

assign

覆盖原内容(多出补空白符)

insert

添加/截取内容(从pos开始的len个)“位置”

有挪动数据

erase

删除内容(从pos开始的len个)有挪动数据

replace

替换内容(从pos开始的len个)有挪动数据

c_str(重点)

返回C格式字符串

find + npos(重点)

从字符串pos位置开始往后找字符c,返回该字符在字符串中的位置

rfind

从字符串pos位置开始往前找字符c,返回该字符在字符串中的位置

substr

在str中从pos位置开始,截取n个字符,然后将其返回

注意:

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

5.string类非成员函数

operator+

尽量少用,因为传值返回,导致深拷贝效率低

operator>> (重点)

输入运算符重载

operator (重点)

输出运算符重载

getline (重点)

获取一行字符串

relational operators (重点)

大小比较

 注意:string使用cin>>str输入时遇到空格之会保存空格之前的

扩展:

find_first_of 查找一个字符串中出现某一字符的位置并返回

例子:

int size_pos=strText.find_first_of(strSeparator(查找字符), size_pos())

find_last_of 查找一个字符串中出现某一字符最后出现的位置并返回 int size_pos=strText.find_last_of(strSeparator(查找字符), size_pos()))

substr 截取字符串中的“内容“

例子:string strResult = strText.substr(size_prev_pos(头), size_pos-size_prev_po(尾)); 

1.3. 牛刀小试

了解了这么多的接口,试着用一两的写写下面的题,理解理解用法、注意事项。

1.仅仅反转字母

 仅仅反转字母 

class Solution {
public:
 bool isLetter(char ch)
 {
 if(ch >= 'a' && ch <= 'z') return true; if(ch >= 'A' && ch <= 'Z') return true;
 return false;
 }
 string reverseOnlyLetters(string S) { if(S.empty()) return S;
 
 size_t begin = 0, end = S.size()-1; while(begin < end)
 {
 while(begin < end && !isLetter(S[begin]))
 ++begin;
 
 while(begin < end && !isLetter(S[end]))
 --end;
 swap(S[begin], S[end]);
 ++begin;
 --end;
 }
 return S;
 }
};

2.找字符串中第一个只出现一次的字符

找字符串中第一个中出现一次的字符

class Solution {
public:
 int firstUniqChar(string s) {
 
 // 统计每个字符出现的次数 int count[256] = {0};
 int size = s.size(); for(int i = 0; i < size; ++i) count[s[i]] += 1; 
 // 按照字符次序从前往后找只出现一次的字符 for(int i = 0; i < size; ++i) if(1 == count[s[i]]) return i; 
 return -1;
 }
};

3.字符串里面最后一个单词的长度

符串里面最后一个单词的长度 

#include<iostream>
#include<string>
using namespace std;
int main()
{
 string line;
 // 不要使用cin>>line,因为会它遇到空格就结束了 // while(cin>>line)
 while(getline(cin, line))
 {
 size_t pos = line.rfind(' '); cout<<line.size()-pos-1<<endl; } return 0;
}

  • 22
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值