【C++ 语言】C++字符串 ( string 类 | 创建方法 | 控制台输出 | 字符串操作 | 栈内存字符串对象 | string* )(二)

简介: 【C++ 语言】C++字符串 ( string 类 | 创建方法 | 控制台输出 | 字符串操作 | 栈内存字符串对象 | string* )(二)

C++ 字符串 方法调用 ( 堆内存对象 )


C++ 字符串 堆内存对象 方法调用 :


1.堆内存对象创建 : 使用 new 创建的 string 对象 需要在堆内存中为其分配内存 , 返回的是 string* 指针类型 ;

//使用 new 创建的对象 , 就不能使用 . 访问其方法和变量 , 需要使用 -> 符号进行访问 
  // -> 相当于 (*). 运算, 先读取指针内容 , 然后访问其方法或变量
  string* string_c_plus_7 = new string(" New String ");


1.获取字符串长度 : 调用 string 对象的 size() 方法 , 即可返回字符串长度 ; 需要使用 -> 代替 . 操作 ;

//① 获取字符串长度 : 
  int string_c_plus_7_size = string_c_plus_7->size();


2.判断字符串是否为空 : 调用 string 对象的 empty() 方法 , 判断字符串是否为空 ; 需要使用 -> 代替 . 操作 ;

//② 判断字符串是否为空 : 
  bool string_c_plus_7_empty = string_c_plus_7->empty();


3.使用指针的好处 :

① 扩大作用域 : 如果在栈内存中使用 , 有作用域限制 , 出了栈内存 作用域 , 该对象就无效了 ;

② 方便参数传递 : 指针 大小为 4 ( 32 位系统 ) 或 8 ( 64 位系统 ) 个字节 , 其当做参数传递 比直接传递对象 ( 动辄几十上百字节甚至更高 ) 效率更高 ;

4.代码示例 :

//使用 new 创建的对象 , 就不能使用 . 访问其方法和变量 , 需要使用 -> 符号进行访问 
  // -> 相当于 (*). 运算, 先读取指针内容 , 然后访问其方法或变量
  string* string_c_plus_7 = new string(" New String ");
  //① 获取字符串长度 : 
  int string_c_plus_7_size = string_c_plus_7->size();
  cout << "string_c_plus_7 : " << string_c_plus_7_size << endl;
  //② 判断字符串是否为空 : 
  bool string_c_plus_7_empty = string_c_plus_7->empty();
  cout << "string_c_plus_7_empty : " << string_c_plus_7_empty << endl;
  //释放堆内存
  delete string_c_plus_7;
  //使用指针的好处 : 
  // ① 如果在栈内存中使用 , 有作用域限制 , 出了栈内存 作用域 , 该对象就无效了 ; 
  // ② 指针 大小为 4 ( 32 位系统 ) 或 8 ( 64 位系统 ) 个字节 , 
  //   其当做参数传递 比直接传递对象 ( 动辄几十上百字节甚至更高 ) 效率更高


5.运行结果 :

string_c_plus_7 : 12
string_c_plus_7_empty : 0


image.png



C / C++ 字符串 完整代码示例



