C语言常用内存函数的深度解析

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: C语言常用内存函数的深度解析

前言


  • 内存函数的使用广泛度大于常用字符串函数的使用广泛度,因为字符串函数只适用于与字符相关的,而内存函数适用于各个类型,因为他是从内存出发,对内存进行修改,因此,学会内存函数,可谓收获满满呀。
  • 这些内存函数的头文件是<string.h>


memcpy


  • 该函数的功能是内存拷贝,相当于字符串函数strncpy的功能,只不过memcpy的运用范围更宽。
  • 该函数是在内存中一对字节一对字节的拷贝。


该函数的函数参数:


913eb17fda7846f986853b09094f849a.png

922586fb9cf749b89dc473575f3afb0c.png


可以看到,对于重叠的拷贝,memcpy是做不到的(也就是一个数组arr[] = {1,2,3,4,5,6,7,8,9,10},1,2,3,4,5要拷贝到3,4,5,6,7上去,这样是不行的),此时应该用memmove,但是vs的memcpy超额完成了任务,也可以进行重叠拷贝。


memcpy函数的使用

#include <stdio.h>
#include <string.h>
int main()
{
  int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
  int arr2[] = { 9,9,9,9,9 };
  memcpy(arr1, arr2, sizeof(int) * (sizeof(arr2) / sizeof(arr2[0])));
  // sizeof(int) * (sizeof(arr2) / sizeof(arr2[0])) arr2的总字节个数;
  for (int i = 0; i < sizeof(arr1) / sizeof(arr1[0]); i++)
  {
    printf("%d ", arr1[i]);
  }
  return 0;
}



2592772e22ee43c29edce345e4054bb1.png

运行结果:9 9 9 9 9 6 7 8 9 10

memcpy对字符串也是一样:

#include <stdio.h>
#include <string.h>
int main()
{
  char arr1[] = "xxxxxxxxxx";
  char arr2[] = "abcdef";
  // 这里拷贝6个字节,注意最好拷贝的时候不要把\0弄没了,不然打印会出错
  printf("%s\n", (char*)memcpy(arr1, arr2, 6));
  // 因为返回的是 void* 所以最好强转一下 
  return 0;
}


运行结果:abcdefxxxx


memcpy函数的自我实现


代码实现的核心在于,如何一个字节一个字节的拷贝,如何拷贝完一个字节找到后一个字节。

#include <stdio.h>
#include <assert.h>
void* my_memcpy(void* dest, const void* src, size_t num)
{
  assert(dest && src);
  // 先保存目的地的开头地址,以便于返回
  void* ret = dest;
  while (num--) // 当num为0拷贝字节数已够,拷贝结束
  {
    // 强转成char*的指针,因为这样,才是一个字节一个字节的拷贝
    *(char*)dest = *(char*)src;
    // 强转成char*指针++向后走跳过一个字节,为下一次的字节拷贝做准备;
    ++(char*)dest;
    ++(char*)src;
  }
  return ret;
}
int main()
{
  int a1[] = { 1,2,3,4,5,6,7,8,9,10 };
  int a2[] = { 9,9,9,9,9 };
  my_memcpy(a1, a2, 20);
  // 20:a2的总字节数
  for (int i = 0; i < sizeof(a1) / sizeof(a1[0]); ++i)
  {
    printf("%d ", a1[i]);
  }
  return 0;
}


运行结果为:9 9 9 9 9 6 7 8 9 10

如果我们用自我实现得功能来进行重叠拷贝:


91782bbc7e81471bb0b7fe7f8601562c.png


运行结果为:


  • 为什么会这样呢?
  • 12拷贝过去,此时arr原有的第三个和第四个元素(34所在位置)也被改为了12,当拷贝第三个元素时,是将1拷贝过去而不是原先的3了。
  • 但如果是memmove就不会有这样的情况.


memmove

该函数的功能也相当于是“内存拷贝”,它包含了memcpy的功能,同时比memcpy函数更为的强大,它可以对重叠的内容进行拷贝,也就是一个数组arr[] = {1,2,3,4,5,6,7,8,9,10},1,2,3,4,5要拷贝到3,4,5,6,7上去,最终数组内容变为1,2,1,2,3,4,5,8,9,10



3be837bc345647eba4cb1f4681d11ea6.png

