肥宅钓鱼网
当前位置: 首页 钓鱼百科

c语言函数程序设计的思路(一起学C程序设计第七课)

时间:2023-07-02 作者: 小编 阅读量: 2 栏目名: 钓鱼百科

递归调用所谓函数的递归调用就是返回值里有部分构成是函数自身。在写递归函数需要注意的点是递归是有限的,一定要有终止的逻辑,让递归函数不再调用自身从而结束递归;否则无限递归下去也是一种“死循环”。static局部变量当static关键字作用在内部变量上,意味着该局部变量不随着所在函数调用完毕后释放内存,即存储方式由动态存储变为静态存储。

C语言函数

在使用visual studio练习编程的时候,可能会出现这样的情况:做了很多的语法练习,既有单独一行语句的练习,也有解决一个课后习题这样一个完整的程序,它们都在同一个main()主函数中,在调试中可能要注释掉一些代码,单独去看当前的一段代码,这样操作有点繁琐,而且主函数越来越“臃肿”显得结构混乱,那怎样能解决这个问题呢?如果把要调试的这段代码独立出来,和其他代码互不影响,需要的时候再去主函数里调用,这样就会使得练习调试方便的多,这段独立代码的形式就是今天的主题——函数。

注意,请认真学习完《C程序设计(第五版)》第七章后再阅读本文会有更大的收获。

函数

我们可以把函数理解成一个迷你型的程序,它有输入输出,也实现了某些算法逻辑。

函数输入——参数

一般基础类型参数,如整型、浮点、字符等,实参的值传递给形参,它们互不影响。

数组类型的参数,由于数组实参传递的是数组首个元素的存储地址,所以形参实际上还是指向同一个数组的,如果形参的值发生变化,相应的实参也会受影响。

函数输出——返回值

通常,我们会统一函数定义的数据类型和return值的数据类型,应该避免“强制数据类型转换”的情况,让程序更加的严谨。

当然,严谨也就意味着失去一些“自由度”,其他的语言比如PHP、Python没有这样的要求,它们甚至没有函数数据类型这样的定义。毕竟每种语言的设计初衷和理念都不太相同,在今后的学习使用中去体会设计者的思路。

函数体

函数体是函数的核心,函数功能的实现都在这里进行编码。在构建函数体的过程中,推荐的做法是先写出框架,再补齐逻辑代码。比如函数体内有一个for循环,那就先出for循环的框架,然后再写continue以及break跳出循环的判断节点,最后再来补齐这里面的逻辑。

函数调用

自定义函数调用

当前源文件调用,在自定义函数之前调用要声明,之后则不用声明。

其他源文件调用,要用extern进行声明。

系统函数调用

使用系统函数前要包含进来相应的头文件,比如使用数学函数sqrt()要在文件开头包含math.h头文件:#include <math.h>,前面还用过字符串相关的系统函数,以及最普遍的系统标准输入输出函数。

嵌套调用

函数A的函数体内调用函数B,函数B的函数体内调用函数C……需要注意的是不能“反调用”,比如A调用B,B又调用了A,或者A调用B,B调用C,C里又调用A,这样就形成了“死循环”。

递归调用

所谓函数的递归调用就是返回值里有部分构成是函数自身。在写递归函数需要注意的点是递归是有限的,一定要有终止的逻辑,让递归函数不再调用自身从而结束递归;否则无限递归下去也是一种“死循环”。

如果递归过程中要存储一些过程中的数据,这时候可以使用static局部变量使得静态化存储,或者以形参数组的形式在函数内部使用。

函数作用域

C语言中,默认自定义的函数是外部函数,即全局函数,可以被其他源文件调用;而内部函数则是相对于函数所在的源文件来界定的,只能在当前源文件内被调用。

所以,函数作用域最小范围是其所在的源文件,除此之外就是作用于全局(所有源文件)。

变量作用域和存储类型

书中的表7.2总结得很全面,了解作用域和存储类型,使得我们在写程序的时候更加严谨规范,不随意定义全局的变量,使程序数据更加“安全”。

对于初学者来说,这些偏理论的知识往往都比较难理解,也很枯燥。笔者推荐在学习的过程中结合visual studio进行实操练习,通过代码辅助增强对这些理论知识的理解;做到知其然也知其所以然,培养好的编程习惯,从底层了解变量的作用域和存储特性,也为学习其他的高级语言做铺垫。

static关键字

static外部变量/外部函数

当static关键字作用在外部变量/外部函数上,意味着该外部变量/外部函数只能在当前的源文件内进行使用,无法跨文件调用。

static局部变量

当static关键字作用在内部变量上,意味着该局部变量不随着所在函数调用完毕后释放内存,即存储方式由动态存储变为静态存储。

实战编程

首先,看一下练习编程的目录结构和多个源文件互相调用运行的规则,顺便把刚学习的知识运用上。

练习目录结构

在源文件目录下创.c结尾的源文件,程序运行的入口文件是main.c,从main()函数开始执行。笔者把每一章节的程序练习单独建立对应的源文件如lesson7.c,在lesson7.c里再创建lesson7()函数,这个函数相当于它所在源文件的主函数,然后在main()函数里调用lesson7(),具体结构如下:

注意,这里的单个源文件中定义的练习函数都加了static关键词表示仅在当前源文件中调用,这样避免与其他源文件同名时造成冲突。

写两个函数分别求两个整数的最大公约数和最小公倍数

这个题目在之前的练习中已经做过,具体算法不再分析,参考一下:

static int maxCommonDivisor(int a, int b) {int divisor = 1;for (int i = 1; i <= a; i){if (a % i == 0 && b % i == 0) {divisor = i;}}return divisor;}static int minCommonMultiple(int a, int b) {int multiple = 1;int min = a > b ? b : a;int i = 1;while (1){multiple = min * i;if (multiple % a == 0 && multiple % b == 0) {return multiple;}i;}}

写一个函数,使给定的一个3x3的二维数组转置,即行列互换。

这个题目很简单,比较直接的实现思路是把给定的数组拷贝一份,然后循环把数组的元素重新赋值即可,赋值逻辑为:array[i][j] = arrayCopy[j][i]。为了加深理解函数的嵌套调用,我们增加两个函数:一个是输出二维数组函数,做转置前后的输出对比;另一个是拷贝二维数组的函数,供转置函数去调用,代码参考如下:

static void transpose(int array[3][3]);static void printfArray(int array[3][3]);int array[3][3] = {{1, 2, 3},{4, 5, 6},{7, 8, 9}};printfArray(array, 3);transpose(array);printf("=====================\n");printfArray(array, 3);static void printfArray(int array[][3], int length) {for (int i = 0; i < length; i) {for (int j = 0; j < length; j){printf("%d\t", array[i][j]);}printf("\n");}}static void transpose(int array[3][3]) {int length = 3;int array_copy[3][3];static void copyArray(int dest_array[][3], int origin_array[][3], int length);copyArray(array_copy, array, length);for (int i = 0; i < length; i) {for (int j = 0; j < length; j){array[i][j] = array_copy[j][i];}}}static void copyArray(int dest_array[][3], int origin_array[][3], int length) {for (int i = 0; i < length; i) {for (int j = 0; j < length; j){dest_array[i][j] = origin_array[i][j];}}}

用递归法将一个整数n转换成字符串。例如输入483,应输出字符串“483”。n的位数不确定,可以是任意位数的整数。

本题明确使用递归函数,函数的输入是一个n位数,输出是个位数(末位数)。算法的核心就是求每一位的数字:

  1. n位数直接对10取余,得到个位上的数字
  2. n位数除以10取整,然后再对10取余,得到十位上的数字
  3. n位数除以100取整,然后再对10取余,得到百位上的数字

总结一下这个算法就是:对于输入的数据算出末位数字并输出,然后把输入数据的末位截掉变成下次调用的输入值,只要当前数据不是1位数,那就继续递归调用下去。

递归调用的条件是至关重要的,这里通过判断对10取余的结果是不是他自身来决定是否一直递归调用下去,代码参考:

static void printLastnumber(long number) {int n = number % 10;if (n != number) {printLastNumber(number / 10);}n = (n < 0 && n != number) ? -n : n;printf("%d", n);}

PS:如果输入值为负数,则判断一下不是取到最后一位就把负数转成正数输出,不判断会把“-123”输出成“-1-2-3”。

总结

函数的定义和设计参考以下几点:

  1. 复用性,在项目里有很多模块,如果模块1中要按照某种算法计算两个数的关联性,模块2中也要用同样的算法计算两个数的关联性,这样我们就可以把计算两个数关联性的代码提取出来作为单独的函数,再去供不同的模块调用。
  2. 可扩展,一个函数根据用户数据做商品推荐,随着用户行为的不断丰富,函数体的推荐算法也不断的优化,在不改变函数的输入和输出的前提下去做功能性的扩展和补充。
  3. 解耦合,一个函数A处理输入的时候又要用函数B来预处理输入,或者经过计算后再用函数C去输出,这里A依赖其他函数B和C,使它们之间耦合性增强,牵一发而动全身,这样的函数设计是不提倡的。
  4. 抽象,在之前的文章也提到过“要把业务问题抽象成数学问题”,这一点在后面做实际的项目中才能有深刻的认知。

在熟练掌握C语言函数的用法和一些特性之后,就可以组织构建模块进行常规的项目开发了,后续笔者会做一些简单的但是模块体系相对完善的项目演练,我们一起通过具体项目的开发练习来把C语言这门工具用起来。

往期文章

一起学《C程序设计》第六课——数组、字符串及实战练习

一起学《C程序设计》第五课——循环控制及实战练习

一起学《C程序设计》第四课——if语句、switch语句及实战练习

一起学《C程序设计》第三课——数据结构、运算符、表达式和语句

一起学《C程序设计》第二课——算法

一起学《C程序设计》第一课——C语言概述和学习前的准备、意识

