首页 > 系统相关 >c语言回顾-内存操作函数

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

时间:2024-07-04 22:03:09浏览次数:3  
标签:函数 回顾 int void char dest 内存 memcpy

目录

前言

1.memcpy 函数

1.1函数介绍

1.2与strcpy的区别

1.3memcpy的模拟

2.memmove 函数

2.1函数介绍和使用

2.2函数的模拟

3.memset函数

3.1函数介绍

3.2函数的模拟

4.memcmp函数

4.1函数的使用

4.2函数的模拟

结束语


前言

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

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:仅适用于字符串,复制的是字符,直到遇到空字符为止。
  2. 复制内容

    • memcpy:不会检查内容,只复制指定字节数,即使其中包含空字符。
    • strcpy:遇到空字符时停止复制,并将空字符包含在内。
  3. 使用场景

    • memcpy:适用于需要复制特定大小内存块的场景,例如结构体、数组等。
    • strcpy:适用于字符串复制的场景,确保字符串以空字符结尾。
  4. 安全性

    • 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啦,本节的干货也彻底上完了,相信大家多内存函数有了进一步的认识,

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

标签:函数,回顾,int,void,char,dest,内存,memcpy
From: https://blog.csdn.net/2302_79376097/article/details/140112428

相关文章

  • NumPy函数汇总
    一、创建ndarray        1.使用np.array()由pythonlist创建        2.使用np的常规函数创建                1)np.ones(shape,dtype=None,order='C')                2)np.zeros(shape,dtype=float,order='C')......
  • Pandas函数汇总
    一、Pandas数据结构        1.Series                1)Series的创建                        s=Series(a)  #a:列表或NumPy数组或字典        2)Series的属性                  ......
  • Matplotlib函数汇总
    一、基本图形绘制        1. 绘制基本图形                plt.plot()        2. 样式和颜色        样式:'-','--','-.',':','.',',',,o,^,v,<,>,s,+,x,D,d,1,2,3,                    4,h,H......
  • SQL246 获取employees中的first_name (RIGHT和LEFT函数)
    题目连接selectfirst_namefromemployeesorderbyright(first_name,2);MySQL中的RIGHT()和LEFT()函数是用于从字符串中提取子字符串的内置函数。LEFT()函数用于从字符串的左侧开始提取指定数量的字符。其基本语法如下:LEFT(str,len)str是要从中提取字符的原......
  • 泛语言计划 第二弹 函数
    开始之前,我们先复习一下数学中是怎么定义函数的。数集A和B,如果A所有数在B中都存在且只有一个数与之对应,则称B是A的函数由此,我们可以看出一个关键词:对应。编程中的函数也一样,只是编程中不一定是数,也可以是布尔值、字符串,甚至操作。这个A集,我们叫他参数,B集,叫他返回值。如果B是操......
  • 7.1.SQL注入-基于函数报错的方式来利用updatexml()
    基于函数报错的方式来进行利用-字符型(本页updatexml())前提条件是后台数据库没有屏蔽数据库语法报错信息updatexml()方法详解注释:第一个参数,意思就是xml文档的名称第二个参数,意思就是定位到xml文档中指定的某一个位置的更新第三个参数,意思就是将文档中指定某一个位置......
  • 7.2.SQL注入-基于函数报错extractvalue(),floor()
    注入基于函数报错extractvalue(),floor()-字符型基于extractvalue()爆出数据库版本payload语句:kobe'andextractvalue(0,concat(0x7e,version()))#爆出数据库版本基于floor()floor()函数就是取整数爆出数据版本信息kobe'and(select2from(selectcount(*),......
  • YOLOv5改进 | 损失函数 | EIoU、SIoU、WIoU、DIoU、FocuSIoU等多种损失函数
    秋招面试专栏推荐 :深度学习算法工程师面试问题总结【百面算法工程师】——点击即可跳转   ......
  • C++内存管理
    内存管理C/C++内存分布静态区(数据段)全局变量:在整个程序运行期间都存在的变量,包括没有显式声明为static的全局变量。静态局部变量:在函数内部声明为static的变量。它们在函数首次调用时初始化,并在程序整个运行期间保持其值。静态全局变量:在文件作用域中声明为static的变量。......
  • 一、内存分区模型
    1.概述C++程序在执行时,将内存方向划分为4个区域。代码区:存放函数体的二进制代码,由操作系统进行管理全局区:存放全局变量,静态变量,常量栈区:由编译器自动分配释放,存放函数的参数值,局部变量堆区:由程序员分配和释放,若程序员不释放,程序结束时由操作系统回收2.程序运行前编译后......