21eaee4f2c9347b1b0d75119ac7e6e46.png995b8d64a22f4a05b688417265c11ffc.png


memmove函数的使用

#include <stdio.h>
#include <string.h>
int main()
{
  int a1[] = { 1,2,3,4,5,6,7,8,9 };
  int a2[] = { 6,6,6,6,6 };
  memmove(a1, a2, 20);
  for (int i = 0; i < sizeof(a1) / sizeof(a1[0]); ++i)
  {
    printf("%d ", a1[i]);
  }
  return 0;
}


运行结果为:


8a8d3df23ba34fb8aed9e6ef6de975d0.png

  • 可以看到,这种拷贝的功能与memcpy是一样的,接下来进行不一样的重叠拷贝:
#include <stdio.h>
#include <string.h>
int main()
{
  int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
  memmove(arr + 2, arr, 20);
  // 将 1 2 3 4 5 拷贝到 3 4 5 6 7 上面去
  for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); ++i)
  {
    printf("%d ", arr[i]);
  }
  return 0;
}



cde3d751d76b4637907a1eb93b7ac63d.png


把arr + 2 当作dest,arr当作src,也就是说此时src要小于dest,即得出当src < dest时由后往前拷贝;


这里是memmove(arr + 2, arr, 20),如果是memmove(arr, arr + 2, 20)(将3,4,5,6,7拷贝到1,2,3,4,5上面去)呢?此时arr + 2为src,arr为dest,这时候就需要从前向后拷贝了,也就是src > dest的情况,即得出当src > dest时由前向后拷贝。


还有一种情况:memmove(arr, arr + 5, 20)是把6,7,8,9,10拷贝到1,2,3,4,5上去,这时虽然src大于dest,但是src与dest的字节差值大于等于了num,也就是说无论是从前向后拷贝还是从后向前拷贝都是一样的,此时的功能就可以理解为是memcpy的功能了。


memmove函数的自我实现


  • 通过上面的使用和分析,有三种情况都是要考虑到:
    1.从前向后拷贝src > dest
  • 2.从后向前拷贝src < dest
  • 3.从后向前还是从前向后都是可以的sizeof(type) * |(src - dest)| >= num


9965709913e5422aab5c74dc91e50e74.png


代码实现:

#include <stdio.h>
#include <assert.h>
void* my_memmove(void* dest, const void* src, size_t num)
{
  assert(dest && src);
  void* ret = dest;
  // 落在第三区间总会进一种拷贝方式
  if (src < dest)
  {
    // 从后往前
    while (num--) // 因为判断后 num-- 一次,第一次进来如果num开始为20,
    {             // 进来后等于19找到最后一个字节拷贝
      // 随着num--,拷贝的字节从后往前依次跳一个字节
      *((char*)dest + num) = *((char*)src + num);
    }
  }
  else
  {
    // 从前往后
    while (num--)
    {
      // 这里跟上面的memcpy的自我实现差不多,但是memcpy也可以以从后往前的方式实现
      *(char*)dest = *(char*)src;
      ++(char*)dest;
      ++(char*)src;
    }
  }
  return ret;
}
int main()
{
  int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
  my_memmove(arr + 2, arr, 20);
  for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); ++i)
  {
    printf("%d ", arr[i]);
  }
  return 0;
}



运行结果为:1 2 1 2 3 4 5 8 9 10

实现了重叠拷贝,当然其它案例也行,这里大家自行测试了。


memcmp

  • 该函数是比较函数,与strncmp的功能相同,只不过memcmp的对比类型不单只是字符类型了。
  • memcmp是一个字节一个字节的对比。

函数参数如下:


e65d8b6122924e398f5e8fd7d9c1dbaa.png

3ddecc57e2474ec382757df8754bc26f.png


06803a926a2449d88d57569cf1dec5f1.png



函数的返回值:如果前num个字节ptr1ptr2都相等,则返回0,如果找到第一个不相等的字节,返回ptr1的这个字节减去ptr2的这个字节的差值。


memcmp函数的使用

1.

#include <stdio.h>
#include <string.h>
int main()
{
  int a1[] = { 1,2,3,4,5,6,7,8 };
  int a2[] = { 1,2 };
  printf("%d\n", memcmp(a1, a2, 8));
  // 8个字节,比较前两个元素
  return 0;
}



