91欧美超碰AV自拍|国产成年人性爱视频免费看|亚洲 日韩 欧美一厂二区入|人人看人人爽人人操aV|丝袜美腿视频一区二区在线看|人人操人人爽人人爱|婷婷五月天超碰|97色色欧美亚州A√|另类A√无码精品一级av|欧美特级日韩特级

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

C語(yǔ)言中的高內(nèi)聚低耦合講解

multisim ? 來(lái)源:CSDN博客 ? 作者:迂者-賀利堅(jiān) ? 2021-08-16 14:06 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

編程時(shí),我們講究的是高內(nèi)聚低耦合,在協(xié)同開發(fā)、代碼移植、維護(hù)等環(huán)節(jié)都起到很重要的作用。

一、原理篇而低耦合,是指模塊之間盡可能的使其獨(dú)立存在,模塊之間不產(chǎn)生聯(lián)系不可能,但模塊與模塊之間的接口應(yīng)該盡量少而簡(jiǎn)單。這樣,高內(nèi)聚從整個(gè)程序中每一個(gè)模塊的內(nèi)部特征角度,低耦合從程序中各個(gè)模塊之間的關(guān)聯(lián)關(guān)系角度,對(duì)我們的設(shè)計(jì)提出了要求。

程序設(shè)計(jì)和軟件工程發(fā)展過(guò)程中產(chǎn)生的很多技術(shù)、設(shè)計(jì)原則,都可以從內(nèi)聚和耦合的角度進(jìn)行解讀。作為C語(yǔ)言程序設(shè)計(jì)的初學(xué)者,結(jié)合當(dāng)前對(duì)于函數(shù)的理解可達(dá)到的程度,我們探討一下如何做到高內(nèi)聚低耦合。

針對(duì)低耦合。耦合程度最低的是非直接耦合,指兩個(gè)函數(shù)之間的聯(lián)系完全是通過(guò)共同的調(diào)用函數(shù)的控制和調(diào)用來(lái)實(shí)現(xiàn)的,耦合度最弱,函數(shù)的獨(dú)立性最強(qiáng)。但一組函數(shù)之間沒(méi)有數(shù)據(jù)傳遞顯然不現(xiàn)實(shí),次之追求數(shù)據(jù)耦合,調(diào)用函數(shù)和被調(diào)用函數(shù)之間只傳遞簡(jiǎn)單的數(shù)據(jù)參數(shù),例如采用值傳遞方式的函數(shù)。

有些函數(shù)數(shù)在調(diào)用時(shí),利用形式參數(shù)傳地址的方式,在函數(shù)體內(nèi)通過(guò)指針可以修改其指向的作用域以外的存儲(chǔ)單元,這構(gòu)成了更強(qiáng)的耦合,稱為特征耦合,在這里,使函數(shù)之間產(chǎn)生聯(lián)系的是地址這樣的特征標(biāo)識(shí)。另外,有兩個(gè)函數(shù)可能會(huì)打開同一個(gè)文件進(jìn)行操作,這也構(gòu)成了特征耦合的一種形式。

更強(qiáng)的耦合是外部耦合,這里,一組模塊都訪問(wèn)同一全局變量,而且不通過(guò)參數(shù)表傳遞該全局變量的信息,當(dāng)發(fā)現(xiàn)程序執(zhí)行結(jié)果異常時(shí),很難定位到是在哪個(gè)函數(shù)中出了差錯(cuò)。不少初學(xué)者覺得參數(shù)傳遞麻煩,將要處理的數(shù)據(jù)盡可能地定義為全局變量,這樣,函數(shù)之間的接口簡(jiǎn)單了,但形成的是耦合性很強(qiáng)的結(jié)構(gòu)。

