【C语言航路】第十一站:字符串、字符和内存函数(上)

简介: 【C语言航路】第十一站:字符串、字符和内存函数

一、字符串函数

1.strlen

(1)strlen的库函数文档

如下图所示,是strlen在库函数中的文档。

它的参数是const char*类型的,这是考虑到它不会被改变的原因

返回类型是size_t类型的,这是考虑到它计算的是长度不会是负数的原因

这个函数返回的是一个字符串的长度,传入一个地址,计算的是\0字符之前的长度

(2)strlen的模拟实现

在这里我们采用三种方式来实现:循环计数、递归、指针减指针

#include<stdio.h>
#include<assert.h>
//循环计数
int my_strlen1(const char* str)
{
  assert(str);
  int count = 0;
  while (*str != '\0')
  {
    count++;
    str++;
  }
  return count;
}
//递归
int my_strlen2(const char* str)
{
  assert(str);
  if (*str == NULL)
  {
    return 0;
  }
  return 1 + my_strlen2(str + 1);
}
//指针减指针
int my_strlen3(const char* str)
{
  assert(str);
  const char* start = str;
  while (*str != '\0')
  {
    str++;
  }
  return str - start;
}
int main()
{
  char arr[] = "abcdef";
  int len = my_strlen3(arr);
  printf("%d\n", len);
  return 0;
}

(3)strlen的注意事项

1.字符串已经 '\0' 作为结束标志,strlen函数返回的是在字符串中 '\0' 前面出现的字符个数(不包含 '\0' )。

2.参数指向的字符串必须要以 '\0' 结束

3.注意函数的返回值为size_t,是无符号的

关于第三点注意事项,我们在这里着重说一下

#include<stdio.h>
#include<string.h>
int main()
{
  if (strlen("abc") - strlen("abcdef") > 0)
  {
    printf(">\n");
  }
  else
  {
    printf("<=\n");
  }
  return 0;
}

这段代码的运行结果是

肯定与我们的设想是不一样的,因为strlen("abc")计算出来的是3,strlen("abcdef")计算出来的是6,虽然看上去好像3-6是-3,应该是<=,但是要主要这里的都是无符号的,计算出来的结果也是无符号的。所以这个数其实在内存中来看是一个很大的数。还是大于0的,所以是>

2.strcpy

(1)strcpy的库函数文档

它的参数是char* destination 和const char* source,destination的意思是目标空间的地址,

source的意思是源头空间的地址,这个地址里面的值是不可以被修改的。

功能是将source处的字符串拷贝到destination处

返回类型是char*,意思是将destination处的地址返回

(2)strcpy的使用以及注意事项

1.源字符串必须以 '\0' 结束。

2.会将源字符串中的 '\0' 拷贝到目标空间。

3.目标空间必须足够大,以确保能存放源字符串。

4.目标空间必须可变。

在下面这段代码中

#include<stdio.h>
#include<string.h>
int main()
{
  char arr1[] = "xxxxxxxxxxxxxxxxxx";
  char arr2[] = "hello world";
  strcpy(arr1, arr2);
  printf("%s\n", arr1);
  return 0;
}

我们直接从调试里面查看数组的值

在下面这段代码中

#include<stdio.h>
#include<string.h>
int main()
{
  char arr1[] = "xxxxxxxxxxxxxxxxxx";
  char arr2[] = "hello world";
  strcpy(arr2, arr1);
  printf("%s\n", arr1);
  return 0;
}

出现了错误,这是因为,arr2的空间不是足够大。

在下面这段代码中

#include<stdio.h>
#include<string.h>
int main()
{
  char* p = "xxxxxxxxxxxxxxxxxx";
  char arr2[] = "hello world";
  strcpy(p, arr2);
  printf("%s\n", p);
  return 0;
}

程序直接崩溃,这是因为char* p是不可以被修改的

(3)strcpy的模拟实现

#include<stdio.h>
#include<assert.h>
char* my_strcpy(char* dest, const char* src)
{
  assert(dest && src);
  char* ret = dest;
  while (*dest++ = *src++)
  {
    ;
  }
  return ret;
}
int main()
{
  char arr1[] = "xxxxxxxxxxxxx";
  char arr2[] = "hello world";
  char* p = my_strcpy(arr1, arr2);
  printf("%s\n", p);
  return 0;
}

运行结果为

3.strcat

