从C语言到C++_11(string类的常用函数)力扣58和415(下)

简介: 从C语言到C++_11(string类的常用函数)力扣58和415

从C语言到C++_11(string类的常用函数)力扣58和415(中):https://developer.aliyun.com/article/1513667

8. string相关OJ

现在关于OJ题建议刷《剑指offer》和力扣上string的题,

用牛客一道题演示下上面提到的geline函数:看看IO型和接口型C++的区别

字符串最后一个单词的长度_牛客题霸_牛客网 (nowcoder.com)

简单  通过率:35.00%  时间限制:1秒  空间限制:32M


描述


计算字符串最后一个单词的长度,单词以空格隔开,字符串长度小于5000。(注:字符串末尾不以空格为结尾)


输入描述:


输入一行,代表要计算的字符串,非空,长度小于5000。


输出描述:


输出一个整数,表示输入字符串最后一个单词的长度。


示例1


输入:


hello nowcoder


输出:


8


说明:


最后一个单词为nowcoder,长度为8  

#include <iostream>
using namespace std;
 
int main() {
    int a, b;
    while (cin >> a >> b) { // 注意 while 处理多个 case
        cout << a + b << endl;
    }
}
// 64 位输出请用 printf("%lld")

代码解析:

这题直接用 cin 就不行,因为有空格,所以我们用 getline 输入一行字符,查文档:

虽然不包string头文件有些编译器也能过,但还是包上好:

#include <iostream>
#include <string>
using namespace std;
 
int main()
{
    string str;
    getline(cin,str);
    size_t pos = str.rfind(' ');
    if(pos != str.npos)
    {
        cout << str.size() - pos - 1 << endl;// -1是减去空格
    }
    else //找不到直接输出size
    {
        cout << str.size();
    }
    return 0;
}

看看力扣是这么写的:(力扣就检查用例较多,且此题多加了最后有空格的情况)

58. 最后一个单词的长度 - 力扣(LeetCode)

难度简单给你一个字符串 s,由若干单词组成,单词前后用一些空格字符隔开。返回字符串中 最后一个 单词的长度。

单词 是指仅由字母组成、不包含任何空格字符的最大子字符串。

示例 1:

输入:s = "Hello World"

输出:5

解释:最后一个单词是“World”,长度为5。

示例 2:

输入:s = "   fly me   to   the moon  "

输出:4

解释:最后一个单词是“moon”,长度为4。

示例 3:

输入:s = "luffy is still joyboy"

输出:6

解释:最后一个单词是长度为6的“joyboy”。

提示:

  • 1 <= s.length <= 10^4
  • s 仅有英文字母和空格 ' ' 组成
  • s 中至少存在一个单词

代码解析:

只是比牛客的那题多了一个条件,所以直接放代码:

class Solution {
public:
  int lengthOfLastWord(string s) {
        while(s[s.size()-1] == ' ')
        {
            s.erase(s.size()-1,1);
        }
    size_t pos = s.rfind(' ');
    if (pos != s.npos)
    {
      return s.size() - pos - 1;// -1是减去空格
    }
    else //找不到直接输出size
    {
      return s.size();
    }
  }
};

力扣415. 字符串相加

415. 字符串相加 - 力扣(LeetCode)

给定两个字符串形式的非负整数 num1num2 ,计算它们的和并同样以字符串形式返回。

你不能使用任何內建的用于处理大整数的库(比如 BigInteger), 也不能直接将输入的字符串转换为整数形式。

示例 1:

输入:num1 = "11", num2 = "123"

输出:"134"

示例 2:

输入:num1 = "456", num2 = "77"

输出:"533"

示例 3:

输入:num1 = "0", num2 = "0"

输出:"0"

提示:

  • 1 <= num1.length, num2.length <= 10^4
  • num1num2 都只包含数字 0-9
  • num1num2 都不包含任何前导零
class Solution {
public:
    string addStrings(string num1, string num2) {
 
    }
};

代码解析:

这题首先想到从字符串末尾开始加,加后考虑进位后头插到一个新的字符串。需要用到头插接口:

查文档发现string没有头插,只有任意位置插入删除,凑合用用: 法一:头插

class Solution {
public:
  string addStrings(string num1, string num2) {
    int end1 = num1.size() - 1, end2 = num2.size() - 1;
    int next = 0;// 表示进位,这里只会是0或1
    string ret;// 把加后的字符弄到这
    while (end1 >= 0 || end2 >= 0)// 只要有一个没加完就继续,加完的看做数字0
    {
      int val1 = end1 >= 0 ? num1[end1] - '0' : 0;// -'0'转化为数字
      int val2 = end2 >= 0 ? num2[end2] - '0' : 0;
      int add = val1 + val2 + next;
      next = add > 9 ? 1 : 0;// 处理进位
      //写到这想到头插:查文档:
      ret.insert(0, 1, (add % 10) + '0');// 在ret第0个位置插入1个字符
      --end1;
      --end2;
    }
    if (next)// 提交后看测试用例有进位没处理到的处理一下
    {
      ret.insert(0, 1, '1');
    }
    return ret;
  }
};

