c语言内存函数

简介: memcpy函数,memcpy函数模拟实现,memmove函数,从后向前拷贝:从前向后拷贝:随意拷贝:memmove函数模拟实现,关于以上两个函数的总结: ,memset函数,memset函数模拟实现,memcmp函数

memcpy函数

函数原型:void * memcpy ( void * destination, const void * source, size_t num );

头文件: <string.h>

作用:从source的起始位置开始向后复制num个字节的数据到destination指向的内存空间

注意事项:

1、返回的是目标空间的起始地址

2、源内存空间大小 <= 目标内存空间大小

3、memcpy函数只能处理源内存空间与目标内存空间不重叠时的数据拷贝

(arr+2,arr,20)是想将12345拷贝至34567的内存空间,希望的输出结果是12123458910,但是实际结果为12121218910,这就是因为使用memcpy函数时出现了以下的情况:


当我们完成1和2与3和4的拷贝时,3和4的内存空间其实已经变成1和2了,所以当我们想把原来3和4位置上的数拷贝给5和6时其实是将1和2重新拷贝给了5和6,此时5和6的内存空间也变成了1和2,7和8的情况也是一样的,只不过num=20所以8侥幸逃过一劫。

memcpy函数模拟实现

#define _CRT_SECURE_NO_WARNINGS 1
#include<ctype.h>
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<errno.h>
my_memcpy(void* dest, const void* src, size_t num)
{
  void* ret = dest;
  assert(dest && src);
  while (num--)
  {
    *(char*)dest = *(char*)src;
    dest = (char*)dest + 1;
    src = (char*)src + 1;
  }
  return ret;
}
int main()
{
  int arr1[10] = { 0 };
  int arr2[] = { 1,2,3,4,5,6,7,8 };
  my_memcpy(arr1, arr2, 20);
  int i = 0;
  for (i = 0; i < 10; i++)
  {
    printf("%d ", arr1[i]);
  }
  return 0;
}

memmove函数

函数原型:void * memmove ( void * destination, const void * source, size_t num );

头文件: <string.h>

作用:从source的起始位置开始向后复制num个字节的数据到destination指向的内存空间

注意事项:

1、返回目标空间的起始地址

2、源内存空间大小 <= 目标内存空间大小

3、当两内存空间存在重叠时,拷贝顺序分为从前向后、从后向前拷贝和随意拷贝三种情况

从后向前拷贝:

具体流程:5->7、 4->6、 3->5、 2->4、 1->3

从前向后拷贝:

具体流程: 3->1、 4->2、 5->3、 6->4、 7->5

随意拷贝:


具体流程:7->8、6->9、5->10  或  3->8、4->9、5->10(大致意思如下,后面没画完)

针对上面的三种情况,我们可以得到memmove函数模拟实现要用到的判断条件:      



①if dest < src                                            //从前向后


②if dest >= src && dest <= src + num;     //从后向前


③if dest >src + num || dest + num < src;  //任意方向均可

memmove函数模拟实现

#include <stdio.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
void* my_memmove(void* dest, const void* src, size_t n) {
    unsigned char* d = (unsigned char*)dest;
    const unsigned char* s = (const unsigned char*)src;
    if (d == s) {
        return d;
    }
    if (s + n < d || d + n < s) {  // 没有内存重叠
        while (n--) {
            *(d++) = *(s++);
        }
    }
    else {  // 存在内存重叠
        if (d < s) {  // 目标地址在源地址之前,从前向后拷贝
            while (n--) {
                *(d++) = *(s++);
            }
        }
        else {  // 目标地址在源地址之后,从后向前拷贝
            d += n - 1;
            s += n - 1;
            while (n--) {
                *(d--) = *(s--);
            }
        }
    }
    return dest;
}
int main()
{
    int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
    //my_memmove(arr+2, arr, 20);
    my_memmove(arr, arr + 2, 20);
    int i = 0;
    for (i = 0; i < 10; i++)
    {
        printf("%d ", arr[i]);
    }
    return 0;
}


