C语言----C语言内存函数

简介: C语言----C语言内存函数

1.memcpy--内存拷贝--使用和模拟实现

//memcpy基本格式:
//               目标空间地址       原空间地址        被拷贝的字节个数
//void *memcpy(void * destination, const void * source,size_t num);
//因为内存拷贝拷贝的数据有:整型数据、结构体数据、结构体数据。那么我们就要用到void*进行接收任意类型的数据
//void*指针可以接收任意空间
 
//size_t num这个是要被拷贝的字节个数
//
 
 
//返回的是目标空间的起始地址
 
 
int main()
{
    int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
    int arr2[20] = { 0 };
    memcpy(arr2, arr1 + 2, 20);//因为我们要拷贝的是3~7,那么就是5个整型,20个字节
 
        //因为我们要从3开始进行拷贝
        //恰好因为数组名是首元素的地址,那么3的地址就是arr+2
    //将arr1中的3 4 5 6 7拷贝放到arr2中
    for (int i = 0; i < 20; i++)
    {
        printf("%d ", arr2[i]);
    }
    return 0;
}
//我们要满足目标空间能进行修改
//目标空间足够大
 
//这个memcpy函数是不会考虑\0的,我让他拷贝多少个字节他就拷贝多少个字节

函数memcpy从source的位置开始复制num个字节的数据到destination指向的内存位置。

1.这个函数在遇到\0的时候并不会停下来

2.如果source和destination有任何的重叠,复制的结果都是未定义的

memcpy函数最终返回的是目标空间的起始地址

//函数的一种写法:
这个函数最终返回的是目标空间的起始地址
//void* my_memcpy(void *dest, const void*src, size_t num)//目标空间是要进行修改的,但是源头是不能被修改的,所以我们加上const
//{
//    assert(src && dest);//防止两个指针是空指针
//    void* ret = dest;//我们需要先创建一个指针存放最开始的dest,通过ret我们可以找到dest指向的数组
//    //这里的src指向的是数组中3的位置
//    //dest指向的是arr2数组的首元素的位置
//    //拷贝20个字节
//    //void*的指针不能进行直接计算,void*的指针是无具体类型的指针
//    for (int i = 0; i < num; i++)//一对字节一对字节
//    {
//        *(char*)dest = *(char*)src;//强制类型转换再进行解引用就将那一对字节进行交换
//        dest = (char*)dest + 1;//先将dest强制类型转换,在进行+1操作,因为上面的强制类型转换只是暂时的,为了能让dest进行++操作,我们就这样操作
//        src = (char*)src + 1;
//    }
//    return ret;
//}
 
//函数的另一种写法:
//函数的一种写法:
//这个函数最终返回的是目标空间的起始地址
void* my_memcpy(void *dest, const void*src, size_t num)
{
    assert(src && dest);
    void* ret = dest;
    while (num--)//也是循环num次
    {
        *(char*)dest = *(char*)src;
        dest = (char*)dest + 1;
        src = (char*)src + 1;
    }
    return ret;
}
 
 
 
 
int main()
{
    int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
    int arr2[20] = { 0 };
    int *p=my_memcpy(arr2, arr1 + 2, 20);//这个函数的作用就是我们在arr1中选择的那部分粘贴在arr2中
    for (int i = 0; i < 20; i++)
    {
        printf("%d ", p[i]);
    }
    return 0;
}
 
 
//memcpy函数不负责重叠内存的拷贝,非要使用,结果就是未定义的
 
//只负责不重叠的内存

函数的返回值是void*类型的数据

这个memcpy函数有三个数据

2.memmove--内存移动--使用和模拟实现

2.memmove--内存移动--使用和模拟实现

