【卡码网C++基础课 11.句子缩写】

news2024/11/22 18:22:07

目录

  • 题目描述与分析
  • 一、字符大小的比较
  • 二、代码编写
  • 三、函数的使用
  • 四、形参和实参
  • 五、引用


题目描述与分析

题目描述:
输出一个词组中每个单词的首字母的大写组合。

输入描述:
输入的第一行是一个整数n,表示一共有n组测试数据。(输入只有一个n,没有多组n的输入)
接下来有n行,每组测试数据占一行,每行有一个词组,每个词组由一个或多个单词组成;每组的单词个数不超过10个,每个单词有一个或多个大写或小写字母组成;
单词长度不超过10,由一个或多个空格分隔这些单词。

输出描述:
请为每组测试数据输出规定的缩写,每组输出占一行。

输入示例:

1
ad dfa     fgs

输出示例:

ADF

一、字符大小的比较

字符串是由一个个字符组合而成的, 比如字符串"hello", 是由字符(char)类型’h’、‘e’、‘l’、‘l’、'0’组成的,我们可以通过索引来访问每一个字符。

在这里插入图片描述
字符的大小是根据它们在字符集中的ASCII码值来确定的。ASCII码是一种字符编码标准,将每个字符映射到一个唯一的整数值。比如小写字母’a’对应的ASCII码值是97, 小写字母’b’对应的ASCII码值是98,所以可以基于ASCII码值来比较字符的大小。
在这里插入图片描述
从上表可以看到,小写字母从 ‘a’ 到 ‘z’ 对应的ASCII码值是从 97 到 122,而大写字母从 ‘A’ 到 ‘Z’ 对应的ASCII码值是从 65 到 90, 大小写字母之间的差值为32,所以可以通过数学运算将大写字符加上32后转换为小写字符。

二、代码编写

按照题目要求,我们依旧先把代码的基础结构写出来。

#include<iostream>
#include<string>
using namespace std;
int main() {
  int n;
  cin >> n; // 接收n行测试数据
  getchar(); // 从输入中获取一个字符
}

注意这里使用了 getchar() 函数来吸收一个回车符,因为在输入 n 之后通常需要输入回车符才会输入下一行。

之后仍然可以使用while循环来处理n行数据,每一行数据使用getline(cin, n) 来进行接收。

string s; // 定义变量s, 接收每行输入的字符串
while(n--) {
   getline(cin, s); // 从标准输入读取一行字符串,存储在字符串变量 s中
   string result; //  定义变量result, 用来输出经过处理后的结果
}

在这里插入图片描述
题目要求是输出一个词组中每个单词的首字母的大写组合, 那每行的第一个字符的大写形式肯定会被拼接到result中, 如果第一个字符是小写字符,则将之转换成大写字符,再进行拼接。

那如何判定字符是不是小写字符呢,在之前我们已经提到过小写字母从 ‘a’ 到 ‘z’ 对应的ASCII码值是从 97 到 122,且是连续的,也就是说如果输入的字符大于等于a, 并且小于等于z, 则一定在a~z的范围之内,也就是一定是小写字母。

// 如果第一个字符是小写字符
if (s[0] >= 'a' && s[0]  <= 'z') {

}

大小写字符之间的差值是32, 将小写字符做减法运算,就可以得到对应的大写字符。

// 如果输入的字符是小写字符,则进行转换
if (s[0] >= 'a' && s[0]  <= 'z') {
  // ASCII码中小写字母和大写字母相差32
    s[0] -= 32;
}
result += s[0];// 将每行的第一个字符转换成大写后添加到result中

然后从 s 的第二个字符开始遍历到倒数第二个字符。在循环中,检查当前字符是否为空格,并且下一个字符不是空格(这是为了跳过单词之间的多个空格),如果满足条件,则说明遇到了一个新的单词,下一个字符是这个单词的首字母,然后将下一个字符经过处理后添加到 result 中。

