編程學(xué)習(xí)過(guò)程中,我曾經(jīng)我犯過(guò)一個(gè)錯(cuò)誤(我想多數(shù)人也跟我一樣心急)?!皩W(xué)”完C語(yǔ)言后緊接著學(xué)C++,等稍微有基礎(chǔ)了之后開(kāi)始接觸C++ GUI Qt編程等。但學(xué)習(xí)Qt圖形化編程的過(guò)程中又發(fā)現(xiàn)寫(xiě)程序的關(guān)鍵點(diǎn)無(wú)外乎編寫(xiě)函數(shù)或方法。自己對(duì)函數(shù)編寫(xiě)的理解不太深入而導(dǎo)致不會(huì)寫(xiě)“自己”的C/C++函數(shù)(或方法)。雖然我們能夠?qū)⑿枨蠓纸獬啥鄠€(gè)模塊或函數(shù),但這并不意味著我們也能將需求編寫(xiě)成函數(shù)(或方法)來(lái)讓程序正常運(yùn)行。因此,本文簡(jiǎn)要總結(jié)一下有關(guān)函數(shù)的一些概念及個(gè)人學(xué)習(xí)體會(huì)。
圖1 人腦將概念分解成“屬性”和“函數(shù)”的過(guò)程
1、內(nèi)存中用一維空間來(lái)表示多維世界
也許我們活在一維世界里,只是我們認(rèn)為這個(gè)世界是三維(3D)或更多維的復(fù)雜多樣。 為什么這么說(shuō)呢?因?yàn)橛?jì)算機(jī)的世界就是由一維數(shù)組升級(jí)成多維數(shù)組形式給我們演變出了多維畫(huà)面和模型。比如,內(nèi)存存最小地址標(biāo)識(shí)單位為字節(jié)(byte),然后我們將一維內(nèi)存通過(guò)多維形式標(biāo)識(shí)并在此基礎(chǔ)上借助數(shù)組、指針、結(jié)構(gòu)體等數(shù)據(jù)結(jié)構(gòu)構(gòu)造了更復(fù)雜的模型和需求,最終能夠在內(nèi)存的一維空間中完成多維現(xiàn)實(shí)世界的“模擬”需求。如下圖2所示一維數(shù)組多種標(biāo)識(shí)方法就是其中的具體案例之一。計(jì)算機(jī)內(nèi)存其實(shí)一維的,一維空間來(lái)表示多維空間是我們不可否認(rèn)的事實(shí)。
圖2 內(nèi)存中一維數(shù)組來(lái)表示多維數(shù)組
2、為什編寫(xiě)函數(shù)是程序員的基本功
雖然我們通過(guò)一維內(nèi)存空間來(lái)表示出多維現(xiàn)實(shí)世界,但其現(xiàn)實(shí)過(guò)程需要較理解不同數(shù)據(jù)類(lèi)型及其在內(nèi)存中的布局、程序的運(yùn)行原理、內(nèi)存中多種數(shù)據(jù)結(jié)構(gòu)的融融應(yīng)用等基礎(chǔ)概念及原理。我之前對(duì)編程感到恐怖,因?yàn)榭吹侥切┐a時(shí)腦袋的形成的形象是空白的(或者說(shuō)一維的),腦海里沒(méi)有形成這些代碼在磁盤(pán)、內(nèi)存、寄存器、CPU等之間“流”進(jìn)“流”的過(guò)程和狀態(tài)。
在這樣的狀態(tài)下,我們無(wú)論學(xué)C或面向?qū)ο蟮腃++及Qt等其他編程語(yǔ)言及工具,最終編寫(xiě)函數(shù)或?qū)ο蟮姆椒ǖ倪^(guò)程中總會(huì)感覺(jué)到無(wú)從下手。我想這也是我們覺(jué)得C/C++難學(xué)原因之一,因此我回頭重學(xué)了C語(yǔ)言的基礎(chǔ)知識(shí)。在這個(gè)過(guò)程中,兩個(gè)內(nèi)容的回顧對(duì)我?guī)?lái)了意想不到的收獲。其中之一是C語(yǔ)言面向?qū)ο缶幊?,尤其是用C語(yǔ)言實(shí)現(xiàn)封裝和繼承特性。其二是用C語(yǔ)句描述算法的相關(guān)解釋說(shuō)明。
圖3 程序在內(nèi)存中布局
3、用C語(yǔ)句描述算法
程序就是對(duì)計(jì)算機(jī)要執(zhí)行的一組操作序列的描述。高級(jí)語(yǔ)言源程序的基本組成單位是語(yǔ)句。語(yǔ)句按功能可以分為兩類(lèi): 一類(lèi)用于描述計(jì)算機(jī)要執(zhí)行的操作運(yùn)算(如賦值語(yǔ)句),另一類(lèi)是控制上述操作運(yùn)算的執(zhí)行順序(如循環(huán)控制語(yǔ)句) 。前一類(lèi)稱(chēng)為操作運(yùn)算語(yǔ)句,后一類(lèi)稱(chēng)為流程控制語(yǔ)句。
C語(yǔ)言是一種表達(dá)式語(yǔ)言,所有的操作運(yùn)算都通過(guò)表達(dá)式來(lái)實(shí)現(xiàn)。由表達(dá)式組成的語(yǔ)句稱(chēng)為表達(dá)式語(yǔ)句,它由一個(gè)表達(dá)式后接一個(gè)分號(hào)組成(注意,沒(méi)有分號(hào)的不是語(yǔ)句)。表達(dá)式語(yǔ)句可以分為以下三種基本類(lèi)型:
(1) 賦值語(yǔ)句:由賦值表達(dá)式加一個(gè)分號(hào)組成。例如:i=1;
(2) 函數(shù)調(diào)用語(yǔ)句:
(3) 空語(yǔ)句
高級(jí)語(yǔ)言一般以?xún)煞N形式提供流程控制:
(1)形成流程控制結(jié)構(gòu)(如if、while、for語(yǔ)句)。
(2)簡(jiǎn)單的流程轉(zhuǎn)向。
控制結(jié)構(gòu)分為順序、選擇和循環(huán)等三種基本結(jié)構(gòu),大多數(shù)高級(jí)語(yǔ)言都提供這三種控制結(jié)構(gòu)。準(zhǔn)確地說(shuō),是后兩種。因?yàn)轫樞蛐褪亲匀恍纬傻?,無(wú)須在程序中加以專(zhuān)門(mén)的控制。
圖4 if或switch語(yǔ)句模擬多路選擇結(jié)構(gòu)的開(kāi)關(guān)電路
限定轉(zhuǎn)向語(yǔ)句(簡(jiǎn)單的流程轉(zhuǎn)向)不形成控制結(jié)構(gòu),只是簡(jiǎn)單地使流程從其所在處轉(zhuǎn)向另一處。但是它不允許用戶(hù)自己指定轉(zhuǎn)向,而是按系統(tǒng)事先規(guī)定的原則向某一點(diǎn)轉(zhuǎn)移,用戶(hù)不必指定轉(zhuǎn)向。C語(yǔ)言中屬于這類(lèi)的語(yǔ)句有三種:
(1) break 語(yǔ)句:它的功能是把流程從所在處轉(zhuǎn)向所在的循環(huán)結(jié)構(gòu)或多路選擇結(jié)構(gòu)之后,或者說(shuō)是中止執(zhí)行這些結(jié)構(gòu)(見(jiàn)圖5)。
(2) continue 語(yǔ)句:使本次循環(huán)體的執(zhí)行提前結(jié)束(不再執(zhí)行continue下面的語(yǔ)句),然后再根據(jù)循環(huán)條件是否滿(mǎn)足,決定是否進(jìn)入下次循環(huán)(見(jiàn)圖5)。
圖5 限定轉(zhuǎn)向語(yǔ)句(簡(jiǎn)單的流程轉(zhuǎn)向)
(3) 函數(shù)調(diào)用和返回: 函數(shù)調(diào)用的功能是使流程轉(zhuǎn)向所調(diào)用的函數(shù)體。return語(yǔ)句的功能是使流程從被調(diào)用函數(shù)返回主調(diào)函數(shù)。這兩種流程控制都可能伴隨有參數(shù)傳遞。
綜前所述,函數(shù)編寫(xiě)的關(guān)鍵在理解和善用操作語(yǔ)句(賦值語(yǔ)句)和控制語(yǔ)句,把C語(yǔ)言中的基本語(yǔ)句歸納如下:
圖6 C語(yǔ)言中的基本語(yǔ)句
4、理解指針(地址)和結(jié)構(gòu)體
指針是一個(gè)特殊的變量,它里面存儲(chǔ)的數(shù)值被解釋成為內(nèi)存里的一個(gè)地址。要搞清一個(gè)指針需要搞清指針的四方面的內(nèi)容:指針的類(lèi)型、指針?biāo)赶虻念?lèi)型、指針的值或者叫指針?biāo)赶虻膬?nèi)存區(qū)、指針本身所占據(jù)的內(nèi)存區(qū)。
結(jié)構(gòu)體是構(gòu)造復(fù)雜數(shù)據(jù)類(lèi)型的最有效的工具,對(duì)這個(gè)概念還不了解,基本上無(wú)法構(gòu)造數(shù)據(jù)模型,一般能日常使用的程序中沒(méi)有一個(gè)業(yè)務(wù)體是完全使用原生數(shù)據(jù)類(lèi)型來(lái)完成的,如下圖6所示。設(shè)計(jì)數(shù)據(jù)模型的時(shí)候,一般先把頭文件中的結(jié)構(gòu)體數(shù)據(jù)整理出來(lái)。然后設(shè)計(jì)好功能函數(shù)的參數(shù),以及名字,然后才真正開(kāi)始寫(xiě)C源碼。
圖7 用C語(yǔ)言來(lái)封裝屬性和函數(shù)
其實(shí)C語(yǔ)言也能編寫(xiě)面向?qū)ο缶幊田L(fēng)格的程序,如附件1所示的封裝特性演示代碼就采用封裝特性 ,還有繼承特性的實(shí)現(xiàn),篇幅原因不再贅述。當(dāng)看懂了這段代碼后,我突然明白了函數(shù)指針、結(jié)構(gòu)體、面向?qū)ο缶幊讨械膖his(或self),及構(gòu)造函數(shù)等等的來(lái)歷。也領(lǐng)悟到了將函數(shù)封裝到的類(lèi)(class或?qū)ο螅├镏螅ㄟ^(guò)點(diǎn)或指針訪(fǎng)問(wèn)函數(shù)(方法)來(lái)實(shí)現(xiàn)對(duì)結(jié)構(gòu)體成員訪(fǎng)問(wèn)和修改在內(nèi)存中的實(shí)現(xiàn)過(guò)程。
5、結(jié)束語(yǔ)
函數(shù)、指針、結(jié)構(gòu)體這三大塊硬骨頭是學(xué)習(xí)C語(yǔ)言(或編程)的絆腳石 ,下功夫拿掉基本上C語(yǔ)言的大動(dòng)脈就打通了,如果想開(kāi)發(fā)實(shí)際能用到的程序,那么也需要了解 文件和數(shù)據(jù)庫(kù)的讀寫(xiě)等第四塊常被我們忽視的骨頭。 尤其是每當(dāng)通過(guò)new來(lái)創(chuàng)建對(duì)象或定義結(jié)構(gòu)體變量來(lái)創(chuàng)建數(shù)據(jù)模型時(shí),我們會(huì)感覺(jué)到計(jì)算機(jī)世界里創(chuàng)建資源和使用資源時(shí)多么“簡(jiǎn)單”和“直接”的。如果在現(xiàn)實(shí)世界,從0到1的價(jià)值創(chuàng)造(不是資源調(diào)配)是多么難的事情,需要我們要么用金錢(qián)換來(lái)或其他方式對(duì)等交換。因此,雖然編程很“難”(與其說(shuō)難、不如說(shuō)我們貪),但相對(duì)于現(xiàn)實(shí)世界的種種困難,讓我們?cè)谟?jì)算機(jī)虛擬世界里擁有無(wú)限的資源和可能性。
圖8 電腦對(duì)“程序”的理解和處理過(guò)程
參考資料:
C語(yǔ)言程序設(shè)計(jì)教程 譚浩強(qiáng) 張基溫 唐永炎 高等教育出版社
C語(yǔ)言游戲?qū)崙?zhàn)教程
附件1:C語(yǔ)言封裝特性演示代碼
#define _CRT_SECURE_NO_WARNINGS
#include
#include
#include
struct student {
void (*setStudentId)(struct student* s, int year, int classNum, int serialNum);
const char* (*getGender)(struct student* s);
void (*setGender)(struct student* s, const char* strGender);
int id; // 學(xué)號(hào)
char name[20]; // 姓名
int gender; // 性別
int mark; // 分?jǐn)?shù)
};
void setStudentId(struct student* s, int year, int classNum, int serialNum)
{
char buffer[20];
sprintf(buffer, "%d%d%d", year, classNum, serialNum);
int id = atoi(buffer);
s->id = id;
}
const char* getGender(struct student* s)
{
if (s->gender == 0)
{
return "女";
}
else if (s->gender == 1)
{
return "男";
}
return "未知";
}
void setGender(struct student* s, const char* strGender)
{
int numGender;
if (strcmp("男", strGender) == 0)
{
numGender = 1;
}
else if (strcmp("女", strGender) == 0)
{
numGender = 0;
}
else
{
numGender = -1;
}
s->gender = numGender;
}
void initStudent(struct student* s)
{
s->setStudentId = setStudentId;
s->getGender = getGender;
s->setGender = setGender;
}
int main()
{
struct student stu;
// 初始化student
initStudent(&stu);
// 學(xué)號(hào):202212326
// 姓名:小明
// 性別: 男
// 分?jǐn)?shù):98
stu.setStudentId(&stu, 2022, 123, 26);
strcpy(stu.name, "小明");
stu.setGender(&stu, "男");
stu.mark = 98;
// 打印這些數(shù)值
printf("學(xué)號(hào):%d\\n", stu.id);
printf("姓名:%s\\n", stu.name);
const char* gender = stu.getGender(&stu);
printf("性別:%s\\n", gender);
printf("分?jǐn)?shù):%d\\n", stu.mark);
return 0;
}
收錄于合集 **#**軟件工程
9個(gè)
上一篇編程思想-模塊化程序設(shè)計(jì)案例DCIM(3)下一篇編程思想-軟件產(chǎn)品的定義及定價(jià)
-
計(jì)算機(jī)
+關(guān)注
關(guān)注
19文章
7532瀏覽量
88435 -
內(nèi)存
+關(guān)注
關(guān)注
8文章
3048瀏覽量
74209 -
模型
+關(guān)注
關(guān)注
1文章
3296瀏覽量
49042
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論