提交通过后发现效率很低(虽然力扣效率不太准确,但是你网比较好且提交几次还是很慢就应该真是代码效率低了)哪里会低?string是连续的空间,所以任意位置插入需要挪动数据,效率就低了(O(N^2))。那能不能用效率高的尾插?而且 += 很方便。答案是可以的,头插和尾插最后的差别就是字符串反过来了,那我们以前写过逆置字符串就能派上用场,更方便的是C++已经提供了逆置字符串的函数,(string没提供,头文件algorithm提供了)在这个旧版网站搜一下:cplusplus.com - The C++ Resources Network

或者在msdn应用搜一下:(能搜到就行)

和我们以前写的差不多,左闭右开的区间,直接用,(法二,尾插+逆置):

class Solution {
public:
    string addStrings(string num1, string num2) {
        int end1 = num1.size() - 1, end2 = num2.size() - 1;
        int next = 0;// 表示进位,这里只会是0或1
        string ret;
        while(end1 >= 0 || end2 >= 0)// 只要有一个没加完就继续,加完的看做数字0
        {
            int val1 = end1 >= 0 ? num1[end1] - '0' : 0;// -'0'转化为数字
            int val2 = end2 >= 0 ? num2[end2] - '0' : 0;
            int add = val1 + val2 + next;
            next = add > 9 ? 1 : 0;// 处理进位
 
            //ret.insert(0,1,(add % 10) + '0');// 在ret第0个位置插入1个字符
            ret += (add % 10) + '0';// 尾插
            --end1;
            --end2;
        }
        if(next)// 提交后看测试用例有进位没处理到的处理一下
        {
            //ret.insert(0,1,'1');// 头插
            ret += '1';// 尾插
        }
 
        reverse(ret.begin(),ret.end());
        return ret;
    }
};

本章完。

学了这么多接口函数,又刷了两道OJ,后面也差不多就会刷了,

后面更一篇string相关的OJ题,再后一篇就模拟实现string,

后面的STL学习基本都是按照这个顺序走,只是内容少了点,因为接口函数都是类似的。

目录
相关文章
|
2月前
|
编解码 Java 开发者
Java String类的关键方法总结
以上总结了Java `String` 类最常见和重要功能性方法。每种操作都对应着日常编程任务,并且理解每种操作如何影响及处理 `Strings` 对于任何使用 Java 的开发者来说都至关重要。
308 5
|
3月前
|
存储 C语言
`scanf`是C语言中用于按格式读取标准输入的函数
`scanf`是C语言中用于按格式读取标准输入的函数,通过格式字符串解析输入并存入指定变量。需注意输入格式严格匹配,并建议检查返回值以确保读取成功,提升程序健壮性。
1009 0
|
5月前
|
安全 C语言
C语言中的字符、字符串及内存操作函数详细讲解
通过这些函数的正确使用,可以有效管理字符串和内存操作,它们是C语言编程中不可或缺的工具。
325 15
|
6月前
|
对象存储 C++ 容器
c++的string一键介绍
这篇文章旨在帮助读者回忆如何使用string,并提醒注意事项。它不是一篇详细的功能介绍,而是一篇润色文章。先展示重载函数,如果该函数一笔不可带过,就先展示英文原档(附带翻译),最后展示代码实现与举例可以直接去看英文文档,也可以看本篇文章,但是更建议去看英文原档。那么废话少说直接开始进行挨个介绍。
135 3
|
6月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
174 0
|
6月前
|
存储 编译器 C语言
关于string的‘\0‘与string,vector构造特点,反迭代器与迭代器类等的讨论
你真的了解string的'\0'么?你知道创建一个string a("abcddddddddddddddddddddddddd", 16);这样的string对象要创建多少个对象么?你知道string与vector进行扩容时进行了怎么的操作么?你知道怎么求Vector 最大 最小值 索引 位置么?
161 0
|
6月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
266 0
|
8月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
308 12
|
9月前
|
缓存 安全 Java
《从头开始学java,一天一个知识点》之:字符串处理:String类的核心API
🌱 **《字符串处理:String类的核心API》一分钟速通!** 本文快速介绍Java中String类的3个高频API:`substring`、`indexOf`和`split`,并通过代码示例展示其用法。重点提示:`substring`的结束索引不包含该位置,`split`支持正则表达式。进一步探讨了String不可变性的高效设计原理及企业级编码规范,如避免使用`new String()`、拼接时使用`StringBuilder`等。最后通过互动解密游戏帮助读者巩固知识。 (上一篇:《多维数组与常见操作》 | 下一篇预告:《输入与输出:Scanner与System类》)
258 11
|
9月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。