C与指针随笔

C与指针 随笔

第三章 数据

const* 与 *const

int const* p;

*p是一个整型常量。所以p是这样一个指针,它所指向的变量的值不可修改,但是p的值可修改,也就是说p可以指向其他变量。

int *const p;

p是一个常量指针,也就是说p不可以再改变指向,它只能指向它一开始指向的变量。但是可以通过p修改它所指向的变量的值。数组正是这样的一种指针。

第四章 语句

空循环体

while((ch=getchar())!=EOF&&ch!='\0')
    ;

这样做的好处是清楚地显示了循环体是空的,不至于使人误会程序接下来地一条语句才是循环体。

判断闰年:先从最为特殊的情况开始

if(year%400==0){
    leap_year=1;
}
else if(year%100==0){
    leap_year=0;
}
else if(year%4==0){
    leap_year=1;
}
else{
    leap_year=0;
}

getchar()的返回值

首先认为语句

a=x=y+3;

中a和x的值相等这件事是错误的。因为假如x是char型,a、y是int型,那么将 y+3 赋给x,x便会截取赋值,此时a的值将不会等于 y+3。而 getchar() 函数的返回值也是int,则语句

char ch;
......
while((ch=getchar())!=EOF){
    statements;
}

是有一定几率发生错误的。比如说当你输入的为 \377 字符时,被截取后再提上来和 EOF 相等。

逗号运算符

ch=getchar();
while(ch!='\n'){
    ......
    ch=getchar();
}

while(ch=getchar(),ch!='\n'){
    ......
}

等价。因为表达式的真假只和最后一个逗号后面的表达式有关。

第五章 操作符与表达式

优先级问题

语句

x=f()+g()+z();

中,虽然按照优先级后一个加法是在前一个加法之后进行。但是对于函数调用的顺序却没有一个确定的规则可循。假如函数都对某一个 全局变量 进行调用修改,那么x的最终结果就将无法确定。从而此语句不具有可移植性。最好改为

temp=f();
temp+=g();
temp+=z();

第六章 指针

指针二则运算

如果一个指针减去一个整数后,运算结果产生的指针所指向的位置在数组第一个元素之前,那么它是 非法的 。加法运算稍有不同,结果指针指向数组最后一个元素后面的那个内存位置仍是 合法 (但不能对这个指针执行间接访问操作),不过再往后就不合法了。

第七章 函数

函数的缺省认定

当函数调用一个无法见到原型的函数时,编译器便会认为该函数返回一个整型值。

函数的参数

C语言的规则很简单:

所有参数都是传值调用

数组参数的传址调用看起来似乎和传值调用相悖。但是,此处其实并无矛盾之处——数组名的值实际上是一个指针,传递给函数的就是这个指针的一份拷贝。

第八章 数组

下标引用

array[2]2[array]*(array+2) 是等价的。因为前两者都会以第三种方式解释。
对二维数组,假如申请了一个名为 array[M][N] 的数组,
array[2][3] 会被解释成 * ( *(array+2)+3) ,这也就是为什么在函数传递二维数组时必须标明 纵轴 最大值(如:Func(a[][N]))。

各类数组声明

以下声明为旧式风格,忽略了函数的参数。

char a[MAX];//元素类型为字符
int a[MAX];//元素类型为整数
float a[MAX];//元素类型为浮点数

char *a[MAX];//元素类型为指向字符型变量的指针
int *a[MAX];//元素类型为指向整型变量的指针
float *a[MAX];//元素类型为指向浮点型变量的指针

char (*a)[MAX]()//元素类型为函数指针,这个函数指针指向一个返回值为字符型的函数
int (*a)[MAX]();//元素类型为函数指针,这个函数指针指向一个返回值为整型的函数
float (*a)[MAX]();//元素类型为函数指针,这个函数指针指向一个返回值为浮点型的函数

char *(*a)[MAX]()//元素类型为函数指针,这个函数指针指向一个返回值为字符型指针的函数
int *(*a)[MAX]();//元素类型为函数指针,这个函数指针指向一个返回值为整型指针的函数
float *(*a)[MAX]();//元素类型为函数指针,这个函数指针指向一个返回值为浮点型指针的函数

第九章 字符串、字符和字节

strlen()函数的返回值

以下两个语句不是等价的:

if(strlen(x)>=strlen(y)){...}
if(strlen(x)-strlen(y)>=0){...}

由于strlrn()函数的返回值是size_t(定义在stdlib.h中,和unsigned类型等价)型,所以strlen(x)-strlen(y)的结果将永远为无符号整型,亦即上述第二个语句将永远为真。其实该函数返回值为无符号整形也是有优点的,它能表示字符串更大的长度。

strncpy()的第三个参数

char *strncpy(char *dst,char const *src,size_t len);

strcpy()函数一样,strncpy()把源字符串的字符复制到目标数组。然而,他总是正好向dst写入 len个字符。如果strlen(dst)的值小于lendst数组就用额外的NUL字节填充到len长度。如果strlen(dst)的值大于或等于len,那么只有len个字符被复制到dst注意:它的结果将不会以NUL字节结尾

字符分类函数

举个例子,用

if(isalpha(ch)){...}

代替

if(ch>='A'&&ch<='Z'||ch>='a'&&ch<='z'){...}

明显更简洁。其中isXXX()类函数包含在ctype.h头文件中。

第十三章 高级指针话题

函数指针

int f(int);
int (*pf)(int)=&f;
int ans;
//ans=f(25);
//ans=(*pf)(25);
//ans=pf(25);

