C语言进阶技术:同事这些操作把我惊呆了!
扫描二维码
随时随地手机看文章
壹
#include"xxx.c"
参考demo:
1//FileName :main
2#include
3#include
4
5/***************************
6 * .c文件声明区域
7 **************************/
8#include"module1.c"
9#include"module2.c"
10
11/***************************
12 * Fuction: main
13 * Author :(最后一个bug)
14 **************************/
15int main(int argc, char *argv[]) {
16
17 Fuction1();
18 Fuction2();
19 printf("欢迎关注公众号:最后一个bug\n");
20 return 0;
21}
1//FileName: Module1.c
2#include
3/***************************
4 * Fuction: Fuction1
5 * Author :(最后一个bug)
6 **************************/
7void Fuction1()
8{
9 printf("Run Fuction1\n");
10}
1//FileName: Module2.c
2#include
3/***************************
4 * Fuction: Fuction2
5 * Author :(最后一个bug)
6 **************************/
7void Fuction2()
8{
9 printf("Run Fuction2\n");
10}
输出结果:
分析一下:
看来这波操作可行,似乎还省去了.h文件,之前bug菌说过,分析.h文件的时候直接把.h文件在对应的.c文件中的位置处展开然后进一步分析即可,其实这.c文件也是如此,接着往下看。
参考demo:
1//FileName :main
2#include
3#include
4
5char * cBug1 = "bugNo1"; //这里是位置1
6char * cBug2 = "bugNo2";
7/***************************
8 * .c文件声明区域
9 **************************/
10#include"module1.c"
11#include"module2.c"
12
13//char * cBug1 = "bugNo1";//这里是位置2
14//char * cBug2 = "bugNo2";
15
16/***************************
17 * Fuction: main
18 * Author :(最后一个bug)
19 **************************/
20int main(int argc, char *argv[]) {
21
22 Fuction1();
23 Fuction2();
24 printf("欢迎关注公众号:最后一个bug\n");
25 return 0;
26}
1//FileName: Module2.c
2#include
3/***************************
4 * Fuction: Fuction1
5 * Author :(最后一个bug)
6 **************************/
7void Fuction1()
8{
9 printf("Run Fuction1\n");
10 printf("%s\n",cBug1);
11}
1//FileName: Module2.c
2#include
3/***************************
4 * Fuction: Fuction2
5 * Author :(最后一个bug)
6 **************************/
7void Fuction2()
8{
9 printf("Run Fuction2\n");
10 printf("%s\n",cBug2);
11}
输出结果:
分析一下:
我们在位置1进行两个变量的定义,成功编译运行得到如上的结果,符合我们的预期,然而当我们去掉位置1进行位置2的定义,程序却无法进行编译,看来跟我们预期在编译过程中直接展开.c文件是一致的。
有些历史悠久的项目经过了N多位大佬的蹂躏,说实在的代码结构上已经非常可怕了,往往每个源文件内容非常之长,为了保持代码原样,会采用#include"xxx.c"把这几的相关文件嵌入进去,也便于自己后期维护。
在前期进行软件调试的时候可能自己会在不同的文件中安插不同测试功能函数,通过这样方法可以方便的引入和剔除。
比如说你需要对源文件中的一些静态变量进行相关的监控处理,然而又不想在本文件中增加测试代码,于是便可以在#include"xxx.c"中进行测试函数的编写来供使用,比如 :
1//FileName :main
2#include
3#include
4
5static int a = 5;
6/***************************
7 * .c文件声明区域
8 **************************/
9#include"module1.c"
10
11/***************************
12 * Fuction: main
13 * Author :(最后一个bug)
14 **************************/
15int main(int argc, char *argv[]) {
16
17 Fuction1();
18 printf("main %d\n",a);
19 printf("欢迎关注公众号:最后一个bug\n");
20 return 0;
21}
1//FileName: Module2.c
2#include
3/***************************
4 * Fuction: Fuction1
5 * Author :(最后一个bug)
6 **************************/
7void Fuction1()
8{
9 printf("Run Fuction1\n");
10 printf("Fuction1 %d\n",a);
11}
贰
void
参考void*用法:
1#include
2#include
3#include
4
5#define NUM 10
6/*************************************
7 * Fuction:了解一下void*的使用
8 * Author : (最后一个bug)
9 *************************************/
10int main(int argc, char *argv[]) {
11 int *p1 = (int *)malloc(NUM*sizeof(int));
12 int *p2 = (int *)malloc(NUM*sizeof(int));
13 int i = 0;
14
15 //初始化p1
16 for(i = 0;i < NUM;i++)
17 {
18 *(p1+i) = i;
19 }
20 //进行内存copy
21 memcpy(p2,p1,NUM*sizeof(int));
22
23 //输出另外一个分配的内存
24 for(i = 0;i < NUM;i++)
25 {
26 printf("%d,",*(p2+i));
27 }
28 //释放内存
29 free(p1);
30 free(p2);
31 return 0;
32}
运行结果:
参考实例:
1#include
2#include
3/**********************************
4 * Fuction : add
5 * descir : 加法的相关数据及处理办法
6 * Author : (最后一个bug)
7 **********************************/
8typedef struct _tag_Add
9{
10 int a;
11 int b;
12 int result;
13}sAdd;
14
15void Add(void *param)
16{
17 sAdd *p = (sAdd *) param;
18 p->result = p->a + p->b;
19}
20/**********************************
21 * Fuction : add
22 * descir : 乘法的相关数据及处理办法
23 * Author : (最后一个bug)
24 **********************************/
25typedef struct _tag_Mul
26{
27 float a;
28 float b;
29 float result;
30}sMul;
31
32void Mul(void *param)
33{
34 sMul *p = (sMul *) param;
35 p->result = p->a * p->b;
36}
37
38/*************************************
39 * Fuction : sCal
40 * descir : 公共的调用接口
41 * Author : (最后一个bug)
42 ************************************/
43void sCal(void *param,void *fuc)
44{
45 ((void (*)(void*))fuc)(param);
46}
47
48/**********************************
49 * Fuction : main
50 * descir : 应用接口实例
51 * Author : (最后一个bug)
52 **********************************/
53int main(void)
54{
55 sAdd stAdd;
56 sMul stMul;
57
58 //数据初始化
59 stAdd.a = 10;
60 stAdd.b = 20;
61
62 stMul.a = 5;
63 stMul.b = 5;
64 //接口直接用
65 sCal(&stAdd,Add);
66 sCal(&stMul,Mul);
67 //对应的输出
68 printf("a + b = %d\n",stAdd.result);
69 printf("a * b = %f\n",stMul.result);
70 printf("公众号:最后一个bug\n");
71 return 0;
72 }
运行结果:
分析一下:
叁
“ 逗号表达式 ”
一个逗号表达式的实例:
1#include
2#include
3/******************************************
4 * Fuction: Main
5 * Descir : 测试一个逗号表达式
6 * Author :(最后一个bug)
7 *****************************************/
8int main(int argc, char *argv[]) {
9 int Val = 1;
10
11 Val = ++Val,Val+10,Val*10; //逗号表达式
12
13 printf("Val = %d",Val);
14
15 return 0;
16}
分析一下:
三点搞定:
逗号表达式从表达式1开始顺序从左向右执行;
其逗号表达式最后的值为最后一个表达式的值;
逗号运算的优先级最低,也就说明与其他运算符结合使用的时候,在没有括号的情况下逗号运算符最后才执行。
上面例子的结果:
1/******************************************
2 * Fuction: 非逗号表达式书写
3 * Descir :
4 * Author :(最后一个bug)
5 *****************************************/
6if(IsOk())
7{
8 sOkProc();
9 return GetOkCode();
10}
11else
12{
13 sNoProc();
14 return GetNoCode();
15}
16/******************************************
17 * Fuction: 采用逗号表达式书写
18 * Descir :
19 * Author :(最后一个bug)
20 *****************************************/
21return (IsOk())?(sOkProc(),GetOkCode()):(sNoProc(),GetNoCode());
分析一下:
上面是两种代码书写方式,第一种占据了多行,而第二种进占据一行,这样同样一个屏幕所容纳的有效代码第一种就明显少于第二种方式,所以很多程序员都会选择使用一种大长屏或者多屏进行开发。
第二种方式似乎很多小伙伴觉得代码不够美观,也不便于维护,其实这仅仅只是一种习惯罢了,就好像编码的时候 : 第一个大括号是否需要另外起一行,或者是使用==号一定要像if( 1== b)这样把数据放左边,当你习惯了这种编码风格也会觉得用第二方式来得直接。
3、逗号表达式常用的地方
下面为大家介绍几个用逗号表示式比较多的地方:
参考demo:
1#include
2#include
3#define ROW_NUM (5)
4#define LINE_NUM (5)
5/******************************************
6 * Fuction: Main
7 * Descir :for 遍历查找
8 * Author :(最后一个bug)
9 *****************************************/
10int main(int argc, char *argv[]) {
11 int i = 0,j = 0;
12 int Matrix[ROW_NUM][LINE_NUM] ={{1,1,1,1,1},\
13 {2,2,2,2,2},\
14 {3,3,3,3,3},\
15 {4,4,4,4,4},\
16 {5,5,5,5,5},\
17 };
18
19 for(i = 0,j = 0;(i < ROW_NUM)&&(j < LINE_NUM);i++,j += 2)
20 {
21 printf("Matrix[%d][%d] = %d\n",i,j,Matrix[i][j]);
22 }
23 printf("公众号:最后一个bug\n");
24 return 0;
25}
分析一下:
上面在for循环中遍历相关数据几比较常规的处理,也是逗号表达式经常出现的地方,这样的表现形式让代码更加简单明了。
其结果如下:
参考Demo
1#include
2#include
3/******************************************
4 * Fuction: Main
5 * Descir :弱化++前后问题
6 * Author :(最后一个bug)
7 *****************************************/
8int main(int argc, char *argv[]) {
9 int i = 0;
10
11 //1、常规操作
12 i = 0;
13 while(++i < 3)
14 {
15 printf(" i = %d\n",i);
16 }
17 printf("*****************\n");
18
19 i = 0;
20 while(i++ < 3)
21 {
22 printf(" i = %d\n",i);
23 }
24 printf("*****************\n");
25
26 //2、逗号表达式处理一下
27 i = 0;
28 while( i++,i < 3)
29 {
30 printf(" i = %d\n",i);
31 }
32 printf("*****************\n");
33
34 i = 0;
35 while( ++i,i < 3)
36 {
37 printf(" i = %d\n",i);
38 }
39 printf("*****************\n");
40
41 printf("公众号:最后一个bug\n");
42 return 0;
43}
44
分析一下:
当使用逗号表达式以后,不管++在前还是在后,其都会自增加1,然后再进行右边表达式的处理,这样就不用担心是不是多记了一次,导致各种问题。
运行结果:
参考demo
1#include
2#include
3
4#define GET_INDEX(a ,b) ( a+= 2,a + b)
5/******************************************
6 * Fuction: Main
7 * Descir : 简化宏
8 * Author :(最后一个bug)
9 *****************************************/
10int main(int argc, char *argv[]) {
11 int i = 0,Val = 0;
12 int Param1 = 0, Param2 = 0;
13 int Matrix[5] ={5,5,5,5,5};
14
15 printf(" Matrix = %d\n",Matrix[GET_INDEX(Param1,Param2)]);
16 printf("公众号:最后一个bug\n");
17 return 0;
18}
分析一下:
逗号表达式最终还是一个表达式,所以它可以直接用在几乎所有变量可以用的地方,这是和语句不同的。
所以逗号表达式左边的表达式可以预先进行各种处理,其最右边的表达式相当于返回最后的结果,从而减少函数的封装和调用。
逗号表达式其实就是横向编码的一种方式,能够让程序员更好的利用一行的空间,使得代码更加紧凑,所以使用逗号表达式并没炫技,而是增强了代码的灵活度,不过话说回来逗号表达式在C混乱编码大赛上的使用频度是非常之高的。
-END-
来源 | 最后一个bug
作者 | bug菌
| 整理文章为传播相关技术,版权归原作者所有 |
| 如有侵权,请联系删除 |
【1】单片机为何既有Flash又有EEPROM?
【2】“史上”最烂代码!
【3】goto语句对与错
【4】C语言内存操作的陷阱!你踩过坑吗?
【5】RTOS 是如何进行任务划分的?
免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!