内存函数<C语言>

server/2024/9/23 18:45:49/

前言

        前面两篇文章介绍了字符串函数,不过它们都只能用来处理字符串,C语言中也内置了一些内存函数来对不同类型的数据进行处理,本文将介绍:memcpy()使用以及模拟实现,memmove()使用以及模拟实现,memset()使用,memcmp()使用。


目录

前言

memcpy()使用以及模拟实现

memmove()使用以及模拟实现

memset()使用

 memcmp()使用


memcpy()使用以及模拟实现

        函数参数及其返回类型:

void* memcpy(void* destination, const void* source ,size_t num);
//返回值为目标空间的起始地址

作用:

从源地址起复制num个字节到目标地址

注意点:

①memcpy函数不负责重叠内存的情况(如果源地址和目的地址有任何重叠,结果都是未定义的)

②头文件<string.h>

使用举例:

 模拟实现:

//memcpy模拟实现
#include<assert.h>
void* my_memcpy(void* s1, const void* s2, size_t num) {assert(s1 && s2);//防止传入空指针void* p1 = s1;//使用一个值保存首地址,等下好返回首地址while (num--) {*(char*)s1 = *(char*)s2;(char*)s1 = (char*)s1 + 1;//不能使用(char*)s1++,因为(char*)(强制类型转换)和++(自增)都是表达式//因为第一个表达式执行后是临时的,转换结果没被保存下来(char*)s2 = (char*)s2 + 1;}return p1;
}
int main() {int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };int arr1[10] = { 0 };int i = 0;my_memcpy(arr1, arr, sizeof(arr));for (i = 0; i < sizeof(arr) / sizeof(arr[0]); i++) {printf("%d ", arr1[i]);}return 0;
}

运行结果:


memmove()使用以及模拟实现

         函数参数及其返回类型:

void* memmove(void* destination, const void* source ,size_t num);
//返回值为目标空间的起始地址

作用:

从源地址起复制num个字节到目标地址(memmove函数能完全代替memcpy函数),尽量使用memmove函数

注意点:

①memcpy函数和memmove函数最大区别就是,memmove函数能实现源地址和目的地址有重叠的拷贝,所以memmove函数能完全代替memcpy函数

②头文件<string.h>

使用举例:

 模拟实现:

前面我们已经实现了memcpy函数的模拟实现,现在我们面临唯一问题就是如何解决,源地址和目的地址有重叠的拷贝,经过分析我们发现重叠其实有两种情况:

①源头地址大于目标地址时

        源地址从后往前拷贝

②目的地址大于源头地址时

        源地址从前往后拷贝

 代码:

//memmove模拟实现
#include<string.h>
#include<assert.h>
void* my_memmove(void* s1, const void* s2, size_t num) {assert(s1 && s2);void* ret = s1;if (s2 >= s1) {//①源头地址大于目标地址时while (num--) {*((char*)s1+num) = *((char*)s2+num);}}else {//②目的地址大于源头地址时while (num--) {*(char*)s1 = *(char*)s2;(char*)s1 = (char*)s1 + 1;(char*)s2 = (char*)s2 + 1;}}return ret;
}
int main() {int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };my_memmove(arr + 5, arr, sizeof(arr[0]) * 5);for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); i++) {printf("%d ", arr[i]);}return 0;
}

运行结果:


memset()使用

         函数参数及其返回类型:

void* memset(void* ptr, int value ,size_t num);
//返回值为被填充空间的起始地址

作用:

用来设置内存,将内存空间以字节为单位设置成想要内容

注意点:

①是以字节为单位设置的,当为int类型设置时可能出错

②头文件<string.h>

使用举例:


 memcmp()使用

         函数参数及其返回类型:

int memcmp(const void* ptr1, const void* ptr2,size_t num);
//返回值是大于:大于0的数,等于:0,小于:小于0的数

作用:

比较两块空间前num个字节

注意点:

①是以字节为单位比较

②头文件<string.h>

使用举例:


http://www.ppmy.cn/server/45163.html

相关文章

如何配置才能连接远程服务器上的 redis server ?

文章目录 Intro修改点 Intro 以阿里云服为例。 首先&#xff0c;我在我买的阿里云服务器中以下载源码、手动编译的方式安装了 redis-server&#xff0c;操作流程见&#xff1a;Ubuntu redis 下载解压配置使用及密码管理 && 包管理工具联网安装。 接着&#xff0c;我…

机器学习之二分类提升决策树(Two-class Boosted Decision Tree)

二分类提升决策树(Two-class Boosted Decision Tree)是一种常用的机器学习方法,主要用于分类任务。该方法结合了决策树模型和提升(boosting)算法的优点,通过多个弱分类器(通常是简单的决策树)来构建一个强分类器。下面是关于二分类提升决策树的主要概念和工作流程: 1…

关于《Java并发编程之线程池十八问》的补充内容

一、写在开头 在上一篇文章我们写《Java并发编程之线程池十八问》的时候,鉴于当时的篇幅已经过长,很多内容就没有扩展了,在这篇文章里对一些关键知识点进行对比补充。 二、Runnable vs Callable 在创建线程的时候,一般会选用 Runnable 和 Callable 两种方式。 【源码对…

最长递增子序列,交错字符串

第一题&#xff1a; 代码如下&#xff1a; int lengthOfLIS(vector<int>& nums) {//dp[i]表示以第i个元素为结尾的最长子序列的长度int n nums.size();int res 1;vector<int> dp(n, 1);for (int i 1; i < n; i){for (int j 0; j < i; j){if (nums[i]…

Kubectl 的使用——k8s陈述式资源管理

一、kebuctl简介: kubectl 是官方的CLI命令行工具&#xff0c;用于与 apiserver 进行通信&#xff0c;将用户在命令行输入的命令&#xff0c;组织并转化为 apiserver 能识别的信息&#xff0c;进而实现管理 k8s 各种资源的一种有效途径。 对资源的增、删、查操作比较方便&…

长安链使用Golang编写智能合约教程(一)

长安链是分2.1.和2.3.两个版本&#xff0c;本节面说的是2.1.的版本 需要2.3.版本的合约&#xff0c;请看教程&#xff08;二&#xff09;&#xff01; 教程&#xff08;二&#xff09;我会写如何查历史数据 教程二&#xff1a;&#xff08;长安链2.3.的版本的智能合约编写&…

代码随想录算法训练营第22天(py)| 二叉树 | 669. 修剪二叉搜索树、108.将有序数组转换为二叉搜索树、538.把二叉搜索树转换为累加树

669. 修剪二叉搜索树 力扣链接 给定一个二叉搜索树&#xff0c;同时给定最小边界L 和最大边界 R。通过修剪二叉搜索树&#xff0c;使得所有节点的值在[L, R]中 (R>L) 思路 如果当前节点元素小于low&#xff0c;递归右子树&#xff0c;返回符合条件的头节点 如果当前节点元…

6.S081的Lab学习——Lab5: xv6 lazy page allocation

文章目录 前言一、Eliminate allocation from sbrk() (easy)解析&#xff1a; 二、Lazy allocation (moderate)解析&#xff1a; 三、Lazytests and Usertests (moderate)解析&#xff1a; 总结 前言 一个本硕双非的小菜鸡&#xff0c;备战24年秋招。打算尝试6.S081&#xff0…