int main()
{
    //将123456拷贝放到34567上面
    int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
    memmove(arr1 + 2, arr1, 20);
    for (int i = 0; i < 10; i++)
    {
        printf("%d ", arr1[i]);
    }//输出结果是1 2 1 2 3 4 5 8 9 10
    return 0;
}
//memmove就是用来处理这些重叠内存的拷贝
void* my_memmove(void *dest, const void*src, size_t num)
{
    void* ret = dest;//将目标空间起始地址记录下来
 
    assert(dest && src);//保证指针的有效性
    //最开始的dest指向的是3,src指向的是1,字节是20
    if (dest < src)//前-->后
        //目标空间的首元素<拷贝的起始元素--就相当与我们把34567拷贝到12345
        //那么12的位置就是dest<src的位置
 
    {
        while (num--)//总共拷贝num个字节,那么就是num--,等所有字节拷贝完循环就停止了
        {
            *(char*)dest = *(char*)src;
            //拷贝完一对字节就往后走,进行+1操作
            dest = (char*)dest + 1;
            src = (char*)src + 1;
        }
    }
    else//后-->前剩下的区域---dest > src
    {
        while (num--)
        {
            //进入循环的时候,num 已经是19了
            //*((char*)dest + num)---找到最后一个字节的位置
            *((char*)dest + num)    =       *((char*)src + num);
        //目标的最后一个位置的字节------    起源空间的最后一个位置
            //将起源的最后一个字节先开始挪动,从最后一个位置依次挪动
            //就是将我们要挪动的那块区域,我们从最右边开始挪动
        }
    }
    return ret;
}
int main()
{
    //将123456拷贝放到34567上面
    int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
    my_memmove(arr1 + 2, arr1, 20);
    for (int i = 0; i < 10; i++)
    {
        printf("%d ", arr1[i]);//1 2 1 2 3 4 5 8 9 10
    }
    return 0;
}
 
//总之:这个拷贝是分三块区域的,最前面的一块区域满足dest<src
//我们只能从前往后进行拷贝,不然会出错误
 
//而剩下的两块区域可以同时从后往前进行拷贝,那么我们就将这两块区域放在一起

在C语言标准中,明确规定了memcpy只要能实现不重叠的拷贝就行,重叠的拷贝交给memmove

我们发现vs上面的库函数memcpy函数也能实现重叠内存的拷贝

我们在以后的拷贝中,我们可以用memmove,因为不管是重叠的还是不重叠的都能搞定

3.memset--内存设置--函数的使用

memset是用来设置内存的,将内存中的值以字节单位设置为想要的内容

基本格式:

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

ptr就是指向要被填充的内存块的指针

value就是我们想要设置的值是什么

num是设置的字节的大小

int main()
{
    char arr[] = "hello world";
    //将world改成5个x
    memset(arr + 6, 'x', 5);//数组首元素的地址+6就是指向的是w
    printf("%s\n", arr);
    return 0;
}

4.memcmp--内存比较--函数的使用

4.memcmp--内存比较--函数的使用

内存的比较

什么类型的都能进行比较

int main()
{
    int arr1[] = { 1,2,3,4,5 };
    int arr2[] = { 1,2,3,6,5 };
    int ret=memcmp(arr1, arr2, 16);
    printf("%d\n", ret);//输出的数据就是-1
    //就是说明arr1<arr2
    return 0;
}


相关文章
|
1月前
|
存储 编译器 程序员
【C语言】内存布局大揭秘 ! -《堆、栈和你从未听说过的内存角落》
在C语言中,内存布局是程序运行时非常重要的概念。内存布局直接影响程序的性能、稳定性和安全性。理解C程序的内存布局,有助于编写更高效和可靠的代码。本文将详细介绍C程序的内存布局,包括代码段、数据段、堆、栈等部分,并提供相关的示例和应用。
48 5
【C语言】内存布局大揭秘 ! -《堆、栈和你从未听说过的内存角落》
|
1月前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
66 10
|
1月前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
52 9
|
1月前
|
存储 Unix Serverless
【C语言】常用函数汇总表
本文总结了C语言中常用的函数,涵盖输入/输出、字符串操作、内存管理、数学运算、时间处理、文件操作及布尔类型等多个方面。每类函数均以表格形式列出其功能和使用示例,便于快速查阅和学习。通过综合示例代码,展示了这些函数的实际应用,帮助读者更好地理解和掌握C语言的基本功能和标准库函数的使用方法。感谢阅读,希望对你有所帮助!
41 8
|
1月前
|
C语言 开发者
【C语言】数学函数详解
在C语言中,数学函数是由标准库 `math.h` 提供的。使用这些函数时,需要包含 `#include <math.h>` 头文件。以下是一些常用的数学函数的详细讲解,包括函数原型、参数说明、返回值说明以及示例代码和表格汇总。
50 6
|
1月前
|
存储 C语言
【C语言】输入/输出函数详解
在C语言中,输入/输出操作是通过标准库函数来实现的。这些函数分为两类:标准输入输出函数和文件输入输出函数。
268 6
|
1月前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
64 6
|
1月前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
42 5
|
2月前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
65 1
|
7月前
|
程序员 C语言 C++
【C语言基础】:动态内存管理(含经典笔试题分析)-2
【C语言基础】:动态内存管理(含经典笔试题分析)