1.如何用C语言实现一个函数,传递两个整形数,返回两个数的和?

#include<stdio.h>

int add(int a,int b)
{
return a+b;
} void main()
{
int a=,b=;
printf("a=%d\nb=%d\nResult=%d",a,b,add(a,b));
getchar();
}

2.同时计算两个数的和与差?

#include<stdio.h>

int add(int a,int b)
{
return a+b;
} int sub(int a,int b)
{
return a-b;
} void main()
{
int a=,b=;
printf("a=%d\nb=%d\na+b=%d\n\n",a,b,add(a,b));
printf("a=%d\nb=%d\na-b=%d\n\n",a,b,sub(a,b));
getchar();
}

  现在你所写软件已经具备加法和减法功能。要知道,人的欲望无穷无尽……

3.提供一个函数,使得我在加法完成之后做一些事情(必须要在函数加法函数返回之前做),比如打印结果,修改结果。 

#include <stdio.h>

int add(int a,int b)
{
return a+b;
} //提供一个可以在add之后随意做点事情的函数
//为了修改值,所以需要传递变量的指针
//函数传递了一个函数指针,其原型为:int (*)(int *)
int addAfterDoSomething(int a,int b,int (*AddAfter)(int *result))
{
int resutl = add(a,b);
AddAfter(&resutl);
return resutl;
} //要在add之后修改其值
int changeValue(int * value)
{
printf("Value Before Change : %d\n",*value);
*value = *value -;
printf("Value After Change : %d\n",*value);
} void main()
{
int a=,b=;
printf("a=%d\nb=%d\nResult=%d",a,b,addAfterDoSomething(a,b,changeValue));
getchar();
}

4.提供一个函数,使得我在加法完成之后做一些事情(必须要在函数加法函数返回之前做)之后,再做一些事情。

//是的正如你所想,局势在函数最后面再加一个函数指针
int addAfterDoSomething(int a,int b,int (*AddAfter)(int *result),int (*AddAfterAfter)(int *result))

5.提供一个函数,使得我在加法完成之后做一些事情(必须要在函数加法函数返回之前做)之后,再做一些事情之后,再做一些事情。

int addAfterDoSomething(int a,int b,int (*AddAfter)(int *result),int (*AddAfterAfter)(int *result),int (*AddAfterAfterAfter)(int *result))

  正如你所看,对于一个追求完美的人,上面黄色背景的那一串函数原型,太碍眼,太丑陋,看起来太复杂,太让人接受不了。

6.美化第3步的代码

//省略若干

#ifndef DoSomethingAfterAdd
#define DoSomethingAfterAdd int (*AddAfter)(int *result)
#endif //省略若干
//将
int addAfterDoSomething(int a,int b,int (*AddAfter)(int *result))
//替换为
int addAfterDoSomething(int a,int b,DoSomethingAfterAdd);

7.美化第四步的代码

//省略若干

#ifndef DoSomethingAfterAdd
#define DoSomethingAfterAdd int (*AddAfter)(int *result)
#endif //省略若干
//将
int addAfterDoSomething(int a,int b,int (*AddAfter)(int *result))
//替换为
int addAfterDoSomething(int a,int b,DoSomethingAfterAdd,DoSomethingAfterAdd);
//很明显,这样通不过编译!
//Error:重复的参数名

 正确的方法如下:

//定义一个带有参数的宏
#ifndef DoAfter(X)
#define DoAfter(X) int (*X)(int *result)
#endif //在编译代码时,编译器会将DoAfter(AddAfter)翻译为int (*AddAfter)(int *result)
//同理,DoAfter(AddAfterAfter) ==int (*AddAfterAfter)(int *result)
int addAfterDoSomething(int a,int b,DoAfter(AddAfter),DoAfter(AddAfterAfter));

   参考:https://gcc.gnu.org/onlinedocs/cpp/Macro-Arguments.html

8.美化第5步