for (int i = 1; i < s.size() - 1; i++) { // 单词之间可能有多个空格
  // 如果当前字符是空格,并且下一个字符不是空格
      if (s[i] == ' ' && s[i + 1] != ' ') {
      // 判定下一个字符是否是小写字母,如果是则转换成大写
      if (s[i + 1] >= 'a' && s[i + 1]  <= 'z') {
        s[i + 1] -= 32;
      }
      // 将下一个字符拼接到结果result中
      result += s[i + 1]
    }
}

最后,通过 cout 将处理后的结果输出到标准输出,每个字符串占据一行。

cout << result << endl;

完整代码如下:

#include<iostream>
#include<string>
using namespace std;
int main() {
    int n;
    string s;
    cin >> n;
    getchar(); // 吸收一个回车,因为输入n之后,要输入一个回车
    while (n--) {
        getline(cin, s); // 接收新的一行
        string result;
      // 如果首字母是小写,转换成大写
        if (s[0] >= 'a' && s[0]  <= 'z') {
            s[0] -= 32;
        }
      // 将首字母拼接到结果集中
        result += s[0];
      // 遍历整个字符串
        for (int i = 1; i < s.size() - 1; i++) {
          // 如果当前字符是空格,下一个字符不是空格,说明下一个字符是新单词的首字母
            if (s[i] == ' ' && s[i + 1] != ' ') {
                // 判定新单词的首字母是否是小写,小写则转换成大写
              if (s[i + 1] >= 'a' && s[i + 1]  <= 'z') {
                s[i + 1] -= 32;
              }
              // 将新单词首字母拼接到结果集中
              result += s[i + 1];
            }
        }
        cout << result << endl;
    }
}

三、函数的使用

上述代码虽然满足了题目要求,但是题目中判断字符是否是小写的形式,如果是小写则转换成大写,这段内容重复了。

if (s[0] >= 'a' && s[0]  <= 'z') {
    s[0] -= 32;
}
 if (s[i + 1] >= 'a' && s[i + 1]  <= 'z') {
  s[i + 1] -= 32;
}

如果后面还需要将小写字符转换成大写,那我们还需要再写一次,代码就会显得有些冗余,更致命的是,如果这是一段很长的代码,并且在多个地方应用,当我们有了新的需求,需要对这一段代码进行修改时,我们需要一处处的找出再修改,这种情况下我们可以使用第一节中提到的函数,将代码模块化,并在合适的地方重用,从而增加代码的复用性和可维护性。

下面我们就尝试写一个将小写字符转换成大写字符的函数。

我们先来复习一遍函数的定义方式:
在这里插入图片描述
一个 C++ 函数定义包括以下几个部分:

1.返回类型:函数可以返回一个值,返回类型是函数计算后返回给调用者的数据类型。如果函数不返回任何值,则返回类型为 void。
2.函数名称:这是函数的实际名称,用于在调用函数时标识它。
3.参数列表(可选):在括号内,可以声明零个或多个参数,参数是传递给函数的变量,用于执行函数体内的操作。
4.函数体:在大括号内,包含一个或多个语句,这些语句定义了函数的执行操作。

在这道题中,
1.返回类型:返回结果是经过转换后的大写字符,所以返回类型为char
2.函数名:可以自定义,这里使用changeChar作为函数的名称
3.形参列表:之前的形参列表为空,而一般的形参列表通常包括参数类型和参数名称。参数类型表示参数的数据类型,可以是内置数据类型(例如整数、字符、浮点数等)、用户自定义的数据类型。参数名称通常是用来描述参数的有意义的名称,可以在函数体内部使用,这里接收一个字符作为输入,所以形参列表类型为char, 参数名称可以用a表示, 引用传递&意味着函数可以修改传递给它的参数。

// char代表返回类型,changeChar代表函数名称,char a表示函数接收一个字符a作为参数
char changeChar(char &a) {
  if (a >= 'a' && a <= 'z') a -= 32;
  // 返回类型为char, 最终还要返回a
  return a;
}

上面的函数 changeChar成功实现了将传递给它的字符从小写转换为大写,现在再将它运用到我们之前写的代码中,看看是不是更简洁一点。

