C语言深入理解指针(非常详细)(五)

2023-11-02

回调函数

回调函数就是一个通过函数指针调用的函数

如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数
时,被调用的函数就是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条
件发生时由另外的一方调用的,用于对该事件或条件进行响应

举个例子:

//使⽤回调函数改造前
#include <stdio.h>
int add(int a, int b)
{
return a + b;
}
int sub(int a, int b)
{
return a - b;
}
int mul(int a, int b)
{
return a * b
}
int div(int a, int b)
{
return a / b;
}
int main()
{
int x, y;
int input = 1;
int ret = 0;
do
{
printf("******************)
printf(" 1:add,2:sub\n)
printf(" 3:mul,4:div\n)
printf("******************)
printf("请选择:");
scanf("%d", &input);
switch (input)
{
case 1:
printf("输⼊操作数:");
scanf("%d %d", &x, &y)
ret = add(x, y);
printf("ret = %d\n", ret)
break;
case 2:
printf("输⼊操作数:");
scanf("%d %d", &x, &y)
ret = sub(x, y);
printf("ret = %d\n", r
break;
case 3:
printf("输⼊操作数:");
scanf("%d %d", &x, &y)
ret = mul(x, y);
printf("ret = %d\n", ret)
break;
case 4:
printf("输⼊操作数:");
scanf("%d %d", &x, &y)
ret = div(x, y);
printf("ret = %d\n", ret)
break;
case 0:
printf("退出程序\n");
break;
default:
printf("选择错误\n");
break;
}
} while (input);
return 0;
}
//使⽤回到函数改造后
#include <stdio.h>
int add(int a, int b)
{
return a + b;
}
int sub(int a, int b)
{
return a - b;
}
int mul(int a, int b)
{
return a * b;
}
int div(int a, int b)
{
return a / b;
}
void calc(int(*pf)(int, int))
{
int ret = 0;
int x, y;
printf("输⼊操作数:");
scanf("%d %d", &x, &y);
ret = pf(x, y);
printf("ret = %d\n", ret);
}
int main()
{
int input = 1;
do
{
printf("******************)
printf(" 1:add,2:sub\n)
printf(" 3:mul,4:div\n)
printf("******************)
printf("请选择:");
scanf("%d", &input);
switch (input)
{
case 1:
calc(add);
break;
case 2:
calc(sub);
break;
case 3:
calc(mul);
break;
case 4:
calc(div);
break;
case 0:
printf("退出程序\n");
break;
default:
printf("选择错误\n");
break;
}
} while (input);
return 0;
}

qsort使用举例

#include <stdio.h>
//qosrt函数的使⽤者得实现⼀个⽐较函数
int int_cmp(const void * p1, const void * p2)
/*void*p为任意类型的指针,
因为创造这个函数的人不知道使用者会传入什么类型的指针,
因此用viod类型的指针就可以解决这个问题*/
{
return (*( int *)p1 - *(int *) p2);
/*返回值是用两个指针变量进行相减,其中int*为强制类型转换,将p1和p2强制转换为整形类型的指针,再通过解引用判断大小*/
}
int main()
{
int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };
int i = 0;
qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof (int), int_cmp);
/*qsort函数需要数组的首元素地址,还有多少个元素,一个数组元素有多少字节,比较大小的函数*/
/*因为需要比较两元素的大小,然后进行排序,所以需要一个比较大小的元素*/
for (i = 0; i< sizeof(arr) / sizeof(arr[0]); i++)
{
printf( "%d ", arr[i]);
}
printf("\n");
return 0;
}

qsort函数的模拟实现

使用回调函数,模拟实现qsort(采用冒泡的方式)

#include <stdio.h>
//比较大小的函数
int int_cmp(const void* p1, const void* p2)
{
	return (*(int*)p1 - *(int*)p2);
}
//交换的函数
void _swap(void* p1, void* p2, int size)
{
	int i = 0;
	for (i = 0; i < size; i++)
	{
		char tmp = *((char*)p1 + i);
		*((char*)p1 + i) = *((char*)p2 + i);
		*((char*)p2 + i) = tmp;
		//*((char*)p1 + i)就是数组的元素一点i个字节,因为是强制类型转换为char*类型的指针,又由于char*类型的指针大小为一个字节,所以移动i个字节就是移动i个数组元素
	}
}
//模拟实现的qsort函数
void bubble(void* base, int count, int size, int(*cmp)(void*, void*))
{
	int i = 0;
	int j = 0;
	for (i = 0; i < count - 1; i++)
	{
		for (j = 0; j < count - i - 1; j++)
		{
			if (cmp((char*)base + j * size, (char*)base + (j + 1) * size) > 0)
			//判断指针大小是否大于0,如果大于0就会交换
			{
				_swap((char*)base + j * size, (char*)base + (j + 1) * size, size);
			}
		}
	}
}

int main()
{
	int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };
	//char *arr[] = {"aaaa","dddd","cccc","bbbb"};
	int i = 0;
	bubble(arr, sizeof(arr) / sizeof(arr[0]), sizeof(int), int_cmp);
	for (i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
	{
		printf("%d ", arr[i]);
	}
	printf("\n");
	return 0;
}

sizeof和strlen的对比

sizeof

在学习操作符的时候,我们学习了 sizeof , sizeof 计算变量所占内存内存空间大小的,单位是字节,如果操作数是类型的话,计算的是使用类型创建的变量所占内存空间的大小
sizeof 只关注占用内存空间的大小,不在乎内存中存放什么数据比如:

#inculde <stdio.h>
int main()
{
int a = 10;
printf("%d\n", sizeof(a));
printf("%d\n", sizeof a);
printf("%d\n", sizeof(int));
return 0;
}

strlen

strlen 是C语言库函数,功能是求字符串长度。函数原型如下:

size_t strlen ( const char * str );
这里随便说一下size_t是无符号整形类型

统计的是从 strlen 函数的参数 str 中这个地址开始向后, \0 之前字符串中字符的个数
strlen 函数会一直向后找 \0 字符,直到找到为止,所以可能存在越界查找(找不到\0)

#include <stdio.h>
int main()
{
char arr1[3] = {'a', 'b', 'c'};
char arr2[] = "abc";
printf("%d\n", strlen(arr1));
/*strlen(arr1)就是越界查找的典型例子
因为数组里都是字符,并没找到\0
因此就会越界查找,甚至找不到*/
printf("%d\n", strlen(arr2));
/*每个字符串末尾都有\0,
因此strlen的大小就是abc中有多少个字符*/
printf("%d\n", sizeof(arr1));
printf("%d\n", sizeof(arr1));
return 0;
}

sizeof和strlen的对比

sizeof

  1. sizeof是操作符
  2. sizeof计算操作数所占内
    存的大小,单位是字节
  3. 不关注内存中存放什么数

strlen

  1. strlen是库函数,使用需要包含头头件 string.h
  2. srtlen是求字符串长度的,统计的是 \0 之前字符的隔个数
  3. 关注内存中是否有 \0 ,如果没有 \0 ,就会持续往后找,可能
    会越界

一道关于sizeof的题

#include<stdop.h>
int main()
{
short s=10;
in**加粗样式**t i=2;
int n=sizeof(s=i+4);
printf("%d\n",s);
printf("%d\n",n);
return 0;
}

s作为短整型是占用两个字节,而i是整形则占4个字节,在int n=sizeof(s=i+4)中,很多人会认为i+4会赋值给s,但是我们要注意二者的类型是不同的,一个是短整型,另一个是整形,因此会发生截断,由于s始终为短整形,因此sizeof(s)就是s的内存大小,就是2
对于n为什么是等于10,其实sizeof的操作数如果是一个表达式,那么表达式是不会参与计算的,因为我们已经知道了s是短整型,那么是没有必要再去计算的,所以s赋的值仍然是10.
表达式有两种属性
1:类型属性
2:值属性
所以我们以后只需要判断sizeof中的表达式类型就可以知道大小

在这里插入图片描述

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C语言深入理解指针(非常详细)(五) 的相关文章

随机推荐

  • 使用git时出现fatal: Authentication failed for

    出现如下错误 fatal Authentication failed for 这是因为我使用的git账号的密码修改了 但是之前已经保存了原来的密码 所以就是密码错误导致无法登陆 但是又不会弹出来让我重新输入密码的界面 这就尴尬了 后来找到了
  • MSYS2 介绍、下载与安装、Pacman常用命令

    一 MSYS2 介绍 MSYS2 官网 MSYS2 MSYS2 Minimal SYStem 2 是一个集成了大量的GNU工具链 工具和库的开源软件包集合 它提供了一个类似于Linux的shell环境 可以在Windows系统中编译和运行许
  • odoo15设置定时任务

    概念 主要是 ir cron模型 它有以下属性 ir actions server id 服务端动作 cron name 任务名称 user id 调度程序用户 默认是当前环境用户 active 默认True interval number
  • KNN最近邻算法——原理和Python代码

    KNN最近邻算法是一种监督学习算法 基本思想是取距离测试数据最近的K个点 这K个点训练数据属于某一类型的数量多 则将该测试数据点也判断为该类 距离可以取 1 欧氏距离 2 曼哈顿距离 算法 1 计算测试数据与各个训练数据之间的距离 2 按照
  • java基础:Object类和Objects工具类的介绍

    1 Object 1 说明 Object类是所有类 数组 枚举类的父类 java允许把任何类型的对象赋给Object类型的变量 当定义一个类时没有使用extends为它显式指定父类 则该类默认继承Object父类 2 Object类的常用方
  • Servlet的Cookie与Seesion的介绍

    什么是Cookie Cookie是服务器在本地机器上存储的小段文本 它是一个简单的字符串形式的键值对 网络服务器用HTTP头向客户端发送cookie 一般情况下Cookie是由服务端生成的 当然js也能操作cookie 在客户端 浏览器解析
  • Python将numpy(.npy文件)存储为.ply文件

    Python将numpy npy文件 存储为 ply文件 安装Open3D From NumPy to Ply Open3D文档 安装Open3D pip install open3d From NumPy to Ply Pass xyz
  • [561]TensorFlow练习1: 对评论进行分类

    TensorFlow是谷歌2015年开源的一个深度学习库 到现在正好一年 和TensorFlow类似的库还有Caffe Theano MXNet Torch 但是论火爆程度 TensorFlow当之无愧 短短一年就在Github就收获了4万
  • ubuntu安装docker及在docker中使用gpu

    ubuntu安装docker和nvidia docker2 docker19以下版本 安装docker 安装nvidia docker2 docker19以上版本 安装docker 安装nvidia container runtime do
  • CSS布局

    CSS布局 div标签 概念 div div 本身是HTML4的标签 因其特殊性而一直作为CSS布局的核心标签 特点 div div 本身默认样式极少 只有默认块级元素这一项 没有太多默认属性 默认宽度为100 总结 本身不具有太多默认的样
  • CTFshow--web--红包题第六弹

    提示了不是sql注入 需要找源码 查看网页源代码
  • 【Hadoop生态圈】1.Hadoop入门教程及集群环境搭建

    文章目录 1 简介 2 环境准备 3 安装hadoop 3 修改Hadoop配置文件 3 1 hadoop env sh配置 3 2 core site xml配置 3 3 hdfs site xml配置 3 4 mapred site x
  • 实操----mybatis自动生成实体类/mapper.java/mapper.xml:mybatis-generator-config

    填错信息导致的常见问题 一 pom里面添加配置 SpringBoot项目创建完成后 在SpringBoot项目的src main resources目录下创建generatorConfiguration xml文件 有时候出现报错找不到文件
  • c++中怎么控制浮点型小数点后输出的位数

    关于这个问题 涉及到的是流的格式化 大家都知道 关于c 的小数输出 小数点后面的尾数全部省略 那么该怎么办呢 我们需要了解一部分关于c 输出时格式的知识 首先 需要有 include
  • 高并发系统设计40问-缓存篇

    系统中使用缓存的优缺点 优点 缓存加快访问速度 便于横向扩展 可分为静态 动态 热点缓存 缺点 增加系统的复杂性 可能导致数据不一致 增加成本
  • html箭头随着点击而移动,Vue实现点击箭头上下移动效果的方法

    Vue实现点击箭头上下移动效果的方法 发布时间 2020 07 02 15 38 11 来源 亿速云 阅读 229 作者 清晨 这篇文章主要介绍Vue实现点击箭头上下移动效果的方法 文中示例代码介绍的非常详细 具有一定的参考价值 感兴趣的小
  • slambook2(ch10)—— Ubuntu18.04安装g2o_viewer + 例程演示

    slambook2 ch10 Ubuntu18 04安装g2o viewer 例程演示 一 安装g2o viewer 1 检查是否安装g2o viewer 2 g2o viewer安装 二 例程演示 1 pose graph g2o SE3
  • fragment组件

    在vue2中 组件必须有一个根标签 在vue3中 组件可以没有根标签 因为内部会将多个标签包含在一个fragment虚拟元素中 优点 减少标签层级 减小内存占用
  • Python搭建自己[IP代理池]

    IP代理是什么 ip就是访问网页数据服务器位置信息 每一个主机或者网络都有一个自己IP信息 为什么要使用代理ip 因为在向互联网发送请求中 网页端会识别客户端是真实用户还是爬虫程序 在今天以互联网为主导的世界中 数据一种资源 谁能得到更多有
  • C语言深入理解指针(非常详细)(五)

    目录 回调函数 qsort使用举例 qsort函数的模拟实现 sizeof和strlen的对比 sizeof strlen sizeof和strlen的对比 一道关于sizeof的题 回调函数 回调函数就是一个通过函数指针调用的函数 如果你