C语言库函数 — 内存函数(含模拟实现内存函数)

简介: C语言库函数 — 内存函数(含模拟实现内存函数)

前言

内存操作函数是一类计算机程序设计中的函数库,旨在提供对内存进行常见操作的函数

本期介绍内存函数memcpy()memmove()memcmp()memset()

包含 模拟实现以上的内存函数

一、内存操作函数

内存操作函数是一类计算机程序设计中的函数库,旨在提供对内存进行常见操作的函数。 这些函数通常包括将数据从一个内存位置复制到另一个内存位置、在内存中查找特定数据、比较内存中的两个数据块等。内存操作函数是编写任何计算机程序所必需的基本工具之一,它可以使程序员更轻松地操作内存和数据。在C语言中,一些常用的内存操作函数包括memcpy()memset()memcmp()。在其他编程语言中,也有类似的内存操作函数库可供使用。

memcpy

函数介绍

memcpy函数是C标准库中的一个函数,用于将一段内存中的数据拷贝到另一段内存中。

它的定义如下:

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

其中,dest表示目标内存的地址,src表示源内存的地址,n表示需要拷贝的字节数。

该函数的功能是将源内存中的前n个字节拷贝到目标内存中。如果源内存和目标内存有重叠,那么函数的行为是未定义的。

memcpy函数通常用于实现内存数据复制、内存数据搬运等操作。由于它的实现方式是直接对内存进行操作,因此效率较高,在一些需要高效处理大量数据的场景中得到广泛应用。

模拟实现memcpy函数

#include<stdio.h>
#include<string.h>
#include<assert.h>
void* 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[] = { 1,2,3,4,5,6,7 };
  int arr2[20] = { 0 };
  my_memcpy(arr2, arr1, 28);
  int i = 0;
  for (i = 0; i < 10; i++)
  {
    printf("%d ", arr2[i]);
  }
  return 0;
}

memmove

函数介绍

memmove是 C 语言中的一个函数,用于在内存区域之间移动一定数量的字节。

memmove 可以处理重叠内存区域的情况,因此比 memcpy 更加通用。函数的原型如下:

void* memmove(void* dest, const void* src, size_t n);

其中,dest是目标内存区域的指针,src 是源内存区域的指针,n 是要移动的字节数。memmove 函数将源内存区域的内容复制到目标内存区域中,如果两个内存区域有重叠部分,则函数能够正确处理,保证数据的正确性。

例如,以下代码段演示了使用 memmove 进行内存拷贝:

#include <stdio.h>
#include <string.h>
int main()
{
    char str1[] = "Hello, world!";
    char str2[20];
    memmove(str2, str1, strlen(str1) + 1);
    printf("str2: %s\n", str2);
    return 0;
}

在这个例子中,我们将字符串 str1 的内容复制到了 str2 字符数组中,通过 memmove 函数实现。函数的第一个参数是目标地址,第二个参数是源地址,在这个例子中分别是 str2str1。第三个参数是要复制的字节数,我们通过 strlen(str1) + 1 来计算。

需要注意的是,由于 memmove 函数可以处理重叠区域的情况,因此在某些情况下可能会比较慢。如果确保不会出现重叠的情况,可以使用 memcpy 函数代替。

模拟实现memmove函数

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

memcmp

函数介绍

memcmp函数是C语言库函数之一,用于比较两段内存区域的内容。其原型如下:

int memcmp(const void *s1, const void *s2, size_t n);

其中,s1和s2分别是两个待比较内存区域的起始地址,n是要比较的字节数。这个函数会将s1和s2的前n个字节进行比较,返回值表示它们的相对大小,具体定义如下:

  • 如果s1小于s2,则返回一个负数;
  • 如果s1等于s2,则返回0;
  • 如果s1大于s2,则返回一个正数。

该函数通常用于排序或查找等场景中,可以通过比较内存中的内容来确定它们的相对大小关系,以便进行排序或查找。

需要注意的是,该函数只比较内存中的内容,不考虑数据类型等其他因素。因此,比较不同类型的数据时可能会得到不正确的结果。同时,如果n超过了s1或s2指向的内存空间大小,可能会造成内存泄漏或越界访问等问题。因此,在使用memcmp函数时需要注意传入的参数是否合法。

模拟实现memcmp函数

int my_memcmp(const void *s1, const void *s2, size_t n)
{
    const unsigned char *p1 = s1, *p2 = s2;
    
    for (size_t i = 0; i < n; ++i) {
        if (p1[i] != p2[i]) {
            return (p1[i] < p2[i]) ? -1 : 1;
        }
    }
    
    return 0;
}

memset

函数介绍

memset函数是C语言中的一个库函数,用于给一段内存空间赋值。其原型如下:

void *memset(void *s, int c, size_t n);

其中,s表示待赋值的内存空间起始地址,c表示要赋的值(通常为0),n表示要赋值的字节数。

该函数的作用是将一段内存空间赋值为指定的值,可以用来初始化数组、清空内存等。例如,将一个整型数组清空可用以下代码:

int arr[10];
memset(arr, 0, sizeof(arr));

该代码会将arr数组的所有元素都赋值为0。

需要注意的是,该函数只能处理基本数据类型,无法处理复杂数据类型。同时,由于该函数没有对边界进行检查,如果n超过了s指向的内存空间大小,可能会造成内存泄漏或越界访问等问题。因此,在使用memset函数时需要注意传入的参数是否合法。

模拟实现memset函数

void *memset(void *s, int c, size_t n) {
    unsigned char *p = s;
    while (n--) {
        *p++ = (unsigned char)c;
    }
    return s;
}

如这篇博客对大家有帮助的话,希望 三连 支持一下 !!! 如果有错误感谢大佬的斧正 如有 其他见解发到评论区,一起学习 一起进步。

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