news 2026/4/23 18:34:20

string*的使用

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
string*的使用

在 C++ 中,string是标准库提供的字符串类,指向string的指针(string*用法和指向普通自定义类型的指针逻辑一致,但需结合string的特性理解。下面从核心概念、常见用法、避坑点三方面详细讲解。

一、核心概念:string*是什么?

string*指向string对象的指针,存储的是string对象在内存中的首地址,而非string存储的字符串内容。

类比理解:

  • string s = "hello";→ 定义了一个string对象s,它在内存中占据一块空间(存储字符串、长度、容量等);
  • string* p = &s;→ 指针p指向s的内存地址(相当于 “记录了s的门牌号”)。

二、string*的核心用法(结合代码场景)

1. 定义与初始化

string*必须指向已存在的string对象(避免野指针),常见初始化方式:

cpp

运行

#include <iostream> #include <string> using namespace std; int main() { // 1. 定义string对象,指针指向它(最常用) string str = "apple"; string* p1 = &str; // &取str的地址,赋值给string*指针 // 2. 指针数组:存储多个string对象的地址(你代码中的核心场景) string str1 = "banana", str2 = "cherry", str3 = "date"; string* arr[3] = {&str1, &str2, &str3}; // 指针数组,每个元素是string* // 3. 动态分配string对象(较少用,需手动释放) string* p2 = new string("grape"); // 堆上创建string对象,p2指向它 return 0; }
2. 解引用:通过指针访问string对象

指针本身只是地址,必须通过解引用(*才能访问指向的string对象,进而操作字符串内容。

操作场景语法示例说明
访问字符串内容cout << *p1;解引用指针,输出p1指向的string对象的字符串(输出:apple
比较字符串内容if (*p1 > *p2)解引用后用string重载的>比较字符串内容(而非比较指针地址)
调用string成员函数(*p1).size()/p1->size()访问string的成员(长度、拼接、查找等),->是解引用 + 成员访问的简写
修改字符串内容*p1 = "orange";解引用后直接赋值,修改指向的string对象的内容

示例代码:

cpp

运行

int main() { string str = "apple"; string* p = &str; // 1. 解引用访问字符串内容 cout << *p << endl; // 输出:apple // 2. 调用string成员函数(两种写法等价) cout << (*p).size() << endl; // 输出:5(解引用后用.访问成员) cout << p->size() << endl; // 输出:5(推荐!-> 是指针的成员访问符) // 3. 修改字符串内容 *p = "banana"; cout << *p << endl; // 输出:banana // 4. 拼接字符串 p->append(" cherry"); // 等价于 (*p).append(" cherry") cout << *p << endl; // 输出:banana cherry return 0; }
3. 指针数组:string* arr[]

string* arr[]存储string*指针的数组,每个元素指向一个独立的string对象,排序时只需交换指针地址(无需拷贝字符串内容,效率高)。

cpp

运行

// 排序函数:指针数组传参,n是元素个数 void sortStrings(string* arr[], int n) { string* temp; // 临时指针,用于交换地址 for (int i = 0; i < n-1; i++) { for (int j = 0; j < n-1-i; j++) { // 关键:解引用指针,比较字符串内容(而非地址) if (*arr[j] > *arr[j+1]) { // 仅交换指针地址,字符串内容不动 temp = arr[j]; arr[j] = arr[j+1]; arr[j+1] = temp; } } } } int main() { string a = "pear", b = "apple", c = "grape"; string* arr[3] = {&a, &b, &c}; // 指针数组指向3个string对象 sortStrings(arr, 3); // 排序指针数组 // 遍历指针数组,输出结果 for (int i = 0; i < 3; i++) { cout << *arr[i] << endl; // 解引用arr[i](string*),输出字符串 } // 输出:apple → grape → pear return 0; }
4. 动态数组:string*指向string数组

若需要动态创建多个string对象,可让string*指向string数组的首地址:

cpp

运行

int main() { // 静态string数组,指针指向数组首元素 string strs[] = {"cat", "dog", "bird"}; string* p = strs; // 数组名strs等价于首元素地址&strs[0] // 遍历数组(指针方式) for (int i = 0; i < 3; i++) { cout << *p << " "; // 输出:cat dog bird p++; // 指针后移,指向下一个string对象 } // 动态string数组(堆上分配) string* pArr = new string[3]{"red", "green", "blue"}; for (int i = 0; i < 3; i++) { cout << pArr[i] << " "; // 等价于*(pArr+i),输出:red green blue } delete[] pArr; // 动态数组需手动释放 return 0; }

三、避坑点(新手最容易错)

1. 不要直接把string对象赋值给string*

错误写法(你最初的问题):

cpp

运行

string str = "apple"; string* p = str; // ❌ 错误!str是string对象,不是地址,类型不匹配

正确写法:

cpp

运行

string* p = &str; // ✅ 取str的地址,赋值给string*
2. 避免野指针(指针指向已销毁的string对象)

cpp

运行

string* p; { string temp = "hello"; p = &temp; // temp是局部变量,出括号后销毁 } cout << *p; // ❌ 未定义行为!p指向已释放的内存(野指针)
3. 指针比较 vs 字符串内容比较

string*直接用>/<比较的是内存地址,而非字符串内容,必须解引用后比较:

cpp

运行

string s1 = "banana", s2 = "apple"; string* p1 = &s1, *p2 = &s2; if (p1 > p2) { cout << "指针地址p1 > p2"; // 可能成立(地址随机),但无业务意义 } if (*p1 > *p2) { cout << "字符串s1 > s2"; // 成立("banana" > "apple"),符合业务逻辑 }
4. 范围 for 遍历指针数组的注意事项

for(string* p : arr)中,arr必须是固定长度的数组(编译器能推导长度),若数组长度通过参数传入,需改用普通 for 循环:

cpp

运行

// 通用写法(适配任意长度n) void printStrings(string* arr[], int n) { for (int i = 0; i < n; i++) { cout << *arr[i] << endl; } }

四、总结:string*核心口诀

  1. 定义:string* p = &str;(指针指向string对象的地址);
  2. 访问:*p取对象,p->成员调函数(如p->size());
  3. 比较:解引用后比内容(*p1 > *p2),不要直接比指针;
  4. 数组:string* arr[]存多个string地址,排序交换地址更高效;
  5. 避坑:不赋野指针、不直接赋值string对象给string*
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 12:29:26

揭秘Docker Offload技术内幕:如何实现云端任务秒级卸载?

第一章&#xff1a;揭秘Docker Offload技术内幕&#xff1a;如何实现云端任务秒级卸载&#xff1f;Docker Offload 技术是一种将容器化工作负载从边缘节点或本地主机快速迁移至云端执行的机制&#xff0c;其核心在于动态资源调度与轻量级容器镜像的协同优化。该技术通过智能判断…

作者头像 李华
网站建设 2026/4/23 12:29:21

人工智能行业迎来重大突破:全新模型架构引领技术革新新浪潮

人工智能行业迎来重大突破&#xff1a;全新模型架构引领技术革新新浪潮 【免费下载链接】LFM2-1.2B-Extract 项目地址: https://ai.gitcode.com/hf_mirrors/LiquidAI/LFM2-1.2B-Extract 在当今数字化时代&#xff0c;人工智能技术正以前所未有的速度重塑着各个行业的发…

作者头像 李华
网站建设 2026/4/23 12:29:33

Wan2.2-T2V-A14B支持多摄像头视角切换的实现机制

Wan2.2-T2V-A14B 多摄像头视角切换机制深度解析 在影视创作中&#xff0c;镜头语言是叙事的灵魂。一个精准的推拉摇移、一次恰到好处的主客视角转换&#xff0c;往往能极大增强情节张力与观众代入感。然而&#xff0c;在AI生成视频领域&#xff0c;大多数模型仍停留在“固定机位…

作者头像 李华
网站建设 2026/4/23 11:12:30

Windows右键菜单大扫除:ContextMenuManager极简使用手册

Windows右键菜单大扫除&#xff1a;ContextMenuManager极简使用手册 【免费下载链接】ContextMenuManager &#x1f5b1;️ 纯粹的Windows右键菜单管理程序 项目地址: https://gitcode.com/gh_mirrors/co/ContextMenuManager 你是否曾经为Windows右键菜单中那些杂乱无章…

作者头像 李华
网站建设 2026/4/23 3:18:46

零基础掌握镜像烧录:Balena Etcher新手快速上手指南

零基础掌握镜像烧录&#xff1a;Balena Etcher新手快速上手指南 【免费下载链接】etcher Flash OS images to SD cards & USB drives, safely and easily. 项目地址: https://gitcode.com/GitHub_Trending/et/etcher 想要轻松将系统镜像写入SD卡或USB驱动器吗&#…

作者头像 李华
网站建设 2026/4/23 15:33:49

考试场景下的MCP AI Agent容灾设计(专家级高可用部署方案曝光)

第一章&#xff1a;考试场景下MCP AI Agent容灾设计概述在高并发、强一致性的考试系统中&#xff0c;MCP&#xff08;Mission-Critical Processing&#xff09;AI Agent承担着实时监考、异常行为识别与应急响应等关键任务。一旦AI Agent出现故障&#xff0c;可能导致监考中断、…

作者头像 李华