在C語(yǔ)言中,還可以通過(guò)靜態(tài)局部變量,在同一個(gè)程序的兩次調(diào)用之間共享數(shù)據(jù),這也可以視為是一種外部耦合,只不過(guò)靜態(tài)局部變量的作用域限于函數(shù)內(nèi)部,其影響也只在函數(shù)內(nèi)部,耦合程度比使全局變量也還是弱很多。由此,我們可以理解前述在使用全局變量、靜態(tài)局部變量時(shí)提出的“用在合適的時(shí)候,不濫用”的原則。

針對(duì)高內(nèi)聚。內(nèi)聚程度最高的是功能內(nèi)聚,模塊內(nèi)所有元素的各個(gè)組成部分全部都為完成同一個(gè)功能而存在,共同完成一個(gè)單一的功能,模塊已不可再分。這樣的函數(shù)功能非常清晰、明確,一般出現(xiàn)在程序結(jié)構(gòu)圖的較低被調(diào)用的層次上。

次之的是順序內(nèi)聚,一個(gè)函數(shù)中各個(gè)處理元素和同一個(gè)功能密切相關(guān),通常前一個(gè)處理元素的輸出是后一個(gè)處理元素的輸入。對(duì)于這樣的函數(shù),如果不致于產(chǎn)生高耦合的話,可以分開兩個(gè)函數(shù)實(shí)現(xiàn)。

有的函數(shù),其中的不同處理功能僅僅是由于都訪問(wèn)某一個(gè)公用數(shù)據(jù)而發(fā)生關(guān)聯(lián),這稱為通信內(nèi)聚和信息內(nèi)聚,內(nèi)聚程度進(jìn)一步下降。內(nèi)聚程度再低的情況就不再一一列舉,最差的偶然內(nèi)聚中,一個(gè)函數(shù)內(nèi)的各處理元素之間沒(méi)有任何聯(lián)系,只是偶然地被湊到一起。

可以想像這樣的模塊東一榔頭西一錘子,類似一個(gè)毫無(wú)凝聚力的團(tuán)伙,對(duì)應(yīng)的是低質(zhì)量??傊?,在解決問(wèn)題劃分函數(shù)時(shí),要遵循“一個(gè)函數(shù),一個(gè)功能”的原則,盡可能使模塊達(dá)到功能內(nèi)聚。

要做到高內(nèi)聚低耦合,重點(diǎn)是要在寫代碼之前花些時(shí)間做好設(shè)計(jì)。在下面的例子中,將討論結(jié)合具體的問(wèn)題,如何將以上的因素考慮進(jìn)去。

二、示例篇本例受裘宗燕老師《從問(wèn)題到程序——程序設(shè)計(jì)與C語(yǔ)言引論啟發(fā)》。

任務(wù)輸出200以內(nèi)的完全平方數(shù)(一個(gè)數(shù)如果是另一個(gè)整數(shù)的完全平方,那么我們就稱這個(gè)數(shù)為完全平方數(shù),也叫做平方數(shù)),要求每隔5個(gè)數(shù)據(jù)要輸出一個(gè)換行。

解決方案及點(diǎn)評(píng)對(duì)于這個(gè)簡(jiǎn)單任務(wù),我們?cè)谝粋€(gè)main函數(shù)中完成了任務(wù)。程序如方案1:

//方案1:內(nèi)聚性較高的單模塊實(shí)現(xiàn)方案

#include 《stdio.h》

int main()

{

int m, num=0;

for (m = 1; m * m 《= 200; m++)

{

printf(“%d ”, m * m);

num++;

if (num%5==0)

printf(“

”);

}

return 0;

}

由于任務(wù)本身簡(jiǎn)單,將之在一個(gè)main函數(shù)中實(shí)現(xiàn)后,這個(gè)函數(shù)的內(nèi)聚程度接近功能內(nèi)聚,已經(jīng)相當(dāng)高了,就任務(wù)本身,不需再進(jìn)行分解。為使讀者能深入理解模塊質(zhì)量方面的技術(shù),我們將試圖將內(nèi)聚程序再提高一些,然后考察耦合程度不同的各種解決方案。