/ 001_CMake_1.cpp: 定义应用程序的入口点。
//
#include "001_CMake_1.h"
#include "c_extern.h"
using namespace std;
//定义方法接收 int& 引用类型变量
//并在方法中修改该变量的值
void quote(int& b) {
  b = 888;
}
int main()
{
  cout << "Hello CMake。" << endl;
  //1. C C++ 兼容
  //博客地址 : https://hanshuliang.blog.csdn.net/article/details/98840708
  //调用 c_extern.h 头文件中定义的方法
  //该方法定义在了 C 语言文件中
  add(1, 2);
  //2. 引用数据类型
  //博客地址 : https://hanshuliang.blog.csdn.net/article/details/99239635
  //代码 展示 流程 : 
  //① 定义 普通 类型 变量
  int a = 8;
  //② 定义 引用类型变量, 格式 : 类型名称& 变量名 = 对应类型变量名称 ;
  int& b = a;
  //③ 调用函数传入引用类型参数 : 将引用类型传给接收引用类型的方法
  quote(b);
  //④ 打印引用数据类型的修改结果 , 结果是 b 被修改成了 888
  cout << b << endl;
  //引用数据类型定义与使用 : 
  // ① 引用数据类型定义 : 类型名称& 变量名 = 对应类型变量名称 ;
  // ② 引用数据类型的使用方法 : 直接当做原来的变量使用即可, 可以替换原来变量的位置使用
  //引用类型解析 : 
  // ① int& 是引用数据类型 , b 是 a 的引用 
  // ② 分配一块内存存放 int 类型数据 8 , 将该内存赋予一个别名 a
  // ③ 同时又给该内存赋予另外一个别名 b 
  //3. 字符串使用
  //C 字符串 表示方法 : 
  // ① 字符数组 : 本质是 字符 数组 char[] , 这里注意字符数组要以 NULL 或 '\0' 结尾; 
  char string_c[] = { 'h', 'e', 'l', 'l', 'o', '\0' };
  // ② 指针 : 使用指针形式表示字符串 , 默认末尾增加 '\0' ;
  char* string_c_p = "hello";
  //字符串打印 : 
  // ① 打印字符串 , cout 后的 << 后可以打印 字符串 , 也可以打印变量
  // ② 输出 cout << 字符串或变量1 << 字符串或变量2 ... << endl 可以拼接 输出信息
  cout << "string_c : " << string_c << endl;
  cout << "string_c_p : " << string_c_p << endl;
  //C 语言中的字符串操作
  //拷贝字符串 
  char string_c_copy_destination[6];
  char string_c_copy_source[] = { 'h', 'e', 'l', 'l', 'o', '\0' };
  // ① 参数 : strcpy 方法是拷贝字符串的方法 , 第一个参数是目标字符串 , 第二个参数是源字符串
  // ② 作用 : 该方法是将 源字符串 拷贝到 目标字符串中
  strcpy(string_c_copy_destination, string_c_copy_source);
  // ③ 打印拷贝结果 : 
  cout << "string_c_copy_destination : " << string_c_copy_destination << endl;
  //拼接字符串 
  //① 定义目标字符串 : 拼接字符串的目标字符串的大小一定要大于等于要拼接的两个字符串大小之和, 否则会报错
  char string_c_cat_destination[30] = " cat dst ";
  char string_c_cat_source[] = " cat src ";
  //② 拼接字符串方法参数 : 第一个参数是目标字符串 , 第二个参数是源字符串
  //③ 目标字符串大小 : 注意 目标字符串的 大小一定要大于 两个字符串实际大小
  strcat(string_c_cat_destination, string_c_cat_source);
  //④ 打印字符串拼接结果 : 
  cout << "string_c_cat_destination : " << string_c_cat_destination << endl;
  //获取字符串长度
  //① 参数 : 传入要获取的字符串 , 该长度不含 '\0' 结尾标志
  //② 作用 : 获取实际的字符串长度 , 即自动识别 '\0' 位置 , 获取其长度 , 与所占用的内存大小无关
  char string_c_len[] = { 'h', 'e', 'l', 'l', 'o', '\0' };
  char string_c_len2[20] = { 'h', 'e', 'l', 'l', 'o', '\0' };
  char * string_c_len3 = "hello";
  //① 字符数组长度 , 测量从开始到 '\0' 的长度, 不包括 '\0'
  int len1 = strlen(string_c_len);
  //② 指定大小的字符数组长度 , 结果不是指定的大小的值 , 获取的是实际字符串长度
  int len2 = strlen(string_c_len2);
  //③ 指针表示的字符串 , 其获取的大小是实际字符串大小, 不包含自动添加的 '\0' 
  int len3 = strlen(string_c_len3);
  //④ 打印 三个字符串大小
  cout << "len1 : " << len1
  << " len2 : " << len2
  << " len3 : " << len3
  << endl;
  //字符串比较
  // ① 参数说明 : 参数是需要比较的两个字符串 , 第一个参数 str1 , 第二个参数 str2
  // ② 对比规则 : str1 和 str2 两个字符串 , 从左到右 逐个对比 ASCII 码 大小 ; 
  //  a. 如果 str1 等于 str2 , 返回 0; 
  //  b. 如果 str1 > str2 , 返回值 大于 0 ;
  //  c. 如果 str1 < str2 , 返回值 小于 0 ;
  //定义需要比较的字符串
  char* string_c_comp_1 = "Hello";
  char* string_c_comp_2 = "Hello";
  char* string_c_comp_3 = "hello";
  // ① 两个字符串相等
  int cmp_result_1_2 = strcmp(string_c_comp_1, string_c_comp_2);
  // ② "Hello" 字符串 (H 对应 ASCII 72) 小于 "hello" 字符串 (h 对应 ASCII 104) , 返回值 小于 0
  int cmp_result_1_3 = strcmp(string_c_comp_1, string_c_comp_3);
  // ③ "hello" 字符串 (h 对应 ASCII 104) 大于 "Hello" 字符串 (H 对应 ASCII 72) , 返回值 大于 0
  int cmp_result_3_1 = strcmp(string_c_comp_3, string_c_comp_1);
  //输出字符串对比结果
  cout << "cmp_result_1_2 : " << cmp_result_1_2 
  << " cmp_result_1_3 : " << cmp_result_1_3
  << " cmp_result_3_1 : " << cmp_result_3_1
  << endl;
  // C++ string 类 : 该类定义在 iostream 头文件中
  //创建 string 类型对象有三种方法 : 
  //① 直接使用字符串赋值 
  //② 调用构造方法赋值 
  //③ 最后可以调用 new 为字符串分配一块内存
  //① 使用字符串赋值
  string string_c_plus_1 = " Hello ";
  //② 调用构造方法 初始化字符串
  string string_c_plus_2(string_c_plus_1);
  string string_c_plus_3(" World ");
  //上面的三种字符串不需要释放 , 因为其定义在栈内存中 , 下面使用 new 创建字符串的情况需要 delete 释放内存 ; 
  //③ 使用 new 申请的内存 , 需要使用 delete 释放
  string *string_c_plus_4 = new string(" New ");
  delete string_c_plus_4;
  //使用 new [] 申请的数组 , 需要使用 delete[] 释放
  //使用 malloc 申请的内存 , 需要使用 free 释放
  //C++ 字符串输出
  //字符串对象不能直接在 cout 中输出, cout << string string_c_plus_5 << endl; 是错误的
  //cout << string_c_plus_2 << endl;
  //要将 string 对象打印到控制台上, 需要将其转为 C 字符串 , char* 或 char[] 才能输出
  cout << string_c_plus_1.c_str() << endl;
  //C++ 字符串拼接
  //① "+" : 操作符重载 , 重新定义了 加号运算符的行为 , 这里加号可以实现字符串拼接 , 与 Java 类似
  //② 调用 string 对象的 append 方法 , 拼接字符串
  string string_c_plus_5 = string_c_plus_1 + string_c_plus_3;
  string string_c_plus_6 = string_c_plus_1.append( string_c_plus_3 );
  //输出拼接的字符串
  cout << string_c_plus_5.c_str() << endl;
  cout << string_c_plus_6.c_str() << endl;
  //获取 C++ 字符串长度 : 调用 string 对象的 size() 方法 , 获取字符串长度
  int string_c_plus_1_size = string_c_plus_1.size();
  cout << "string_c_plus_1_size : " << string_c_plus_1_size << endl;
  //判断 C++ 字符串是否为空 : 调用 string 对象的 empty() 方法 ; 
  bool string_c_plus_1_empty = string_c_plus_1.empty();
  cout << "string_c_plus_1_empty : " << string_c_plus_1_empty << endl;
  //使用 new 创建的对象 , 就不能使用 . 访问其方法和变量 , 需要使用 -> 符号进行访问 
  // -> 相当于 (*). 运算, 先读取指针内容 , 然后访问其方法或变量
  string* string_c_plus_7 = new string(" New String ");
  //① 获取字符串长度 : 
  int string_c_plus_7_size = string_c_plus_7->size();
  cout << "string_c_plus_7 : " << string_c_plus_7_size << endl;
  //② 判断字符串是否为空 : 
  bool string_c_plus_7_empty = string_c_plus_7->empty();
  cout << "string_c_plus_7_empty : " << string_c_plus_7_empty << endl;
  //释放堆内存
  delete string_c_plus_7;
  //使用指针的好处 : 
  // ① 如果在栈内存中使用 , 有作用域限制 , 出了栈内存 作用域 , 该对象就无效了 ; 
  // ② 指针 大小为 4 ( 32 位系统 ) 或 8 ( 64 位系统 ) 个字节 , 
  //   其当做参数传递 比直接传递对象 ( 动辄几十上百字节甚至更高 ) 效率更高 
  return 0;
}



