• C#预处理命令


    C#预处理命令
    与C++不同,C#没有独立的预处理器。C#中的预处理指令(pre-processing directives)仅仅用来与C保持一致,而并不是编译器开始编译代码之前的一个单独的处理步骤,它是作为

    词法解析的一部分来执行的。
    #if

    #else

    #elif

    #endif

    # define

    #undef

    #warning

    #error

    #line

    #region

    #endregion

    #pragma

    #pragma warning

    #pragma checksum

    虽然编译器没有单独的预处理器,但在处理该节中描述的指令时如同存在一个单独的预处理器;这些指令用于辅助条件编译。与 C 和 C++ 指令不同,不能使用这些指令创建宏。

    预处理器指令必须是行上的唯一指令。
    详细参考msdn


    扩展
    一,C语言预处理命令
    1)预处理不是C语言本身的组成部分,在程序编译之前完成预处理;
    2)预处理一般来说有三种功能:宏定义,文件包含和条件编译;
    3)预处理的本质是将特定的代码加载到源程序中,对于宏定义来说就是替换,对于文件包含来说就是将所要包含的文件加载到预处理定义的部分,而条件编译则表示是否将某段程

    序编译,本质上就是将需要的程序进行编译,而不需要的程序不编译

    二,C中的预处理命令
    C中的预处理命令是由ANSIC统一规定的,但它不是C语言的本身组成部分,不能直接对它们进行编译,因为编译程序无法识别它们。必须对程序进行通常的编译(包括词法和语法分

    析,代码生成,优化等)之前,先对程序中这些特殊的命令进行“预处理”,例如:如果程序中用#include命令包含一个文件“stdio.h”,则在预处理时,将stdio.h文件中的实

    际内容代替该命令。经过预处理后的程序就像没有使用预处理的程序一样干净了,然后再由编译程序对它进行编译处理,得到可供执行的目标代码。现在的编译系统都包括了预处

    理,编译和连接部分,在进行编译时一气呵成。我们要记住的是预处理命令不是C语言的一部分,它是在程序编译前由预处理程序完成的。

    C提供的预处理功能主要有三种:宏定义,文件包含,条件编译。它们的命令都以“#”开头。
    一 , 宏定义 :用一个指定的标识符来代表一个字符串,它的一般形式为:
    #define 标识符 字符串
    #define PI 3.1415926
    我们把标识符称为“宏名”,在预编译时将宏名替换成字符串的过程称为“宏展开”,而#define是宏定义命令。

    几个应该注意的问题:

    1,  是用宏名代替一个字符串,也就是做简单的置换,不做正确性检查,如把上面例子中的1写为小写字母l,预编译程序是不会报错的,只有在正式编译是才显示出来。

    2,  宏定义不是C语句,不必在行未加分号,如果加了分号则会连分号一起置换。

    3,  #define语句出现在程序中函数的外面,宏名的有效范围为定义命令之后到本源文件结束,通常#define命令写在文件开头,函数之前,作为文件的一部分,在此文件范围内有

    效。

    4,  可以用#undef命令终止宏定义的作用域。如:

    #define PI 3.1415926

    main(){

    }

    #undef PI

    mysub(){
    }

    则在mysub中PI 不代表3.1415926。

    5,  在进行宏定义时,可以引用已定义的宏名,可以层层置换。

    6,  对程序中用双撇号括起来的字符串内的字符,即使与宏名相同,也不进行置换。

    7,  宏定义是专门用于预处理命令的一个专有名词,它与定义变量的含义不同,只做字符替换不做内存分配。

    带参数的宏定义,不只进行简单的字符串替换,还进行参数替换。定义的一般形式为:#define 宏名(参数表)字符串

    如:#define S(a,b) a*b,具体使用的时候是int area; area=(2,3);

    对带参数的宏定义是这样展开置换的:在程序中如果有带参数的宏(如area=(2,3)),则按#define命令行中指定的字符串从左到右进行置换。如果串中包含宏中的形参(如a,b)

    ,则将程序语句中的相关参数(可以是常量,变量,或表达式)代替形参。如果宏定义中的字符串中的字符不是参数字符(如上*),则保留,这样就形成了置换的字符串。

     带参数的宏与函数有许多相似之处,在调用函数时也是在函数名后的括号内写实参,也要求实参与形参的数目相等,但它们之间还有很大的不同,主要有:

    1,       函数调用时,先求出实参表达式的值,然后代入形参,而使用带参的宏只是进行简单的字符替换。

    2,       函数调用是在程序运行时处理的,为形参分配临时的内存单元。而宏展开则是在编译前进行的,在展开时并不分配内存单元,不进行值的传递处理,也没有返回值的概

    念。

    3,       对函数中的实参和形参都要定义类型,二者的类型要求一致,如不一致,应进行类型转换;而宏不存在类型问题,宏名无类型,它的参数也无类型,只是一个符号代表

    ,展开时代入指定的字符串即可。宏定义时,字符串可以是任何类型的数据。

    4,       函数调用只可得到一个返回值,而用宏可以设法得到几个结果。

    5,       使用宏次数多时,宏展开后源程序长,因为没展开一次都使程序增长,而函数调用不会这样。

    6,       宏替换不占运行时间,只占编译时间,而函数调用则占运行时间(分配单元,保留现场,值传递,返回)。

    二 , 文件包含:一个源文件可以将另一个源文件的全部内容包含进来,即将另外的文件包含到本文件中。

      #include <文件名> 或 #include“文件名”

       感觉它像JAVA中的包,而它的作用像在J2EE中我们可以用*.xml做配置文件,然后各个模块调用这个文件,但这个文件如果修改后,凡使用(包含)此文件的所有文件(因为使

    用时是拷贝了原来的一份)有都需要从新编译,好像又失去了灵活的意义。

    在#include命令中,文件名可以用“”或<>括起来,它们的区别是用<>时,系统到存放在用户当前目录中寻找要包含的文件,若找不到,再按照标准方式查找(即按尖括号的方式

    查找)。一般说来,如果是为调用库函数而用#include命令来包含相关的头文件,则用<>,以节省查找时间。如果要包含的是用户自己编写的文件(这种文件一般都在当前目录中

    ),一般用“”,若文件不在当前目录中,“”内可给出文件路径。

    三,条件编译

    一般情况下,源程序中的所有行都参加编译。但有时希望对其中一部分内容只在满足一定条件才进行编译,也就是对一部分内容指定编译的条件,这就是条件编译。

    1,#indef 标识符

          程序段1

       #else

          程序段2

       #endif

    当所指定的标识符已经被#include命令定义过,则在程序编译阶段只编译程序1,否则编译程序段2。

    2,#if 表达式

          程序段1

    #else

    程序段2

    #endif

      优点:采用条件编译,可以减少被编译的语句,从而减少目标程序的长度,减少运行时间,当条件编译段比较多时,目标程序长度可大大减少。

    转自:http://dev.csdn.net/article/37/37021.shtm

  • 相关阅读:
    [转载]在sharepoint里增加PDF图标显示收藏主题
    CuratorFramework开源Zookeeper快速开发框架介绍
    Zookeeper和CuratorFramework实践之:分布式消息队列
    Zookeeper和CuratorFramework实践系列之: 配置管理
    JS 文字向上滚动代码
    sql递归查询问题
    双线单IP和双线双IP机房的区别
    js获取标准北京时间
    ftp 21端口被占用解决办法
    "由于没有远程桌面授权服务器可以提供许可证..."不能远程桌面解决方法
  • 原文地址:https://www.cnblogs.com/sutengcn/p/1206611.html
Copyright © 2020-2023  润新知