#include<iostream>
#include<string>
using namespace std;
// 将小写字母转换成大写字母的函数
char changeChar(char a) { 
    if (a >= 'a' && a <= 'z') a -= 32;
    return a;
}
int main() {
    int n;
    string result, s;
    cin >> n;
    getchar();
    while (n--) {
        result = "";
        getline(cin, s);
        result += changeChar(s[0]); // 将s[0]传递到参数进行处理,转换成大写字母
        for (int i = 1; i < s.size() - 1; i++) { 
            if (s[i] == ' ' && s[i + 1] != ' ') {
              result += changeChar(s[i + 1]); // 将s[i+1]传递到参数进行处理,转换成大写字母
            }
        }
        cout << result << endl;
    }
}

四、形参和实参

在 C++ 程序设计中,理解形式参数(形参)和实际参数(实参)的区别和作用是非常重要的,特别是在函数调用和定义中。

形参是在函数定义时在函数签名中声明的变量。它们作为占位符存在,在函数调用时不具有实际的值,而是用于定义应该如何处理传递给函数的值或引用。形参在函数体内部作为局部变量使用,它们的作用域限定在函数内。
在下面的函数定义中,x 和 y 是形参:

int add(int x, int y) {
    return x + y;
}

实参是在调用函数时传递给函数的实际值。实参可以是常量、变量、表达式或者更复杂的类型。在函数调用过程中,实参的值或引用被传递给相应的形参。实参传递的值必须与函数的形参类型匹配,否则会发生类型错误。

int main() {
    int x = 5, y = 3;
    printSum(x, y); // 这里进行了函数调用,x和y是实参
      cout << x << y << endl;
    return 0;
}

当函数执行时,实参的值会复制一份给形参,因此在函数中的修改只会影响形参的值,不会影响传递的实参。

void printSum(int a, int b) { // a和b是传递的形参
 // 当函数执行时,实参的值会复制一份给a, b, a的值为5,y的值为3
  int sum = a + b;
  a++;  // a的值为6
  b--; // b的值为2
  cout << a << endl; // 6
  cout << b << endl; // 2
}
int main() {
    int x = 5, y = 3;
    printSum(x, y); // 这里进行了函数调用,x和y是实参,实参的值并没有改变
      cout << x << endl; // 5
      cout << y << endl; // 3
    return 0;
}

五、引用

在定义函数的时候,我们写形参列表的时候使用了&a的形式,这里的&表示参数a是一个引用,这里的“引用”是什么意思呢?

引用实际上是为变量起了另外一个名字,并且在引用上执行的操作会影响到引用所指向的原始变量。在声明引用时,需要在变量名前使用 & 符号。引用必须在声明时进行初始化,并且一旦初始化,就不能改变引用的目标。

int a = 10;
int &ref = a; // 声明并初始化引用为a

引用常常作为函数参数来使用,以便在函数内部修改函数调用时传递的变量,比如下面的示例:
当函数传递参数时,通常会创建参数的副本。使用引用参数可以避免不必要的参数复制,这对于大型数据结构(如数组)尤其有用,因为复制这些数据结构会产生显著的开销。

#include <iostream>
using namespace std;

void changeValue(int & a) {
    a = 100; // 修改调用方传递的变量的值
}

int main() {
    int x = 10;
    cout << x << endl;
    changeValue(x); // 传递x的引用给函数
    cout << x << endl; // x的值被函数修改为100
    return 0;
}

需要注意的是,引用的作用域(作用范围)通常是在声明它的函数或代码块内,超出作用域后引用将无效。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

[数据集][目标检测]电力场景输电线悬垂线夹检测数据集VOC+YOLO格式2538张2类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;2538 标注数量(xml文件个数)&#xff1a;2538 标注数量(txt文件个数)&#xff1a;2538 标注…

照片逼真肖像动画的音频驱动合成——AniPortrait翻译与调试

