博客專欄

EEPW首頁 > 博客 > 向上取整的一點(diǎn)技巧 | C語言

向上取整的一點(diǎn)技巧 | C語言

發(fā)布人:魚鷹談單片機(jī) 時(shí)間:2021-12-19 來源:工程師 發(fā)布文章

做底層或者說驅(qū)動(dòng)開發(fā)時(shí),經(jīng)常會(huì)遇到一些硬件限制,比如 STM32 有些寄存器要求 16 bit 訪問、32 bit 訪問,否則可能引發(fā) Hardfault 中斷;再比如擦除 flash 扇區(qū)時(shí),會(huì)要求整塊擦除。這些都是硬件決定的,必須這樣做。

假設(shè)我們需要 32 bit 訪問一個(gè)地址 dst 時(shí),如果要寫入 10 個(gè)字符(兩次寫入 8 字節(jié),所以需要 3 次才能完全寫入),一般我們是怎么做的呢?

char     *str = "0123456789";
uint32_t  len = strlen(str);
uint32_t *src = str; 
uint32_t *dst = 0x20000000;
#define SIZE    (4)
if((len % SIZE) != 0) // 判斷是否整除
{
  len = len / SIZE + 1;
}
else
{
  len = len / SIZE;
}
for(int i = 0; i < len; i++)
{
   dst [i] = src [i];  //  32 bit 訪問
}

我們可以看到,上面的寫法是最容易想到的一種,也是新手最常寫的,但是有經(jīng)驗(yàn)的老手會(huì)怎么寫呢?

char     *str = "0123456789";
// 設(shè)置 10 字節(jié)寫入,當(dāng)然一般情況下是以函數(shù)參數(shù)的形式傳入
uint32_t  len = strlen(str);
uint32_t *src = str; 
uint32_t *dst = 0x20000000;
#define SIZE    (4)
len = (len + SIZE - 1) / SIZE;
for(int i = 0; i < len; i++)
{
   dst [i] = src [i];  //  32 bit 訪問
}
len = (len + SIZE - 1) / SIZE;   //重點(diǎn)

就上面一條代碼,簡單高效直接。

一般看到這種代碼,你就會(huì)覺得這個(gè)作者有水平,不會(huì)是新手。

事實(shí)上,魚鷹看了不少開源代碼,發(fā)現(xiàn)只要會(huì)這么操作的,代碼水平一般不會(huì)太差的。當(dāng)然不是說寫出最上面代碼的水平一定不怎么樣,就像你不能說一個(gè)人寫的字很難看,然后斷定他作文水平不行是一樣的道理。

只是能寫出上面代碼的,應(yīng)該可以稱之為經(jīng)驗(yàn)豐富了吧。

繼續(xù)探討上面的代碼。

你會(huì)發(fā)現(xiàn)上面的代碼非常巧妙,寫入長度 + 對(duì)齊長度 - 1,這樣一來即使寫入長度只多一個(gè),也一定會(huì)導(dǎo)致最終結(jié)果 + 1,這樣就巧妙的避免了 if 判斷。

而且這條語句沒有限制說一定是偶數(shù)或奇數(shù),而是任意正整數(shù)(0 和 1 除外),可謂巧妙至極。

所以你進(jìn)行固件升級(jí)時(shí),如果要計(jì)算一共需要的扇區(qū)大小,不如使用該方式吧(當(dāng)然對(duì)于扇區(qū)大小不同的不行)。這會(huì)讓你的同事覺得你很有水平的。

向下取整就簡單多了:

len /= SIZE;


*博客內(nèi)容為網(wǎng)友個(gè)人發(fā)布,僅代表博主個(gè)人觀點(diǎn),如有侵權(quán)請(qǐng)聯(lián)系工作人員刪除。



關(guān)鍵詞: 編程

技術(shù)專區(qū)

關(guān)閉