//当然非常简单了……
int addAfterDoSomething(int a,int b,DoAfter(AddAfter),DoAfter(AddAfterAfter),DoAfter(AddAfterAfterAfter))
{
int resutl = add(a,b);
AddAfter(&resutl);
AddAfterAfter(&resutl);
AddAfterAfterAfter(&resutl);
return resutl;
}

9.美化第八步

//定义一个自定义类型FunAddAfter
//类型的值为一个指向函数的指针
typedef int(* FunAddAfter)(int *Result); //可以像使用int类型一样使用FunAddAfter
int addAfterDoSomething(int a,int b,FunAddAfter AddAfter,FunAddAfter AddAfterAfter,FunAddAfter AddAfterAfterAfter)
{
int resutl = add(a,b);
AddAfter(&resutl);
AddAfterAfter(&resutl);
AddAfterAfterAfter(&resutl);
return resutl;
} void main()
{
int a=,b=;
//在使用自定义类型时,需要像其他类型一样声明,赋值,然后使用 -.-废话
FunAddAfter change;
//函数名其实也就是一个指针,指向函数的地址
  change = changeValue;
printf("a=%d\nb=%d\nResult=%d",a,b,addAfterDoSomething(a,b,change));
getchar();
}

  到这里如果说teypdef的作用也就是美化的话,那么#define完全可以胜任所有的工作。

10.实现一个函数,使得可以在加法运算后执行若干个,未知个数个其他事情  

/定义一个自定义类型FunAddAfter
//类型的值为一个指向函数的指针
typedef int(* FunAddAfter)(int *Result); typedef struct _funList
{
FunAddAfter node;
struct _funList * nextNode;
} FunList,*PFunList; //仅仅打印值
int printfValue(int * value)
{
printf("the Value is :%d\n",*value);
return *value;
} //计算一个数的平方
int Squre(int * value)
{
printf("Squre value :%d is %d\n",*value,(*value)*(*value));
return *value=(*value)*(*value);;
} int DoManythingAfterAdd(int a, int b ,PFunList funlist)
{
int result = add(a,b);
while (NULL!=funlist)
{
result = funlist->node(&result);
funlist = funlist->nextNode;
}
return result;
} void main()
{
int a=,b=;
//在使用自定义类型时,需要像其他类型一样声明,赋值,然后使用 -.-废话
FunAddAfter change,squreValue,printvalue;
PFunList funlistHead,funListNode1,funListNode2;
//函数名其实也就是一个指针,指向函数的地址
change = changeValue;
squreValue=Squre;
printvalue = printfValue;
funlistHead =(FunList *) malloc(sizeof(FunList));
funListNode1 =(FunList *) malloc(sizeof(FunList));
funListNode2 =(FunList *) malloc(sizeof(FunList));
funlistHead->nextNode = funListNode1;
funlistHead->node = change;
funListNode1->nextNode = funListNode2;
funListNode1 ->node = printvalue;
funListNode2 ->node = squreValue;
funListNode2 ->nextNode = NULL;
printf("a=%d\nb=%d\nResult=%d",a,b,DoManythingAfterAdd(a,b,funlistHead));
getchar();
}

总结typedef的作用

1.满足Geek的简洁代码需求——正如你所看到的,函数的功能其实也可以通过一长串的函数原型来实现,或者使用宏定义,我也可以不使用typedef来完成功能,但是对于追求卓越的人,容忍不了看起来太长和复杂的函数定义

2.满足Geek的控制欲——除了C标准提供的基本类型,我也要定义一个自己的公司内部使用的类型,在公司不得使用int,而一定要使用tianchaoweida

typedef int tianchaoweida;

3.因为tepedef能定义自定义类型,所以就能够超越#Define,有自己独特存在的理由!如下:

FunAddAfter change,squreValue,printvalue;

源码下载

