<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>

新聞中心

EEPW首頁(yè) > 嵌入式系統 > 設計應用 > keil c51編譯器對內存分配一些使用心得

keil c51編譯器對內存分配一些使用心得

作者: 時(shí)間:2016-11-28 來(lái)源:網(wǎng)絡(luò ) 收藏
現在的存儲器已經(jīng)不像七八年前那樣昂貴了,但是ram相對于rom和eeprom的價(jià)格還是不可同樣看待的,所以程序中節省內存在現在看來(lái)還是非常關(guān)鍵的。原因有以下幾點(diǎn):

1.ram的存取速度相對于eeprom的存取速度要快很多倍,不在一個(gè)數量級上,主要是因為eeprom的存儲要想寫(xiě)入就必須先擦除,而且eeprom的擦出需要成塊擦除(這是由于eeprom的擦除原理是場(chǎng)效應管的極上電擦除的,為了節省成本廠(chǎng)家一般都是8Bytes/page 64Bytes/page),所以使用ram來(lái)處理中間的數據是能夠符合速度要求的。
2.無(wú)論是xram還是eeprom都是外部存儲器,在負值時(shí)都要用到16bit地址空間(8位機),這樣無(wú)形中就增大了程序的code的體積并且使得速度上也受到影響,所以盡量把indata區的ram用到極限是非常有意義的。
本人總結了一些節省內存的規律,提供給大家討論一下,看看是否可行。
1.內存分配的基本原理:
keil與其他的c語(yǔ)言編譯器我認為從內存分配的原理上是基本相同的??偨Y起來(lái),其實(shí)很簡(jiǎn)單,就是選最長(cháng)的路徑進(jìn)行編譯(話(huà)糙理不糙),例如下面的兩段程序
program 1:
unsigned char a();
void b();
void main()
{
unsigned char byte1;
unsigned char byte2;
byte1 = byte2 = 3;
if( a() == 3)
{
b();
}
a();
return;
}
// a function
unsigned char a()
{
unsigned char byte_a1;
unsigned char byte_a2;
byte_a1 = byte_a2 = 3;
byte_a1 = 4;
byte_a2 = 5;
return byte_a1;
}
// b function
void b()
{
unsigned char byte_b1;
unsigned char byte_b2;
unsigned char byte_b3;
byte_b1 = byte_b2 = byte_b3 = 3;
return;
}
program 2:
void a();
void b();
void main()
{
unsigned char byte1;
unsigned char byte2;
byte1 = byte2 = 3;
a();
return;
}
// a function
void a()
{
unsigned char byte_a1;
unsigned char byte_a2;
byte_a1 = byte_a2 = 3;
if (byte_a1 == 3)
{
b();
}
byte_a1 = 4;
byte_a2 = 5;
return;
}
// b function
void b()
{
unsigned char byte_b1;
unsigned char byte_b2;
unsigned char byte_b3;
byte_b1 = byte_b2 = byte_b3 = 3;
return;
}
兩段程序的作用是相同的,都是先執行函數a,然后根據byte_a1的值判斷去執行b程序,但是用keil編譯的結果卻不相同program 1 編譯的結果是data:14 code:48,而program 2 編譯的結果是data:16 code:56,可見(jiàn)program 1 比 program 2 即節省了code又節省了內存。
看一下反匯編代碼,就可以了解原因了,在a函數中調用b函數,a函數定義的byte_a1和byte_a2變量沒(méi)有被釋放,所以program 2 的內存分配是 8(SFR) + 1(STACK) + 2(MAIN FUNC) + 2(A FUNC) + 3(B FUNC) = 16 Bytes,而program 1 的內存分配是 8(SFR) + 1(STACK) + 3(B FUNC) = 14Bytes, 由于B函數和A函數是并行的,所以節省了a函數需要的2個(gè)字節。
這樣總結看來(lái)程序不要串行,應盡量并行,充分利用有限的ram資源,這樣既可以使code區變小,也可以使速度變快。