要提高上面解決方案中函數(shù)(僅main一個(gè)函數(shù))的內(nèi)聚程度,我們考察程度的功能“找出完全平方數(shù)并輸出”——“找出完全平方數(shù)”和“輸出”這本身就是兩個(gè)功能,再細(xì)分輸出時(shí)還有“要求5個(gè)數(shù)據(jù)在一行”的要求,這些功能的實(shí)現(xiàn)細(xì)節(jié)都在一個(gè)函數(shù)當(dāng)中,可見是有余地再提高內(nèi)聚程度的。

在實(shí)現(xiàn)的應(yīng)用中,幾乎所有的處理都可以分解為“輸入-計(jì)算-輸出”的模式,優(yōu)秀的解決方案往往至少要將這三個(gè)模塊都獨(dú)立出來(lái),對(duì)于“計(jì)算”模塊而言,其內(nèi)部不再包括輸入輸出,專門接受輸入的數(shù)據(jù),計(jì)算完成后返回結(jié)果即可。當(dāng)然,對(duì)于復(fù)雜的問(wèn)題,在各個(gè)環(huán)節(jié)上可能還需要再做分解。

下面,我們探討將“找出完全平方數(shù)輸出”和“每5個(gè)數(shù)據(jù)后換行”分開實(shí)現(xiàn)的方案。這樣的分解有助于提高內(nèi)聚性,與此同時(shí),分解后的兩個(gè)模塊間的耦合程度,成為我們要關(guān)注的焦點(diǎn)。

現(xiàn)在將“找出完全平方數(shù)并輸出”的功能仍放在main函數(shù)中(獨(dú)立成為單獨(dú)的函數(shù)也可以,但不必要了),而“每5個(gè)數(shù)據(jù)后換行”的功能,設(shè)計(jì)一個(gè)名稱為format的函數(shù),它每調(diào)用一次就輸出一個(gè)空格作為兩個(gè)完全平方數(shù)間的分隔,而每調(diào)用到第5次時(shí),輸出的是一個(gè)換行。

這兩個(gè)模塊之間,需要有一個(gè)“現(xiàn)在是第幾次調(diào)用”的信息需要傳遞,不可能用耦合程度最松散的非直接耦合。我們考慮數(shù)據(jù)耦合,用簡(jiǎn)單形式參數(shù)傳值,得到方案2。

//方案2:一個(gè)耦合度低,但不能完成功能要求的解決方案

#include 《stdio.h》

void format(int);

int main()

{

int m, num=0;

for (m = 1; m * m 《= 200; m++)

{

printf(“%d”, m * m);

format(num);

}

return 0;

}

void format(int n)

{

n++;

if (n%5==0)

printf(“

”);

else

printf(“ ”);

return;

}

在這個(gè)程序結(jié)構(gòu)中,format與main函數(shù)的耦合程度為數(shù)據(jù)耦合。在main中定義了局部變量num,在一次都未輸出時(shí),置初值為0是合理的。在調(diào)用format時(shí),將num傳遞來(lái)的表示第幾次輸出(第幾個(gè)完全平方數(shù))的形式參數(shù)n,n自增1,然后再控制輸出空格或換行。

然而分析和運(yùn)行程序發(fā)現(xiàn),“每隔5個(gè)數(shù)據(jù)輸出一個(gè)換行”的功能并未實(shí)現(xiàn)。因?yàn)樾问絽?shù)n在函數(shù)format內(nèi)的改變對(duì)應(yīng)的實(shí)在參數(shù)num占不同的內(nèi)存空間,n++修改的結(jié)果,對(duì)num無(wú)任何的影響,導(dǎo)致了在下一次調(diào)用時(shí),丟失了“輸出的是第幾個(gè)”的重要信息。

一個(gè)補(bǔ)救的方法,是由format將變化后的n值作為返回值,再傳回給main函數(shù),得到如下方案3的程序:

