C++--String类

系列文章目录


文章目录

目录

系列文章目录

文章目录

前言

一、为什么要学习string

1.c语言的字符串

2.OJ上的使用

二、string类的接口介绍

1.string简介

2.string构造成员函数

3.operator=函数

4.string容器size和length

5.重载operator[]和引用返回的意义

        5.1 operator[]

        5.2 at访问

        5.3.front 和 back

6.迭代器Iterators

范围for遍历

7.反向迭代器iterator

8.插入和修改

8.1 push_back尾插

8.2 append追加

8.3  +=常用​编辑

8.4 pop_back和erase

8.5 assign 分配

8.6 insert(重点)

8.7 erase  删除

8.8 replace 替换

8.9 swap交换

9.其他容器(resize,reserve····)

9.1 capacity、max_size

9.2 reserve、resize

1 reserve

2 resize

3 shrink_to_fit

10. 字符串操作

        10.1 c_str

        10.2 find

rfind的应用:

        10.3 find_first_of

        10.4 find_last_of

        10.5 find_not_first_of和find_not_last_of

        10.6 compare

11.非成员函数

     11.1 operator+

     11.2 relational operator

12.字符串和其他数组类型转换

三、字符串题目:

        1.仅仅反转字母

2.字符串中的第一个唯一字符

3.验证回文串

4.字符串相加

5.字符串最后一个单词的长度

6.getline

四、字符排序sort


前言

        STL(standard template libaray-标准模板库):是C++标准库的重要组成部分,不仅是一个可复用的组件库,而且是一个包罗数据结构与算法的软件框架。网上有句话说:“不懂STL,不要说你会C++”。STL是C++中的优秀作品,有了它的陪伴,许多底层的数据结构以及算法都不需要自己重新造轮子,站在前人的肩膀上,健步如飞的快速开发。


一、为什么要学习string

1.c语言的字符串

        C语言中,字符串是以'\0'结尾的一些字符的集合,为了操作方便,C标准库中提供了一些str系列的库函数,但是这些库函数与字符串是分离开的,不太符合OOP的思想,而且底层空间需要用户自己管理,稍不留神可能还会越界访问。

2.OJ上的使用

        在OJ中,有关字符串的题目基本以string类的形式出现,而且在常规工作中,为了简单、方便、快捷,基本都使用string类,很少有人去使用C库中的字符串操作函数。

二、string类的接口介绍

       在学习类时离不开接口的查询,所以我们通过文档来介绍string:
stringicon-default.png?t=N7T8https://legacy.cplusplus.com/reference/string/string/

1.string简介

        

string实际上是个类模板basic_string<template>. 这里涉及编码,我们后面在讲解。接下来我们开始介绍接口,将string理解为char类型的顺序表即可。

2.string构造成员函数

        构造函数:

我们介绍最常用的几个,分别是(1)(2)(4):

	string s1;                 //(1)string s2("hello string!");//(4)string s3(s2);             //(2)

我们使用cout和cin输出输入string实例化的对象

#include <iostream>
#include <string>
using namespace std;void test_string01()
{string s1;string s2("hello string!");string s3(s2);cout << s1 << endl;cout << s2 << endl;cout << s3 << endl;cin >> s1;cout << s1 << endl;
}int main()
{test_string01();return 0;
}

注意:string的输入和输出是被重载过的所以我们能使用。

对于这个我们可以猜想,pos时postion(位置)的缩写,是部分拷贝。关于npos我们也可以查找

无符号的整形-1,按照补码储存,整形最大值,由于一个数组开4G不现实,使用npos实际上就是结束位置。所以如果len > 字符长度或者不传时就是npos

void test_string01()
{string s1;string s2("hello string!");string s3(s2);string s4(s2, 6);cout << s4 << endl;string s5(s2, 6, 12);cout << s5 << endl;
}

结果:

下面这样也可以:

string s6("hello string!", 6, 12);

这样实际上是隐式类型转换,字符串"hello string!"构造临时对象,然后通过函数进行构造。

优化后是直接构造:

//构造
string s1("hello string!");
//隐式类型转换
string s2 = "hello string!";
//隐式类型转换有临时对象产生
const string& s3 = "hello string!";

