• ios从入门到放弃之C基础巩固宏定义、条件编译、文件包含、typedef、const关键字


    接着上一次https://www.cnblogs.com/webor2006/p/15468778.html继续

    宏定义:

    不带参数宏定义:

    预处理指令的概念:

    • C语言在对源程序进行编译之前,会先对一些特殊的预处理指令作解释(比如之前使用的#include文件包含指令),产生一个新的源程序(这个过程称为编译预处理),之后再进行通常的编译。

    • 为了区分预处理指令和一般的C语句,所有预处理指令都以符号“#”开头,并且结尾不用分号。

    • 预处理指令可以出现在程序的任何位置,它的作用范围是从它出现的位置到文件尾。习惯上我们尽可能将预处理指令写在源程序开头,这种情况下,它的作用范围就是整个源程序文件。

    • C语言提供供了多种预处理功能,如宏定义、文件包含、条件编译等。合理地使用预处理功能编写的程序便于阅读、修改、移植和调试,也有利于模块化程序设计。

    实践:

    比如通常我们遍历数组是这样写的对吧:

    其中对于数组长度如果不想动态计算,就可以使用宏定义,如下:

    也就是宏定义的格式为:

    #define 标识符 字符串

    其中的“#”表示这是一条预处理命令。凡是以“#”开头的均为预处理命令。“define”为宏定义命令。“标识符”为所定义的宏名。“字符串”可以是常数、表达式、格式串等。它会在程序编译成0和1之前,将所有宏名替换为宏的值。

    另外关于宏在使用中有一个非常容易犯的错误点,就是不要在后面加分号,比如咱们加个分号你会发现程序就报错了:

    另外宏定义的作用域是从定义的第一行开始,一直到文件末尾,但是如果你想提前终止宏定义可以使用:

    而它的使用场景一般可以对不变的东东进行提取,比如API地址的访问,通常项目的API地址可能域名是一样的,所以可以用宏定义把baseurl提取出来,如下:

    带参数的宏定义:

    对于宏定义还可以定义参数的,比如求两数的和我们通常是这样定义函数的:

    接下来咱们可以将这个求和的功能进行宏定义,如下:

    其中对于宏一定要明白,无论是有参数还是没参数的宏,它们都是不会做任何计算的,仅仅是在翻译成0和1之前做一个简单的“替换”。

    那对于有参数的宏定义啥时候用比较好呢?如果函数的功能比较简单,仅仅是做一些简单的运算则可以使用宏定义,使用宏定义效率更高,运行速度更快【因为其实不是代码替换,不像函数还得去函数地址中找存储空间,再给形参分配空间,再运算再返回】,但是如果函数比较复杂,不仅仅是一些简单的运算,那么还是得使用函数。

    下面再来定义一个宏,有一个细节需要揭露:

    呃,貌似不如预期呀,预期应该是(5 + 5) * (4 + 4),而现在变成了5 + 5 * 4 + 4了,此时宏定义则需要这样修改:

    接下来再来看一个问题:

    呃,又不如预期了,预期应该是PF(2),也就是4嘛,这里分析一下原因,其实就是将宏定义展开就知道了:

    按照从左至右的顺序来算出来,是不是刚好就是等于16?而预期应该展开是这样:

     

    其实解决起来也很简单,如下:

    也就是对于带参数的宏定义有如下两个注意点:

    1、一般情况下建议写带参数的宏的时候,给每个参数加上一个();

    2、一般情况下建议写带参数的宏的时候,给结果也加上一个();

    条件编译: 

    基本概念:

    在很多情况下,我们希望程序的其中一部分代码只有在满足一定条件时才进行编译,否则不参与编译(只有参与编译的代码最终才能被执行),这就是条件编译。

    为什么要使用条件编译?

    1、按不同的条件去编译不同的程序部分,因而产生不同的目标代码文件。有利于程序的移植和调试。

    2、条件编译当然也可以用条件语句来实现。 但是用条件语句将会对整个源程序进行编译,生成 的目标代码程序很长,而采用条件编译,则根据条件只编译其中的程序段1或程序段2,生成的目 标程序较短。

    #if-#else 条件编译指令:

    这个条件编译跟if...else写法非常类似,先来看一下我们平常写的:

    而改成条件编译就是这样了:

    是不是神似,对于这个条件编译有两个细节可以发现:

    1、条件编译的代码都是顶格写的,有别于咱们正常写的代码;

    2、为啥没有输出“牛逼”呢?这其实也很好理解,因为这是预处理指令,此时还在编译之前,是不会执行score变量的赋值的,当然就输出else的语句喽。 

    所以,这里有一个注意点:条件编译是不能用来判断变量的,因为在不同的生命周期,它一般会和宏定义结合使用,比如:

    对于通常的条件语句我们可以是多条对吧,对于条件编译也是一样的可以:

    那对于if和条件编译之间有啥区别呢,下面来总结一下:

    共同点:

    都可以对给定的条件进行判断,添加满足或者不满足都可以执行特定的代码。

    区别:

    1、生命周期不同:if是在运行时,而#if是在编译之前;

    2、#if需要一个明确的结束符号#endif,为啥呢?因为如果省略掉#endif,那么系统就不知道条件编译的范围,则会将满足条件之后的第二个条件之后的所有内容都删除。

    3、if会将所有的代码都编译到二进制当中,而#if只会将满足条件的部分一直到下一个条件的部分编译到二进制当中。

    条件编译的优点:缩小应用程序的大小,因为是部分编译。

    另外条件编译也是可以有多个else的,比如:

    使用条件编译指令调试bug:

    最典型的是你在debug时是需要输出一些调试信息的,但是到了release包时这些调试信息是不需要的对吧,此时就可以使用条件编译来控制这个日志的输出,比如:

    而如果发布上线了,改一下类型:

    这里,‘...’指可变参数。这类宏在被调用时,##表示成零个或多个参数,包括里面的逗号,一直到到右括弧结束为止。当被调用时,在宏体(macro body)中,那些符号序列集合将代替里面 的VA_ARGS标识符。

    其它写法:

    #ifdef 条件编译指令:

    格式为:

    它的功能是,如果标识符已被#define命令定义过则对程序段1进行编译;否则对程序段2进行编译。如果没有程序段2(它为空),本格式中的#else可以没有,即可以写为:

    下面来试一下:

    修改一下:

    #ifndef 条件编译指令:

    这个跟上面ifdef相关,比较简单:

    文件包含:

    关于文件包含其实就是用#include,天天在用:

    但是它还是有一些值得学习的细节,关于#include有两种写法:

    1、#include <>它会先去编译器环境下查找,如果找不到则再去系统的环境下查找,通常系统的库用的就是它;

    2、#include ""会先在当前文件查找,找不到再去编译器环境下查找,如果再找不到则再去系统的环境下查找。

    通常在使用这个包含指令时,会遇到如下两个问题。

    重复包含问题:

    它会将待包含的文件内容完整的拷贝过来,接着就有一个重复包含的问题了,下面看一下:

    而由于函数是可以重复声明,所以如果不小写包含了多次,编译运行是完全不会有问题的,比如:

    程序木影响,但是重复包含会降低编译效率,因为每遇到一个include都需要进行翻译成代码的过程, 所以为了防止重复包含,一般在头文件中会加入如下条件编译代码,如下:

    对于这些只做了解既可,因为在IDE中新建c文件时会自动帮你加上这些判断。

    循环包含问题:

    对于上面的重复包含头文件是不影响程序运行的对吧,但是如果是循环包含那就会编译出错,下面来还原一下循环包含错误出现的整个过程。这里再新建一个文件,里面定义一个减法函数:

    好,接下来有一个需求,就是在我们做减少操作之前 ,需要先进行加法运算,那对于加法运算不是在other.c已经封装好了么?所以:

    好,接下来回到main中调用一下:

    嗯,一切都木毛病, 接下来又来需求了,需要在加法之前,先做减法,也就是:

    一编译报错了,主要是报在两个头文件中了,如下:

     

    然后错误详情就是循环拷贝了:

    其实也很好理解,因为other.c中需要使用minus.h,而minus.h中也需要使用other.c,是不是相互依赖死循环了?那如何解决呢?也很简单,只需要单方面拷贝既可,像这样改:

    最后,还有一个间接拷贝的含义,现在这个场景正好可以揭示:

    这里只包含了other.h对吧,而other.h中又间接地包含了minus.h:

     

    所以,等于main.c中包含了两个.h文件,minus.h就是其间接包含,了解一下。

    typedef:

    关于它其实就是取别名,这里直接上代码把一些关键点过一下。

    给基本数据类型取别名:

      

    其中它还可以给自定义的类型再取别名,如:

    给结构体类型取别名:

    此时就可以给它取别名让其输写更加便捷:

     

    将Person的定义提到外面既可:

    而由于结构体的定义有三种形式,所以对应的取别名还有另外两种形式,看一下:

    以上三种方式一定都要熟悉,因为未来都会用得到的。

    给枚举取别名:

    同样还有其它定义方式:

    上面三种全是先定义枚举类型再给它取别名对吧,其实还可以在定义的同时取别名,如下:

    给指针取别名:

    普通指针:

    可能看着这指针*有点晕,于是乎可以给它也定义一个别名:

    函数指针:【需掌握】

    接下来还可以给函数指针取别名,这块需要好好掌握一下,本身写起来不是那么容易:

    好,接下来又来了一个函数指针:

    其中对于这两个函数指针的声明:

    有木有发现这俩函数指针的定义就除了指类名称不一样,其它都一模一样对吧,此时就可以定义一个别名:

    这个在实际中也是用得比较多的。

    typedef和宏定义区别:

    其实对于typedef定义别名,用宏定义也能达到类似的效果,下面看一下:

    那。。这俩到底有啥区别呢?这里有一个原则 :一般情况下如果要给数据类型起一个别名建议用typedef,而不要用#define,下面再看一个例子:

    这貌似没看出啥问题对吧,好,下面再看:

    接下来再用宏定义:

    为啥会报错呢?其实也很容易想明白,宏定义本身就是字符替换,对于咱们这个程序替换之后的效果其实是:

    很明显这样是有问题,所以记住这个原则既可,下面来总结一下区别。

    typedef与函数的区别:

    从整个使用过程可以发现,带参数的宏定义,在源程序中出现的形式与函数很像。但是两者是有本质区别的:

    1> 宏定义不涉及存储空间的分配、参数类型匹配、参数传递、返回值问题

    2> 函数调用在程序运行时执行,而宏替换只在编译预处理阶段进行。所以带参数的宏比函数具有更高的执行效率 

    typedef和#define的区别:

    用宏定义表示数据类型和用typedef定义数据说明符的区别。

    • 宏定义只是简单的字符串替换,是在预处理完成的

    • typedef是在编译时处理的,它不是做简单的代换,而是对类型说明符重新命名。被命名的标识符具有类型定义说明的功能

    const关键字:

    对于const也是实际会用得比较多的,所以也需要好好掌握。

    基本概念:

    const是一个类型修饰符

    • 使用const修饰变量则可以让变量的值不能改变。
    • 常类型是指使用类型修饰符const说明的类型,常类型的变量或对象的值是不能被更新的。

    const有什么主要的作用?

    1、可以定义const常量,具有不可变性。 例如:

    2、便于进行类型检查,使编译器对处理内容有更多了解,消除了一些隐患。

    编译器就会知道i是一个常量,不允许修改。

    3、可以避免意义模糊的数字出现,同样可以很方便地进行参数的调整和修改。 同宏定义一样,可以做到不变则已,一变都变!如(1)中,如果想修改Max的内容,只需要:const int Max=you want;即可。

    4、可以保护被修饰的东西,防止意外的修改,增强程序的健壮性。 还是上面的例子,如果在 函数体内修改了i,编译器就会报错:

    5、可以节省空间,避免不必要的内存分配:

    6、提高了效率。编译器通常不为普通const常量分配存储空间,而是将它们保存在符号表中,这使得它成为一个编译期间的常量,没有了存储与读内存的操作,使得它的效率也很高。

    使用const:

    1、修饰一般常量一般常量是指简单类型的常量。这种常量在定义时,修饰符const可以用在类型说明符前,也可以用在类型说明符后。

    2、修饰常数组(值不能够再改变了)定义或说明一个常数组可采用如下格式:

    3、修饰函数的常参数const修饰符也可以修饰函数的传递参数,格式如下: void Fun(const int Var); 告诉编译器Var在函数体中的无法改变,从而防止了使用者的一些无 意的或错误的修改。

    4、修饰函数的返回值: const修饰符也可以修饰函数的返回值,是返回值不可被改变,格式如下:

    5、修饰常指针:【重点】

    对于const使用最难的就是用在指针声明上了,下面好好学一下这块:

    此时定义的指针内容是能够被更改对吧,那如果你不想让更改,此时就可以加上const关键字了,如下:

    这是因为如果const写在指针类型的左边,则指针的指向可以变,但是指向的内存空间中的值是不能改变的,为了更好的理解,下面换一个程序,将其拆解一下:

    好,接下来加const啦:

    是不是这里可以说明:如果const写在指针类型的左边, 那么意味着指向的内存空间中的值不能改变, 但是指针的指向可以改变。

    接下来继续修改:

    同样报错,这里又说明:如果const写在指针的数据类型和*号之间, 那么意味着指向的内存空间中的值不能改变, 但是指针的指向可以改变。

    接下来再来修改:

    说明:如果const写在指针的右边(数据类型 * const), 那么意味着指针的指向不可以改变, 但是指针指向的存储空间中的值可以改变

    所以对以上情况总结一下:

    1、如果const写在指针类型的左边, 那么意味着指向的内存空间中的值不能改变, 但是指针的指向可以改变;
    2、如果const写在指针的数据类型和*号之间, 那么意味着指向的内存空间中的值不能改变, 但是指针的指向可以改变;
    3、如果const写在指针的右边(数据类型 * const), 那么意味着指针的指向不可以改变, 但是指针指向的存储空间中的值可以改变;

    规律:
    1、如果const写在指针变量名的旁边, 那么指针的指向不能变, 而指向的内存空间的值可以变;
    2、如果const写在数据类型的左边或者右边, 那么指针的指向可以改变, 但是指向的内存空间的值不能改变;

    其实有一种比较简单的记法就是:

    只要const修饰的是指针变量名,那么就代表指针的指向是不能变,但其指向的存储空间的值可以变,除这种情况之外,都是相反的。

    总结:

    至此,终于把C基础相关的知识点基本给学完了,接下来则开启跟IOS相关的OC语言的学习了,回顾整个学习过程,说实话很多都不是很难,之前都有学过,但是也有很多新的知识点是之前不知道的,这就是所谓的温故知新吧,虽说整个学习的节奏非常慢【光C就学了一年多。。】,但是整个学习是比较踏实的,戒骄戒躁是学习任何技能的大前提,期待下次OC的学习之旅~~

  • 相关阅读:
    U3D shaderlab 相关指令开关
    CCF NOI1073
    CCF NOI1185
    CCF NOI1077(自然数的拆分问题 )
    CCF NOI1070(汉诺塔)
    CCF NOI1069
    2018年全国多校算法寒假训练营练习比赛(第一场)G.圆圈
    poj1941(递归)
    Codeforce914B (Conan and Agasa play a Card Game)
    Codeforce916B
  • 原文地址:https://www.cnblogs.com/webor2006/p/15824904.html
Copyright © 2020-2023  润新知