(1)strcat的库函数文档

1.它的参数是char* destination和const char* source,与strcpy是一样的。

2.它的功能是将source处的字符串追加到destination后面

3.它的返回类型是char*,返回destination的地址

(2)strcat的使用以及注意事项

1.源字符串必须以 '\0' 结束。

2.目标空间必须有足够的大,能容纳下源字符串的内容。

3.目标空间必须可修改。

4.字符串自己给自己追加,会陷入死循环

#include<stdio.h>
#include<string.h>
int main()
{
  char arr[20] = "hello ";
  char* p = strcat(arr, "world");
  printf("%s\n", arr);
  return 0;
}

运行结果为

(3)strcat的模拟实现

思路就两步:

1.找到\0字符的位置

2.追加

#include<stdio.h>
#include<assert.h>
char* my_strcat(char* dest, const char* src)
{
  assert(dest && src);
  char* ret = dest;
  //1.找到'\0'字符
  while (*dest != '\0')
  {
    dest++;
  }
  //2.追加
  while (*dest++ = *src++)
  {
    ;
  }
  return ret;
}
int main()
{
  char arr[20] = "hello ";
  char* p = my_strcat(arr, "world");
  printf("%s\n", arr);
  return 0;
}

(4)strcat自己对自己追加

这个其实是不可以的,会造成死循环的现象

假如说下面的字符串自己对自己追加的话

\0会被修改为a,后面的都会覆盖过去,这样我们就永远也找不到\0了,最终导致分配的空间用完, 使用了未分配的空间而崩溃

下图是我们模拟出来的strcat的自己对自己追加后的效果

如果是官方的库里面的话,似乎没有出现这个问题,这个是visual studio对这个问题进行了优化,在一些其他的ide上可能就会出现这个死循环问题,导致程序出现bug

相关文章
|
3月前
|
存储 编译器 程序员
【C语言】内存布局大揭秘 ! -《堆、栈和你从未听说过的内存角落》
在C语言中,内存布局是程序运行时非常重要的概念。内存布局直接影响程序的性能、稳定性和安全性。理解C程序的内存布局,有助于编写更高效和可靠的代码。本文将详细介绍C程序的内存布局,包括代码段、数据段、堆、栈等部分,并提供相关的示例和应用。
78 5
【C语言】内存布局大揭秘 ! -《堆、栈和你从未听说过的内存角落》
|
3月前
|
存储 算法 C语言
【C语言】字符常量详解
字符常量是C语言中处理字符数据的重要工具。通过单引号括起一个字符,我们可以方便地使用字符常量进行字符判断、字符运算和字符串处理等操作。理解字符常量的表示方法、使用场景和ASCII码对应关系,对于编写高效的C语言程序至关重要。
229 11
|
3月前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
102 10
|
3月前
|
存储 C语言 开发者
【C语言】格式化输出占位符及其标志字符详解(基于ISO/IEC 9899:2024)
在C语言中,格式化输出通过 `printf` 函数等格式化输出函数来实现。格式说明符(占位符)定义了数据的输出方式,标准ISO/IEC 9899:2024(C23)对这些格式说明符进行了详细规定。本文将详细讲解格式说明符的组成部分,包括标志字符、宽度、精度、长度修饰符和类型字符,并适当增加表格说明。
80 6
|
3月前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
126 6
|
4月前
|
传感器 人工智能 物联网
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发,以及面临的挑战和未来趋势,旨在帮助读者深入了解并掌握这些关键技术。
94 6
|
4月前
|
存储 算法 C语言
C语言中常见的字符串处理技巧,包括字符串的定义、初始化、输入输出、长度计算、比较、查找与替换、拼接、截取、转换、遍历及注意事项
本文深入探讨了C语言中常见的字符串处理技巧,包括字符串的定义、初始化、输入输出、长度计算、比较、查找与替换、拼接、截取、转换、遍历及注意事项,并通过案例分析展示了实际应用,旨在帮助读者提高编程效率和代码质量。
198 4
|
4月前
|
存储 算法 程序员
C 语言指针详解 —— 内存操控的魔法棒
《C 语言指针详解》深入浅出地讲解了指针的概念、使用方法及其在内存操作中的重要作用,被誉为程序员手中的“内存操控魔法棒”。本书适合C语言初学者及希望深化理解指针机制的开发者阅读。
|
4月前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
102 1