注意,以上注释部分是等价的。第一条语句简单地调用函数f(),但它的执行过程可能和你想象的不太一样。函数名f首先被转换为一个函数指针,该指针指定函数在内存中的位置。然后,函数调用操作该函数,执行开始于这个地址的代码。第三条语句和前面两条语句效果是一样的。间接访问操作并非必需,因为编译器需要的是一个函数指针。

字符串常量

*("xyz"+1)
//"xyz"[1]

的结果作为右值就是y

*"xyz"

的结果作为右值就是x

操作符 ->

-> 前面的变量必须为指向结构的指针。所以:

typedef struct{
    int x;
    int y;
}Point;

Point p;
Point *a=&p;
Point **b=&a;

根据以上声明,则

b->x
b->a
*b->a
*b->x
b->a>x
(*b)->a

都是非法的。

第十四章 预处理器

#define替换

将宏参数转换为一个字符串的技巧是使用 "#argument" ,其中argument是宏参数,如此argument将不会以变量而是字符串的形式表出。如:

#define PRINT(FORMAT,VALUE)  \
        printf("The value of "#VALUE" is "#FORMAT"",VALUE)
......
PRINT(%d,x+3);

输出:

The value of x+3 is 25

宏与函数

一般而言,宏非常频繁地用于执行简单的计算。因为宏相对于函数没有函数调用/返回的开销。比如getchar()就是用宏定义的。但是当程序长度过长时或需要用递归时,宏就不再实用。宏还有一个优点就是宏与类型无关。比如:

#define add(x,y) ((x)+(y))
//在那些对表达式求值的宏中,每个宏参数出现的地方都应该加上括号,并且在整个宏定义的两边也要加上括号。

其中x、y可以是int型,也可以是float型和long型等等等等。再一个:

#define MALLOC(n,type)  \
        ((type *)malloc((n)*sizeof(type)))
......
pi=MALLOC(25,int);
//=>pi=((int *)malloc((25)*sizeof(int)))

在这个例子中,函数无法实现type参数功能。但是,并不是说宏就没有副作用,当宏参数在宏定义中出现不止一次时,就很可能有副作用。比如:

#define MAX(x,y) ((x)>(y)?(x):(y))
......
x=5;
y=8;
z=MAX(x++,y++);
printf("x=%d y=%d z=%d",x,y,z);

输出为:

x=6 y=10 z=9

因为

z=MAX(x++,y++);
//=>z=((x++)>(y++)?(x++):(y++));

而函数则没有这个问题,因为函数参数只在函数调用时求值一次,它的结果值传给函数。

#undef

这条预指令用于移除一个宏定义。例子:

#include<stdio.h>
#define MAX 100
int main(int argc,char *argv[]){
    printf("%d ",MAX);
    #undef MAX
    int MAX=10;
    printf("%d",MAX);
    return 0;
}

输出为:

100 10

条件编译

#if constant-expression1
    statements1
#elif constant-expression2
    statements2
#else
    statements3
#endif

当constant-expression1为真时编译statements1,当constant-expression2为真时编译statements2,否则编译statements3。它和

if constant-expression1
    statements1
else if constant-expression2
    statements2
else
    statements3

相比优点在于后者全部都要编译进去,而前者只需要编译一部分。所以前者效率会更高。更多关于条件编译的知识请戳这里

嵌套文件包含

嵌套#include文件的一个不利之处在于它使我们很难判断源文件之间真正的依赖关系,另一个不利之处在于一个头文件可能会被多次包含。如:

#include "x.h"
#include "y.h"

如果x.h和y.h中都包含了z.h这个头文件,那么z.h就被重复包含了。为了解决这个问题,可以这样做:

//在z.h头文件中
#ifndef _Z_H
#define _Z_H 1
//1可有可无
/*
*All the stuff that you want in the header file.
*/
#endif

注意预处理和正式编译的处理顺序

以下代码是错的:

#if sizeof(int)==2
    typedef long int32;
#else
    typedef int int32;
#endif

因为sizeof是在预处理器完成工作后而发挥作用的。而在预处理阶段无法处理sizeof等字段。

后记

《C与指针》这本书暂时先告一段落,毕竟第一遍看不能完全看懂也不能吸收全部精华。总的来说寒假虽然天天打游戏,但是总算做了点正事。寒假弯道虽没超车,但也幸好没翻车。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,530评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,403评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,120评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,770评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,758评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,649评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,021评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,675评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,931评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,751评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,410评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,004评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,969评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,042评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,493评论 2 343

推荐阅读更多精彩内容

  • 指针是C语言中广泛使用的一种数据类型。 运用指针编程是C语言最主要的风格之一。利用指针变量可以表示各种数据结构; ...
    朱森阅读 3,423评论 3 44
  • Lua 5.1 参考手册 by Roberto Ierusalimschy, Luiz Henrique de F...
    苏黎九歌阅读 13,729评论 0 38
  • 多线程、特别是NSOperation 和 GCD 的内部原理。运行时机制的原理和运用场景。SDWebImage的原...
    LZM轮回阅读 2,003评论 0 12
  • 那是两年前,我还在念大三的时候,一堂叫做“英汉翻译”的课的期末考试,内容是现场翻译一篇学术论文。虽然有电子词典,但...
    李子李子短信阅读 1,209评论 0 17
  • 一、结构化面试 1.形象包装很重要,着装稳重中不失活泼,有教师的规范,给人一种端庄的感觉,有活力,比如(系着蝴蝶结...
    范秀红阅读 380评论 0 0