//方案3:利用了返回值使耦合度增大,但功能得以實(shí)現(xiàn)的方案

#include 《stdio.h》

int format(int);

int main()

{

int m, num=0;

for (m = 1; m * m 《= 200; m++)

{

printf(“%d”, m * m);

num = format(num);

}

return 0;

}

int format(int n)

{

n++;

if (n%5==0)

printf(“

”);

else

printf(“ ”);

return n;

}

維持原函數(shù)返回值為void,而將參數(shù)改為傳地址,得到下面的方案4。這個(gè)方案的耦合度更高一些,但功能還是能夠?qū)崿F(xiàn)的。

//方案4:傳地址實(shí)現(xiàn)功能的方案,耦合度更大

#include 《stdio.h》

void format(int*);

int main()

{

int m, num=0;

for (m = 1; m * m 《= 200; m++)

{

printf(“%d”, m * m);

format(&num);

}

return 0;

}

void format(int *p)

{

(*p)++;

if ((*p)%5==0)

printf(“

”);

else

printf(“ ”);

return;

}

一定有人想到了用全局變量的解決方案。這樣,可以將num定義為全局變量,num的生存周期不再依賴于函數(shù)調(diào)用,其值也能在函數(shù)的調(diào)用之間保持不變(只要其間沒(méi)有另外給它賦值),從而可以完成傳遞信息的任務(wù)。這時(shí),format因?yàn)闊o(wú)需參數(shù)傳遞,可以設(shè)計(jì)為無(wú)參函數(shù),得到如下方案5的程序:

//方案5:耦合度最高的全局變量方案

#include 《stdio.h》

void format();

int num=0;

int main()

{

int m ;

for (m = 1; m * m 《= 200; m++)

{

printf(“%d”, m * m);

format();

}

return 0;

}

void format()

{

num++;

if (num%5==0)

printf(“

”);

else

printf(“ ”);

return;

}

這是解決這個(gè)問(wèn)題的耦合程度最高的一個(gè)方案。將num定義為外部變量,意味著如果還有其他函數(shù),num是可以被任何函數(shù)修改的,當(dāng)發(fā) format 計(jì)數(shù)錯(cuò)誤時(shí),尋找錯(cuò)誤困難,而修改后又可能會(huì)帶來(lái)其他地方的錯(cuò)誤。在這么一個(gè)短小的程序中,這種方案可能尚可接受,當(dāng)程度的規(guī)模稍變大,可能帶來(lái)的問(wèn)題必須高度重視。因此,在實(shí)際應(yīng)用中,強(qiáng)調(diào)全局變量要慎用(不是不用)。

考慮到num是在format中應(yīng)用的私用數(shù)據(jù)——只有format才關(guān)心這到底是第幾個(gè)數(shù)據(jù),main本來(lái)都不用關(guān)心的。這樣,可以考慮將num定義為format中的局部靜態(tài)變量,得到方案6的程序:

//方案6:用靜態(tài)局部變量,耦合度偏高但封裝性最好的方案

#include 《stdio.h》

void format();

int main()

{

int m ;

for (m = 1; m * m 《= 200; m++)

{

printf(“%d”, m * m);

format();

}

return 0;

}

void format()

{

static int num=0;

num++;

if (num%5==0)

printf(“

”);

else

printf(“ ”);

return;

}