此时,当我们此时将模拟函数切换成memcpy函数我们会发现:



明显的,这次居然成功了!!!这就很奇怪了🙄,这是因为:


某些版本的VS标准库中的memcpy函数也能实现重叠内存拷贝,但还是不建议使用

关于以上两个函数的总结:

二者都是用于在内存之间进行数据拷贝的函数,但它们在处理重叠内存区域时有所不同:

对于memcpy函数:

  • 不会检查源地址和目标地址是否重叠
  • 当源和目标内存区域重叠时,可能会导致未定义行为,结果不可预测

对于memmove函数:

  • 能正确处理源和目标内存区域重叠情况。即使两个内存区域有部分或完全重叠
  • 在处理重叠情况时,使用了一种更加安全且效率较低的算法来执行拷贝操作

结论:

  1. 若确定两块内存空间不重叠,则可以使用 memcpy函数,它的执行速度更快
  2. 若无法确定两块内存空间是否重叠,或者确实存在重叠情况,应使用memmove函数来保证拷贝操作的正确性

在使用这些函数时,请确保目标内存区域足够大以容纳源数据,并进行适当的错误处理

memset函数

函数原型:void * memset ( void * ptr, int value, size_t num );

头文件: <string.h>

作用:将内存中的值以字节为单位设置成想要的内容

注意事项:

1、返回目标空间的起始地址。

2、value表示想要更改的内容,ptr表示要更改的目标空间,num表示要操作的个数

3、memset函数是以字节为单位设置内存值。

#define _CRT_SECURE_NO_WARNINGS 1
#include<ctype.h>
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<errno.h>
int main()
{
  char ch[] = "hello world";
  memset(ch, 'x', 5);
  printf("%s\n", ch);
  return 0;
}

memset函数模拟实现

这次没有模拟实现🤡,这次要利用监视和内存窗口来更好的理解,它在内存中的操作


#define _CRT_SECURE_NO_WARNINGS 1
#include<ctype.h>
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<errno.h>
int main()
{
  int arr[] = {1,2,3,4,5,6,7,8,9,10};
  memset(arr,0,20);
  return 0;
}

  运行开始前我们点击”调试“-”窗口“-”内存“,随机选取一个内存窗口,同时打开监视窗口,当程序运行至return 0时我们会发现内存中会发生改变。

每次分配的地址都会发生改变,所以每次查询首元素地址时直接输入数组名然后回车即可

内存中的变化:




监视中的变化:



~通过以上变化我们就可以更充分的认识“memset函数是以字节为单位设置内存值”这句话的意思~

memcmp函数

函数原型:int memcmp ( const void * ptr1, const void * ptr2, size_t num );

头文件: <string.h>

作用:从ptr1和ptr2指向的位置开始,向后比较num个字节(str1和str2分别指向两个内存块)

注意事项:

1、ptr1小于ptr2时,返回小于零的整数

2、ptr1等于prt2时,返回零

3、ptr1大于ptr2时,返回大于零的整数

#define _CRT_SECURE_NO_WARNINGS 1
#include<ctype.h>
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<errno.h>
int main()
{
  char arr1[] = {1,2,3,4,5};
  char arr2[] = { 1,3,5,7,9 };
  int ret = memcmp(arr1,arr2,5);
  printf("%d ", ret);
  return 0;
}


当我们展开八列的时候我们会发现,数组1的前五个字节为01 00 00 00 02,而数组2的前五个字节为01  00  00  00  03,所以数组一的内存块儿是小于数组二的内存块,返回结果为:


相关文章
|
2月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
121 26
|
2月前
|
安全 C语言
C语言中的字符、字符串及内存操作函数详细讲解
通过这些函数的正确使用,可以有效管理字符串和内存操作,它们是C语言编程中不可或缺的工具。
233 15
|
8月前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
367 23
|
7月前
|
人工智能 Java 程序员
一文彻底搞清楚C语言的函数
本文介绍C语言函数:函数是程序模块化的工具,由函数头和函数体组成,涵盖定义、调用、参数传递及声明等内容。值传递确保实参不受影响,函数声明增强代码可读性。君志所向,一往无前!
171 1
一文彻底搞清楚C语言的函数
|
8月前
|
C语言
【C语言程序设计——函数】亲密数判定(头歌实践教学平台习题)【合集】
本文介绍了通过编程实现打印3000以内的全部亲密数的任务。主要内容包括: 1. **任务描述**:实现函数打印3000以内的全部亲密数。 2. **相关知识**: - 循环控制和跳转语句(for、while循环,break、continue语句)的使用。 - 亲密数的概念及历史背景。 - 判断亲密数的方法:计算数A的因子和存于B,再计算B的因子和存于sum,最后比较sum与A是否相等。 3. **编程要求**:根据提示在指定区域内补充代码。 4. **测试说明**:平台对代码进行测试,预期输出如220和284是一组亲密数。 5. **通关代码**:提供了完整的C语言代码实现
142 24
|
8月前
|
存储 C语言
【C语言程序设计——函数】递归求斐波那契数列的前n项(头歌实践教学平台习题)【合集】
本关任务是编写递归函数求斐波那契数列的前n项。主要内容包括: 1. **递归的概念**:递归是一种函数直接或间接调用自身的编程技巧,通过“俄罗斯套娃”的方式解决问题。 2. **边界条件的确定**:边界条件是递归停止的条件,确保递归不会无限进行。例如,计算阶乘时,当n为0或1时返回1。 3. **循环控制与跳转语句**:介绍`for`、`while`循环及`break`、`continue`语句的使用方法。 编程要求是在右侧编辑器Begin--End之间补充代码,测试输入分别为3和5,预期输出为斐波那契数列的前几项。通关代码已给出,需确保正确实现递归逻辑并处理好边界条件,以避免栈溢出或结果
358 16
|
8月前
|
存储 编译器 C语言
【C语言程序设计——函数】分数数列求和2(头歌实践教学平台习题)【合集】
函数首部:按照 C 语言语法,函数的定义首部表明这是一个自定义函数,函数名为fun,它接收一个整型参数n,用于指定要求阶乘的那个数,并且函数的返回值类型为float(在实际中如果阶乘结果数值较大,用float可能会有精度损失,也可以考虑使用double等更合适的数据类型,这里以float为例)。例如:// 函数体代码将放在这里函数体内部变量定义:在函数体中,首先需要定义一些变量来辅助完成阶乘的计算。比如需要定义一个变量(通常为float或double类型,这里假设用float。
196 3
|
8月前
|
存储 算法 安全
【C语言程序设计——函数】分数数列求和1(头歌实践教学平台习题)【合集】
if 语句是最基础的形式,当条件为真时执行其内部的语句块;switch 语句则适用于针对一个表达式的多个固定值进行判断,根据表达式的值与各个 case 后的常量值匹配情况,执行相应 case 分支下的语句,直到遇到 break 语句跳出 switch 结构,若没有匹配值则执行 default 分支(可选)。例如,在判断一个数是否大于 10 的场景中,条件表达式为 “num> 10”,这里的 “num” 是程序中的变量,通过比较其值与 10 的大小关系来确定条件的真假。常量的值必须是唯一的,且在同一个。
167 2
|
8月前
|
存储 编译器 C语言
【C语言程序设计——函数】回文数判定(头歌实践教学平台习题)【合集】
算术运算于 C 语言仿若精密 “齿轮组”,驱动着数值处理流程。编写函数求区间[100,500]中所有的回文数,要求每行打印10个数。根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码。如果操作数是浮点数,在 C 语言中是不允许直接进行。的结果是 -1,因为 -7 除以 3 商为 -2,余数为 -1;注意:每一个数据输出格式为 printf("%4d", i);的结果是 1,因为 7 除以 -3 商为 -2,余数为 1。取余运算要求两个操作数必须是整数类型,包括。开始你的任务吧,祝你成功!
141 1