c语言回顾-内存操作函数

简介: c语言回顾-内存操作函数

前言

在动态内存的章节中小编详细讲解了动态内存分配函数,在这里小编将补充内存操作函数的讲解!

1.memcpy 函数

1.1函数介绍

原型:void *memcpy(void *dest, const void *src, size_t n);

功能:将 n 字节的数据从源内存区域 src 复制到目标内存区域 dest

  • 参数:
  • dest:指向目标内存区域的指针。
  • src:指向源内存区域的指针。
  • n:要复制的字节数。
  • 返回值:返回指向目标内存区域的指针。

char src[10] = "Hello";
char dest[10];
memcpy(dest, src, 5); // 将前5个字节从 src 复制到 dest

又比如将数组中的指定的数据放到另一个数组中

#include <stdio.h>
//#include <stdlib.h>
//#include <assert.h>
#include <string.h>
int main()
{
int arr1[10]={1,2,3,4,5,6,7,8,9,10};
int arr2[10]={0};
memcpy(arr2,arr1,20);
for(int i=0;i<10;i++)
{
printf("%d ",arr2[i]);
}
return 0;
}

输出结果:1 2 3 4 5 0 0 0 0 0

1.2与strcpy的区别

  1. 数据类型
  • memcpy:适用于任意类型的数据,复制的是指定的字节数。
  • strcpy:仅适用于字符串,复制的是字符,直到遇到空字符为止。
  1. 复制内容
  • memcpy:不会检查内容,只复制指定字节数,即使其中包含空字符。
  • strcpy:遇到空字符时停止复制,并将空字符包含在内。
  1. 使用场景
  • memcpy:适用于需要复制特定大小内存块的场景,例如结构体、数组等。
  • strcpy:适用于字符串复制的场景,确保字符串以空字符结尾。
  1. 安全性
  • memcpy:需要手动管理字符串结尾,容易出现内存越界问题。
  • strcpy:自动处理字符串结尾,但在源字符串未包含空字符时容易造成缓冲区溢出。

注意:如果source和destination有任何的重叠,复制的结果都是未定义的。但是vs,vscode 等运行出来的结果却是对的,或许提升了性能。

1.3memcpy的模拟

因为我们不知道所要复制的数据类型是什么,所以参数类型我们设置为void*,接受各种类型,

用因为void*不能解应用,所以强制转化类型操作,强制转化为char*

#include <stdio.h>
void*my_memcpy(void*dest,const void*src,size_t count)
{
    void*ret=dest;
    while(count--)
    {
        *(char*)dest=*(char*)src;//强制转化类型
        dest=(char*)dest+1;
        src=(char*)src+1;
    }
    return ret;
}
int main()
{
int arr1[10]={0};
int arr2[10]={1,2,3,4,5,6,7,8,9,10};
my_memcpy(arr1, arr2,20); //5个整型复制占20个字节
for(int i=0;i<10;i++)
{
    printf("%d ",arr1[i]);
}
return 0;
}

运行结果:

在 my_memcpy 函数中使用

*(char*)dest = *(char*)src ;dest = (char*)dest + 1;src = (char*)src + 1

是因为我们希望逐字节复制数据。使用 char 类型指针可以确保每次操作都是一个字节,这对于实现 memcpy 函数的逐字节复制行为是必要的。

为什么使用 `char`

逐字节复制:char 类型指针每次移动一个字节,这是我们想要的行为,因为 memcpy 需要逐字节复制数据。

通用性:char类型适用于任何类型的数据,无论是整数、浮点数还是结构体,因为它操作的是最小单位的字节。

如果使用 int或double其他类型指针,则会面临对齐问题和类型限制,无法实现 memcpy函数的通用性。

2.memmove 函数

2.1函数介绍和使用

void * memmove ( void * destination, const void * source, size_t num );

和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。

如果源空间和目标空间出现重叠,就得使用memmove函数处理。

使用:


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

运行结果:

2.2函数的模拟


#include <stdio.h>
#include <string.h>
void *my_memmove(void *dest, const void *src, size_t n)
{
    char *csrc = (char *)src;
    char *cdest = (char *)dest;
    if(csrc<cdest)
    {
        for(int i=n-1;i>=0;i--)
        {
            cdest[i]=csrc[i];
        }
    }
    else
    {
        for(int i=0;i<n;i++)
        {
            cdest[i]=csrc[i];
        }
    }
    return dest;
}
int main()
{
int arr1[10]={1,2,3,4,5,6,7,8,9,10};
my_memmove(arr1, arr1+4,20); 
for(int i=0;i<10;i++)
{
    printf("%d ",arr1[i]);
}
return 0;
}

my_memove(arr1+4,arr1,20)时

画图解释:

3.memset函数

3.1函数介绍

void * memset ( void * ptr, int value, size_t num );

memset是用来设置内存的,将内存中的值以字节为单位设置成想要的内容。 函数用于将指定的值设置到一段内存区域中。它在初始化数组或结构体时非常有用。


#include <stdio.h>
#include <string.h>
int main ()
{
 char str[] = "hello world";
 memset (str+2,'x',6);
 printf(str);
 return 0;
}

输出结果:hexxxxxxrld

3.2函数的模拟


#include <stdio.h>
#include <string.h>
void*my_memset(void*dest, int c, size_t count)
{
    char*str = (char*)dest;
    while(count--)
    {
        *str++ = (char)c;
    }
    return dest;
}
int main ()
{
 char str[] = "hello world";
 my_memset (str+2,'x',6);
 printf("%s\n", str);
 int arr[10]={1,2,3,4,5,6,7,8,9,10};
 my_memset(arr,0,20);
for(int i=0;i<10;i++)
{
    printf("%d ",arr[i]);
}
 return 0;
}

