嵌入式入门学习笔记,遇到的问题以及心得体会!
DAY16
概述:
1.工程目录的搭建
2.Make & Makefile
3.条件编译
笔记:
1.工程目录的搭建
工程名目录
--------->bin:该目录主要存储工程的可执行文件
--------->obj:该目录主要存储工程中的目标文件
--------->include:该目录主要存储工程目录中的头文件
--------->src:该目录主要存储工程中的原文件
C工程源文件中的文件类型:
源文件:纯代码文件
头文件:
以.h为后缀的文件
原文件:
以.c为后缀的文件
头文件的书写格式:
#ifndef 宏名 //如果没有定义这个宏
#define 宏名 //定义这个宏
… //…是宏的内容
#endif //#ifxxx都必须和#endif配套使用
宏名:建议下划线加大写字母组成
建议和头文件名保持一致
#ifndef __MYTEST_H__
#define __MYTEST_H__
#endif
#if 0 //如果假
xxxx
#endif
头文件中写的内容:
1.预处理指令
->防止头文件重复包含的预处理指令
->宏定义
2.类型定义
typedef
3.函数声明
使用头文件的目的:
给函数调用时,做语法检查
提供类型定义,宏定义
注意:
任何C语言工程中,有且仅有一个main函数。
在C语言中,凡是看到以#开头的指令,大多都是预处理指令
GCC编译器的编译流程:
1.预处理:
作业:处理预处理指令
预处理指令:
常用的:
#include
#define
#if
#ifdef
#endif
#else
#elif
#ifndef
不常用的:
#pragam
#error
#if开头的常用于代码过滤和条件编译
预处理的命令:
gcc -E xxx.c -o xxx.i
2.编译
作用:将.c/.i原文件编译成.s汇编文件
编译指令:
gcc -S xxx.c/i
不用-o重命名,默认生成.s文件
可以通过汇编代码分析程序执行过程和过程中产生的错误。
3.汇编
作用:将汇编指令翻译成机器指令(二进制指令)
汇编指令:
gcc -c xxx.c/i/s
不用-o重命名,默认生成.o文件(我们称为目标文件)
目标文件是不可执行的二进制文件,因为没有链接库,所以不能执行。
4.链接
链接库
vi快速跳转:
一定是在命令行模式下:
光标快速回到文件开头:gg
光标快速回到文件末尾:shift g
光标快速移动到某一行:首先保证光标在第一行,输入想要跳转的行数,回车
函数调用的格式:
<函数返回值类型的变量> = <被调函数名>(<实参列表>);
<被调函数名>(<实参列表>);
例子:
int main()
{
man();
return 0;
}
int main()
{
int ret = man();
return 0;
}
搭建工程目录:
mkdir myproname
cd myproname
mkdir bin obj include src
工程管理工具:
make
工程管理文件:Makefile/makefile
make遵循Makefile中编写的规则来管理工程
make的使用
make:编译工程
make clean:执行clean规则,清空编译产生的文件
make -C:进入制定的目录执行make
makefile的编写:
Makefile是一个文件,文件名为Makefile/makefile
目标: 依赖
规则
例子:
mystrlen.out : main.o myopt.o
gcc main.o myopt.o -o mystrlen.out
rm *.o
main.o : main.c
gcc -c main.c -o main.o
myopt.o : myopt.c
gcc -c myopt.c -o myopt.o
clean:
rm mystrlen.out
例子1:
myman : main.o man.o
gcc main.o man.o -o myman
rm *.o
main.o : main.c
gcc -c main.c -o main.o
man.o : man.c
gcc -c man.c -o man.o -D VER1
生成工程时:make
清空工程时:make clean
makefile可以自己推导编译命令:
例如:
mystrlen.out : main.o myopt.o
gcc main.o myopt.o -o mystrlen.out
rm *.o
main.o : main.c
myopt.o : myopt.c
clean:
rm mystrlen.out
makefile中的变量:
obj= main.o myopt.o
mystrlen.out : $(obj)
gcc main.o myopt.o -o mystrlen.out
rm $(obj)
main.o : main.c
myopt.o : myopt.c
clean:
rm mystrlen.out
常见的自动变量:
$*:不包含扩展名的目标文件
$+:所有的依赖文件,以空格分开,并以出现的先后为序
$<:第一个依赖文件的名称
$?:所有时间戳比目标文件晚的依赖文件,并以空格分开
$@:目标文件的完整名称
$^:所有不重复的目标依赖文件,以空格隔开
$%:如果目标是归档成员,则该变量表示目标的归档成员名称
例子:
mystrlen.out : main.o myopt.o
gcc $+ -o $@
rm $+
main.o : main.c
myopt.o : myopt.c
.PHONY : clean
clean:
rm mystrlen.out
注意://.PHONY:用来修饰clean,使得此clean与其他clean区分开来,让执行make clean执行其下面的命令。
工程目录下嵌套的makefile
humpro-------------bin
|
|----------obj------------------Makefile
|
|----------include
|
|----------src------------------Makefile
|
|----------Makefile
工程目录中:
Makefile:
1.分别进入src和obj目录中执行make
2.管理整个工程
src中的Makefile:
生成目标文件,并将目标文件移动到obj目录中
obj中的Makefile:
生成可执行文件,并将可执行文件移动到bin目录下
makefile中加入注释:
以#开头:
例子: #注释的内容
条件编译:
什么时候用:
需要选择性编译代码中一部分功能时使用
实现:
通过预处理指令,在预处理阶段进行代码过滤,进而实现条件编译
条件编译时编译的命令:
gcc xxx.c -o xxx -D 编译条件(通常为宏)
#include <stdio.h>
int wstar()
{
/*如果定义了V1这个宏,则编译时,只编译下面的这个内容*/
#ifdef V1
puts("Liu shi shi");
#elif V2
puts("Zhao li ying");
#elif V3
puts("Jia ling");
#elif V4
puts("Lin yun er");
#elif V5
puts("Li zhi en");
#endif
return 0;
}
int main()
{
wstar();
return 0;
}