C程序设计(谭浩强)——第五版和第三版对比

    推荐阅读
  • 商业模式包括哪些内容(什么是商业模式)

    简单来说,商业模式就是一种维持生存的方式。如果从操作层面来看,商业模式就是生产价值的内部过程。商业模式的基本内容例如,B2C就是一种常见的电子商务模式,是企业直接面向消费者销售产品和服务的商业零售模式,如天猫、京东、网易考拉、苏宁易购等大型互联网企业采用的都是这种商业模式。京东会从有品质保证的品牌厂商进货,然后通过线上渠道向用户销售,通过“自营品牌”的独特商业模式来保持竞争力。

  • 什么玉是好玉怎么判断(判断玉的好坏的方法)

    就是陈性《玉记》中所说的,要体如凝脂,精光内蕴,质厚温润,脉理坚密。和田玉又和翡翠不同,翡翠要求鲜明光亮光泽外射,而和田玉则要求精光内蕴,光泽蕴含在里面,而不是透在外边。和田玉因颜色不同,有白玉、黄玉、青玉、碧玉、墨玉之分,一般来说,在质地相同或相近的情况下,白玉为贵,黄玉次之,而青玉和青白玉等价值就要低些。所谓玉不琢不成器,雕工对于和田玉有着极其重要的意义。

  • 在家自制大肉肠的做法(好吃不贵的肉肠做法)

    以下内容希望对你有帮助!在家自制大肉肠的做法将鳕鱼去皮,肉取下,刺弄干净。将鱼肉放入料理机打成肉蓉。将肉馅、淀粉、白胡椒,味精放入,打入一个鸡蛋,搅拌均匀。最后放入盐,再次拌匀。将盐渍肠衣水洗,泡开,灌水试验看有没有漏点。将肠衣套在小漏斗上,开始灌制。灌好后,两端系紧,入锅蒸20分钟,中间注意观察,间隔用针放气,以免肠衣涨破。熟透出锅,过凉水。凉凉,切片装盘。

  • 新年祝福贺卡加合适的文字(不落俗套可写进贺卡的新年祝福)

    不落俗套可写进贺卡的新年祝福01.新年快乐“祝你永远开心”的意思也不是要你一丝烦恼都没有,那样不太现实只是希望你不用那么在乎是否快乐,希望你拥有想哭就哭,想笑就笑的情绪自由也希望你不管怎样都能有面对负面情绪的能力和重新开始的勇气。

  • 怎么查询个人详细征信报告(如何查询个人征信报告)

    可以说征信报告就是一台高清无码的照妖镜,把我们每个人照的干干净净、体无完肤。奉劝有准备结婚的同志,一定要一起查询一下彼此的信用报告,以防坠入火坑。注册的时候需要个人身份证号、银行卡或信用卡。第三种方法就是打开各大手机银行App,上面会提供个人征信报告查询的,下面主要给大家讲一下光大银行的。打开光大银行手机App,登录后点击底部“贷款”,进入页面后,拉到最底下“贷款服务”,里面有个“个人信用报告”。

  • 水果沙拉怎么做窍门(你水果沙拉的正宗做法)

    ,今天小编就来聊一聊关于水果沙拉怎么做窍门?接下来我们就一起去研究一下吧!水果沙拉怎么做窍门把水果洗净去皮切块状再把水果倒进盆子里倒入酸奶搅拌均匀,再倒在盘子上淋上沙拉酱就搞定了。

  • 怎么卸载打印机驱动(怎么卸载打印机驱动重新安装)

    3、任意选择一个打印机,然后点击“打印服务器属性”。

  • dnf活动增幅器使用方法(原来还有这么多增幅器可以拿)

    前言10月14日版本更新,除了职业平衡之外,作为惯例,还额外更新了两个活动,一是最强大作战,二是地下城采矿大亨。活动期间,每周通关三次推荐地下城可领取一次签到奖励,签到的次数可以领取对应的阶段奖励,两种奖励礼盒均为账号绑定。获取奖励的途径主要分为两种,一种是小游戏内的熔炉抽奖,另一类是商店兑换奖励,以上所有奖励均为账号绑定。

  • 新产品能申请实用新型专利吗(计算机程序申请实用新型专利要注意什么)

    新产品能申请实用新型专利吗针对实用新型专利而言,由于申请的项目多通过率也较高,不可避免的出现了许多并不符合专利要求的产品去申请实用新型专利。该项案例涉及了计算机的程序,我们在申请实用新型专利时学会判断在提交的权利要求中是否既包含产品特征有包括改进方案,如果是的话,这样产品本身就不属于实用新型专利保护的客体哦。

  • 鲁迅现代诗大全100首(鲁迅经典诗作十首)

    鲁迅的情诗我的失恋我的所爱在山腰;想去寻她山太高,低头无法泪沾衣。注释:这首诗创作于1924年10月3日,最初发表于1924年12月8日《语丝》周刊第4期。作者鲁迅在《野草·英文译本》序中说:“因为讽刺当时盛行的失恋诗,作《我的失恋》”。未敢翻身已碰头。横眉冷对千夫指,俯首甘为孺子牛。旧时迷信,以为人的命运中犯了华盖星,运气就不好。