c語(yǔ)言實(shí)現(xiàn)向上取整計(jì)算方法
c語(yǔ)言向上取整計(jì)算
用整數(shù)N 除以 M,要求向上取整數(shù)
int n = (N + M -1) / M ;
簡(jiǎn)化后就是:
int n= (N - 1) / M +1;
最笨的辦法
int n = N / M + ((N % M != 0) ? 1 : 0 )
c語(yǔ)言向上取整的一點(diǎn)技巧
做底層或者說(shuō)驅(qū)動(dòng)開(kāi)發(fā)時(shí),經(jīng)常會(huì)遇到一些硬件限制,比如 STM32 有些寄存器要求 16 bit 訪問(wèn)、32 bit 訪問(wèn),否則可能引發(fā) Hardfault 中斷;再比如擦除 flash 扇區(qū)時(shí),會(huì)要求整塊擦除。這些都是硬件決定的,必須這樣做。
假設(shè)我們需要 32 bit 訪問(wèn)一個(gè)地址 dst 時(shí),如果要寫(xiě)入 10 個(gè)字符(兩次寫(xiě)入 8 字節(jié),所以需要 3 次才能完全寫(xiě)入),一般我們是怎么做的呢?
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 訪問(wèn)
}可以看到,上面的寫(xiě)法是最容易想到的一種,也是新手最常寫(xiě)的,但是有經(jīng)驗(yàn)的老手會(huì)怎么寫(xiě)呢?
char ? ? *str = "0123456789";
// 設(shè)置 10 字節(jié)寫(xiě)入,當(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 訪問(wèn)
}
len = (len + SIZE - 1) / SIZE; ? //重點(diǎn)就上面一條代碼,簡(jiǎn)單高效直接。
一般看到這種代碼,你就會(huì)覺(jué)得這個(gè)作者有水平,不會(huì)是新手。
事實(shí)上,魚(yú)鷹看了不少開(kāi)源代碼,發(fā)現(xiàn)只要會(huì)這么操作的,代碼水平一般不會(huì)太差的。當(dāng)然不是說(shuō)寫(xiě)出最上面代碼的水平一定不怎么樣,就像你不能說(shuō)一個(gè)人寫(xiě)的字很難看,然后斷定他作文水平不行是一樣的道理。
只是能寫(xiě)出上面代碼的,應(yīng)該可以稱(chēng)之為經(jīng)驗(yàn)豐富了吧。
繼續(xù)探討上面的代碼。
你會(huì)發(fā)現(xiàn)上面的代碼非常巧妙,寫(xiě)入長(zhǎng)度 + 對(duì)齊長(zhǎng)度 - 1,這樣一來(lái)即使寫(xiě)入長(zhǎng)度只多一個(gè),也一定會(huì)導(dǎo)致最終結(jié)果 + 1,這樣就巧妙的避免了 if 判斷。
而且這條語(yǔ)句沒(méi)有限制說(shuō)一定是偶數(shù)或奇數(shù),而是任意正整數(shù)(0 和 1 除外),可謂巧妙至極。
所以你進(jìn)行固件升級(jí)時(shí),如果要計(jì)算一共需要的扇區(qū)大小,不如使用該方式吧(當(dāng)然對(duì)于扇區(qū)大小不同的不行)。這會(huì)讓你的同事覺(jué)得你很有水平的。
向下取整就簡(jiǎn)單多了:
len /= SIZE;
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
C語(yǔ)言構(gòu)建動(dòng)態(tài)數(shù)組完整實(shí)例
這篇文章主要介紹了C語(yǔ)言構(gòu)建動(dòng)態(tài)數(shù)組完整實(shí)例,幫助讀者加深對(duì)C語(yǔ)言數(shù)組及指針的理解,需要的朋友可以參考下2014-07-07
QT5實(shí)現(xiàn)TTS文本語(yǔ)音朗讀功能
TTS?語(yǔ)音朗讀?是開(kāi)發(fā)中常用的功能,Qt已經(jīng)給封裝完成,我們只需要調(diào)用即可,本文就為大家介紹了QT5如何調(diào)用實(shí)現(xiàn)文本朗讀功能的,需要的可以參考一下2023-05-05
C語(yǔ)言中四種取整方式,取余/取模運(yùn)算以及負(fù)數(shù)取模問(wèn)題詳解
這篇文章主要介紹了C語(yǔ)言中四種取整方式及負(fù)數(shù)取模問(wèn)題,包括了算法的分析與改進(jìn),是很多程序設(shè)計(jì)競(jìng)賽中常見(jiàn)的算法,需要的朋友可以參考下2021-09-09
C語(yǔ)言數(shù)據(jù)的存儲(chǔ)超詳細(xì)講解下篇浮點(diǎn)型在內(nèi)存中的存取
使用編程語(yǔ)言進(jìn)行編程時(shí),需要用到各種變量來(lái)存儲(chǔ)各種信息。變量保留的是它所存儲(chǔ)的值的內(nèi)存位置。這意味著,當(dāng)您創(chuàng)建一個(gè)變量時(shí),就會(huì)在內(nèi)存中保留一些空間。您可能需要存儲(chǔ)各種數(shù)據(jù)類(lèi)型的信息,操作系統(tǒng)會(huì)根據(jù)變量的數(shù)據(jù)類(lèi)型,來(lái)分配內(nèi)存和決定在保留內(nèi)存中存儲(chǔ)什么2022-04-04
C++中拷貝構(gòu)造函數(shù)的總結(jié)詳解
深拷貝和淺拷貝可以簡(jiǎn)單理解為:如果一個(gè)類(lèi)擁有資源,當(dāng)這個(gè)類(lèi)的對(duì)象發(fā)生復(fù)制過(guò)程的時(shí)候,資源重新分配,這個(gè)過(guò)程就是深拷貝,反之,沒(méi)有重新分配資源,就是淺拷貝2013-09-09
一般函數(shù)指針和類(lèi)的成員函數(shù)指針深入解析
以下是對(duì)一般函數(shù)指針和類(lèi)的成員函數(shù)指針進(jìn)行了介紹。需要的朋友可以過(guò)來(lái)參考下2013-08-08
關(guān)于C++中由于字節(jié)對(duì)齊引起內(nèi)存問(wèn)題定位分析
前幾天遇到一個(gè)稀奇古怪的問(wèn)題,在創(chuàng)建對(duì)象的時(shí)候程序異常退出,查找代碼發(fā)現(xiàn)結(jié)構(gòu)體數(shù)組問(wèn)題,最終把問(wèn)題簡(jiǎn)化得到解決方法,下面小編把我的問(wèn)題及解決方案分享到腳本之家平臺(tái)供大家參考下2021-06-06

