8、C语言指针专题:指针与字符串

news2026/5/22 3:41:39
在C语言中字符串本质是“以空字符\0结尾的字符序列”而指针是操作字符串最灵活、高效的工具。字符串的存储、访问、修改、排序及各类处理都可以通过指针实现且指针操作相比数组下标操作更节省内存、执行效率更高。本文将围绕给定的核心知识点结合具体程序示例逐点拆解指针与字符串的关联用法每个知识点及细分点均配套程序段帮助大家吃透指针操作字符串的本质突破C语言字符串学习的难点。一、字符串常量的指针表示字符串常量是指用双引号括起来的字符序列如hello在内存中以只读形式存储在常量区其本质是一个字符数组且自动在末尾添加空字符\0作为结束标志。而字符串常量的指针表示就是用字符指针char *指向这个字符串常量的起始地址通过指针访问字符串中的每个字符。核心要点字符指针指向字符串常量时指针存储的是字符串首字符的地址且字符串常量是只读的不能通过指针修改其内容否则会导致程序崩溃。1.1 字符串常量的指针声明与访问字符串常量的指针声明有两种常见方式本质一致均是让字符指针指向字符串常量的起始地址下面通过程序段演示两种方式及访问方法。代码解析两种声明方式本质相同都是让字符指针指向字符串常量hello world的首字符h的地址。printf函数的%s格式符会自动从指针指向的地址开始遍历字符直到遇到\0因此可以直接打印字符指针。需要重点注意字符串常量存储在常量区是只读的通过指针修改其内容会触发内存访问错误实际开发中需严格规避。1.2 字符串常量指针的本质与字符数组的关联字符串常量的指针本质是指向“只读字符数组”的指针其长度由字符串常量的长度1\0决定且指针本身可以重新赋值指向其他字符串常量但不能修改指向的内容。代码解析字符指针str可以重新赋值从指向hello改为指向world这是因为指针本身存储的是地址修改指针的值地址不会影响原字符串常量。通过临时指针p遍历字符串直到遇到\0可以计算出字符串的长度这也印证了“字符串以\0结尾”的核心特性。二、字符指针与字符数组的区别字符指针char *和字符数组char arr[]是操作字符串的两种常用方式初学者极易混淆但二者在内存存储、赋值方式、可修改性、内存分配等方面有本质区别这也是C语言的高频考点。下面通过对比表格和程序段清晰区分二者的核心差异。2.1 核心区别对比表格程序段对比项字符指针char *str字符数组char arr[]内存存储指向字符串常量区或堆区指针本身存储地址栈区或全局区存储字符串的所有字符含\0赋值方式可直接赋值字符串常量如str hello可重新赋值只能初始化赋值如char arr[] hello不能直接用重新赋值可修改性指向常量区时不可修改内容指向堆区/栈区时可修改栈区/全局区的字符数组内容可修改内存分配指针占4/8字节32位/64位系统字符串单独存储数组大小字符串长度1存储所有字符程序段示例直观对比二者区别代码解析这段程序清晰展示了字符指针与字符数组的核心区别。字符数组arr存储在栈区内容可修改但不能直接用重新赋值字符指针str指向常量区的字符串内容不可修改但指针本身可以重新赋值指向其他字符串常量。此外sizeof(arr)计算的是数组的总大小含\0而sizeof(str)计算的是指针本身的大小与系统位数相关。2.2 易混淆场景解析避坑重点场景1字符指针指向栈区字符数组——此时指针指向的内容可修改与字符数组的区别仅在于“指针存储地址数组存储字符”。代码解析当字符指针指向栈区的字符数组时指针指向的内容不再是只读的可以通过指针修改字符数组的内容这是易混淆的关键点。此时字符指针的作用只是“指向字符数组的首地址”本质是通过指针操作字符数组。三、字符串指针数组字符串指针数组是“存储字符指针的数组”即数组的每个元素都是char*类型的指针每个指针指向一个字符串字符串常量或动态分配的字符串。其核心优势是无需存储完整的字符串仅存储字符串的地址节省内存且便于管理多个字符串如字符串排序、查找。声明格式char *指针数组名[数组长度];如char *strArr[5]数组的每个元素都是char*指针可指向不同长度的字符串。3.1 字符串指针数组的初始化与访问字符串指针数组的初始化有两种常用方式直接初始化指向字符串常量、动态初始化指向动态分配的字符串下面通过程序段演示两种方式及访问方法。#include stdio.h#include stdlib.h#include string.hint main() {// 方式1直接初始化每个元素指向字符串常量char *strArr1[3] {apple,banana,orange};// 方式2动态初始化每个元素指向动态分配的字符串char *strArr2[3];// 为每个指针分配内存并赋值strArr2[0] (char *)malloc(10 * sizeof(char));strArr2[1] (char *)malloc(10 * sizeof(char));strArr2[2] (char *)malloc(10 * sizeof(char));if (strArr2[0] NULL || strArr2[1] NULL || strArr2[2] NULL) {printf(内存申请失败\n);exit(1);}strcpy(strArr2[0], cat);strcpy(strArr2[1], dog);strcpy(strArr2[2], bird);// 访问字符串指针数组两种方式printf(方式1直接初始化的字符串指针数组\n);for (int i 0; i 3; i) {printf(strArr1[%d] %s 地址%p\n, i, strArr1[i], strArr1[i]);}printf(\n方式2动态初始化的字符串指针数组\n);for (int i 0; i 3; i) {printf(strArr2[%d] %s 地址%p\n, i, strArr2[i], strArr2[i]);}// 释放动态分配的内存方式2需手动释放for (int i 0; i 3; i) {free(strArr2[i]);strArr2[i] NULL;}return 0;}代码解析strArr1是直接初始化的字符串指针数组每个元素指向一个字符串常量存储在常量区strArr2是动态初始化的字符串指针数组每个元素指向堆区动态分配的内存通过strcpy赋值字符串。访问时直接通过数组下标strArr[i]即可打印字符串因为strArr[i]是char*指针printf的%s格式符会自动遍历到\0。3.2 字符串指针数组的核心优势与二维字符数组对比字符串指针数组与二维字符数组char arr[3][10]都能存储多个字符串但字符串指针数组更节省内存因为它仅存储字符串的地址而二维字符数组需要分配固定大小的内存即使字符串长度不足。代码解析字符串指针数组strArr的大小是24字节3个char*指针每个8字节而二维字符数组arr的大小是30字节3行10列明显更节省内存。此外字符串指针数组的元素可灵活指向不同长度的字符串而二维字符数组的长度固定灵活性较差这也是字符串指针数组的核心优势。四、字符串排序指针交换法对多个字符串进行排序时常用的方法有“字符串交换法”和“指针交换法”。其中指针交换法是效率最高的方式——无需交换字符串本身避免大量字符拷贝节省时间仅交换指向字符串的指针交换地址就能实现字符串的排序。核心逻辑用字符串指针数组存储所有字符串的地址排序时比较两个字符串的内容用strcmp函数若顺序不符交换两个指针的地址而非字符串内容最终实现排序效果。4.1 字符串排序指针交换法升序#include stdio.h#include string.h// 指针交换法排序升序按字符串ASCII码从小到大void sortStrings(char *strArr[], int n) {char *temp; // 临时指针用于交换两个指针的地址// 冒泡排序核心交换指针不交换字符串内容for (int i 0; i n - 1; i) {for (int j 0; j n - 1 - i; j) {// strcmp(a, b)a b 返回负数a b 返回0a b 返回正数if (strcmp(strArr[j], strArr[j1]) 0) {// 交换指针地址仅交换指向不移动字符串temp strArr[j];strArr[j] strArr[j1];strArr[j1] temp;}}}}int main() {// 字符串指针数组存储待排序的字符串char *strArr[] {banana, apple, orange, grape, pear};int n sizeof(strArr) / sizeof(strArr[0]); // 计算数组长度5个元素printf(排序前的字符串\n);for (int i 0; i n; i) {printf(%s , strArr[i]);}// 调用排序函数指针交换法sortStrings(strArr, n);printf(\n\n排序后的字符串升序\n);for (int i 0; i n; i) {printf(%s , strArr[i]);}return 0;}代码解析排序函数sortStrings接收字符串指针数组和数组长度使用冒泡排序算法核心是通过temp临时指针交换两个字符串指针的地址而非交换字符串本身。strcmp函数用于比较两个字符串的大小按ASCII码顺序当strArr[j]大于strArr[j1]时交换二者的指针地址最终实现升序排序。这种方式效率极高尤其适合字符串较长的场景避免了大量字符拷贝的开销。4.2 字符串排序指针交换法降序只需修改strcmp函数的比较条件即可实现降序排序核心逻辑与升序一致仅交换比较后的判断条件。#include stdio.h#include string.h// 指针交换法排序降序按字符串ASCII码从大到小void sortStringsDesc(char *strArr[], int n) {char *temp;for (int i 0; i n - 1; i) {for (int j 0; j n - 1 - i; j) {// 修改比较条件strcmp返回负数时交换a bif (strcmp(strArr[j], strArr[j1]) 0) {temp strArr[j];strArr[j] strArr[j1];strArr[j1] temp;}}}}int main() {char *strArr[] {banana, apple, orange, grape, pear};int n sizeof(strArr) / sizeof(strArr[0]);printf(排序前的字符串\n);for (int i 0; i n; i) {printf(%s , strArr[i]);}sortStringsDesc(strArr, n);printf(\n\n排序后的字符串降序\n);for (int i 0; i n; i) {printf(%s , strArr[i]);}return 0;}代码解析与升序排序相比仅修改了strcmp函数的比较条件——当strArr[j]小于strArr[j1]strcmp返回负数时交换两个指针的地址即可实现降序排序。这体现了指针交换法的灵活性只需修改比较逻辑就能切换排序方式。五、字符串处理函数的指针实现C语言标准库string.h提供了丰富的字符串处理函数如strlen、strcpy、strcmp、strcat这些函数的底层实现均基于指针操作。掌握这些函数的指针实现原理能更深入理解指针与字符串的关联同时也能灵活自定义字符串处理函数。下面逐一实现常用的字符串处理函数每个函数均采用指针操作配套程序段演示与标准库函数的功能保持一致。5.1 字符串长度函数my_strlen功能计算字符串的长度不含\0核心逻辑用指针遍历字符串直到遇到\0统计遍历的次数。代码解析my_strlen函数接收char*指针str用临时指针p遍历字符串每遍历一个字符len自增1直到*p \0时停止返回len字符串长度不含\0。该实现与标准库strlen函数的逻辑完全一致核心是通过指针遍历字符串。5.2 字符串复制函数my_strcpy功能将源字符串src复制到目标字符串dest包括\0核心逻辑用指针遍历源字符串将每个字符逐一复制到目标字符串的对应位置。#include stdio.h#include stdlib.h// 自定义strcpy函数指针实现将src复制到destchar *my_strcpy(char *dest, const char *src) {if (dest NULL || src NULL) { // 避免空指针printf(空指针异常\n);exit(1);}char *p dest; // 保存目标字符串的起始地址用于返回// 指针遍历复制每个字符直到src的\0while ((*dest *src) ! \0) {dest; // 目标指针自增src; // 源指针自增}return p; // 返回目标字符串的起始地址}int main() {char dest[20]; // 目标字符数组需足够大避免溢出char *src hello world;// 调用自定义my_strcpy函数my_strcpy(dest, src);printf(复制后的字符串%s\n, dest);// 复制动态分配的字符串char *dest2 (char *)malloc(20 * sizeof(char));if (dest2 NULL) {printf(内存申请失败\n);exit(1);}my_strcpy(dest2, c language);printf(动态复制后的字符串%s\n, dest2);// 释放动态内存free(dest2);dest2 NULL;return 0;}代码解析my_strcpy函数接收目标指针dest和源指针srcconst修饰避免修改源字符串用临时指针p保存dest的起始地址用于返回通过指针自增将src的每个字符逐一复制到dest直到复制到\0为止。需要注意dest的内存空间必须足够大否则会导致内存溢出。5.3 字符串比较函数my_strcmp功能比较两个字符串的大小按ASCII码顺序核心逻辑用指针遍历两个字符串逐一比较对应位置的字符直到遇到不同字符或\0。返回值规则与标准库strcmp一致两个字符串相等返回0str1 str2返回负数差值str1字符ASCII - str2字符ASCIIstr1 str2返回正数差值str1字符ASCII - str2字符ASCII。代码解析my_strcmp函数通过指针遍历两个字符串逐一比较对应位置的字符若遇到不同字符立即返回两个字符的ASCII差值若其中一个字符串先遍历到\0则返回两个字符串剩余字符的ASCII差值即长度差。该实现与标准库strcmp函数完全一致能准确判断两个字符串的大小关系。5.4 字符串拼接函数my_strcat功能将源字符串src拼接在目标字符串dest的末尾覆盖dest原来的\0并在拼接后的字符串末尾添加新的\0核心逻辑用指针找到dest的\0再将src的字符逐一复制到dest的\0之后。代码解析my_strcat函数分为两步首先通过指针遍历dest找到其末尾的\0然后将src的字符逐一复制到dest的\0之后直到复制到src的\0完成拼接。需要注意dest的内存空间必须足够大能容纳原dest和src的所有字符含两个\0最终只保留一个否则会导致内存溢出。指针是C语言操作字符串的核心工具其核心优势是灵活、高效能极大简化字符串的存储、访问和处理。本文核心知识点总结如下字符串常量的指针表示字符指针指向字符串常量的首地址字符串常量只读不能通过指针修改内容字符指针与字符数组的区别核心在于内存存储、可修改性和赋值方式字符指针灵活但需注意只读属性字符数组可修改但灵活性差字符串指针数组存储字符指针的数组仅存储字符串地址节省内存便于管理多个字符串字符串排序指针交换法不交换字符串内容仅交换指针地址效率高适合大量、长字符串排序字符串处理函数的指针实现strlen、strcpy、strcmp、strcat等函数的底层均基于指针遍历掌握其实现原理能深入理解指针与字符串的关联。掌握指针与字符串的关键是“理清指针的指向”和“理解字符串以\0结尾”的特性多写程序、多调试熟练运用指针操作字符串既能提升代码效率也能突破C语言字符串学习的难点。实际开发中需注意字符串的只读属性、内存溢出、空指针等问题避免程序异常。

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

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

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…