在這里,靜態(tài)局部變量num的作用域是局部的,定義在函數(shù)體里,封裝性在所有方案里是最好的,從而能保證信息的隱蔽性,避免其他函數(shù)無(wú)意的越權(quán)訪問(wèn);不過(guò),num的生存期是全局的,可以跨越函數(shù)的不同次調(diào)用,在兩次調(diào)用間傳遞信息,耦合程度(自己和自己的耦合)要高一些,但使main函數(shù)和format函數(shù)的耦合達(dá)到了最理想的程度,既保證了功能的正確,又保證了局部數(shù)據(jù)的安全性,表現(xiàn)出靜態(tài)局部變量的優(yōu)勢(shì)。綜上所述,在解決一個(gè)問(wèn)題時(shí),存在著諸多的方案。方案1可以接受,但希望提高內(nèi)聚性而做出改進(jìn);方案2用簡(jiǎn)單的參數(shù)傳值方式實(shí)現(xiàn)耦合程度低,但很可惜不能完成功能;在其他方案中,對(duì)于這個(gè)問(wèn)題,選擇的優(yōu)先順序是:

?

方案6、方案3 》 方案4 》 方案5

?建議讀者回顧前面的內(nèi)容,想一想這樣排序的理由。在上述探討各個(gè)方案的過(guò)程中,我們應(yīng)該體會(huì)到在程序設(shè)計(jì)能力提高的過(guò)程中,不斷地學(xué)習(xí)新的技術(shù),懂得新的評(píng)判標(biāo)準(zhǔn),這也就是一個(gè)不斷拓寬眼蜀的過(guò)程。在稍后的練習(xí)中,不妨多想一些方案,也能夠從專業(yè)的角度評(píng)判方案的優(yōu)劣,最終做到的,就是出手就是最佳方案的專業(yè)水平。

責(zé)任編輯:haq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • C語(yǔ)言
    +關(guān)注

    關(guān)注

    183

    文章

    7644

    瀏覽量

    145602
  • 編程
    +關(guān)注

    關(guān)注

    90

    文章

    3716

    瀏覽量

    97191

原文標(biāo)題:用C語(yǔ)言實(shí)例描述程序中的內(nèi)聚和耦合

