C和指针 第十三章 高级指针话题
高级声明:
int (*f)();
这里声明有两个括号,第二个括号是函数调用,第一个括号是聚组作用。(*f)是一个函数,所以f是指向返回整型的函数的指针。程序中的每个函数都位于,内存中某个位置,所以存在指向那个位置的指针。
int *f[];
下标的优先级高,所以f是个数组,数组中元素为指向int的指针。
int (*f[])()
首先按照优先级进行分析,括号内(*f[])先求值,所以*f[], f是数组,数组元素是指向某种类型的指针,结合外面的看,f是一个数组,数组元素是指向返回int类型的函数的指针。
函数指针:
指针在使用前需要进行初始化,对于函数指针,访问前需要初始化为指向摸个函数。
int fuc(int); int (*fPtr)(int) = &fuc;
//在函数初始化之前需要具有fuc的原型,否则,编译器无法检查fuc的类型是否与fPtr一致。
初始表达式中的&操作符是可选的,因为函数名被使用时,总是由编译器把它转换为函数指针,&操作符只是显示的说明了编译器将隐式执行任务。
函数指针被声明后有三种方式调用函数:
int result; result = func(25);
result = (*fPtr)(25);
result = fPtr(25);
第一种方法,直接调用,执行时函数名首先被转换成一个函数指针,该指针指向函数在内存中的位置。然后,函数调用操作符调用该函数,执行位于这个地址的代码
第二种方法,先间接访问,把函数指针转换成函数名,这个转换时不必要的,执行时还会像第一种方法一样被转换到函数指针。
第三种方法,间接访问时不必要的,编译器需要的是一个函数指针,这里直接使用函数指针。
函数指针也可以作为值,传入其他函数,叫做回调函数。例如下面的类型无关的链表查询函数。
#include <stdio.h>
typedef struct Node {
struct Node *next;
int value;
}Node;
/*
* 这里值以及比较函数均是void类型,需要在使用时根据类型自己进行转换
* */
Node *search_list(Node *root, void const *value, int (*compare)(void const *, void const *))
{
Node *current = root;
while(current != NULL){
//回调函数进行比较
if(compare(¤t -> value, value) == 0){
break;
}
current = current -> next;
}
return current;
}
//声明为void *类型,和compare保持一致
int int_compare(void const *num1, void const * num2){
//根据类型进行转换
if(*(int *)num1 == *(int *)num2){
return 0;
}else {
return 1;
}
}
int main()
{
Node third = {NULL, 3};
Node second = {&third, 2};
Node first = {&second, 1};
int target = 2;
Node *result = search_list(&first, &target, int_compare);
if(result == NULL){
printf("No Found\n");
}else{
printf("got it: %d", result -> value);
}
return 0;
}
运行:

如果希望在value为字符的链表中查找,只需要将比较函数改为字符类型既可以。
指针数组的第二个应用就是转移表:
#include <stdio.h> double add(double, double);
double sub(double, double);
double mul(double, double);
double div(double, double); //转移表中保存函数的指针,确保指针类型相同
double (*transTable[])(double, double) = {add, sub, mul, div}; int main()
{
printf("%g\n", transTable[0](1, 1));
printf("%g\n", transTable[1](2, 1));
printf("%g\n", transTable[2](3, 5));
printf("%g\n", transTable[3](4, 3)); return 0;
} double add(double a, double b)
{
return a + b;
} double sub(double a, double b)
{
return a - b;
} double mul(double a, double b)
{
return a * b;
} double div(double a, double b)
{
return a / b;
}
运行结果:

要注意转移表使用时,越界访问。
命令行参数:
命令行参数是指向指针的指针的另一个用武之地,参数传给C程序main时包含两个形参,argc和argv,argc是命令行参数的数目,argv是保存参数指针的数组。第一个是指向程序名称,末尾是一个NULL指针。
#include <stdio.h> int main(int argc, char **argv)
{
int count = argc;
printf("count: %d\n", count); for(int idx = 0; idx < count; idx++)
{
printf("[%s]\n", argv[idx]);
} return 0;
}
编译选项添加两个参数

运行结果:

字符串常量:
字符串常量出现于表达式中,它的值是一个指针常量,编译器把这些指定字符拷贝一份储存在内存的某个位置,并储存一个指向第一个字符的指针。我们可以对他进行下标引用。
char *ch = "abcd" + 1;
ch为指向b的指针。可以把字符串常量和数组名一样看待。
C和指针 第十三章 高级指针话题的更多相关文章
- C和指针 第十三章 习题
1,1标准输入读入字符,统计各类字符所占百分比 #include <stdio.h> #include <ctype.h> //不可打印字符 int isunprint(int ...
- 《C和指针》 读书笔记 -- 第13章 高级指针话题
1.函数指针 int (*f)(); int *(*f[])(); 用途: [1]回调函数 e.g. /*在一个单链表中查找指定值*/ Node *search_list(Node *node,voi ...
- 【WPF学习】第三十三章 高级命令
前面两章介绍了命令的基本内容,可考虑一些更复杂的实现了.接下来介绍如何使用自己的命令,根据目标以不同方式处理相同的命令以及使用命令参数,还将讨论如何支持基本的撤销特性. 一.自定义命令 在5个命令类( ...
- 《C与指针》——高级指针话题
指针真是让人又爱又恨..... 首先还是先来看一下C语言中的高级指针声明.不要被表面迷惑最重要. /* ** <C和指针>——高级指针话题 */ int i; //定义一个整型变量 int ...
- C++(指针和高级指针)-上篇
[在指针中存储地址] int *pAge=nullptr; //将PAge声明为int指针,即用于存储int变量的地址 如果将指针初始化为0或者NUll,以后必须将变量的地址赋给它,如下例代码: ; ...
- C和指针 第六章 数组名与指针
指针的算术运算符是指针和数组之间的一种关联,但不是唯一关联: 可以使用数组名作为指向数组第一个元素的指针,但是不可以给数组名赋新的值. //如下声明a int a[10]; //用a作为指向数组第一个 ...
- 联合与枚举 、 高级指针 、 C语言标准库(一)
1 输入一个整数,求春夏秋冬 1.1 问题 在实际应用中,有的变量只有几种可能取值.如人的性别只有两种可能取值,星期只有七种可能取值.在 C 语言中对这样取值比较特殊的变量可以定义为枚举类型.所谓枚举 ...
- JavaScript高级程序设计:第十三章
第十三章 一.理解事件流 事件流描述的是从页面中接收事件的顺序. 1.事件冒泡 IE的事件流叫做事件冒泡,即事件开始时由最具体的元素接收,然后逐级向上传播到较为不具体的节点.以下面的HTML页面为例: ...
- C和C指针小记(十三)-数组
1.1 一维数组 一维数组的声明: int a[10]; 这里a就是一个数组. 数组a的类型就是一个指向整型的常量指针. 但是数组和指针是**不相同**的. **数组具有特定数量的元素,而指针只是一个 ...
随机推荐
- 网格弹簧质点系统模拟(Spring-Mass System by Fast Method)附源码
弹簧质点模型的求解方法包括显式欧拉积分和隐式欧拉积分等方法,其中显式欧拉积分求解快速,但积分步长小,两个可视帧之间需要多次积分,而隐式欧拉积分则需要求解线性方程组,但其稳定性好,能够取较大的积分步长. ...
- 洛谷10月月赛Round.3
Rank11:260=60+100+100 P2409 Y的积木 题目背景 Y是个大建筑师,他总能用最简单的积木拼出最有创意的造型. 题目描述 Y手上有n盒积木,每个积木有个重量.现在他想从每盒积木中 ...
- Child <- many-to-one ->Parent
网上找到个描述的很精妙的例子 Child <- many-to-one ->Parent class Child { private ...
- 判断.net中在windows系统下的字节序
字节序,是指字节在内存中的存储顺序.其又分为大端字节(Big-Endian)序和小端字节序(Little-Endian). 以下摘自百度百科: a) Little-Endian就是低位字节排放在内存的 ...
- [CareerCup] 18.1 Add Two Numbers 两数相加
18.1 Write a function that adds two numbers. You should not use + or any arithmetic operators. 这道题让我 ...
- npoi批量导入实现及相关技巧
批量导入功能对于大部分后台系统来说都是不可或缺的一部分,常见的场景-基础数据的录入(部门,用户),用批量导入方便快捷.最近项目需要用到批量导入,决定花点时间写套比较通用的Excel导入功能.经过考虑, ...
- 小巧数据库 Apache Derby 使用攻略
1. Derby 介绍 将目光放在小 Derby 的原因是纯绿色.轻巧.内存占用小,分分钟在你机子跑起来,自己做点需要连接数据库的代码实践非常方便. 虽然 Mysql 也可以,多一种选择,不是也挺好么 ...
- 基于C/S架构的3D对战网络游戏C++框架_02系统设计(总体设计、概要设计)
本系列博客主要是以对战游戏为背景介绍3D对战网络游戏常用的开发技术以及C++高级编程技巧,有了这些知识,就可以开发出中小型游戏项目或3D工业仿真项目. 笔者将分为以下三个部分向大家介绍(每日更新): ...
- c++调用lua环境配置
安装完luaforwindows后进行配置 配置项目属性
- JS:offsetWidth\offsetleft 等图文解释
网页可见区域宽: document.body.clientWidth;网页可见区域高: document.body.clientHeight;网页可见区域宽: document.body.of ...