现在的位置: 首页 > 综合 > 正文

对c 和 c++ 的几点感悟

2017年09月02日 ⁄ 综合 ⁄ 共 1979字 ⁄ 字号 评论关闭

在家是专门离线记录的,现在在公司,就在这写写,自己能看懂就行。

1. 资源是有限的,不是无限的,因此资源是需要被分享的,这从根本上导致了程序bug,导致了程序的复杂性。

 eg.:钓鱼岛之争,因为钓鱼岛背后的能源,能源有限,却由两个不同利益团体需要。

 eg.:内存地址空间是有限的,却被不同的进程或线程公用, 内存管理函数来了,操作系统做的事情就麻烦了。

    eg.:   以arm920t为例,每一种模式能用的寄存器最多也就18个(r0-r15, cpsr, spsr),而这18个寄存器被所有函数公用,这么少的资源,既然能组成这么庞大的嵌入式世界,本身 就很神奇,所以写汇编麻烦,不容易懂,高级语言厉害之一就在于不用程序员去管理那些麻烦的寄存器。 sp(r13)的共享导致栈溢出是个经常出现的问题。

    eg.:栈溢出: 一个函数破坏了别的函数的栈,因为同一个进程下地址空间是一致的,栈的分配也确定了,栈被该进程下所有函数共享,栈溢出导致bug。

    eg.:慎用全局变量的原因,因为该全局变量被不同源文件,不同函数共享等等,在有优先级的多线程程序中,全局变量所以最好加保护,就像领土国家要派士兵保卫一样。

2. 在c中有很多是 成对(pair)的, 即其生命周期中执行了2次,另外成对的也必须执行两次。

    eg.:是执行,而不是出现,典型的是内存分配和释放,简化考虑在一个函数中实现,在一个函数中执行了成功两次malloc, 在该函数执行完成前必须两次执行free内存(非出现),因为有可能出现if等分支导致free出现多次,但真正只执行两次。 当然free 后要 使指针为NULL避免野指针。

    eg.: c和c++对于if等成对控制分支没有Verilog那么严格,Verilog中if 和 else 必须成对,且if中用到的变量在else中也必须赋值,如果不这样很容易导致锁存器,导致bug。

    eg.:   变量(对象)属性成对:从c来看为  值(类型)  + 地址。 从汇编来看为 地址 + 变量占用的内存字节(以及排列方式);

3. c/c++中一个变量的两个必要属性, 值与地址。 地址本身是无法单独存在的,其存在的价值是能够找到其对应的变量的值, 所以c中函数有所谓的值传递,地址传递。 比如指针,如果指针未被正确赋值,就成了野指针(如执行free后未置NULL)。 怎么解释呢,感觉绕口, 指针本身是个变量,所以指针变量也有地址和值,只是其值存的是其它变量的地址,通过指针变量的值访问到其它变量的地址,进而访问到对应变量的值。 

   eg.:  这是今天主要想写的,贴代码说,因为也是弄了半天才真正明白。

#include <stdio.h>
typedef signed long acpi_native_int; //32 位,4字节
#define  _AUPBND                (sizeof (acpi_native_int) - 1)
#define  _ADNBND                (sizeof (acpi_native_int) - 1)
#define _bnd(X, bnd)            (((sizeof (X)) + (bnd)) & (~(bnd)))

#define _bnd(X, bnd)            (((sizeof (X)) + (bnd)) & (~(bnd)))  // 下一个变量地址(必须按对齐规则,本例中是4字节对齐)
#define va_arg(ap, T)           (*(T *)(((ap) += (_bnd (T, _AUPBND))) - (_bnd (T,_ADNBND))))
#define va_end(ap)              (ap = (va_list) NULL)
#define va_start(ap, A)         (void) ((ap) = (((char *) &(A)) + (_bnd (A,_AUPBND))))


char *test(const char *fmt, ...)
{
	char *args = 0;
	va_start(args, fmt);  // 经宏展开得到获得了test函数第一个可选形参的地址,因为test的参数在栈中从高到低为test中从右到左。
	
	int args_value = *(int *)args; // 根据获得args的地址获得args的值
	int *var_addr = (int *)args_value; // args的值同时也是main中传来的var的地址
	int var = *var_addr; // 根据var的地址解析得值
	
	va_arg(args, int *); 
	int num = (*((int *)args));	
	
	printf("var = %d\n", var);
	printf("num = %d\n", num);
		
	return args;
}

int main(void)
{
	int var = 2; 
	test("hello", &var, 20);	 
	return 0;
}
     

如上代码是在看linux的sscanf函数时用来验证的,看注释分析来分析去,也就是在变量的地址和值见绕。

4. 还有好些,是离线记的。

 

抱歉!评论已关闭.