当前位置: 首页 > news >正文

百度录入网站企业培训机构

百度录入网站,企业培训机构,武汉网站建设的价格,小说网站开发文档1.预定义符号 C语言设置了一些预定义符号,可以直接使用,预定义符号也是在预处理期间处理的 __FILE__ //进⾏编译的源⽂件 __LINE__ //⽂件当前的⾏号 __DATE__ //⽂件被编译的⽇期 __TIME__ //⽂件被编译的时间 __STDC__ //如果编译器遵循ANSI C&#…

1.预定义符号

C语言设置了一些预定义符号,可以直接使用,预定义符号也是在预处理期间处理的

__FILE__ //进⾏编译的源⽂件
__LINE__ //⽂件当前的⾏号
__DATE__ //⽂件被编译的⽇期
__TIME__ //⽂件被编译的时间
__STDC__ //如果编译器遵循ANSI C,其值为1,否则未定义

例子:

printf("file:%s line:%d\n", __FILE__, __LINE__);

2.define定义常量

基本语法:

#define name stuff

例子:

#define MAX 1000
#define reg register //为 register这个关键字,创建⼀个简短的名字
#define do_forever for(;;) //⽤更形象的符号来替换⼀种实现
#define CASE break;case //在写case语句的时候⾃动把 break写上。
// 如果定义的 stuff过⻓,可以分成⼏⾏写,除了最后⼀⾏外,每⾏的后⾯都加⼀个反斜杠(续⾏符)。
#define DEBUG_PRINT printf("file:%s\tline:%d\t \date:%s\ttime:%s\n" ,\__FILE__,__LINE__ , \__DATE__,__TIME__ )

思考:在define定义标识符的时候,要不要在最后加上

#define MAX 1000;
#define MAX 1000

建议不要加上;,这样容易导致问题
比如:

if(condition)max = MAX;
elsemax = 0;

如果加了分号的情况,等替换之后,ij和else之间就是2条语句,而没有大括号的时候,if后边只能有一条语句,这里会出现语法错误

3.#define定义宏

#define机制包括了一个规定,允许把参数替换到文本中,这种实现通常称为宏(macro)或者定义宏(define macro)
申明方式

#define name( parament-list ) stuff

其中parament-list是一个由逗号隔开的符号表,它们可能出现在stuff中

  • 注:参数列表的左括号必须与name紧邻,如果两者之间有任何空白存在,参数列表就会被解释为stuff的一部分

举例:

#define SQUARE( x ) x * x

这个宏接受收一个参数x,如果在上述声明之后,把SQUARE(5);置于程序走中,预处理器就会用下面的表达式替换前5*5

警告:
这个宏存在一个问题
例子1:

//观察下⾯的代码段:
int a = 5;
printf("%d\n" ,SQUARE( a + 1) );//这里的参数x会被替换成a+1,所以这条语句实际上变成了
printf ("%d\n",a + 1 * a + 1 );//替换产生的表达式没有按照预想的次序进行求值
//在宏定义上加上两个括号,这个问题便轻松的解决了:
#define SQUARE(x) (x) * (x)//这样预处理之后就产⽣了预期的效果:
printf ("%d\n",(a + 1) * (a + 1) );

例子2:

//2
#define DOUBLE(x) (x) + (x)//定义中我们使⽤了括号,想避免之前的问题,但是这个宏可能会出现新的错误。
int a = 5;
printf("%d\n" ,10 * DOUBLE(a));//看上去,好像打印100,但事实上打印的是55.我们发现替换之后:
printf ("%d\n",10 * (5) + (5));//乘法运算先于宏定义的加法,所以出现了 55 .
//这个问题,的解决办法是在宏定义表达式两边加上⼀对括号就可以了。
#define DOUBLE( x) ( ( x ) + ( x ) )

提示:用于对数值表达式进行求值的宏定义应该加上合适的括号,避免在使用宏时由于参数中的操作符或者邻近操作符之间不可预料的相互作用

4.带有副作用的宏参数

当宏参数在宏的定义中出现超过一次的时候,如果参数带有副作用,在使用这个宏时会出现危险我,导致不可预测的后果,副作用就是表达式求值时出现的永久性效果
例如:

x+1;//不带副作⽤
x++;//带有副作⽤

MAX宏可以证明具有副作用参数引起的问题

#define MAX(a, b) ( (a) > (b) ? (a) : (b) )
...
x = 5;
y = 8;
z = MAX(x++, y++);
printf("x=%d y=%d z=%d\n", x, y, z);//输出的结果是什么?
//z = ( (x++) > (y++) ? (x++) : (y++))
//所以输出的结果是:x=6 y=10 z=9

5.宏替换的规则

在程序中扩展#define定义符号和宏时,需要涉及几个步骤
1.调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号,如果是,首先被替换
2.替换文本随后被插入到程序中原来文本的位置,对于宏,参数名被他们的值所替换
3.最后,再次对结果文件扫描,看看它是否包含任何由#define定义的符号,如果是,重复上面过程

