the c programing language 学习过程4

4Functions and Program Structure

scratch 刮擦 starting over from scratch从头开始 reside驻留 separately 分别的 facilities工具容易 macro arguments宏参数 pattern模式 irrelevant不相干的 sophisticated复杂的 optional sign 可选的标志 handles 处理 presence存在 invocation调用 scope rules 作用域规则

reorganizing 改组 mandatory强制的 受命令的 centralize使集中 peripherally外围的 recursion 递归 implementation贯彻 安装启用 macro宏 pitfalls陷阱 concatenation一系列相关

1UNIX cc main.c getline.c strindex.c把目标文件main.o getline.o strindex.o放到可执行文件a.out中  如果 main.c 错误 可以 cc main.c  getline.o strindex.o同样将main.c编译 和后两者一起载入 cc用.c和.o来区分源文件与目标文件

2return expression 要加括号 但是不是必须的

3dummy( ) { }   这个函数啥也不做啥也不返回 如果定义中缺省了返回类型 那么则为int

4返回非INT型变量可以先声明  如double atof( char s[ ]){  ****************}  可以在main中先声明  double sum, atof ( char [ ] );  声明了一个double 变量sum 和一个函数

5声明时 类型要相同 如果不同 函数和声明在同一个文件会被检测出error 而在多个文件(更常见)内则不会被检测到 比如函数返回double  主调声明int 那函数返回值将被转化为int 就没有意义了

6/* atoi:利用atof函数把字符串s转换成整数 */
int atoi( char s[ ] )
{
double atof(char s[ ]);
return (int) atof ( s );
}

其中如果用return atof ( s );atof也会被强制转换成int 但是编译器会出现警告信息(警告是什么 能吃么)

7逆波兰表示法 运算符号在运算量后面(1 - 2) * (4 + 5)一类的中辍可用逆波兰表示法表示成:1 2 - 4 5 + *  用栈计算就行 运算分量进栈 遇到运算符就pop两个分量出来(如果是二元运算符)运算后进栈 最后把栈顶的值拿出来输出

8+ * 满足交换律 顺序无所谓 可以push(pop()*pop())  而-和/要规定顺序 最好用临时变量调剂下 比如 op=pop(); push(pop()/op)

9c=getch()获取一个字符给c ungetch(c) 把c退回到缓冲区里(假装什么事都没有发生)

#define BUFSIZE 100
char buf[BUFSIZE]; /* 用于unget函数的缓冲区 */
int bufp = 0; /* buf中下一个自由位置 */
int getch(void) /* 取一个字符(可能是推回的字符) */
{
return ( bufp > 0 ) ? buf[--bufp] : getchar( );
}
void ungetch(int c) /* 把字符推回到输入中 */
{
if ( bufp >= BUFSIZE )
printf ( "ungetch: too many characters\n" );
else
buf[bufp++] = c;
}

标准函数里有ungetch

 

10外部变量的作用域从其声明处到函数的末尾

main( ) { }
int sp = 0;
double val[MAXVAL];
void push( double f ) { }
double pop( void ) { }

sp val 在push 和pop中无需声明就可以直接用 但是对main不可见

11如果一个外部变量在定义前就要用到 那就要强制用extern  声明就通报变量(主要声明一个类型) 定义是要有赋值(并不是赋值而是分配内存空间 解释错误)(存储分配)

int sp;
double val[MAXVAL];
这两个说明定义了外部变量 s pv a l,并为之分配存储单元,同时也用作供源文件其余部分
使用的说明。另一方面,如下两行:
extern int sp;
extern double val[MAXVAL];

为源文件剩余部分说明了 s p是一个 i n t类型的外部变量, v a l是一个 d o u b l e数组类型的外部变量

定义数组时要写上大小 extern声明时候就随意了.

12 static variables 静态变量 可以被当前所在源文件的函数调用 不能被其他函数访问 适用于说明外部变量和函数

static char buf[BUFSIZE]; /* ungetch函数使用的缓冲区 */
static int bufp = 0; /* 缓冲区buf的下一个自由位置 */
int getch( void ) { }
void ungetch( int c) { }

 此处 buf和bufp只能被getch和ungetch用到 

静态变量在作为内部变量时和自动变量相似  区别在于不管函数激活还是终止 静态变量的值都存在 自动变量随着函数的调用和结束 会消失

13 register variable 建议编译器 此变量为常用变量 建议放在寄存器内 当然编译器可以不鸟你 只能用在自动变量和实参上

register int x;

f(register unsignedm)

{

register int i;

}

想要在使用寄存器变量的地方加上 register没啥坏处 编译器可以将他们忽视掉 值得注意的是 加了这玩意不管是不是存在寄存器里 地址都不能访问了

14 外面的xy 和里面的xy 没有联系  但是最好也别这么编程 看得蛋疼

int x;

int y;
f ( double x )
{
double y;

}

15 外部变量和静态变量在初始化缺省下 为0  自动和寄存器变量为未定义(垃圾值)纯量变量会在定义时被初始化 

如 int x = 1;  long day = 1000L*60L*60L*24L;

外部变量和静态变量 初始化必须是一个常量式子  初始化变量最好用显示的方式 比较直观

数组的初始化可以这样int days[ ] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };   数组大小缺省时 按元素个数计  这里数组大小为12

15 字符数组初始化可以用字符串 

char pattern [] = "ould";

它是如下虽然长些但却与之等价的定义的缩写:
char pattern [] = { 'o', 'u', 'l', 'd' , '\0' }

数组个数为5

16#include "文件名"

#include <文件名>

前者从源程序找 后者从定义的规则找

17#define 名字 替换文本 如果较长 可以在后面加\可以延续多行  

替换只对单词进行,对括在引号中的字符串不起作用。例如,如果Y E S是一个被定义的名字,那么在 printf ( "YES" ) Y E S M A N中不能进行替换。 

18#define max( A, B ) ( ( A ) > ( B ) ? ( A ) : ( B ) )

x = max( p+q, r+s );
将被替换成
x = ( ( p+q ) > ( r+s ) ? ( p+q ) : ( r+s ) );   形参将直接变替换

当然也有一定的缺陷 

max( i++, j++ ) /* */

又或者#define square( x ) x * x /* */  在square(z+1)的情况下

#undef可以取消定义

#undef getchar

int getchar(void)

{·······}

可以用加#来替换字符串

#define dprint( expr ) printf( #expr " = %g\n", expr )
当用诸如
dprint( x/y );
调用该宏时,该宏就被扩展成
printf( "x/y" " = %g\n", x/y );  等价于printf(“x/y=%g\n”,x/y);

(***较复杂)  ##可以把形参就实参替换

#define paste( front, back ) front ## back
从而宏调用paste(name, 1)的结果是建立单词name1

19条件包含 conditional inclusion #if

例一

#if !defined( HDR )    //如果没有定义HDR 则定义

#define HDR      /* hdr.h文件的内容*/

#endif   

例二

#if SYSTEM == SYSV   //根据不同的SYSTEM 选择不同的头文件
#define HDR "sysv.h"
#elif SYSTEM == BSD
#define HDR "bsd.h"
#elif SYSTEM == MSDOS
#define HDR "msdos.h"
#else
#define HDR "default.h"

#endif
# include HDR   

例三 ifndef 如果没有

#ifndef HDR  //如果没有HDR (例一的变型)
#define HDR
/* hdr.h文件的内容*/
#endif

 

 

 

posted @ 2018-01-09 12:18  Cyborg  阅读(244)  评论(0编辑  收藏  举报