目录
相关文章
|
1月前
|
存储 安全 iOS开发
内存卡怎么格式化?6个格式化方法供你选
随着使用时间的增加,内存卡可能会因为数据积累、兼容性或是文件系统损坏等原因需要进行格式化。那么怎样正确格式化内存卡呢?格式化内存卡的时候需要注意什么呢?本文会给大家提供详细的步骤,帮助大家轻松完成格式化内存卡的操作。
|
3月前
|
监控 JavaScript Java
Node.js中内存泄漏的检测方法
检测内存泄漏需要综合运用多种方法,并结合实际的应用场景和代码特点进行分析。及时发现和解决内存泄漏问题,可以提高应用的稳定性和性能,避免潜在的风险和故障。同时,不断学习和掌握内存管理的知识,也是有效预防内存泄漏的重要途径。
304 62
|
3月前
|
传感器 人工智能 物联网
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发,以及面临的挑战和未来趋势,旨在帮助读者深入了解并掌握这些关键技术。
88 6
|
4月前
|
机器学习/深度学习 算法 物联网
大模型进阶微调篇(一):以定制化3B模型为例,各种微调方法对比-选LoRA还是PPO,所需显存内存资源为多少?
本文介绍了两种大模型微调方法——LoRA(低秩适应)和PPO(近端策略优化)。LoRA通过引入低秩矩阵微调部分权重,适合资源受限环境,具有资源节省和训练速度快的优势,适用于监督学习和简单交互场景。PPO基于策略优化,适合需要用户交互反馈的场景,能够适应复杂反馈并动态调整策略,适用于强化学习和复杂用户交互。文章还对比了两者的资源消耗和适用数据规模,帮助读者根据具体需求选择最合适的微调策略。
1331 5
|
4月前
|
缓存 监控 Java
在使用 Glide 加载 Gif 动画时避免内存泄漏的方法
【10月更文挑战第20天】在使用 Glide 加载 Gif 动画时,避免内存泄漏是非常重要的。通过及时取消加载请求、正确处理生命周期、使用弱引用、清理缓存和避免重复加载等方法,可以有效地避免内存泄漏问题。同时,定期进行监控和检测,确保应用的性能和稳定性。需要在实际开发中不断积累经验,根据具体情况灵活运用这些方法,以保障应用的良好运行。
|
5月前
|
Java 索引
java基础(13)String类
本文介绍了Java中String类的多种操作方法,包括字符串拼接、获取长度、去除空格、替换、截取、分割、比较和查找字符等。
59 0
java基础(13)String类
|
3月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
89 2
|
4月前
|
Java
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
本文深入探讨了Java中方法参数的传递机制,包括值传递和引用传递的区别,以及String类对象的不可变性。通过详细讲解和示例代码,帮助读者理解参数传递的内部原理,并掌握在实际编程中正确处理参数传递的方法。关键词:Java, 方法参数传递, 值传递, 引用传递, String不可变性。
91 1
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
|
4月前
|
安全 Java 测试技术
Java零基础-StringBuffer 类详解
【10月更文挑战第9天】Java零基础教学篇,手把手实践教学!
109 2
|
5月前
|
安全 Java
String类-知识回顾①
这篇文章回顾了Java中String类的相关知识点,包括`==`操作符和`equals()`方法的区别、String类对象的不可变性及其好处、String常量池的概念,以及String对象的加法操作。文章通过代码示例详细解释了这些概念,并探讨了使用String常量池时的一些行为。
String类-知识回顾①