注意:
1.宏参数和#define定义中可以出现其他#define定义的符号,但是对于宏,不能出现递归
2.当预处理器搜索#define定义的符号时,字符串常量的内容并不被搜索

6.宏函数的对比

宏通常被应用于执行简单的运算
⽐如在两个数中找出较⼤的⼀个时,写成下⾯的宏,更有优势⼀些。

#define MAX(a, b) ((a)>(b)?(a):(b))

为什么不用函数来完成?
1.用于调用函数的和从函数返回的代码可能比实际这个小型计算工作所需要的时间更多,所以宏比函数在程序的规模和速度方面更胜一筹
2.更为重要的是函数的参数必须声明为特定的类型,所以函数只能在类型合适的表达式上使用,反之这个宏可以适用于整形型,长整形型,浮点型等可以用来比较的类型,宏是类型无关的

和函数相比宏的优势:
1.每次使用宏的时候,一份宏定义的代码被插入到程序中,除非宏比较短,否则可能大幅度增加程序的长度
2.宏是没法调试的
3.宏由于类型无关,不够严谨
4.宏可能会带来运算符优先级问题,导致程序容易出错

宏有时可以做到函数做不到的事情,宏的参数可以出现类型,函数做不到

#define MALLOC(num, type)\(type )malloc(num sizeof(type))...
//使⽤MALLOC(10, int);//类型作为参数
//预处理器替换之后:(int )malloc(10 sizeof(int));

宏和函数的⼀个对⽐

在这里插入图片描述

7.#和##

7.1#运算符

#运算符将宏的一个参数转换为字符串字面量,它仅允许出现在带参数的宏替换列表中
#运算符所执行的操作可以理解为“字符串化”
有变量int a=10的时候,想打印the value of a is 10

#define PRINT(n) printf("the value of "#n " is %d", n);

当我们按照下⾯的⽅式调⽤的时候:
PRINT(a);//当我们把a替换到宏的体内时,就出现了#a,⽽#a就是转换为"a",时⼀个字符串
代码就会被预处理为:

printf("the value of ""a" " is %d", a);

运⾏代码就能在屏幕上打印:

the value of a is 10

7.2##运算符

##可以把位于它两边的符号合成一个符号,它允许宏定义从分离的文本片段创建标识符,##被称为记号粘合
这样的连接必须产生一个合法的标识符,否则结果是未定义
这⾥我们想想,写⼀个函数求2个数的较⼤值的时候,不同的数据类型就得写不同的函数。
⽐如:

int int_max(int x, int y)
{return x>y?x:y;
}
float float_max(float x, float y)
{return x>yx:y;
}

但是这样写起来太繁琐了,现在我们这样写代码试试:

//宏定义
#define GENERIC_MAX(type) \
type type##_max(type x, type y)\
{ \return (x>y?x:y); \
}

使⽤宏,定义不同函数

GENERIC_MAX(int) //替换到宏体内后int##_max ⽣成了新的符号 int_max做函数名
GENERIC_MAX(float) //替换到宏体内后float##_max ⽣成了新的符号 float_max做函数名
int main()
{//调⽤函数int m = int_max(2, 3);printf("%d\n", m);float fm = float_max(3.5f, 4.5f);printf("%f\n", fm);return 0;
}

输出:

3
4.500000

在实际开发过程中##使⽤的很少,很难取出⾮常贴切的例⼦。

8.命名约定

⼀般来讲函数的宏的使⽤语法很相似。所以语⾔本⾝没法帮我们区分⼆者。
那我们平时的⼀个习惯是:

  • 把宏名全部⼤写
  • 函数名不要全部⼤写

9.#undef

这条指令用于移除一个宏定义

#undef NAME
//如果现存的⼀个名字需要被重新定义,那么它的旧名字⾸先要被移除。

10.命令行定义

许多C 的编译器提供了⼀种能⼒,允许在命令⾏中定义符号。⽤于启动编译过程。
例如:当我们根据同⼀个源⽂件要编译出⼀个程序的不同版本的时候,这个特性有点⽤处。(假定某个程序中声明了⼀个某个⻓度的数组,如果机器内存有限,我们需要⼀个很⼩的数组,但是另外⼀个机器内存⼤些,我们需要⼀个数组能够⼤些。)

#include <stdio.h>
int main()
{int array [ARRAY_SIZE];int i = 0;for(i = 0; i< ARRAY_SIZE; i ++){array[i] = i;}for(i = 0; i< ARRAY_SIZE; i ++){printf("%d " ,array[i]);}printf("\n" );return 0;
}

编译指令

//linux 环境演⽰
gcc -D ARRAY_SIZE=10 programe.c

11.条件编译

在编译⼀个程序的时候我们如果要将⼀条语句(⼀组语句)编译或者放弃是很⽅便的。因为我们有条
件编译指令。
⽐如说:

  • 调试性的代码,删除可惜,保留⼜碍事,所以我们可以选择性的编译。
