本文使用环境:
语言:C/C++
编译环境:都可
写在前面:
这篇文章仅仅只为 自己 的学习。
一、C语言遇到的不算问题的问题
1、宏定义计算
宏定义中必须要加括号。
正确:
#define Period (MAIN_Fosc/100000) //350
//初以1000 转化成khz
#define PWM_Fre (MAIN_Fosc/1000/Period)
错误:
#define Period MAIN_Fosc/100000 //350
//初以1000 转化成khz
#define PWM_Fre MAIN_Fosc/1000/Period
2、结构体指针和结构体变量
struct tm *time_point;
struct tm time_temp;
time_point 和 &time_temp 虽然都是表示的相同结构体的地址,但是不对等。
上述两个变量有着本质的区别。
time_point:表示指针,指向time_point的指针,注意:这里的指针time_point并没有分配地址,即time_point = NULL,因此使用的时候不能对内容直接赋值,必须分配地址:要不申请空间,要不把其他的首地址赋值给它。
time_temp:表示结构体的名字,注意这里已经分配地址了,&time_temp是该结构体的首地址,是实实在在的地址。
总结:两个东西不要搞混淆了。
用指针的形式,可以少申请一段空间,因为可以直接将存在的空间直接赋值给它,但这段空间的数据是两个变量使用,一个变量改变会影响另一个变量。
如果用后一种定义,相当于多使用的一份空间,费空间但可以保证数据的独立性。
2、C语言函数同名
GCC在编译的时候,如果某个文件中有弱定义函数(attribute((weak)) fun()),另一个不相关的文件中有一个相同函数名的函数(fun(int data)),预处理–编译–汇编–链接,四个过程都不会报错,但是在链接过程中会将fun(int链接到弱定义对应的函数中)–即C语言重名函数问题
举例:
main函数
#include <stdio.h>
__attribute__((weak)) int fun(void)
{
;
}
int main()
{
printf("fun entry\n");
fun();
printf("fun end\n");
}
test函数:
#include <stdio.h>
int fun(int data)
{
printf("data:%d\n",data);
}
编译两个文件然后执行,会发现main里面打印data了。
此现象说明,编译过程中因为有弱定义,所以main.c没有报错,但是在ld过程中,编译器会首先找全部的文件,是否有对应的函数名,如果有的话就直接将对应的函数名链接在一起。所以在main里面的fun没有使用若定义的fun,而是使用了test里面的fun。
如果有多个文件里面有同名函数呢?
答案是报错,因为ld的时候会发现有多个函数是相同的名字,导致链接出错。上述情况发生错误,是因为有弱定义,编译器不会报重定义。
fun的参数为什么会是10呢?
这就涉及到另外的问题,函数在调用的时候,默认的参数是存放在几个固定的寄存器中,(32一般是r0-r3,64一般是r0-r8)如果参数超过这些寄存器的值,会将参数入栈。说明之前的r0寄存器中的值刚好是10。
函数的返回值存放在r0中