本文引用地址:http://dyxdggzs.com/article/201611/322596.htm

2.uncalled segment 影響內存分配:
不知道大家是否發(fā)現過(guò)當存在沒(méi)有調用的函數時(shí),內存空間很有可能會(huì )溢出,這個(gè)原因其實(shí)也非常簡(jiǎn)單例如:
program 3:
void a();
void b();
void c(unsigned char byte_input);
void main()
{
unsigned char byte1;
unsigned char byte2;

byte1 = byte2 = 3;

a();

c(3);

return;
}

// a function
void a()
{
unsigned char byte_a1;
unsigned char byte_a2;

byte_a1 = byte_a2 = 3;

if (byte_a1 == 3)
{
b();
}

byte_a1 = 4;
byte_a2 = 5;

return;
}

// b function
void b()
{
unsigned char byte_b1;
unsigned char byte_b2;
unsigned char byte_b3;

byte_b1 = byte_b2 = byte_b3 = 3;

return;
}

void c(unsigned char byte_input)
{
unsigned char byte_c;

byte_c = byte_input;

return;
}

program 3 所示如果再main.c里面調用c(3)編譯后data:16,而如果不調用c(3),編譯后data:17,原因是調用c(3) data = 8(SFR) + 1(STACK) + 2(MAIN FUNC) + 2(A FUNC) + 3(B FUNC) = 16Bytes,而如果不調用c(3) data = 8(SFR) + 1(STACK) + 2(MAIN FUNC) + 2(A FUNC) + 3(B FUNC) + 1(C FUNC) = 17 Bytes。
所以,建議大家如果暫時(shí)不調用的函數最好屏蔽掉,以免影響整體的內存分配。

這次先寫(xiě)到這里吧,希望大家多和我討論討論,下一次我想和大家討論一下有關(guān)keil中data_group的問(wèn)題。

//-------------------------

對于 "uncalled segment 影響內存分配" 這個(gè)問(wèn)題,發(fā)表一點(diǎn)看法(因為彼人也非JSJ畢業(yè)).
程序源文件(c, a51文件),先經(jīng)過(guò)編譯得到obj文件,所謂的目標文件.各個(gè)obj文件就是一個(gè)個(gè)的模塊,每個(gè)模塊基本上都含有代碼段和數據段,也就是說(shuō),代碼在rom里面要占用多少CODE空間,數據在ram里面要占用多少ram空間等等信息.我以為lib文件也和obj文件類(lèi)似,只是文件結構有些不一樣.
obj(lib)文件然后經(jīng)過(guò)l51.exe(bl51.exe),就是說(shuō)把可執行代碼模塊根據連接定位參數地址上連接在一起();數據段也連接在一起,在ram空間中分配.對ram空間的分配中就有一個(gè)連接過(guò)程"覆蓋分析".調用一個(gè)c函數,就會(huì )為這個(gè)函數所使用的ram空間進(jìn)行分配(一些局部變量),這個(gè)函數返回時(shí)再回收分配給他的ram空間,根據函數互相之間的調用前后關(guān)系,編譯器就可以時(shí)實(shí)的知道ram空間的使用情況(其中就存在一個(gè)函數重入的問(wèn)題),作為連接時(shí)ram空間分配的參數. 如果源文件中的函數(模塊)從來(lái)沒(méi)有被任何函數顯示的調用(所謂非顯示調用就是這段代碼,連接器目前還不知道這段代碼什么時(shí)候會(huì )被調用或是否會(huì )被調用),連接時(shí)就會(huì )為它分配永遠有效的ram空間(就象全局變量),不會(huì )被回收.



關(guān)鍵詞: keilc51編譯器內存分

評論


技術(shù)專(zhuān)區

關(guān)閉
国产精品自在自线亚洲|国产精品无圣光一区二区|国产日产欧洲无码视频|久久久一本精品99久久K精品66|欧美人与动牲交片免费播放
<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>