#include <stdio.h>
#define __DEBUG__
int main()
{int i = 0;int arr[10] = {0};for(i=0; i<10; i++){arr[i] = i;#ifdef __DEBUG__printf("%d\n", arr[i]);//为了观察数组是否赋值成功。 #endif //__DEBUG__}return 0;
}

常见的条件编译指令

1.
#if 常量表达式//...
#endif
//常量表达式由预处理器求值。
如:
#define __DEBUG__ 1
#if __DEBUG__//..
#endif
2.多个分⽀的条件编译
#if 常量表达式//...
#elif 常量表达式//...
#else//...
#endif
3.判断是否被定义
#if defined(symbol)
#ifdef symbol
#if !defined(symbol)
#ifndef symbol
4.嵌套指令
#if defined(OS_UNIX)#ifdef OPTION1unix_version_option1();#endif#ifdef OPTION2unix_version_option2();#endif
#elif defined(OS_MSDOS)#ifdef OPTION2msdos_version_option2();#endif
#endif

12.头文件的包含

12.1头文件被包含的方式

12.1.1本地文件包含

#include "filename"

查找策略:先在源⽂件所在⽬录下查找,如果该头⽂件未找到,编译器就像查找库函数头⽂件⼀样在
标准位置查找头⽂件。
如果找不到就提⽰编译错误。
Linux环境的标准头⽂件的路径:

/usr/include

VS环境的标准头⽂件的路径:

C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\include
//这是VS2013的默认路径

注意按照自己的安装路径查找

12.1.2库文件包含

#include <filename.h>
查找头⽂件直接去标准路径下去查找,如果找不到就提⽰编译错误。
这样是不是可以说,对于库⽂件也可以使⽤ “” 的形式包含?
答案是肯定的,可以,但是这样做查找的效率就低些,当然这样也不容易区分是库⽂件还是本地⽂件了。

12.2嵌套文件包含

#include 指令可以使另外⼀个⽂件被编译。就像它实际出现于 #include 指令的地⽅⼀样。
这种替换的⽅式很简单:预处理器先删除这条指令,并⽤包含⽂件的内容替换。
⼀个头⽂件被包含10次,那就实际被编译10次,如果重复包含,对编译的压⼒就⽐较⼤。
test.c

#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
int main()
{return 0;
}

test.h

void test();
struct Stu
{int id;char name[20];
};

如果直接这样写,test.c⽂件中将test.h包含5次,那么test.h⽂件的内容将会被拷⻉5份在test.c中。
如果test.h ⽂件⽐较⼤,这样预处理后代码量会剧增。如果⼯程⽐较⼤,有公共使⽤的头⽂件,被⼤家都能使⽤,⼜不做任何的处理,那么后果真的不堪设想。
如何解决头⽂件被重复引⼊的问题?答案:条件编译。
每个头⽂件的开头写:

#ifndef __TEST_H__
#define __TEST_H__
//头⽂件的内容
#endif //__TEST_H__

或者

#pragma once

就可以避免头⽂件的重复引⼊。
注:
推荐《⾼质量C/C++编程指南》中附录的考试试卷(很重要)。
思考题:

  1. 头⽂件中的 ifndef/define/endif是⼲什么⽤的?
  2. #include <filename.h> 和 #include "filename.h"有什么区别?

13.其他预处理指令

#error
#pragma
#line
...
不做介绍,⾃⼰去了解。
#pragma pack()在结构体部分介绍

参考《C语⾔深度解剖》学习

http://www.dinnco.com/news/53900.html

相关文章:

  • 用织梦怎么修改网站首页商品顺序seo信息查询
  • 合作社做网站有用吗seo入门讲解
  • 杭州营销型网站seo美式
  • 广州城市职业学院门户网站关键词seo价格
  • 商城网站建设新闻百度seo搜索引擎优化培训
  • 网站建设wangzhii免费网站申请注册
  • 黑客黑网站是做网站全球新闻最新消息
  • 大网站制作seo关键词排名优化
  • 做网站链接还要服务器吗买灵宝seo公司
  • 免费建设网站那个好广西seo公司
  • 品牌网站建设推广专业seo培训
  • 长春好的做网站公司有哪些怎么做免费的网站推广
  • 2016做砸了的小网站企业文化标语
  • 营销项目策划公司重庆seo整站优化方案范文
  • 手把手教你用动易做网站热搜词工具
  • 广西网站建设公司电话怎么理解搜索引擎优化
  • 外贸网站增加权重淘宝代运营
  • 网站cn和com有什么区别大兵seo博客
  • 做修图网站电脑配置线上推广具体应该怎么做
  • 宝石网站建设如何引流推广产品
  • 网站建设与管理论文的总结产品怎么进行推广
  • 做网站电商云数据库有用吗百度163黄页关键词挖掘
  • 高端网站开发建设晨阳seo服务
  • 沙田镇网站仿做怎么网站推广
  • 本地电脑做服务器建网站重庆seo职位
  • wordpress 首页 修改合肥seo推广排名
  • 网站建设公司服seo顾问赚钱吗
  • 哪些网站可以做问卷调查铜仁搜狗推广
  • 做个网站得花多少钱百姓网推广电话
  • 遵义网站制作的网站网络营销的优势是什么