4.memcmp函数

4.1函数的使用

int memcmp ( const void * ptr1, const void * ptr2, size_t num );

比较从ptr1和ptr2指针指向的位置开始,向后的num个字节


#include <stdio.h>
#include <string.h>
int main()
{
 char buffer1[] = "DWgaOtP12df0";
 char buffer2[] = "DWGAOTP12DF0";
 int n;
 n = memcmp(buffer1, buffer2,sizeof(buffer1));
 if (n > 0) 
 printf("'%s' is greater than '%s'.\n", buffer1, buffer2);
 else if (n < 0) 
 printf("'%s' is less than '%s'.\n", buffer1, buffer2);
 else
 printf("'%s' is the same as '%s'.\n", buffer1, buffer2);
 return 0;
}

4.2函数的模拟


#include <stdio.h>
#include <string.h>
int my_memcmp(const void *str1, const void *str2, size_t n)
{
 const unsigned char *s1 = (const unsigned char *)str1;
 const unsigned char *s2 = (const unsigned char *)str2;
 /*
 size_t i;
 for (i = 0; i < n; i++)
 {
 if (s1[i] < s2[i])
 return -1;
 if (s1[i] > s2[i])
 return 1;
 }
 return 0;
 */
 while (n--)
 {
 if (*s1 != *s2)
 return (*s1 - *s2);
 s1++;
 s2++;
 }
 return 0;
}
 
int main()
{
 int a[6]={1,2,3,4,5,6};
 int b[6]={1,2,3,4,5,7};
 int n;
 n = my_memcmp(a,b,20);
 if (n > 0)
    printf("Array 'a' is greater than array 'b'.\n");
else if (n < 0)
     printf("Array 'a' is less than array 'b'.\n");
else
    printf("Array 'a' is the same as array 'b'.\n");
 
 return 0;
}

比较前五个数数据:

结束语

OK啦,本节的干货也彻底上完了,相信大家多内存函数有了进一步的认识,

最后友友们动动你们尊贵的手指给小编点点赞,评论一下吧,忠诚感谢大家的支持!!

目录
相关文章
|
7天前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
28 6
|
11天前
|
传感器 人工智能 物联网
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发,以及面临的挑战和未来趋势,旨在帮助读者深入了解并掌握这些关键技术。
31 6
|
24天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
31 6
|
1月前
|
C语言
【c语言】动态内存管理
本文介绍了C语言中的动态内存管理,包括其必要性及相关的四个函数:`malloc`、``calloc``、`realloc`和`free`。`malloc`用于申请内存,`calloc`申请并初始化内存,`realloc`调整内存大小,`free`释放内存。文章还列举了常见的动态内存管理错误,如空指针解引用、越界访问、错误释放等,并提供了示例代码帮助理解。
46 3
|
1月前
|
存储 算法 程序员
C语言:库函数
C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。
|
2月前
|
机器学习/深度学习 C语言
【c语言】一篇文章搞懂函数递归
本文详细介绍了函数递归的概念、思想及其限制条件,并通过求阶乘、打印整数每一位和求斐波那契数等实例,展示了递归的应用。递归的核心在于将大问题分解为小问题,但需注意递归可能导致效率低下和栈溢出的问题。文章最后总结了递归的优缺点,提醒读者在实际编程中合理使用递归。
67 7
|
2月前
|
存储 编译器 程序员
【c语言】函数
本文介绍了C语言中函数的基本概念,包括库函数和自定义函数的定义、使用及示例。库函数如`printf`和`scanf`,通过包含相应的头文件即可使用。自定义函数需指定返回类型、函数名、形式参数等。文中还探讨了函数的调用、形参与实参的区别、return语句的用法、函数嵌套调用、链式访问以及static关键字对变量和函数的影响,强调了static如何改变变量的生命周期和作用域,以及函数的可见性。
33 4
|
2月前
|
存储 C语言
【c语言】字符串函数和内存函数
本文介绍了C语言中常用的字符串函数和内存函数,包括`strlen`、`strcpy`、`strcat`、`strcmp`、`strstr`、`strncpy`、`strncat`、`strncmp`、`strtok`、`memcpy`、`memmove`和`memset`等函数的使用方法及模拟实现。文章详细讲解了每个函数的功能、参数、返回值,并提供了具体的代码示例,帮助读者更好地理解和掌握这些函数的应用。
30 0
|
2月前
|
C语言
【c语言】qsort函数及泛型冒泡排序的模拟实现
本文介绍了C语言中的`qsort`函数及其背后的回调函数概念。`qsort`函数用于对任意类型的数据进行排序,其核心在于通过函数指针调用用户自定义的比较函数。文章还详细讲解了如何实现一个泛型冒泡排序,包括比较函数、交换函数和排序函数的编写,并展示了完整的代码示例。最后,通过实际运行验证了排序的正确性,展示了泛型编程的优势。
23 0
|
2月前
|
算法 C语言
factorial函数c语言
C语言中实现阶乘函数提供了直接循环和递归两种思路,各有优劣。循环实现更适用于大规模数值,避免了栈溢出风险;而递归实现则在代码简洁度上占优,但需警惕深度递归带来的潜在问题。在实际开发中,根据具体需求与环境选择合适的实现方式至关重要。
36 0