字符函数 和 字符串函数超全详解(后续会持续优化)
目录字符分类函数字符转换函数strlen 的使用和模拟实现strcpy /strncpy 的使用和模拟实现strcat /strncat 的使用和模拟实现strcmp /strncmp 的使用和模拟实现strstr 的使用和模拟实现strtok 函数的使用strerror 函数的使用一、字符分类函数核心说明头文件ctype.h规则判断成立返回非 0 整数不成立返回0常用函数速查表函数名功能描述islower(c)判断是否为小写字母a-z 例 islower(a)isupper(c)判断是否为大写字母A-Zisalpha(c)判断是否为字母a-z/A-Zisdigit(c)判断是否为数字字符0-9isspace(c)判断是否为空白字符空格 / 换行 / 制表符isalnum(c)判断是否为字母或数字实战示例#include stdio.h #include ctype.h int main() { char ch 5; // 数字判断 if (isdigit(ch)) { printf(%c 是数字\n, ch); } // 字母判断 printf(A 是否为字母%d\n, isalpha(A)); // 非0成立 return 0; }二、字符转换函数核心函数头文件ctype.h功能大小写字母转换非字母字符返回原字符表格函数名功能描述tolower(int c)大写字母 → 小写字母toupper(int c)小写字母 → 大写字母优雅实现示例#include stdio.h #include ctype.h int main() { char str[] Test String!123; int i 0; while (str[i]) { // 小写转大写 if (islower(str[i])) { str[i] toupper(str[i]); } putchar(str[i]); i; } // 输出TEST STRING!123 return 0; }三、strlen 的使用和模拟实现1. 函数基础原型size_t strlen(const char *str);头文件string.h核心计算字符串有效长度不包含结束符\0关键坑点返回值size_t是无符号整数相减恒≥0因为无符号数相减负数会变成一个巨大的正数。size_t是无符号整数类型永远 ≥ 0不可能是负数2. 经典坑题解析#include stdio.h #include string.h int main() { const char* str1 abcdef; // 长度6 const char* str2 bbb; // 长度3 // 无符号数 3-6 4294967293正数条件成立 if (strlen(str2) - strlen(str1) 0) { printf(str2 str1\n); // 实际执行此句 } return 0; }正确写法if (strlen(str2) strlen(str1))直接比较不要相减。3. 三种模拟实现面试必考方式 1计数器法推荐#include assert.h int my_strlen(const char *str) { assert(str ! NULL); // 断言指针非空避免非法访问 int count 0; while (*str ! \0) { count; str; } return count; }方式 2递归法无临时变量int my_strlen(const char *str) { assert(str); if (*str \0) return 0; return 1 my_strlen(str 1); }方式 3指针 - 指针法指针运算int my_strlen(char *str) { assert(str); char *start str; while (*str) str; return str - start; // 指针相减 元素个数 }四、strcpy /strncpy 的使用和模拟实现字符串拷贝函数1. strcpy基础拷贝原型char *strcpy(char *dest,const char *src);目标----源头核心将源字符串含\0拷贝到目标空间安全要求源串必须以\0结束目标空间足够大、可修改不可是常量字符串存在缓冲区溢出风险2. strncpy安全拷贝原型char *strncpy(char *dest, const char *src, size_t num);核心只拷贝前 num 个字符特殊规则若源串长度 num剩余位置自动补\0若源串长度 ≥ num不自动补\0需手动处理3. 模拟实现my_strcpy#include assert.h char *my_strcpy(char *dest, const char *src) { assert(dest src); // 双指针非空校验 char *ret dest; // 保存目标起始地址 // 拷贝字符遇\0自动停止 while ((*dest *src)) { ; } return ret; }my_strncpychar *my_strncpy(char *dest, const char *src, size_t num) { assert(dest src); char *ret dest; size_t i 0; // 拷贝前num个字符遇\0提前终止 for (i 0; i num src[i] ! \0; i) { dest[i] src[i]; } // 不足num位补\0 for (; i num; i) { dest[i] \0; } return ret; }【五、strcat /strncat 的使用和模拟实现字符串追加函数1. strcat基础追加原型char *strcat(char *dest, const char *src);核心将源字符串含\0追加到目标串末尾核心规则目标串必须有\0作为追加起始点源串必须有\0目标空间足够大可容纳追加后内容禁止自追加覆盖\0导致死循环2. strncat安全追加原型char *strncat(char *dest, const char *src, size_t num);核心追加前 num 个字符自动补\0优势无自追加风险工程开发优先选用3. 模拟实现my_strcat#include assert.h char *my_strcat(char *dest, const char *src) { assert(dest src); char *ret dest; // 1. 找到目标串末尾\0 while (*dest) { dest; } // 2. 追加拷贝同strcpy逻辑 while ((*dest *src)) { ; } return ret; }my_strncatchar *my_strncat(char *dest, const char *src, size_t num) { assert(dest src); char *ret dest; // 找到目标串末尾 while (*dest) dest; // 追加num个字符 size_t i 0; for (i 0; i num src[i] ! \0; i) { dest[i] src[i]; } // 强制补\0保证字符串合法 dest[i] \0; return ret; }六、strcmp /strncmp 的使用和模拟实现字符串比较函数1. strcmp完整比较原型int strcmp(const char *str1, const char *str2);两个字符串的对应位置上的字符ASCII 码值进行比较比较规则按 ASCII 码逐字符对比str1 str2返回0的整数str1 str2返回0str1 str2返回0的整数2. strncmp限制长度比较原型int strncmp(const char *str1, const char *str2, size_t num);核心只比较前 num 个字符规则同 strcmp优势更可控避免无意义的全量比较3. 模拟实现my_strcmp#include assert.h int my_strcmp(const char *str1, const char *str2) { assert(str1 str2); // 逐字符比较遇不同字符或\0停止 while (*str1 *str2) { if (*str1 \0) return 0; // 同时到\0相等 str1; str2; } // 返回字符ASCII差值 return *str1 - *str2; }my_strncmpint my_strncmp(const char *str1, const char *str2, size_t num) { assert(str1 str2); size_t i 0; // 比较前num个字符 for (i 0; i num; i) { if (str1[i] ! str2[i]) { return str1[i] - str2[i]; } if (str1[i] \0) return 0; // 提前到\0相等 } return 0; // 前num个字符完全一致 }七、strstr 的使用和模拟实现在一个字符串中找另外一个字符串 找子字符串1. 函数基础原型char *strstr(const char *str1, const char *str2);功能在str1中查找str2第一次出现的位置返回值找到返回子串起始地址未找到返回NULL面试高频考点必须手写模拟实现2. 实战示例#include stdio.h #include string.h int main() { char str[] This is a simple string; // 查找子串simple char *pch strstr(str, simple); if (pch ! NULL) { printf(找到子串%s\n, pch); } return 0; } // 输出找到子串simple string3. 模拟实现BF 算法暴力匹配#include assert.h char *my_strstr(const char *str1, const char *str2) { assert(str1 str2); // 子串为空默认返回原串起始地址 if (*str2 \0) return (char *)str1; char *cp (char *)str1; // 记录当前匹配起始位置 char *s1, *s2; while (*cp) { s1 cp; s2 (char *)str2; // 逐字符匹配 while (*s1 *s2 *s1 *s2) { s1; s2; } // 子串匹配完成返回起始位置 if (*s2 \0) return cp; cp; // 匹配失败向后偏移一位 } return NULL; // 未找到子串 }八、strtok 函数的使用1. 函数基础原型char *strtok(char *str, const char *sep);头文件string.h功能按指定分隔符切割字符串核心规则sep分隔符集合如. 表示点和空格会修改原字符串使用前建议拷贝副本第一次调用传入待切割字符串地址后续调用传入NULL函数内部保存静态指针记录切割位置切割完成返回NULL2. 实战示例切割 IP 地址#include stdio.h #include string.h int main() { char ip_arr[] 192.168.6.111; // 必须是可修改的字符数组 const char *sep .; char *token NULL; // 循环切割 for (token strtok(ip_arr, sep); token ! NULL; token strtok(NULL, sep)) { printf(%s\n, token); } return 0; } /* 输出 192 168 6 111 */九、strerror 函数的使用1. 函数基础原型char *strerror(int errnum);头文件errno.hstring.h功能将系统错误码errno转换为人类可读的错误信息配套函数perror直接打印错误信息更简洁2. 实战示例文件操作错误处理#include stdio.h #include errno.h #include string.h int main() { // 打开不存在的文件 FILE *pf fopen(unexist_file.txt, r); if (pf NULL) { // 方式1strerror errno printf(错误信息%s\n, strerror(errno)); // 方式2perror直接打印带自定义前缀 perror(打开文件失败); return 1; } fclose(pf); return 0; } /* 输出 错误信息No such file or directory 打开文件失败: No such file or directory */✨ 精辟总结本文系统整合了 C 语言字符操作与字符串核心函数将功能相似的基础 / 安全版本合并讲解覆盖使用规则、避坑要点、模拟实现三大核心字符分类 / 转换函数ctype.h快速判断字符类型、实现大小写转换是字符处理的基础工具基础字符串函数strlen/strcpy/strcat/strcmp入门必备但无长度限制存在溢出风险需谨慎使用安全字符串函数strncpy/strncat/strncmp可控性强、兼容性好是工程开发的优先选择工具类函数strstr/strtok/strerror分别解决子串查找、字符串切割、系统错误解析场景覆盖实战刚需。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2513369.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!