文章出處:【微信號(hào):A1411464185,微信公眾號(hào):multisim】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    講解C語(yǔ)言代碼的實(shí)現(xiàn)過(guò)程

    重點(diǎn)講解C語(yǔ)言代碼的實(shí)現(xiàn)過(guò)程,算法的C語(yǔ)言實(shí)現(xiàn)過(guò)程具有一般性,通過(guò)PID算法的C
    發(fā)表于 01-21 07:58

    c語(yǔ)言中的代碼優(yōu)化

    放在寄存器中,但最終該變量可能由于條件不知足并未成為寄存器變量,而是被放在了存儲(chǔ)器中,但編譯器中并不報(bào)錯(cuò)(在C++語(yǔ)言中有另外一個(gè)\"建議\"型關(guān)鍵字:inline)。   下面
    發(fā)表于 01-12 09:45

    請(qǐng)問(wèn)C語(yǔ)言中整形溢出的解決方法有哪些?

    C語(yǔ)言中整形溢出的解決方法有哪些?
    發(fā)表于 12-29 07:33

    單片機(jī)c語(yǔ)言中定義字節(jié)怎么定義?

    定義位可以用***it但是定義字節(jié)呢?還有就是比如匯編中的DPTR的DPH和DPL在c語(yǔ)言中怎樣定義的呢?
    發(fā)表于 12-29 06:32

    請(qǐng)問(wèn)C語(yǔ)言中整形溢出會(huì)產(chǎn)生哪些異常行為?

    C語(yǔ)言中整形溢出會(huì)產(chǎn)生哪些異常行為?
    發(fā)表于 12-26 07:05

    請(qǐng)問(wèn)C語(yǔ)言中整形溢出對(duì)哪些應(yīng)用場(chǎng)景影響較大?

    C語(yǔ)言中整形溢出對(duì)哪些應(yīng)用場(chǎng)景影響較大
    發(fā)表于 12-24 08:24

    C語(yǔ)言C++的區(qū)別及聯(lián)系

    要的因素。 缺點(diǎn):沒(méi)有面向?qū)ο笠拙S護(hù)、易復(fù)用、易擴(kuò)展。 面向?qū)ο?b class='flag-5'>語(yǔ)言: 優(yōu)點(diǎn):易維護(hù)、易復(fù)用、易擴(kuò)展,由于面向?qū)ο笥蟹庋b、繼承、多態(tài)性的特性,可以設(shè)計(jì)出耦合的系統(tǒng),使系統(tǒng)更加靈活、更加易于維護(hù)
    發(fā)表于 12-24 07:23

    Typedef在C語(yǔ)言中的應(yīng)用

    Typedef 在C語(yǔ)言中頻繁用以聲明一個(gè)已經(jīng)存在的數(shù)據(jù)類型的同義字。也可以用預(yù)處理器做類似的事。例如,思考一下下面的例子: #define dPS struct s * typedef
    發(fā)表于 12-22 13:53

    C語(yǔ)言中可以嵌套匯編語(yǔ)言嗎?

    請(qǐng)問(wèn)C語(yǔ)言中可以嵌套匯編語(yǔ)言嗎?
    發(fā)表于 12-15 08:14

    看懂C語(yǔ)言程序中的內(nèi)耦合

    的設(shè)計(jì)提出了要求。 程序設(shè)計(jì)和軟件工程發(fā)展過(guò)程中產(chǎn)生的很多技術(shù)、設(shè)計(jì)原則,都可以從內(nèi)耦合的角度進(jìn)行解讀。作為C語(yǔ)言程序設(shè)計(jì)的初學(xué)者,結(jié)
    發(fā)表于 12-08 06:37

    C語(yǔ)言的分支結(jié)構(gòu)介紹

    1.簡(jiǎn)單if語(yǔ)句 C語(yǔ)言中的分支結(jié)構(gòu)語(yǔ)句中的if條件語(yǔ)句。 簡(jiǎn)單if語(yǔ)句的基本結(jié)構(gòu)如下: 代碼語(yǔ)言:javascript if(表達(dá)式) { 執(zhí)行代碼塊; } 其語(yǔ)義是:如果表達(dá)式的值為真,則執(zhí)行其后的語(yǔ)句,否則不執(zhí)
    發(fā)表于 11-25 07:48

    C語(yǔ)言的常量介紹

    、-13; 實(shí)型常量:13.33、-24.4; 字符常量:‘a(chǎn)’、‘M’ 字符串常量:”I love china!” 在C語(yǔ)言中,可以用一個(gè)標(biāo)識(shí)符來(lái)表示一個(gè)常量,稱之為符號(hào)常量。符號(hào)常量在使用之前必須先
    發(fā)表于 11-24 07:12

    高速密封輸入電流光耦合器 skyworksinc

    電子發(fā)燒友網(wǎng)為你提供()高速密封輸入電流光耦合器相關(guān)產(chǎn)品參數(shù)、數(shù)據(jù)手冊(cè),更有高速密封輸入電流光耦合器的引腳圖、接線圖、封裝手冊(cè)、中文資料、英文資料,高速密封
    發(fā)表于 07-08 18:31
    高速密封<b class='flag-5'>低</b>輸入電流光<b class='flag-5'>耦合</b>器 skyworksinc

    Windows環(huán)境下32位匯編語(yǔ)言中文資料

    電子發(fā)燒友網(wǎng)站提供《Windows環(huán)境下32位匯編語(yǔ)言中文資料.rar》資料免費(fèi)下載
    發(fā)表于 06-30 15:14 ?0次下載

    深入理解C語(yǔ)言C語(yǔ)言循環(huán)控制

    改變程序的執(zhí)行流程,使代碼更加靈活和可控。本文將詳細(xì)介紹這些語(yǔ)句的作用及其應(yīng)用場(chǎng)景,并通過(guò)示例代碼進(jìn)行說(shuō)明。Part.1break語(yǔ)句C語(yǔ)言中break語(yǔ)句有兩種
    的頭像 發(fā)表于 04-29 18:49 ?2047次閱讀
    深入理解<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>:<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>循環(huán)控制