总结
基础
- 程序结构是三种: 顺序结构、选择结构(分支结构)、循环结构。
- 读程序都要从main()入口,然后从最上面顺序往下读(碰到循环做循环,碰到选择做选择),有且只有一个main函数。
- 计算机的数据在电脑中保存是以二进制的形式,数据存放的位置就是他的地址。
- bit是位,指为0或1。byte是指字节,一个字节=八个位。
- 每个C语言程序写完后,都是先编译,后链接,最后运行。(.c→.obj→.exe)这个过程中注意.c 和.obj 文件是无法运行的,只有.exe文件才可以运行。
常考
-
编译预处理不是C语言的一部分,不占运行时间,不要加分号。
C语言编译的程序称为源程序,它以ASCII数值存放在文本文件中。 -
#define PI 3.1415926;
这个写法是错误的,一定不能出现分号。 - 每个C语言程序中main函数是有且只有一个。
- 在函数中不可以再定义函数。
- 算法:可以没有输入,但是一定要有输出。
- break可用于循环结构和switch语句。
- 逗号运算符的级别最低,赋值的级别倒数第二。
-
int *p 中 *p 和 p 的差别:简单说*p 是数值,p 是地址!
- *p:可以当做变量来用;*的作用是取后面地址 p 里面的数值。
- p:是当作地址来用。可以用在 scanf 函数中:scanf("%d",p);
- *p++ 和 (*p)++的之间的差别:(笔试重点)
- 二级指针:
- *p:一级指针:存放变量的地址。
-
**q:二级指针:存放一级指针的地址。
常考题目:
int x=7; int *p=&x,**q=p;
问你:*p 为多少?*q为多少?**q为多少?
*p=7,*q=p,**q=7
再问:**q=&x 的写法可以吗?不可以,二级指针只能存放一级指针的地址。
- 程序进行编译时,并不为形式参数分配存储空间。只有在被调用时,形式参数才临时地占有存储空间。
形式参数(形参)存储类别默认“auto”
形参用关键字auto作存储类别的声明时,关键字“auto”可以省略,auto 不写则隐含确定为“自动存储类别”,它属于动态存储方式。 -
函数的存储类型是函数定义时函数名前面的数据类型前面的存储类型。
函数的存储类型默认“extern”,表示该函数属外部函数(即 可以被本 C 文件外的其他 C 源程序文件中的函数调用)
重点
-
strlen 和 sizeof 的区别(重点):
-
sizeof:(求实际存储空间)
相当于是个宏一样的东西,因为它只是一个运算符,而不是函数,编译时展开为常数,编译的时候有每个变量的定义表,sizeof通过查表确定变量占用的空间,这是分配内存给process 之前要确定的。
其实可以简单的理解sizeof是针对"类型"的,而非"变量",但此时不能这样看,如:sizeof("HELLO");
括号中不是字符数组const char *,而是一个"字符串",所以结果大小为 5。
但字符数组:char *ps = "HELLO"; //字符数组 sizeof(ps) = 4 //只是指针的大小,即 地址(整数类型占4个字节) sizeof(*ps) = 1 //*ps+0代表数组第一个元素的大小,即ps[0] char as[8]; sizeof(as) = 8 //因为as的类型为 char [8],这个大小的确是8 sizeof(*as) = 1 //*as+0代表数组第一个元素的大小,即as[0] char aa[8][9]; sizeof((char *)aa) = 4 //还是 char * char arr[100] = "HELLO"; sizeof(arr) = 100 //和赋什么值没什么关系,关键是"类型"是什么 int func(char p[100]) { sizeof(p) = 4; //C/C++中不能传数组,只能传指针,所以任何数组都会隐式转成指针形式进行操作。 }
strlen:(求字符串长度,即 字符个数,不包括结束符)
它是一个函数,参数是 const char*,搞清楚它的实现,就是碰到'\0'(字符串结尾,就停止计数,但不包括'\0')。所以它不是看类型而是看变量,取决于变量赋的什么值。
-
- 函数的递归一定会考。
两种重要的数组长度:
char a[]={'a','b','c'};
数组长度为3,字符串长度不定(因为没有'\0'结束符)。sizeof(a)为3。char a[5]={'a','c'};
数组长度为5,字符串长度3。sizeof(a)为5。char a[]={'a','c'};
这是一个字符数组,占3个字节。char a[]="abc";
则不同,它是一个字符串,最后还有一个'\0'结束符,占4个字节。- scanf 和 gets 的区别:
如果输入的是 good good study!-
scanf("%s",a);
只会接收 good。考点:不可以接收空格。(遇到空格或回车就终止)
-
gets(a);
会接收 good good study!考点:可以接收空格。(遇到回车终止)
-
-
指针考点:
char ch[]="iamhandsome";
char *p=ch;
问你:*(p+2) 和 *p+2 的结果是多少?
*(p+2) = 'm',*p+2 = 'k'解析:
第一个是地址+2,所以取m;
第二个则是数值+2,即 ASCII码值+2,ijk,所以取k。 - 字符串的赋值:
C语言中没有字符串变量,所以用数组和指针存放字符串:-
char ch[10]={"abcdefgh"};
对 -
char ch[10]="abcdefgh";
对 -
char ch[10]={'a','c','d','e','f','g','h'};
对 -
char *p="abcdefgh";
对 -
char *p;
对 -
ch="abcdefgh";
错!数组名不可以赋值!(只能一个一个循环赋值) -
char *p={"abcdefgh"};
错!大括号是数组用的,相当于集合,而指针类型不能用大括号
-
-
逗号表达式:
将两个及其以上的式子联接起来,从左往右逐个计算表达式,整个表达式的值为最后一个表达式的值。
(2,4)的表达式的数值就是4。z = (2,4);
(整个是赋值表达式),这个时候z的值为4。z = 2,4;
(整个是逗号表达式,即(z = 2),(3),(4);
),这个时候z的值为2,这个(z = 2,4)式子的值为4。用途:
for(i=0,j=0; i<10; i++)
,像这样联接式子,因为for中的每个区域只能放一个式子,所以需要逗号表达式联接多个式子。
问题:
#define f(x) (x\*x)
和#define f(x) x\*x
之间的差别。
define是C语言中的宏定义关键字,其定义格式如下:
#define [MacroName] [MacroValue]
宏定义分为普通宏定义和带参数的宏定义
普通宏定义:
#define PI (3.1415926)
-
带参数的宏定义 (宏函数):
#define max(a,b) ((a)>(b)? (a),(b))
注意:变量在宏中要用括号括起来。
因为,在C语言中define宏定义在编译时,会被展开,进行“傻瓜式”替换,也称为“字面”替换,如果没有括号有可能会产生歧义。如:
int a,b,c,d,e; a=1; b=2; c=6; d=4; e = f(a+b)*f(c+d) ; //理论值e=9*100=900 #define f(x) (x*x) 替换结果为:e =(a+b*a+b)*(c+d*c+d) = 5*34=170 #define f(x) x*x 替换结果为:e = a+b*a+b*c+d*c+d=1+2+12+24+4=43 #define f(x) ((x)*(x)) 替换结果为:e = ( (a+b)*(a+b)*(c+d)*(c+d) )=3*3*10*10=900
这个才是我们想要的结果!
字符
-
字符数据的合法形式:
'1' 是字符占一个字节,"1"是字符串占两个字节(含有一个结束符号)。
'0' 的 ASCII 数值表示为 48,'A'的 ASCII 数值是 65,'a' 的 ASCII 数值是 97,
一般表示单个字符错误的形式:'65' "1"
字符是可以进行算术运算的,记住:'0'-0=48注意:这个48不用记,用的时候直接'0'-0,就可以转换字符和数字了。
记忆:0Aa:486597,486 看 re0 的应该知道,597 就是486+111。 -
大写字母和小写字母转换的方法:
'A'+32='a'
相互之间一般是相差32。注意:这个32不用记,要用的时候直接'a'-'A',就可以得出转换大小写的数值了。
在ASCII码表中,A比a先,所以ASCII码小
类比:A比a先,所以生日年份比a小。 字符型和整数是近亲:两个具有很大的相似之处
char a = 65;
printf("%c",a);
得到的输出结果:aprintf("%d",a);
得到的输出结果:65
转义字符
转义字符分为一般转义字符、八进制转义字符、十六进制转义字符。
- 一般转义字符:\0、 \n、 \'、 \"、 \\。
-
八进制转义字符'\ddd':(其中d为常数,ddd表示三位8进制数)(以0开头的表示八进制)
'\141'=97
是合法的, 前导的0是不能写的。 -
十六进制转义字符'\xhh':(其中h为常数,hh表示两位16进制数)(以0x开头的表示十六进制)
'\x6d'=109
才是合法的,前导的0不能写,并且x是小写。
进制名称 | 英文 | 缩写 |
---|---|---|
二进制 | Binary | B |
八进制 | Octal | O |
十进制 | Decimal | D |
十六进制 | Hexadecimal | H |
输入输出
格式符
格式说明 | 表示内容 | 注释 |
---|---|---|
%d | 整型十进制 int | Decimal |
%ld | 长整型 long int | Long Decimal |
%f | 浮点型 float | Float |
%lf | 双精度浮点型 double | Long Float |
%% | 输出一个百分号 | % |
%c | 字符 char | Char |
%s | 字符串 | String |
%o | 八进制 | Octal |
%#o | 带前导(0)的八进制 | |
%x | 十六进制 | Hexadecimal |
%#x | 带前导(0x)的十六进制 | |
%p | 指针的值,输出地址符 即 地址 |
Pointer |
%md | 整型,m为总长度(常数)(默认为正数+) |
右对齐(数轴左-右+) 即 不足的话,左边补上空格 |
%-md | 整型,m为总长度(常数)(负数-) |
左对齐 即 不足的话,右边补上空格 |
%m.nf | 浮点型 m为总字符长度(包含小数点) n为小数长度 |
要进行四舍五入 当实际的显示大于m,那就按实际输出,也就是m无意义了 %3d 对应 1234 的话,就是1234 没有空格 |
举例说明:
printf("%2d",123);
第二部分有三位,大于指定的两位,原样输出 123printf("%5d",123);
第二部分有三位,小于指定的五位,左边补两个空格 123printf("%10f",1.25);
小数要求补足 6 位的,没有六位的补 0,。结果为 1.250000printf("%5.3f",125);
小数三位,整个五位,结果为 1.250(小数点算一位)printf("%3.1f",1.25);
小数一位,整个三位,结果为 1.3(要进行四舍五入)
输入
要特别注意的是:
scanf("%d,%d",&x,&y);
scanf 的第二个部分一定要是地址!
指定输入的长度:(一般用空格或回车进行间断)
终端输入:1234567scanf("%2d%4d%d",&x,&y,&z);
x 为 12,y 为 3456,z 为 7
终端输入:1 234567 由于 1 和 2 中间有空格,所以只有 1 位给 xscanf("%2d%4d%d",&x,&y,&z);
x 为 1,y 为 2345,z 为 67字符和整型是近亲:
int x=97;
printf("%d",x);
结果为 97printf("%c",x);
结果为 a输入时候字符和整数的区别:
scanf("%d",&x);
这个时候输入 1,特别注意表示的是整数 1scanf("%c",&x);
这个时候输入 1,特别注意表示的是字符'1',ASCII 为整数 49。-
%*d
: -
x保留n位小数,第n+1位四舍五入:((int)\((x*10^n+0.5)/10.0^n\))
y=(int)(x*100+0.5)/100.0
这个保留两位小数,对第三位小数四舍五入y=(int)(x*1000+0.5)/1000.0
这个保留三位小数,对第四位小数四舍五入y=(int)(x*10000+0.5)/10000.0
这个保留四位小数,对第五位小数四舍五入这个有推广的意义,注意 x = (int)x 这样是把多余的小数部分去掉。
注意:这个只去掉了分子的小数部分,把分子化为了整型,当除以分母时,又恢复了浮点型。
基本表达式
注意:C 语言中是用非 0 表示逻辑真,用 0 表示逻辑假。
-
逗号表达式:
将两个及其以上的式子联接起来,从左往右逐个计算表达式,整个表达式的值为最后一个表达式的值。
(2,j=0; i<10; i++),像这样联接式子,因为for中的每个区域只能放一个式子,所以需要逗号表达式联接多个式子。 -
赋值表达式:(可以连续赋值)
- 计算赋值运算符右侧表达式的值。(“=”为赋值运算符)
- 将赋值运算符右侧表达式的值赋给左侧的变量。
-
将赋值运算符左侧的变量的值作为表达式的值。(注意与逗号表达式的右侧区分)
这就是连续赋值可以的原因。而连续比较大小关系不行,因为它表达式的值是bool类型(即 真假),当连续比较时数值会发生改变(变成0或者1)。
例如:x=y=3
就相当于x=(y=3)
结果都为3,这种连续赋值是可以的。
特别注意:而与之相对的连续关系表达式1<0<2
是不行的,这个从数学的角度出发肯定是错的,但是如果是 C 语言那么就是正确的!因为 1<0 为假得到 0,表达式就变成了 0<2 那么运算结果就是 1,为真!
-
关系表达式:(不能连续比较)
- 表达式的数值只能为 1(表示真),或 0(表示假)。
如 9>8 这个关系表达式是真的,所以 9>8 这个表达式的数值就是 1。 如 7<6 这个关系表达式是假的,所以 7<6 这个表达式的数值就是 0 -
易错:int x=1,y=0,z=2; x<y<z 是真还是假?
带入为 1<0<2,从数学的角度出发肯定是错的,但是如果是 C 语言那么就是正确的!因为 1<0 为假得到 0,表达式就变成了 0<2 那么运算结果就是 1,为真! - 等号和赋值的区别!一定记住"="就是赋值,"=="才是等号。
- 表达式的数值只能为 1(表示真),或 0(表示假)。
-
逻辑表达式:(表达式的数值只能为 1(表示为真),或 0(表示假))
- 共有 &&、||、! 三种逻辑运算符号。
- ! > && > || 优先的级别。
- 表示 x 小于 0 大于 10 的方法:0<x<10 是不行的。
先计算 0<x 得到的结果为 1 或则 0;再用 0,或 1 与 10 比较得到的总是真(为 1)。所以一定要用 (0<x)&&(x<10)表示比 0 大比 10 小。 -
注意短路现象。考试比较喜欢考到。
短路现象:为了要提高效率。在逻辑运算时候,如果值已经能决定整个表达式的值,就不会再往右继续运算了。
例如:在if里有个&&,如果左边的值为假,就不会再算右边的真假。
----------------------------------------------------------------------------------------
短路现象常见的有,短路与(&&)和短路或(||)(当执行结果已经可以确定,则后面的将不再执行了)
若a && b ,如果a 的值为假,则整个表达式的值就为假,它是从左向右计算的。所以执行该表达式后,b 的值还是它的初始值,即不进行运算。
若a || b ,如果a 的值为真,整个表达式的值就为真,执行顺序同上。b 的真假由最初的真假来判断,也就是说,当a 为真时,b 则不进行运算了。例如:(m=a>b)&&(n=c>d),当a b c d 分别为1,2,3,4,m=n=1时,由于 a>b 为0,则m =0。而后面的不再执行,所以n=1而不是0。
- switch 语句:
- 执行的流程一定要弄懂!
- 注意有 break 和没有 break 的差别,
- 没有 break 时候,只要有一个 case 匹配了,剩下的都要执行
- 有 break 则是直接跳出了 swiche 语句。
- switch 只可以和 break 一起用,不可以和 continue 用。
switch(x) { //(x:是整型常量,字符型常量,枚举型数据) case 1: … //Case 中不可以是变量。 case 2: … }
-
循环结构:(先执行while中的表达式(这样才能判断是否循环))
注意:不管循环不循环,都要执行while中的表达式才能判断是否循环,不能说一眼看出0--不循环了,就不--了,计算机还是要--的。
函数
- 函数不可以嵌套定义。但是可以嵌套调用。
- 函数名缺省返回值类型,默认为int。
- C 语言由函数组成,但有且仅有一个 main 函数!是程序运行的开始!
-
请问 sum 的结果是多少?add(int x,int y){return x+y;} main() { int sum; sum=add(add(7,8),9); }
结果为 24 - 一定要注意参数之间的传递
实参和形参之间传数值和传地址的差别。
传数值,形参的变化不会改变实参的变化。
传地址,形参的变化会有可能改变实参的变化。 - 函数声明的考查:
int *fun(int a[],int b[]) { ………… }
已经知道函数是这样。这个函数的正确的函数声明怎么写?
int *fun(int *a,int *b);
这里是函数声明的写法,注意数组就是指针int *fun(int a[],int b[]);
这种写法也是正确的int *fun(int b[],int c[]);
这种写法也是正确的,参数的名称可以随便写int *fun(int *,int *);
这种写法也是正确的,参数的名称可以不写
指针
定义 | 含义 |
---|---|
int i; | 定义整型变量 i |
int *p; | p 为指向整型变量的指针变量 |
int a[n]; | 定义整型数组 a,它有 n 个元素 |
int *p[n]; | 定义指针数组 p,它由 n 个指向整型数据的指针元素组成 |
int (*p)[n]; | p 为指向含有 n 个元素的一元数组的指针变量 |
int f(); | f 为返回整型数值的函数 |
int *p(); | p 为返回一个指针的函数,该指针指向整型数据 |
int (*p)(); | p 为指向函数的指针,该函数返回一个整型值 |
int **p; | p 是一个指针变量,它指向一个指向整型数据的指针变量 |
-
int *p 中 *p 和 p 的差别:简单说*p 是数值,p 是地址!
- *p:可以当做变量来用;*的作用是取后面地址 p 里面的数值。
- p:是当作地址来用。可以用在 scanf 函数中:scanf("%d",p);
- *p++ 和 (*p)++的之间的差别:(笔试重点)
- 二级指针:
- *p:一级指针:存放变量的地址。
-
**q:二级指针:存放一级指针的地址。
常考题目:int x=7;int*p=&x,**q=p;
问你:*p 为多少?*q为多少?**q为多少?
*p=7,*q=p,**q=7
再问:**q=&x 的写法可以吗?不可以,二级指针只能存放一级指针的地址。
- 三名主义:
-
传数值和传地址:
void fun(int a,int b) { int t; t=a;a=b;b=t; } main() { int x=1,y=3; fun(x,y); printf("%d,%d",x,y); }
这个题目答案是 1 和 3。
传数值,fun 是用变量接受,所以 fun 中 的交换不会影响到 main 中的 x 和 y 。
传数值,形参的变化不会影响实参。void fun(int *a,int *b) { int t; t=*a;*a=*b;*b=t; } main() { int x=1,y=3; fun(&x,&y); printf("%d,x,y); }
这个题目的答案就是 3 和 1。
传地址,fun 用指针接受!这个时候 fun中的交换,就会影响到 main 中的 x 和 y。
传地址形参的变化绝大多数会影响到实参! -
函数返回值是地址,一定注意*号。
int *fun(int *a,int *b) { //可以发现函数前面有个*,这个就说明函数运算结果是地址 if(*a>*b) return a; //return a 可以知道返回的是 a 地址。 else return b; } main() { int x=7,y=8,*max; max = fun(&x,&y); //由于 fun(&x,&y)的运算结果是地址,所以用 max 来接收。 printf("%d",*max) }
数组
数组:存放的类型是一致的。多个数组元素的地址是连续的。
- 数组的初始化,一维和二维的,一维可以不写,二维第二个[]一定要写
int a[]={1,2} 合法。 int a[][4]={2,3,4}合法。 但 int a[4][]={2,3,4}非法。
科普:
- '0':代表 字符'0',对应ASCII码值为 0x30(也就是十进制的48);
- '\0':代表 空字符\0(转义字符)(输出为空),对应ASCII码值为 0x00(也就是十进制的0),用作字符串结束符;
- 0:代表 数字0,若把 数字0 赋值给 某个字符,对应ASCII码值为 0x00(也就是十进制0);
- "0":代表 一个字符串,字符串中含有 2个字符,分别是 '0' 和 '\0'。
一维数组
-
当所赋初值少于所定义数组的元素个数时,将自动给后面的元素补以初值0。
例:给a数组中所有元素赋初值0
int a[10]={0}; -
对于字符型数组也同样补以初值 0,即 '\0'。(而不是'0','0'对应的ASCII码为48)
(C语言中,'\0'表示的空字符,则其对应的ASCII码值为0。)
(而NULL为空指针,ASCII码也是0)例如:
char c[5]={'@'};
相当于:char c[5]={'@','\0','\0'};
- 一维数组的初始化:(当所赋初值少于所定义数组的元素个数时,将自动给后面的元素补以初值0。)
int a[5]={1,2,4,5};
合法int a[5]={1,};
合法(未设置的部分为0)int a[5]={1,3};
合法int a[]={1,5};
合法,后面决定前面数组的大小!int a[5]={1,6};
不合法,赋值的个数多余数组的个数了 一维数组的定义;
int a[5];
定义时数组的个数不是变量一定是常量。int a[1+1];
合法,个数是常量 2,是个算术表达式int a[1/2+4];
合法,同样是算术表达式int x=5,int a[x];
不合法,因为个数是 x,是个变量#define P 5 int a[P];
合法,define 后的的 P 是符号常量,只是长得像变量
一维数组的重要概念:
- 对 a[10]这个数组的讨论。
- a 表示数组名,是第一个元素的地址,也就是元素 a[0]的地址。(等价于&a)
- a 是地址常量,所以只要出现 a++,或者是 a=a+2 赋值的都是错误的。
- a 是一维数组名,所以它是列指针,也就是说 a+1 是跳一列。
- 对 a[3][3]的讨论。
- a 表示数组名,是第一个元素的地址,也就是元素 a[0][0]的地址。
- a 是地址常量,所以只要出现 a++,或者是 a=a+2 赋值的都是错误的。
- a 是二维数组名,所以它是行指针,也就是说 a+1 是跳一行。
- a[0]、a[1]、a[2]也都是地址常量,不可以对它进行赋值操作,同时它们都是列指针,a[0]+1,a[1]+1,a[2]+1 都是跳一列。
- 注意 a 和 a[0] 、a[1]、a[2]是不同的,它们的基类型是不同的。前者是一行元素,后三者是一列元素。
二维数组
- 二维数组的初始化
int a[2][3]={1,6};
合法,很标准的二维的赋值。int a[2][3]={1,};
合法,后面一个默认为 0。int a[2][3]={{1,} {4,6}};
合法,每行三个。int a[2][3]={{1,}{3,5}};
合法,第一行最后一个默认为 0。int a[2][3]={1,6,7};
不合法,赋值的个数多于数组的个数。