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] == ' ')
	{
		//替换//跨度为1
		s1.replace(i, 1, "%20");
		i += strlen("%20");
	}
	else
		i++;
}
cout << s1 << endl;

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

	string s1("hello world! and hello string!");
	string s2;//将数据放到另一个对象中
	for (auto s : s1)
	{
		if (s != ' ')//不是空格插入
			s2 += s;
		else
			s2 += "%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_t

int 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--;
            //将左右比较,不相同false
            if(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;
            //有了这一步那么也保证了长短不一:因为超过长度会补0
			int 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会超过字符串范围
            //有了这一步那么也保证了长短不一:因为超过长度会补0
			int 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';
		}
        //进位标志不为0
        if(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;

部分排序:


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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

如何开通微软必应bing国内竞价广告账户?

微软必应Bing作为全球第二大搜索引擎&#xff0c;凭借其特有的用户群和市场定位&#xff0c;为中国广告主开辟了一片潜力无限的蓝海市场。云衔科技通过专业实力&#xff0c;为广告主提供全方位支持&#xff0c;从开户到代运营&#xff0c;助力企业扬帆起航。 一、微软必应bing…

百度云防护502 Bad Gateway原因总结和处理方法

最近&#xff0c;随着原百度云加速用户新接入百度云防护后&#xff0c;很多站长反馈网站打不开&#xff0c;出现了502 Bad Gateway的情况。 为此&#xff0c;百度云这里给大家总结下&#xff0c;出现502的大概几个原因&#xff1a; 1.服务器防火墙拦截了百度云防护的IP节点请求…

(docker)进入容器后如何使用本机gpu

首次创建容器&#xff0c;不能直接使用本机gpu 在系统终端进行如下配置&#xff1a; 1.安装NVIDIA Container Toolkit 进入Nvidia官网Installing the NVIDIA Container Toolkit — NVIDIA Container Toolkit 1.15.0 documentation&#xff0c;安装NVIDIA Container Toolkit …

红黑树(RBTree)认识总结

一、认识红黑树 1.1 什么是红黑树&#xff1f; 红黑树是一种二叉搜索树&#xff0c;与普通搜索树不同的是&#xff0c;在每个节点上增加一个“颜色”变量 —— RED / BLACK 。 通过对各个节点颜色的限制&#xff0c;确保从 根 到 NIL &#xff0c;没有一条路径会比其他路径长出…

R语言软件安装及配置

1、下载 网址&#xff1a;www.r-project.org 1.1 下载R 选择download R 选择清华源进行下载 根据自己系统情况下载&#xff0c;我选择windows系统。 先选择base。 选择最新的版本下载。 1.2 下载RTools 下载好后&#xff0c;返回&#xff0c;选择RTools进入后&#xff0c;选…

替换spring-boot中的组件版本

spring-boot是一个用于简化开发的框架&#xff0c;引入spring-boot后会自动包含spring框架&#xff0c;通过引入xxx-start来完成指定组件的功能。比如&#xff1a; spring-boot-starter-web(嵌入 Tomcat 和 web 开发需要的 servlet 和 jsp 支持)spring-boot-starter-data-jpa(…

python EEL应用程序的启动过程

EEL 启动流程 初始化 EEL (eel.init()): 设定静态文件目录&#xff0c;通常是包含 HTML、CSS、JavaScript 等文件的目录。扫描指定目录下的 JavaScript 文件&#xff0c;寻找通过 eel.expose() 暴露的函数。 启动 Web 服务器 (eel.start()): 基于 Bottle 框架启动一个轻量级的 …

2024年3月 青少年等级考试机器人理论真题四级

202403 青少年等级考试机器人理论真题四级 第 1 题 Arduino UNO/Nano主控板&#xff0c;通过按键开关切换高低电平&#xff0c;电路搭设如下&#xff0c;该电路属于&#xff1f;&#xff08; &#xff09; A&#xff1a;外部上拉电阻电路 B&#xff1a;外部下拉电阻电路 C&a…

防火墙远程桌面端口号修改,通过防火墙修改远程桌面的端口号详细操作步骤

使用防火墙修改远程桌面的端口号是一项涉及系统安全和网络配置的重要任务。 以下是详细的操作步骤&#xff0c;旨在确保您能够安全、有效地完成此操作&#xff1a; 一、准备阶段 1. 了解默认端口号&#xff1a;远程桌面端口号通常是3389&#xff0c;这是一个用于远程访问和控…

五款商用加密软件推荐 | 商用加密软件排行榜

没有网络安全就没有国家安全。信息安全是国家经济社会稳定运行&#xff0c;广大人民群众利益的保障。 对于公司来讲&#xff0c;数据安全同样是企业可持续发展的重要保障&#xff0c;防止内部核心数据、知识产权的泄露是企业数据安全的重要工作。下面是五款企业常用的加密软件…

如何查看centos7是否安装nginx

要查看 CentOS 7 系统上是否安装了 Nginx&#xff0c;您可以使用多种方法来检查。以下是一些常见的方法&#xff1a; 通过 RPM 包管理器查询 在 CentOS 系统上&#xff0c;可以使用 RPM 包管理器来查询已安装的软件包。要查看是否安装了 Nginx&#xff0c;您可以在终端中运行以…

Spring框架概述

目录 1. Spring框架的起源 2. Spring框架的构成 3. Spring的发展历程 4. Spring的开发环境 4.1. Maven安装与配置 &#xff08;1&#xff09;Maven的下载与安装 &#xff08;2&#xff09;配置Maven的环境变量 &#xff08;3&#xff09;本地仓库的配置 &#xff08;4…

使用Baidu Comate五分钟 , 工作时间摸鱼8小时

Baidu Comate&#xff1a;引领智能编码新时代 文章目录 Baidu Comate&#xff1a;引领智能编码新时代一、明日工具&#xff0c;今日领先——百度Comate智能编码助手二、万变不离其宗——适配场景需求三、功能研究3.1 指挥如指掌——指令功能3.2 助手增援——插件功能使用3.3 实…

Raft论文阅读笔记+翻译:In Search of Understandable Consensus Algorithm

In Search of Understandable Consensus Algorithm 摘要 Raft是一种管理复制日志的共识算法。它产生与&#xff08;多&#xff09;Paxos等效的结果&#xff0c;并且与Paxos一样高效&#xff0c;但其结构与Paxos不同。这使得Raft比Paxos更易理解&#xff0c;也为构建实际系统提供…

VS2022 错误 LNK2001 无法解析的外部符号

错误 LNK2001 无法解析的外部符号 “private: static struct std::once_flag ThreadPool::flag_” (?flag_ThreadPool0Uonce_flagstdA) STL D:\VS2019\STL\源.obj 1 错误原因 &#xff1a;链接器无法解析 ThreadPool::flag_ 这个静态成员变量。这通常是因为静态成员变量在声明…

宁夏银川市起名专家的老师颜廷利:死神(死亡)并不可怕,可怕的是...

在中国优秀传统文化之中&#xff0c;汉语‘巳’字与‘四’同音&#xff0c;在阿拉伯数字里面&#xff0c;通常用‘4’来表示&#xff1b; 湖南长沙、四川成都、重庆、宁夏银川最靠谱最厉害的起名大师的老师颜廷利教授指出&#xff0c;作为汉语‘九’字&#xff0c;倘若是换一个…

《中国应急管理》是什么级别的期刊?是正规期刊吗?能评职称吗?

问题解答&#xff1a; 问&#xff1a;《中国应急管理》杂志是核心期刊吗&#xff1f; 答&#xff1a;不是核心期刊&#xff0c;是正规学术期刊 问&#xff1a;《中国应急管理》杂志是电子版期刊吗&#xff1f; 答&#xff1a;不是&#xff0c;是纸质期刊 问&#xff1a;《…

IMDB的电影评论数据pytorh使用lstm

使用lstm对IMDB的电影评论数据进行情感分析&#xff08;pytorch代码&#xff09; 接下来让我们看看如何使用pytorch实现一个基于长短时记忆网络的情感分析模型。在飞桨中&#xff0c;不同深度学习模型的训练过程基本一致&#xff0c;流程如下&#xff1a; 数据处理&#xff1…

d17(154-168)-勇敢开始Java,咖啡拯救人生

目录 方法递归 字符集 编码-解码 IO流 字节流 字节输入流 InputSream FileInputStream 字节输出流 OutputSream FileOutputSream 释放资源的方式 try-catch-finallly try-with-resource 字符流 字符输入流 Reader FileReader 文件字符输出流 Writer FileWriter …

大数据分析案例-基于随机森林算法构建银行贷款审批预测模型

&#x1f935;‍♂️ 个人主页&#xff1a;艾派森的个人主页 ✍&#x1f3fb;作者简介&#xff1a;Python学习者 &#x1f40b; 希望大家多多支持&#xff0c;我们一起进步&#xff01;&#x1f604; 如果文章对你有帮助的话&#xff0c; 欢迎评论 &#x1f4ac;点赞&#x1f4…