一步步理解typedef的更多相关文章

  1. 深入理解typedef

    首先请看看下面这两句:                  typedef int a[10];                  typedef void (*p)(void); 如果你能一眼就看出它 ...

  2. 理解typedef(转)

    // 从别人那转的,调整下格式便于阅读. 首先请看看下面这两句: typedef ]; typedef void (*p)(void); 如果你能一眼就看出它们的意思,那请不要再往下看了.如果你不太理 ...

  3. 如何理解typedef void (*pfun)(void)

    问题: 在刚接触typedef void (*pfun)(void) 这个结构的时候,存在疑惑,为什么typedef后只有一"块"东西,而不是两"块"东西呢?那 ...

  4. 函数指针-如何理解typedef void (*pfun)(void)

    问题: 在刚接触typedef void (*pfun)(void) 这个结构的时候,存在疑惑,为什么typedef后只有一"块"东西,而不是两"块"东西呢?那 ...

  5. 我是这样一步步理解--主题模型(Topic Model)、LDA

    1. LDA模型是什么 LDA可以分为以下5个步骤: 一个函数:gamma函数. 四个分布:二项分布.多项分布.beta分布.Dirichlet分布. 一个概念和一个理念:共轭先验和贝叶斯框架. 两个 ...

  6. 一步步理解linux字符设备驱动框架(转)

    /* *本文版权归于凌阳教育.如转载请注明 *原作者和原文链接 http://blog.csdn.net/edudriver/article/details/18354313* *特此说明并保留对其追 ...

  7. 通过100张图一步步理解CNN

    https://blog.csdn.net/v_july_v/article/details/79434745 Youtube上迄今为止最好的卷积神经网络快速入门教程 https://www.bili ...

  8. NRF51822+STM32bootload——typedef void (*Fun) (void) 理解

    1.typdef 用法如下所示 typedef unsigned char uint8_t; typedef unsigned short int uint16_t; typedef unsigned ...

  9. C语言语法 typedef小结

    在总结typedef之前,先了解一个专业术语: 常量指针(const pointer):常量指针在定义的时候必须被初始化,而且一旦初始化完成,则它的值就不能再改变. int errNumb = 0; ...

随机推荐

  1. mybatis逆向工程工具

    mybatis逆向工程 package com.cxy; import java.io.File; import java.util.*; import org.mybatis.generator.a ...

  2. POJ-2112 Optimal Milking(floyd+最大流+二分)

    题目大意: 有k个挤奶器,在牧场里有c头奶牛,每个挤奶器可以满足m个奶牛,奶牛和挤奶器都可以看成是实体,现在给出两个实体之间的距离,如果没有路径相连,则为0,现在问你在所有方案里面,这c头奶牛需要走的 ...

  3. poj 1664放苹果(递归)

    放苹果 Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 37377   Accepted: 23016 Description ...

  4. Array to List

    List<OisDiscountIndex> discountIndexes = Arrays.asList(new OisDiscountIndex[trades.size()]);

  5. ftp功能深度剖析 + 线程 031

    一 打印进度条 import time for i in range(20): # \r 回到行首打印内容 如果有同一行内容,那么就被抹掉了 n = '>'* i print('\r%s'%n, ...

  6. SQL Server外部链接时报错:Error locating serverInstance specified

    SQL Server外部链接时报错:Error locating server/Instance specified 连接时报错信息: 08001 sql server network interfa ...

  7. SElinux学习记录

    1.SELinux:是一种基于域类型模型的强制访问控制安全系统,由NSA编写设计成内核模块包含到内核中,相应的某些安全相关的应用也被打了SE Linux补丁 查看Selinux: ps -Z #查看S ...

  8. thinkphp Model的使用

    4.1 放在哪儿?项目/模块/Model目录下以本教程为例,Home模块的Model/Home/Model/目录下 4.2 model类文件叫什么?模型名: DemoModel.class.php 4 ...

  9. 第一章javascript简介

    javascript 当诞生于1995 最开始是目的是处理在服务器端进行的表单验证:让其在服务器端验证改为在服务端验证,大大提高速度(当时网络慢) 如今javascript是一门功能全面的语言,包含闭 ...

  10. Python数据可视化--matplotlib

    抽象化|具体化: 如盒形图 | 现实中的图 功能性|装饰性:没有装饰和渲染 | 包含艺术性美学上的装饰 深度表达|浅度表达:深入层次的研究探索数据 | 易于理解的,直观的表示 多维度|单一维度:数据的 ...