C++类和对象(下)-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/Jk_Mr/article/details/138169401?spm=1001.2014.3001.5501上的文章中解释了隐式类型转换

对于(5)查看文档可知,(5)是拷贝字符串"···"前n个字符

对于(6),则是使用n个字符‘’进行构造:

后面几个了解一下。关于(7)涉及迭代器我们后面再将。

3.operator=函数

        赋值可以是字符串、字符或者string对象。

4.string容器size和length

string s1("hello string!\n");
cout << s1.size() << endl;
cout << s1.length() << endl;

sizelength是一样的。为什么呢?

实际上是因为string比STL出来的更早,当时定义长度的就是length。由于要保持接口一致性所以增加了size。

同样为了兼容c语言,字符串的最后一个实际上是\0.size和length都不计算\0.

5.重载operator[]和引用返回的意义

        5.1 operator[]

        

有了这个重载,string就可以像数组一样去使用。它的原理类似于:

class string
{
public://引用返回//1、减少拷贝//2、修改返回值char& operator[](size_t i){assert(i < _size);return _str[i];}
private:char* _str;size_t _size;size_t _capacity;
};

那么就可以像数组一样进行访问:

string s1("hello string!\n");
//可以像数组一样进行访问
for (int i = 0; i < s1.size(); i++)
{cout << s1[i] << ' ';
}

可以像数组一样进行修改:

//可以进行修改
for (int i = 0; i < s1.size(); i++)
{s1[i] = 'x';
}

对象开辟的空间在上。所以可以进行访问和修改。

并且string越界可以检测,使用的是断言assert

关于第二个const修饰的operator=()函数,是因为要访问不可修改的对象

下标[]实际上只有连续的空间能够支持,string、vector(顺序表)。

那么还有别的方式访问string 的内容吗?接下来我们介绍迭代器。

         5.2 at访问

和[]的功能是一样的,但是越界访问有区别。operator[]是断言没有商量的余地。

而at是抛异常,可以捕获。(异常后面讲解)

string s1("hello world!");
s1[50];

string s1("hello world!");
s1.at(50);

	string s1("hello world!");try {s1.at(50);}catch (const exception& e){cout << e.what() << endl;}

        5.3.front 和 back

        front是获取首元素,back是尾元素。

6.迭代器Iterators

        迭代器通常包含指向容器中某个元素的指针或引用,以及一些操作符(如递增、递减等),用于在容器中移动和访问元素。

先看使用方法:
        可以将迭代器理解为指针或者像指针一样的东西(不一定是指针)

string s1("hello string!");
//遍历方式1:下标
for (int i = 0; i < s1.size(); i++)
{cout << s1[i] << ' ';
}
cout << endl;
//遍历方式2
string::iterator it1 = s1.begin();//iterator需要指定类域
while (it1 != s1.end())
{cout << *it1 << ' ';++it1;
}
cout << endl;

可以记住这个使用方式,使用typeid可以看变量的类型:

	cout << typeid(it1).name() << endl;

所以说不能认为it1是指针,只是用法像指针。

同样迭代器也可以修改内容:

迭代器是通用的,链表list、二叉树都有迭代器。可以观察list 的文档。

范围for遍历

//遍历方式3
for (auto s : s1)
{cout << s << ' ';
}

        元素赋值给s,自动往后,自动赋值。

        所有的容器都可以使用范围for,本质上它使用的就是迭代器

注意是赋值拷贝给s所以,不会改变s1的内容:

要改变则需要使用引用auto& s:

string也实现const迭代器:

即对象是一个const对象,使用非const迭代器可以修改,会报错,所以要使用const迭代器。

iterator是可读可写,const_iterator是只读。不是const  iterator,是要保护迭代器指向的内容,而不是保护迭代器本身

实际上是这里使用auto就不用考虑了,auto根据右边进行推导。

7.反向迭代器iterator

        反向迭代器就是从后往前迭代。

它的++是倒着走的。