文章目录 AniPortrait翻译AbstractIntroduction2 Method2.1 Audio2Lmk2.2 Lmk2Video 3 Experiments3.1 Implementation Details3.2 Results 代码调试1 下载github项目2 配置环境3 下载包4 下载权重5 推断 AniPortrait翻译 AniPortrait&#xff1a;照片逼真肖像动画的音频驱动合…

解决使用matplotlib不显示中文的问题

某季度某城市某天11点到12点气温变化图 import random x range(60) y_BeiJing [random.uniform(15,18) for i in x] plt.figure(figsize(20,8),dpi80) plt.plot(x,y_BeiJing) x_label ["11点{}分".format(i) for i in x] plt.xticks(x[::5],x_label[::5]) plt.yt…

【网络】网络层协议-IP协议

网络层协议-IP协议 文章目录 IP协议1.网络层 & IP协议基本概念2.IP协议格式2.1报头各字段简介2.2IP如何将报头与有效载荷进行分离&#xff1f;2.3IP如何决定将有效载荷交付给上层的哪一个协议&#xff1f;2.4源IP与目的IP2.5八位生存时间2.6分片与组装 3.网段划分3.1IP地址…

嵌入式Qt移植前期准备-思维导图-学习笔记-基于正点原子阿尔法开发板

嵌入式Qt移植前期准备 概述 Qt官方在哪些平台/系统有支持&#xff1f; Windows Ubuntu IOS 为什么要进行Qt移植? 自制系统&#xff0c;如BusyBox简易系统&#xff0c;希望能运行Qt 厂家提供的Qt库太大&#xff0c;太全&#xff0c;占空间 版本升级&#xff0c;想用其它…

蒙特卡洛应用:RTX 光线追踪算法 ReSTIR 原理

接上文&#xff1a;蒙特卡洛方法相关论文&#xff1a;《Spatiotemporal reservoir resampling for real-time ray tracing with dynamic direct lighting: 2020》 ReSTIR 算法全称为 GIalgorithm based on Spatiotemporal Reservoir Resampling&#xff0c;是一种在实时路径追…

面试利器:产品经理必备的40+高频面试题集,收藏这一篇就够了

产品经理的面试&#xff0c;一般会问什么问题&#xff1f; 作为公司的产品负责人&#xff0c;大小也面了不下 100 个产品&#xff0c;产品经验横跨了 0~3 年、3~5 年。 我罗列了一些常用的产品面试问题&#xff0c;主要有“开始、产品、中级、考察、结束”等五类&#xff0c;…

【C++ 面试 - 面向对象】每日 3 题(十二)

✍个人博客&#xff1a;Pandaconda-CSDN博客 &#x1f4e3;专栏地址&#xff1a;http://t.csdnimg.cn/fYaBd &#x1f4da;专栏简介&#xff1a;在这个专栏中&#xff0c;我将会分享 C 面试中常见的面试题给大家~ ❤️如果有收获的话&#xff0c;欢迎点赞&#x1f44d;收藏&…

区间预测|基于灰狼优化最小二乘支持向量机的多变量回归区间预测Matlab程序GWO-LSSVM-ABKDE

区间预测|基于灰狼优化最小二乘支持向量机的多变量回归区间预测Matlab程序GWO-LSSVM-ABKDE 文章目录 前言区间预测|基于灰狼优化最小二乘支持向量机的多变量回归区间预测Matlab程序GWO-LSSVM-ABKDE 一、GWO-LSSVM-ABKDE模型1. 灰狼优化算法&#xff08;GWO&#xff09;2. 最小二…

vue.js - 看板娘 Live2d

文中的资源文件在这里&#xff1a;我的资源中&#xff0c;打好包了已经&#xff0c;地址&#xff1a;live2d资源 1、在项目的 src/assets 文件夹中&#xff0c;添加 live2d 的资源文件 2、在 src/components 文件中&#xff0c;编写 live2d的index.vue组件 3、在 App.vue 中…

AI在医学领域:在软组织和骨骼肿瘤放射学成像中的应用综述