f4268b69e64c4d188dd25e3d123c3381.png


2.

#include <stdio.h>
#include <string.h>
int main()
{
  int a1[] = { 1,2,3,4,5,6,7,8 };
  int a2[] = { 1,2,5 };
  printf("%d\n", memcmp(a1, a2, 12));
  // 12个字节,比较前三个元素
  return 0;
}


9a727ed097ee4b3b927af5829196a6b5.png


由于vsmemcmp标准是小于就返回-1,大于就返回1,所以这里的结果为-1

memcmp函数的自我实现

经过上面的介绍,我们大概知道了这个函数的功能,接下来自我实现这个函数。

#include <stdio.h>
#include <assert.h>
int my_memcmp(const void* ptr1, const void* ptr2, size_t num)
{
  assert(ptr1 && ptr2);
    // 比对次数--
  while (num--)
  {
      // 如果有不相同的字节,这里为真,进去返回差值
    if (*(char*)ptr1 - *(char*)ptr2)
      return *(char*)ptr1 - *(char*)ptr2;
    ++(char*)ptr1;
    ++(char*)ptr2;
  }
    // 比对完了没有找到不相同的字节说明都相同,返回0;
  return 0;
}
int main()
{
  int a1[] = { 1,2,3,4,5,6,7,8 };
  int a2[] = { 1,2,5 };
  printf("%d\n", my_memcmp(a1, a2, 8));
  // 12个字节,比较前三个元素
  return 0;
}


运行结果为:-2

memset

该函数的功能是填充内存块,也就是将一个数组里面你指定的内容以修改字节的形式修改成你想要的数据。


0fcd25abea9d4cee95a336a2b893ea92.png


f67948c7e76e43c89b302e61b8b69420.png


这里的value虽然是int,但是字符也可以修改,因为字符类型本身就是整型家族,字符可以通过ASCLL码值来进行转换。

memset函数的使用


#include <stdio.h>
#include <string.h>
int main()
{
  char arr[] = "xxxxxxxxxx";
  printf("%s\n", (char*)memset(arr, 'y', 5));
  return 0;
}


e72a7f9b05584d55bf82f947cf58a97f.png

2.

#include <stdio.h>
#include <string.h>
int main()
{
  int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
  printf("%s\n", (char*)memset(arr, 1, 40));
  // 40表示整个arr的字节个数
  return 0;
}

运行结果为:

16843009 16843009 16843009 16843009 16843009 16843009 16843009 16843009 16843009 16843009

为什么运行结果会是这样子呢?


我们将数组arr40个字节都改为1,一个整型有4个字节,如果每一个字节上面都是1,在内存当中每个整型的存放是这样的:01 01 01 01,所以当我们从内存中将这个数据读取出来的时候,那将会是一个很大的数,就如上面的运行结果一样的。

所以对于整型数组我们一定要避免出现这样的情况,要牢记这是一个字节一个字节的修改。


memset函数的自我实现


通过上面的认识,下面自我实现memset函数:

#include <stdio.h>
#include <assert.h>
void* my_memset(void* ptr, int value, size_t num)
{
  assert(ptr);
  void* ret = ptr;
  while (num--)
  {
    *(char*)ptr = value;
    ++(char*)ptr;
  }
  return ret;
}
int main()
{
  char a[] = "abcdefgh";
  printf("%s\n", (char*)my_memset(a, '@', 5));
  return 0;
}



运行结果为:@@@@@fgh


写在最后


在C语言中,熟练的使用内存函数可以说对内存的理解也是很不错的,能够以内存的视角观察代码,说明你的水平已经很不错了。


感谢阅读本小白的博客,错误的地方请严厉指出噢!

