学生时代所学的一些 C 语言知识点回顾(1)
- 2019 年 10 月 4 日
- 筆記
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/solaraceboy/article/details/100668093
文章目录
学生时代所学的一些 C 语言知识点回顾(1)
0. 前言
时间过得真快,一眨眼之间已经毕业很多年,C 语言是大学里所学的第一门编程语言。由于很长时间不用,很多知识点都已生疏,最近抽了个空对之前的一些知识点做了一些回顾,谨以此系列文章记录一下这些刚刚回顾过的知识点。
1.整型与浮点型两种数据类型相乘
#include<stdio.h> int main() { int a = 20; float b = 20.34; printf("%fn",a*b); return 0; } //406.799988
#include<stdio.h> int main() { int a = 20; float b = 20.34; printf("%dn",a*b); return 0; } //-1409143032
#include<stdio.h> int main() { int a = 20; float b = 20.34; printf("%dn",(int)(a*b)); return 0; } //406
如果不进行强制类型转换,那么整型无法保存浮点型的数据。
2.命令行传参
#include<stdio.h> #include<stdlib.h> #include<string.h> int main(int argc, char *argv[]) { int memory_num = 0; int i; char *cmd,*temp; for (i = 0; i < argc; i++) memory_num += strlen(argv[i]); cmd = (char *)malloc(memory_num + argc*strlen(" ")); for (i = 1; i < argc; i++) { //printf("argv:%sn",argv[i]); temp = (char *)malloc(strlen(argv[i])+strlen(" ")); strcpy(temp,argv[i]); strcat(temp," "); //printf("argc[%d]:%s|n",i,temp); strcat(cmd,temp); free(temp); } printf("%sn",cmd); system(cmd); return 0; } /* $ ./cmd ls /;tree ~/docker-build/tesseract-ocr/ ls / bin boot dev etc home lib lib64 media mnt opt proc root run sbin srv sys tmp usr var /home/gysl/docker-build/tesseract-ocr/ ├── 124.jpg ├── 124.txt ├── 1.png ├── 2.jpg ├── 3.jpg ├── 4.png ├── 4.txt ├── build │ ├── 4.1.0.tar.gz │ ├── Dockerfile │ ├── leptonica-1.78.0.tar.gz │ └── tesseract_langs.tar.gz ├── gysl.jpg.txt ├── gysl.txt └── langs ├── chi_sim.traineddata ├── chi_sim_vert.traineddata └── eng.traineddata 2 directories, 16 files */
这一段代码演示 main() 函数传参的具体情形,稍加改造即可以用于一些安全性较高的 shell 脚本的加密。
3. 运算符优先级
#include<stdio.h> int main(int argc,char *argv[]) { int a = 5, b, c; c = (b = a + 2) - (a=1); printf("%dt%dn",c,a); return 0; } // 6 1
4. for 语句里面的逗号运算符
#include<stdio.h> int main(int argc,char *argv[]) { int a = 15, b = 26, i, j; for ( ; i < a, j < b;i++, j++) {} printf("%dt%dn",i,j); return 0; } // 26 26
#include<stdio.h> int main(int argc,char *argv[]) { int a = 15, b = 26, i, j; for ( ; i < a, j < b;i++, j++) ;//{} printf("%dt%dn",i,j); return 0; } // 26 26
#include<stdio.h> int main(int argc,char *argv[]) { int a = 15, b = 26, i, j; for ( ; i < a, j < b;i++, j++) //{} printf("%dt%dn",i,j); return 0; } /* 0 0 1 1 2 2 3 3 4 4 5 5 6 6 7 7 8 8 9 9 10 10 11 11 12 12 13 13 14 14 15 15 16 16 17 17 18 18 19 19 20 20 21 21 22 22 23 23 24 24 25 25 */
#include<stdio.h> int main(int argc,char *argv[]) { int a = 15, b = 6, i, j; for ( ; i < a, j < b;i++, j++) {} printf("%dt%dn",i,j); return 0; } // 6 6
#include<stdio.h> int main(int argc,char *argv[]) { int a = 5, b = 6, i, j; for ( ; i < a, j < b;i++, j++) //{} printf("%dt%dn",i,j); return 0; } /* 0 0 1 1 2 2 3 3 4 4 5 5 */
这种情况以后一个变量的值为上限。for 语句在循环结束之后循环体最后一部分仍会继续执行一次。
5. 各数据类型的长度
#include<stdio.h> int main(int argc,char *argv[]) { printf("int:%dnlong int:%dnshort int:%dnunsigned int:%dnunsigned long int:%dnunsigned short int:%dn",sizeof(int),sizeof(long int),sizeof(short int),sizeof(unsigned int),sizeof(unsigned long int),sizeof(unsigned short int)); return 0; } /* int:4 long int:8 short int:2 unsigned int:4 unsigned long int:8 unsigned short int:2 */
6. 结构体
#include <stdio.h> struct stu{ char *name; //姓名 int num; //学号 int age; //年龄 char group; //所在小组 float score; //成绩 }stus[] = { {"Zhou ping", 5, 18, 'C', 145.0}, {"Zhang ping", 4, 19, 'A', 130.5}, {"Liu fang", 1, 18, 'A', 148.5}, {"Cheng ling", 2, 17, 'F', 139.0}, {"Wang ming", 3, 17, 'B', 144.5} }, *ps; int main(){ //求数组长度 int len = sizeof(stus) / sizeof(struct stu); printf("NamettNumtAgetGrouptScoretn"); for(ps=stus; ps<stus+len; ps++){ printf("%st%dt%dt%ct%.1fn", ps->name, ps->num, ps->age, ps->group, ps->score); } return 0; }
7. 一些值得思考的问题
# include<stdio.h> # include<string.h> int main(int argc,char *argv[]) { char e[4] = "20d0"; int i; printf("Sizeof: %dt%dn",(sizeof e),strlen(e)); printf("Sizeof int: %dn",sizeof (int)); printf("Sizeof e: %dn",sizeof (e)); /* for ( i = 0; i < strlen(e); i++) printf("e[%d]: %cn",i,e[i]); */ printf("%lun",((unsigned long)sizeof(int))); return 0; }
以上代码注释部分去掉之后,第一个printf输出的值会出现不一样。
8. 数组的初始化
# include<stdio.h> int main(int argc, char *argv[]){ int a[10] = {1,2,4,5,[0] = 0,6,8,9,3}; int i = 0; for ( i = 0; i < 10; i++) printf("a[%d]: %dn",i, a[i]); return 0; } /* a[0]: 0 a[1]: 6 a[2]: 8 a[3]: 9 a[4]: 3 a[5]: 0 a[6]: 0 a[7]: 0 a[8]: 0 a[9]: 0 */
# include<stdio.h> int main(int argc, char *argv[]){ int a[10] = {1,2,4,5,[3] = 0,6,8,9,3}; int i = 0; for ( i = 0; i < 10; i++) printf("a[%d]: %dn",i, a[i]); return 0; } /* a[0]: 1 a[1]: 2 a[2]: 4 a[3]: 0 a[4]: 6 a[5]: 8 a[6]: 9 a[7]: 3 a[8]: 0 a[9]: 0 */
这样的申明是不合法的。编译器在处理初始化列表是,会记录下一个待初始化的数组元素的位置。正常情况下,下一个元素是刚被初始化的元素后面的那个。但是,当列表种出现初始化式时,下一个元素会被强制为指示符对应的元素,即使该元素已经初始化了。
9. 变长数组的初始化
# include<stdio.h> int main(int argc, char *argv[]){ int n, i, a[n]; scanf("%d",&n); for ( i = 0; i < n; i++){ scanf("%d",&a[i]); } for ( i = 0; i < n; i++){ printf("a[%d]: %dn",i,a[i]); } return 0; } /* 3 5 8 1 a[0]: 5 a[1]: 8 a[2]: 1 */
代码第4行方括号内部的 n 不能省略。变长数组的长度是在执行时计算数组的长度的。变长数组也可以时多维的。
10. 函数的嵌套调用
# include<stdio.h> int main(int argc, char *argv[]){ int array[10] = { 2,1,0,3,6,5,4,7,9,8}; int ModifyNumber(int a[],int i); void show(int a[], int n); show(array,10); ModifyNumber(array,3); printf("The value has been updated. Please check: n"); show(array,10); return 0; } int ModifyNumber(int a[],int i){ void PrintMsg(); PrintMsg(); scanf("%d",&a[i]); return 0; } void PrintMsg(){ printf("Please input a new value:n"); } void show(int array[], int n){ for (int i = 0; i < n; i++){ printf("Array[%d]: %dn", i, array[i]); } } /* Array[0]: 2 Array[1]: 1 Array[2]: 0 Array[3]: 3 Array[4]: 6 Array[5]: 5 Array[6]: 4 Array[7]: 7 Array[8]: 9 Array[9]: 8 Please input a new value: 90 The value has been updated. Please check: Array[0]: 2 Array[1]: 1 Array[2]: 0 Array[3]: 90 Array[4]: 6 Array[5]: 5 Array[6]: 4 Array[7]: 7 Array[8]: 9 Array[9]: 8 */
在 C 语言种,允许定义函数互相调用(在一个函数种调用另外一个函数),但是函数嵌套定义(在一个函数种定义另外一个函数)是不允许的。类似于下面的函数定义,能通过系统编译,但是程序无法正常执行:
# include<stdio.h> int main(int argc,char *argv[]){ void SendMsg(); void PrintMsg(); SendMsg(); PrintMsg(); return 0; } void SendMsg(){ void PrintMsg(); PrintMsg(); printf("This function is SendMsg.n"); } void PrintMsg(){ void SendMsg(); SendMsg(); printf("This function is PrintMsg.n"); }
这是一种间接递归的形式,必须保证两个函数能正常终止。
11. 静态变量
# include<stdio.h> int main(int argc,char *argv[]){ int Sta(int); int n; scanf("%d",&n); printf("First value: %dn",Sta(n)); printf("Second value: %dn",Sta(n)); return 0; } int Sta(int n){ static int b = 0; b = b + n; return b; } /* 6 First value: 6 Second value: 12 */
在 Sta() 函数第二次被调用时,静态变量 b 的值已经是6。也就是说,静态变量对其他函数隐藏数据,为将来再次调用保留这些数据。
12. 局部变量和外部变量
# include<stdio.h> int main(int argc,char argv[]){ void test(); extern int i; test(); printf("%dn",i); return 0; } int i = 1; void test(){ int j = i; int i =2; printf("%dn",j); } // 1 // 1
代码是合法的。如果去掉关键字 extern, 那么第二个打印的值将是0。