第六章不看,直接到第七章
所謂的啟動(dòng)例程用C寫起來一般是這樣,但一般情況這個(gè)會(huì)是匯編來寫
exit(main(argc, argv))
exit函數(shù)
_exit和_Exit立即進(jìn)入kernel,exit則需要做完必要的清理工作,先run注冊過的終止處理程序,然后清理沒有關(guān)閉的流。
exit _Exit是ISO C標(biāo)準(zhǔn),_exit則是Posix,所以頭文件不一樣
前面是stdlib.h 后面是unistd.h
關(guān)于malloc,calloc,realloc
malloc分配的空間初始值不確定;calloc會(huì)幫忙清零;realloc可以動(dòng)態(tài)分配合適的大小空間,如果不夠會(huì)有copy原始數(shù)據(jù)的動(dòng)作,所以對realloc之前的memory記錄指針不是好方法,因?yàn)閞ealloc完就可能失效。
這三個(gè)都是去call sbrk來擴(kuò)大縮小heap
但一般只是擴(kuò)大,在free的時(shí)候 malloc 會(huì)把擴(kuò)大的 memory cache起來(virual memory)
setjmp, longjmp
跳轉(zhuǎn)函數(shù),支持跨越函數(shù)幀的跳轉(zhuǎn),goto是函數(shù)內(nèi)跳轉(zhuǎn)
注意: longjmp到setjmp的點(diǎn)后,一般情況 全局/局部/局部靜態(tài)變量都沒辦法恢復(fù)到setjmp時(shí)的值;register/volatile變量在編譯后(cc -O)會(huì)放到寄存器中,這些都能實(shí)現(xiàn)回滾。這也就說明setjmp的時(shí)候,jmp_buf并沒有保存這些值。
7.1 如果不調(diào)用return 和 exit 函數(shù)返回值用了printf的返回值,Orz...我沒想清楚怎么做的。
7.2 取決于printf面對的設(shè)備類型,行緩沖的話,碰到換行就會(huì)被輸出,如果是全緩沖,那么則要么需要call fflush,要么等stream被close。
7.3 不用參數(shù)傳遞,不用全局變量,Orz...要傳遞argc, argv -->NO
7.4 通常null就是0,一般指針為 null 的時(shí)候意味著空指針引用,一般0都不會(huì)去放數(shù)據(jù),RO的起始位置在一個(gè)更高的地址,0會(huì)是無效的,用來指引程序出錯(cuò)。
7.5
typedef void func(void);
int atexit(func* fp);
或者
typedef void (*func)(void);
int atexit(func fp);
我喜歡下面那種,因?yàn)楹瘮?shù)本身就是個(gè)pointer。
7.6 calloc會(huì)將分配的空間清零,ISO C不保證0值和空指針相等
7.7 size輸出的是靜態(tài)的RO,RW,ZI,像Heap,Stack是runtime才有的。
7.8 理論上不帶debug信息的.out文件應(yīng)該大小都是等于RO+RW+ZI的。
7.9 這就是.a和.so的區(qū)別了,link的時(shí)候靜態(tài)庫會(huì)被直接copy過來,所以會(huì)變大很多。
7.10 變量作用域的故事。