软组织和骨骼肿瘤&#xff08;Soft-tissue and bone tumours&#xff0c;STBT&#xff09;是人体中较为罕见的肿瘤&#xff0c;包括良性和恶性病变。恶性STBT&#xff0c;约占所有肿瘤的1%。这些肿瘤可以发生在任何年龄和几乎所有解剖部位&#xff0c;起源于包括肌肉、脂肪、血…

富唯智能公司和产品介绍

公司介绍 广州富唯智能科技有限公司是一家专注于人工智能与柔性智造领域的高新技术企业&#xff0c;于2022年联合清华大学深圳国际研究生院成立&#xff0c;拥有多名业内顶尖的人工智能专家、智能制造专家以及硕士和博士组成的高水平研发团队。 公司致力于用前沿的人工智能技…

GenAI大模型应用方法选择深度解析: 模型训练,微调,检索增强RAG和提示工程

重点摘要 每种生成式人工智能学习方法都有其独特的优势和理想应用场景&#xff1a; 模型训练(Model Training)&#xff1a;需要大量的数据和计算资源来从头构建一个人工智能模型。它具有高度的可定制性和可扩展性&#xff0c;但耗时较长。 微调(Fine-Tuning)&#xff1a;专注…

win11成功点亮 WSL 创建的 Linux 子系统 jupyter服务 并配合 conda 环境运行代码【保姆级教程】

&#x1f947; 版权: 本文由【墨理学AI】原创首发、各位读者大大、敬请查阅、感谢三连 &#x1f389; 声明: 作为全网 AI 领域 干货最多的博主之一&#xff0c;❤️ 不负光阴不负卿 ❤️ 文章目录 零、前言一、安装 jupyter notebook二、生成jupyter notebook配置文件三、设置J…

力扣top100-链表类题易错点总结-c++实现(更新中)

首先给一个我之前写的双指针在链表类题中的妙用的link&#xff1a;双指针在链表中的妙用 tip1 来自“合并两个有序链表” 题目链接戳这里 这道题注意的就是如果是要返回一个新链表的头结点&#xff0c;一定要新建一个头结点&#xff1a; ListNode* prehead new ListNode…

备考计算机二级Python之Day5

第5章 函数和代码 一、函数的基本使用 函数是一段具有特定功能的、可重用的语句组&#xff0c;通过函数名来表示和调用。 函数的使用包括两部分&#xff1a;函数的定义和函数的使用 1、函数的定义 Python语言通过保留字def定义函数&#xff0c;语法形式如下&#xff1a; …

判别分析2|Bayes判别分析|Fisher判别分析|软件求解

Bayes判别分析 引入先验信息 距离判别只要求知道总体的数字特征&#xff0c;不涉及总体的分布函数 当均值和协方差未知时&#xff0c;就用样本的均值和协方差矩阵做估计值。距离判别方法简单实用&#xff0c;但没有考虑到每个总体出现的机会大小&#xff0c;即先验概率&#…

数据结构(邓俊辉)学习笔记】优先级队列 09——左式堆:合并算法

文章目录 1. LeftHeap模板类2. 算法3. 实现4. 实例 1. LeftHeap模板类 接下来这节&#xff0c;来讨论左式堆的合并算法。再给出具体算法之前&#xff0c;首先要给出左式堆模板类的定义。 比如这就是一种可能的实现方式&#xff0c;可以看到&#xff0c;我们这里再次利用了 C…

srm供应商一体化招采系统解决方案,需求功能清单以及源码实现(JAVA)

1. 供应商管理 2. 采购需求管理 3. 采购寻源管理 4. 采购合同管理 5. 采购订单管理 6. 采购协同管理 7. 外部商城采购管理 8. 报表查询管理 9. 系统管理 10. 集成管理 资料获取&#xff1a;本文末个人名片。

在Activity中使用Menu

在Activity中使用Menu 手机毕竟和电脑不同&#xff0c;它的屏幕空间非常有限&#xff0c;因此充分地利用屏幕空间在手机界面设计中就显得非常重要了。如果你的活动中有大量的菜单需要显示&#xff0c;这个时候界面设计就会比较尴尬&#xff0c;因为仅这些菜单就可能占用屏幕将…