string s1("hello string!");
auto it1 = s1.begin();
while (it1 != s1.end())
{cout << *it1 << ' ';*it1++;
}
cout << endl;string::reverse_iterator it2 = s1.rbegin();
while (it2 != s1.rend())
{cout << *it2 << ' ';*it2++;
}
cout << endl;

由于string有[]重载,所以比较少用迭代器,但是迭代器是通用的。

这四个实际上可以当作const_iterator迭代器使用。

8.插入和修改

8.1 push_back尾插

        push_back()是插入一个字符:

string s;
s.push_back('a');
cout << s << endl;

8.2 append追加

        append可以尾插一串数据.所有的len是跨度的意思。

string s1("hello");
string s2(" string");
//追加string对象
s1.append(s2);
cout << s1 << endl;
//追加string对象的子串
s1 = "hello";
//从子串的0位置,跨度为2
s1.append(s2, 0, 2);
cout << s1 << endl;
//追加常量字符串的前n个
s1 = "hello";
s1.append(" strnig",4);
cout << s1 << endl;
//追加n个字符c
s1 = "hello";
s1.append(2,'s');
cout << s1 << endl;
//迭代器
s1 = "hello";
s1.append(s2.begin(),s2.end());
cout << s1 << endl;

  结果 :

8.3  +=常用

8.4 pop_back和erase

        pop_back()只支持删除一个。

        erase则支持删除一段:

8.5 assign 分配

        assign实际上和赋值差不多,和append差不多的解释:

8.6 insert(重点)

string s1("hello");
string s2("string ");
//pos位置string对象
s1.insert(0,s2);
cout << s1 << endl;
//pos位置string对象的子串
s1 = "hello";
s1.insert(0, s2, 0,3);
cout << s1 << endl;
//pos位置常量字符串的前n个
s1 = "hello";
s1.insert(0,"strnig ", 4);
cout << s1 << endl;
//pos位置n个字符c
s1 = "hello";
s1.insert(0,2, 's');
cout << s1 << endl;
//迭代器插入n个字符c
s1 = "hello";
s1.insert(s1.begin(), 2, 's');
cout << s1 << endl;
//迭代器
s1 = "hello";
s1.insert(s1.begin(), s2.begin(), s2.end());
cout << s1 << endl;

        可以用insert头插数据,但是insert进行插入,后面的数据都要后移,使用尽量少用,效率较低

8.7 erase  删除

        删出和insert一样要慎用,因为效率低,需要移动数据

string s1("hello world!");
cout << s1 << endl;s1.erase(0, 5);
cout << s1 << endl;s1 = "hello world!";
s1.erase(s1.begin());
cout << s1 << endl;s1 = "hello world!";
s1.erase(s1.begin()+5, s1.end());
cout << s1 << endl;

        

insert和erase的len都可以传很长,因为有npos缺省,但是第一个位置参数不能超出实际范围。

8.8 replace 替换

        有了上面的基础,在观察replace的文档就不难了:

替换从字符位置pos开始并跨越len个字符(或字符串范围[i1,i2)之间的部分)的字符串部分为新内容:注意len是阔度,将长度为len的这一段替换成新的内容str。覆盖并插入。

string s1("hello world!");
string s2("and");
//从5位置开始,跨度为1,即将空格替换成and
s1.replace(5, 1, s2);
cout << s1 << endl;
//从5位置开始,跨度为1,即将空格替换成and
s1 = "hello world!";
s1.replace(s1.begin() + 5, s1.begin() + 6, s2);
cout << s1 << endl;
//将空格替换成%20
s1 = "hello world!";
s1.replace(5, 1, "%20");
cout << s1 << endl;

题目:
将字符s1中所有的空格替换成%20:

string s1("hello world! and hello string!");
for (size_t i = 0; i < s1.size();)
{if (s1[i] == ' '){//替换//跨度为1s1.replace(i, 1, "%20");i += strlen("%20");}elsei++;
}
cout << s1 << endl;

        其中也设计数据挪动效率较低。介绍一种更好的方法:

	string s1("hello world! and hello string!");string s2;//将数据放到另一个对象中for (auto s : s1){if (s != ' ')//不是空格插入s2 += s;elses2 += "%20";//是空格就插入%20;}cout << s2 << endl;

牺牲空间换时间。

8.9 swap交换

        交换涉及深拷贝,后面会讲解,这里要会用。

交换成str。

	string s1("hello");string s2("string!");cout << s1 << endl;cout << s2 << endl;s1.swap(s2);cout << s1 << endl;cout << s2 << endl;

9.其他容器(resize,reserve····)

        9.1 capacity、max_size

	string s1("hello world!");cout << s1.size() << endl;cout << s1.capacity() << endl;cout << s1.max_size() << endl;

        在vs2022(32位下)实际容量大小size大小不一样的.max_size就是string在32位下的最大。实际上也开不了max_size(),因为开一个字符串就到了2G左右,这么大的连续空间时开不出来的。

 观察下面代码,vs下是如何扩容的:

string s1;
size_t sz = s1.capacity();
cout << "capacity changed: " << sz << endl;
cout << "making s1 grow:\n";
for (int i = 0; i < 100; ++i)
{s1.push_back('c');if (sz != s1.capacity()){sz = s1.capacity();cout << "capacity changed:" << sz << endl;}
}

       结果:

        

        结果表明在vs下,实例化对象先开辟了16个空间,然后进行1.5倍扩容。

实际上当字符串小于16字节时,字符串存储在对象里面的buf数组:

当字符过长时则会在堆空间:

capacity指的是有效数据的空间,所以实际上都会多出一个给'\0'.

linux下的结果:

不同的平台扩容是不一样的。但是linux和vs下size和capacity都是指有效空间,实际空间多一个。

9.2 reserve、resize

       1 reserve

        resize意思是是改变size,实际上也会改变capacity。

        reserve则是影响capacity,不会影响size。

	string s1;cout << s1.capacity() << endl;s1.reserve(100);cout << s1.capacity() << endl;

说是给100,实际上vs会给多。

而linux则很老实是多少给多少:

接下来看看vs下reserve是否会缩容:

string s1;
cout << s1.capacity() << endl;
s1.reserve(100);
cout << s1.capacity() << endl;
s1.reserve(20);
cout << s1.capacity() << endl;

        

        默认不会缩容。

而如果当前有效数据个数小于16,并且缩容到大于有效个数小于16则capacity会变成15.将多余的空间释放掉.

如果有效数据个数大于等于16,则不会发生缩容。

而Linux很听话:

可以知道reserve在未知空间进行扩容可能不确定。

所以reserve使用场景是我们确定要靠多大,提前将空间开辟好:

后续就不需要扩容了。

 如果我们reserve超过有效数据的个数,然后访问我们所reserve的超过有效字节的空间是被允许的吗?

	string s1("hello string!");cout << s1.size() << endl;cout << s1.capacity() << endl;s1.reserve(100);cout << s1.capacity() << endl;cout << s1[50] << endl;

结果是不被允许的,因为operator[]重载时只允许访问有效空间:

所以缩容一般不用reserve,要使用reserve最好提前知道要开多少空间。

 2 resize

        如果n小于当前字符串长度,则当前值将缩短为其前n个字符,删除第n个字符之后的字符。

        如果n大于当前字符串长度,则当前内容将被扩展,插入到末尾的字符数将达到n。如果指定了c,则新元素将初始化为c的副本,否则它们将是值初始化的字符(空字符\0)。

vs下缩容:

	string s1("hello world! and hello vs! and hello string!");cout << s1 << endl;cout << s1.size() << endl;cout << s1.capacity() << endl;s1.resize(12);cout << s1 << endl;cout << s1.size() << endl;cout << s1.capacity() << endl;

扩容:已有的不会动。

	string s1("hello world!");cout << s1 << endl;cout << s1.size() << endl;cout << s1.capacity() << endl;s1.resize(20,'!');cout << s1 << endl;cout << s1.size() << endl;cout << s1.capacity() << endl;

注意resize缩容c++并未规定,不同平台缩容结果可能不同。所以缩容我们一般也不用resize。

3 shrink_to_fit

缩容一般用这个。

注意:空间追加和释放都不能分段追加和释放

10. 字符串操作

        10.1 c_str

        为了兼容c语言,对c语言的字符串操作进行使用,c++提供了c_str来获取字符串内容。(不可修改)

返回一个指针,指向一个包含以空字符结尾的字符序列的数组(即一个 C 字符串),该字符序列表示字符串对象的当前值。

        例如在读取文件时:

#define  _CRT_SECURE_NO_WARNINGS 1
#include <iostream>
#include <string>
using namespace std;void test_string()
{string file("test.cpp");FILE* fout = fopen(file.c_str(), "r");char ch = fgetc(fout);while (ch != EOF){cout << ch;ch = fgetc(fout);}
}int main()
{test_string();
}

        10.2 find

        find是查找,从前往后找

与之相对的还有rfind是从后往前找

用法观察函数的参数就可以知道。

rfind的应用:

        获取file的后缀:

	string file("test.cpp");size_t pos = file.rfind('.');//string suffix(file.substr(pos))string suffix(file.substr(pos, file.size() - pos));cout << suffix << endl;

        substr是获取子串:

       取出域名:

    string url("https://blog.csdn.net/Jk_Mr"); //找冒号:size_t pos1 = url.find(':');string url1 = url.substr(0, pos1 - 0);cout << url1 << endl;//找第三个斜杠size_t pos2 = url.find('/',pos1 + 3);string url2 = url.substr(pos1 + 3, pos2 - (pos1 + 3));cout << url2 << endl;//去到最后string url3 = url.substr(pos2 + 1);cout << url3 << endl;

        10.3 find_first_of

        字面上的意思是找到第一个字符,实际上它是找any匹配的字符;从前往后找。

// string::find_first_of
#include <iostream>       // std::cout
#include <string>         // std::string
#include <cstddef>        // std::size_tint main ()
{std::string str ("Please, replace the vowels in this sentence by asterisks.");std::size_t found = str.find_first_of("aeiou");while (found!=std::string::npos){str[found]='*';found=str.find_first_of("aeiou",found+1);}std::cout << str << '\n';return 0;
}

从代码结果可以看出,它是找到与参数任意一个匹配的位置。返回值是第一个匹配值的位置,即第一个e。这和c语言中的strtok函数类似。

        10.4 find_last_of

        从后往前找任何一个相匹配的字符,返回第一个相匹配的字符。

可用于匹配路径Linux和windows

void SplitFilename(const std::string& str)
{std::cout << "Splitting: " << str << '\n';std::size_t found = str.find_last_of("/\\");std::cout << " path: " << str.substr(0, found) << '\n';std::cout << " file: " << str.substr(found + 1) << '\n';
}int main()
{std::string str1("/usr/bin/man");std::string str2("c:\\windows\\winhelp.exe");cout << str2 << endl;SplitFilename(str1);SplitFilename(str2);return 0;
}

        10.5 find_not_first_of和find_not_last_of

               跟find_first_of和find_last_of相反,是和参数都不匹配的

        10.6 compare

11.非成员函数

     11.1 operator+

        为什么要重载成全局函数?

	string s1("hello");string s2("world!");//这样可以重载成成员函数string s3 = s1 + s2;//但是为了支持第一个参数不是隐藏指针this//所以重载成成员函数string s4 = "hello " + s2;

为了支持第一个参数不是隐藏指针this,所以重载成成员函数。

        11.2 relational operator

        同样要支持第一个参数不是this指针,重载成全局函数

       是按照字典序比较的,从第一个开始,如果相同比较下一个的ASCII码值。不相同就返回。

注意:和流提取、流插入的优先级,需要加括号

实际上设计的有点冗余,因为const char* 可以隐式类型转换到const string&。

12.字符串和其他数组类型转换

        c语言中有atoi和itoa用于字符串转换整形整形转换字符串。C++也有类似的接口:

        to_string:

三、字符串题目:

        1.仅仅反转字母

917. 仅仅反转字母 - 力扣(LeetCode)icon-default.png?t=N7T8https://leetcode.cn/problems/reverse-only-letters/description/

给你一个字符串 s ,根据下述规则反转字符串:

  • 所有非英文字母保留在原有位置。
  • 所有英文字母(小写或大写)位置反转。

返回反转后的 s 。

忽略非字母即可,判断是否为字母,是字母就进行反转:

class Solution {
public:bool IsAlpha(char s){if(s >='a' && s<= 'z')return true;if(s >= 'A' && s <= 'Z')return true;return false;}string reverseOnlyLetters(string s) {if(s.empty())return s;int begin = 0 , end = s.size()-1;while(begin < end){//左边边找字符while(begin < end && !IsAlpha(s[begin])){++begin;}//左边边找到了,右边找字符while(begin < end && !IsAlpha(s[end])){--end;}//右边找到了,交换swap(s[begin],s[end]);++begin;--end;}return s;}
};

2.字符串中的第一个唯一字符

387. 字符串中的第一个唯一字符 - 力扣(LeetCode)icon-default.png?t=N7T8https://leetcode.cn/problems/first-unique-character-in-a-string/description/使用计数排序思想:

class Solution {
public:int firstUniqChar(string s) {//范围小可以使用计数排序,统计每个字母出现的次数//然后返回出现一次的字母int count[26] = {0};for(auto ch : s){count[ch -'a']++;//和a的相对位置,a就是下标0位置}//s中的每一个字符已经映射到了数组count上,出现一次的被记录,或者没有会出现一次的//遍历数组查看每个字符的出现次数for(size_t i = 0 ; i < s.size(); i++){if(count[s[i] - 'a'] == 1)//每个字符相对'a'的差值就是数组count映射的位置{return i;}}return -1;}
};

正反查找:

class Solution {
public:int firstUniqChar(string s) {//从首字符开始查找,正查找和逆查找的位置相同则说明该字符只出现了一次//若不相同则说明有俩个或俩个以上的字符。for(int i=0; i<s.size(); ++i){int index = s.find(s[i]);int reverse_index = s.rfind(s[i]);  if(index == reverse_index)return i;}return -1;}
};

3.验证回文串

125. 验证回文串 - 力扣(LeetCode)icon-default.png?t=N7T8https://leetcode.cn/problems/valid-palindrome/逆置比较:

class Solution {
public://判断并转换大写为小写bool IsAlphaOrDigit(char& s){if(s >= 'a' && s<= 'z')return true;if(s >= 'A' && s <= 'Z'){s += 32;return true;}if(s >= '0' && s<= '9')return true;return false;}bool isPalindrome(string s) {string str;//将s逆置放入str,去掉非字母数字字符,大写变小写int end = s.size() - 1;for(int i = end; i >= 0; i--){if(IsAlphaOrDigit(s[i])){str += s[i];//尾插}}string com(str);//逆置比较reverse(com.begin(),com.end());return com == str;}
};//时间:O(N) 空间:O(N)

可以使用tolower来进行转换

双指针比较:

class Solution {
public:bool IsAlphaOrDigit(char& s){if(s >= 'a' && s<= 'z')return true;if(s >= 'A' && s <= 'Z'){s += 32;//转换大小写return true;}if(s >= '0' && s<= '9')return true;return false;}bool isPalindrome(string s) {//双指针法进行判断int begin = 0, end = s.size() - 1;while(begin < end){//左侧找到字母或数字//同时转换大小写while(begin < end && !IsAlphaOrDigit(s[begin]))begin++;//右侧找到字母或数字//同时转换大小写while(begin < end && !IsAlphaOrDigit(s[end]))end--;//将左右比较,不相同falseif(s[begin] != s[end])return false;else{begin++;end--;}}return true;}
};
//时间复杂度: O(N) 空间复杂度:O(1)

4.字符串相加

415. 字符串相加 - 力扣(LeetCode)icon-default.png?t=N7T8https://leetcode.cn/problems/add-strings/

/*
*思路:
* 1. 先逆置数据
* 2. 数据对应为进行相加
* 3. 最后若有进位,还需尾插数据,最后逆置恢复数据
*/
class Solution
{
public:string addStrings(string num1, string num2){//逆置解过进行操作reverse(num1.begin(), num1.end());reverse(num2.begin(), num2.end());//num1和num2的下标int index1 = 0, index2 = 0;int carry = 0;//进位标致int res = 0;//转换数字保存量string ret;//循环进行相加,先考虑俩个数字的长度相同while (index1 < num1.size() || index2 < num2.size() || carry != 0)//若还有进位需要进位{//当进位不为0时index1和index2会超过字符串范围//所以需要判断,然后向前补carry的值res = 0 + 0 + 1;//有了这一步那么也保证了长短不一:因为超过长度会补0int x = index1 < num1.size() ? num1[index1] - '0' : 0;int y = index2 < num2.size() ? num2[index2] - '0' : 0;//单个相加res = x + y + carry;if (res >= 10){carry = 1;ret += res % 10 + '0';}else{carry = 0;ret += res + '0';}index1++;index2++;}//将结果在反过来reverse(ret.begin(), ret.end());return ret;}
};
class Solution
{
public:string addStrings(string num1, string num2){//num1和num2的下标int index1 = num1.size()  - 1, index2 = num2.size()  - 1;int carry = 0;//进位标致int res = 0;//转换数字保存量string ret;//循环进行相加,先考虑俩个数字的长度相同while (index1 >= 0 || index2  >= 0 )//若还有进位需要进位{//当进位不为0时index1和index2会超过字符串范围//有了这一步那么也保证了长短不一:因为超过长度会补0int x = index1 >= 0 ? num1[index1--] - '0' : 0;int y = index2 >= 0 ? num2[index2--] - '0' : 0;//单个相加res = x + y + carry;//处理进位carry = res /10;res = res % 10;//头插//每次都要挪动数据,所以建议尾插然后逆置//ret.insert(0,1,res + '0');ret += res + '0';}//进位标志不为0if(carry != 0){ret += res + '0';}reverse(ret.begin(),ret.end());return ret;}
};

5.字符串最后一个单词的长度

        字符串最后一个单词的长度icon-default.png?t=N7T8https://www.nowcoder.com/practice/8c949ea5f36f422594b306a2300315da?tpId=37&tqId=21224&ru=/exam/oj        cin输入:

#include <iostream>
#include <string>
using namespace std;int main() {string s;cin >> s;int pos = s.rfind(" "), last = s.size() - 1;cout<< last - pos;return 0;
}

为通过,vs上也是5,这说明cin出了问题。通过vs调试可以看出

cin遇到空格和换行就会结束提取,scanf(除了%c遇到空格不停)也是一样。

c++有一个专门的函数解决:

6.getline

getline默认是遇到换行才停止。

#include <iostream>
#include <string>
using namespace std;int main() {string s;getline(cin,s);int pos = s.rfind(" "), last = s.size() - 1;cout<< last - pos;return 0;
}

所以c++中要获取字符串包含空格时,使用getline。

在使用while(cin >> s)时,结束可以用ctrl + C或者 ctrl + Z + 换行。

四、字符排序sort

        如果要对一个字符串进行排序该怎么排,由于string中没有专门的排序,所以我们给出下面的排序头文件,在STL中有一个头文件包含算法:

icon-default.png?t=N7T8https://legacy.cplusplus.com/reference/algorithm/

        在这个头文件中:

sort是用于排序的,它是一个模板:

它的范围规定为必须是左闭右开的:

last为无效的位置。迭代器就可以适配这个范围。

	string s1("hello string!");cout << s1 << endl;//s1 按照字典序排序sort(s1.begin(), s1.end());cout << s1 << endl;

部分排序:


如果你又所收获,可以留下你的点赞和关注谢谢你的观看!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/696739.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用git系统来更新FreeBSD ports源码

FreeBSD跟其它系统相比一大特色就是ports系统。 The Ports Collection is a set of Makefiles, patches, and description files. Each set of these files is used to compile and install an individual application on FreeBSD, and is called a port. By default, the Po…

使用Docker+Jar方式部署微服务工程(前后端分离)看着一篇就够了

本篇教程的使用到的技术有springboot、springcloud、Nacos、Docker、Nginx部署前后端分离访问的微服务。 部署一下Nacos 首先我们需要在服务器中&#xff08;或者本地部署启动一下Nacos&#xff09;&#xff0c;这里我采用服务器的方式进行部署&#xff0c;这里有一点不一样的…

# 从浅入深 学习 SpringCloud 微服务架构(十七)--Spring Cloud config(2)

从浅入深 学习 SpringCloud 微服务架构&#xff08;十七&#xff09;–Spring Cloud config&#xff08;2&#xff09; 一、springcloudConfig 入门案例&#xff1a;搭建 config 服务端 1、登录 码云&#xff1a;https://gitee.com/ 1&#xff09;点击右上角 【】 再点击【新…

算法学习笔记(一)-快速幂

#问题的引入-对于幂次方的求解我们怎么可以最大限度的降低时间复杂度呢 #对于一个基本的幂次运算&#xff0c;c代码如下示例 long long int myPower(int base,int power) {long long int result 1 ;for (int i 1 ; i < power ; i){result * base ;}return result ; } #…

网络实验新境界,PNETLab模拟器部署指南

在网络工程领域&#xff0c;拥有一个可靠的网络实验平台至关重要。PNETLab模拟器是一款功能强大的网络仿真工具&#xff0c;它支持包括华为、华三、锐捷、思科在内的多种设备&#xff0c;并且以开源免费的形式提供&#xff0c;这使得它在业界备受青睐。 软件介绍 PNETLab&am…

K8s:二进制安装k8s(单台master)

一、安装k8s 1、拓扑图 2、系统初始化配置 #所有节点执行 systemctl stop firewalld systemctl disable firewalld iptables -F && iptables -t nat -F && iptables -t mangle -F && iptables -X #永久关闭firewalld并清空iptables所有表规则 setenf…

标准输入输出流(中北大学-程序设计基础(2))

目录 题目 源码 结果示例 题目 输入三角形的三边a,b,c&#xff0c;计算三角形的面积。形成三角形的条件是ab>c,bc>a,ac>b&#xff0c;编写程序&#xff0c;输入a,b,c&#xff0c;检查a,b,c是否满足以上条件&#xff0c;如不满足&#xff0c;由cerr输出有关出错信息…

Spring框架核心:揭秘Java厨房的智能烹饪艺术

前情回顾&#xff1a;Spring框架深度解析&#xff1a;打造你的Java应用梦工厂 六. 实现控制反转 6.1 描述如何在Spring中实现IoC 在Spring Town的厨房里&#xff0c;实现控制反转就像是将食材的采购和准备过程外包给了一个智能系统。这个系统知道每种食材的特性&#xff0c;也…

Java String转JSONObject时保持字段顺序不变

Java String转JSONObject时保持字段顺序不变 问题背景解决方案 问题背景 在业务接口开发过程中&#xff0c;有一个新增接口&#xff0c;需要支持批量新增数据&#xff0c;这时入参就需要用到 json 格式数据&#xff0c;且包含 list 集合&#xff0c;比如这样的数据格式&#x…

Linux-磁盘管理类实训

一、Linux分区和磁盘操作命令 &#xff08;1&#xff09;将系统内所有的分区&#xff08;文件系统&#xff09;列出来&#xff09; &#xff08;2&#xff09;将系统中所有特殊文件格式及名称都列出来 &#xff08;3&#xff09;将/bin下面的可以用的磁盘容量以易读的容量格式…

做抖店如何提高与达人合作的几率?有效筛选+有效推品

我是王路飞。 总是有很多新手商家&#xff0c;找我吐槽&#xff0c;抖音上的达人特别不好找&#xff0c;好不容易加上了&#xff0c;要么是发消息不回复&#xff0c;要么是寄样后就没下文了。 虽然一直都说找达人带货玩法比较简单&#xff0c;但也离不开电商的基本逻辑&#…

【数据可视化01】matplotlib实例介绍1

目录 一、引言二、实例介绍1.柱状图1)简单柱状图2)堆叠柱状图 2.线条形式3.折线图&#xff08;多子图&#xff09;4.散点图5.水平和垂直线条6.饼状图1&#xff09;饼状图2&#xff09;“条形饼”图 一、引言 matplotlib是一个用于绘制数据可视化的Python库。它可以创建各种静态…