相关文章
|
1月前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
62 23
|
1月前
|
算法 C语言
【C语言程序设计——函数】利用函数求解最大公约数和最小公倍数(头歌实践教学平台习题)【合集】
本文档介绍了如何编写两个子函数,分别求任意两个整数的最大公约数和最小公倍数。内容涵盖循环控制与跳转语句的使用、最大公约数的求法(包括辗转相除法和更相减损术),以及基于最大公约数求最小公倍数的方法。通过示例代码和测试说明,帮助读者理解和实现相关算法。最终提供了完整的通关代码及测试结果,确保编程任务的成功完成。
66 15
|
1月前
|
C语言
【C语言程序设计——函数】亲密数判定(头歌实践教学平台习题)【合集】
本文介绍了通过编程实现打印3000以内的全部亲密数的任务。主要内容包括: 1. **任务描述**:实现函数打印3000以内的全部亲密数。 2. **相关知识**: - 循环控制和跳转语句(for、while循环,break、continue语句)的使用。 - 亲密数的概念及历史背景。 - 判断亲密数的方法:计算数A的因子和存于B,再计算B的因子和存于sum,最后比较sum与A是否相等。 3. **编程要求**:根据提示在指定区域内补充代码。 4. **测试说明**:平台对代码进行测试,预期输出如220和284是一组亲密数。 5. **通关代码**:提供了完整的C语言代码实现
60 24
|
1月前
|
存储 C语言
【C语言程序设计——函数】递归求斐波那契数列的前n项(头歌实践教学平台习题)【合集】
本关任务是编写递归函数求斐波那契数列的前n项。主要内容包括: 1. **递归的概念**:递归是一种函数直接或间接调用自身的编程技巧,通过“俄罗斯套娃”的方式解决问题。 2. **边界条件的确定**:边界条件是递归停止的条件,确保递归不会无限进行。例如,计算阶乘时,当n为0或1时返回1。 3. **循环控制与跳转语句**:介绍`for`、`while`循环及`break`、`continue`语句的使用方法。 编程要求是在右侧编辑器Begin--End之间补充代码,测试输入分别为3和5,预期输出为斐波那契数列的前几项。通关代码已给出,需确保正确实现递归逻辑并处理好边界条件,以避免栈溢出或结果
63 16
|
1月前
|
存储 编译器 C语言
【C语言程序设计——函数】分数数列求和2(头歌实践教学平台习题)【合集】
函数首部:按照 C 语言语法,函数的定义首部表明这是一个自定义函数,函数名为fun,它接收一个整型参数n,用于指定要求阶乘的那个数,并且函数的返回值类型为float(在实际中如果阶乘结果数值较大,用float可能会有精度损失,也可以考虑使用double等更合适的数据类型,这里以float为例)。例如:// 函数体代码将放在这里函数体内部变量定义:在函数体中,首先需要定义一些变量来辅助完成阶乘的计算。比如需要定义一个变量(通常为float或double类型,这里假设用float。
36 3
|
1月前
|
存储 算法 安全
【C语言程序设计——函数】分数数列求和1(头歌实践教学平台习题)【合集】
if 语句是最基础的形式,当条件为真时执行其内部的语句块;switch 语句则适用于针对一个表达式的多个固定值进行判断,根据表达式的值与各个 case 后的常量值匹配情况,执行相应 case 分支下的语句,直到遇到 break 语句跳出 switch 结构,若没有匹配值则执行 default 分支(可选)。例如,在判断一个数是否大于 10 的场景中,条件表达式为 “num> 10”,这里的 “num” 是程序中的变量,通过比较其值与 10 的大小关系来确定条件的真假。常量的值必须是唯一的,且在同一个。
19 2
|
1月前
|
存储 编译器 C语言
【C语言程序设计——函数】回文数判定(头歌实践教学平台习题)【合集】
算术运算于 C 语言仿若精密 “齿轮组”,驱动着数值处理流程。编写函数求区间[100,500]中所有的回文数,要求每行打印10个数。根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码。如果操作数是浮点数,在 C 语言中是不允许直接进行。的结果是 -1,因为 -7 除以 3 商为 -2,余数为 -1;注意:每一个数据输出格式为 printf("%4d", i);的结果是 1,因为 7 除以 -3 商为 -2,余数为 1。取余运算要求两个操作数必须是整数类型,包括。开始你的任务吧,祝你成功!
52 1
|
2月前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
92 10
|
2月前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
68 9
|
2月前
|
存储 Unix Serverless
【C语言】常用函数汇总表
本文总结了C语言中常用的函数,涵盖输入/输出、字符串操作、内存管理、数学运算、时间处理、文件操作及布尔类型等多个方面。每类函数均以表格形式列出其功能和使用示例,便于快速查阅和学习。通过综合示例代码,展示了这些函数的实际应用,帮助读者更好地理解和掌握C语言的基本功能和标准库函数的使用方法。感谢阅读,希望对你有所帮助!
62 8

推荐镜像

更多