C语言难点分析整理
C語言難點分析整理(z) 2007-10-25 14:50
###
C語言面試題(zt) 2007-10-07 16:17
| 這篇文章主要是介紹一些在復習C語言的過程中筆者個人認為比較重點的地方,較好的掌握這些重點會使對C的運用更加得心應手。此外會包括一些細節、易錯的地方。涉及的主要內容包括:變量的作用域和存儲類別、函數、數組、字符串、指針、文件、鏈表等。一些最基本的概念在此就不多作解釋了,僅希望能有只言片語給同是C語言初學者的學習和上機過程提供一點點的幫助。 變量作用域和存儲類別: 了解了基本的變量類型后,我們要進一步了解它的存儲類別和變量作用域問題。
換一個角度
extern型的存儲變量在處理多文件問題時常能用到,在一個文件中定義extern型的變量即說明這個變量用的是其他文件的。順便說一下,筆者在做課設時遇到out of memory的錯誤,于是改成做多文件,再把它include進來(注意自己寫的*.h要用“”不用<>),能起到一定的效用。static型的在讀程序寫結果的試題中是個考點。多數時候整個程序會出現多個定義的變量在不同的函數中,考查在不同位置同一變量的值是多少。主要是遵循一個原則,只要本函數內沒有定義的變量就用全局變量(而不是main里的),全局變量和局部變量重名時局部變量起作用,當然還要注意靜態與自動變量的區別。 函數: 對于函數最基本的理解是從那個叫main的單詞開始的,一開始總會覺得把語句一并寫在main里不是挺好的么,為什么偏擇出去。其實這是因為對函數還不夠熟練,否則函數的運用會給我們編程帶來極大的便利。我們要知道函數的返回值類型,參數的類型,以及調用函數時的形式。事先的函數說明也能起到一個提醒的好作用。所謂形參和實參,即在調用函數時寫在括號里的就是實參,函數本身用的就是形參,在畫流程圖時用平行四邊形表示傳參。 函數的另一個應用例子就是遞歸了,筆者開始比較頭疼的問題,反應總是比較遲鈍,按照老師的方法,把遞歸的過程耐心準確的逐級畫出來,學習的效果還是比較好的,會覺得這種遞歸的運用是挺巧的,事實上,著名的八皇后、漢諾塔等問題都用到了遞歸。
數組: 分為一維數組和多維數組,其存儲方式畫為表格的話就會一目了然,其實就是把相同類型的變量有序的放在一起。因此,在處理比較多的數據時(這也是大多數的情況)數組的應用范圍是非常廣的。 具體的實際應用不便舉例,而且絕大多數是與指針相結合的,筆者個人認為學習數組在更大程度上是為學習指針做一個鋪墊。作為基礎的基礎要明白幾種基本操作:即數組賦值、打印、排序(冒泡排序法和選擇排序法)、查找。這些都不可避免的用到循環,如果覺得反應不過來,可以先一點點的把循環展開,就會越來越熟悉,以后自己編寫一個功能的時候就會先找出內在規律,較好的運用了。另外數組做參數時,一維的[]里可以是空的,二維的第一個[]里可以是空的但是第二個[]中必須規定大小。
相關常用的算法還有判斷回文,求階乘,Fibanacci數列,任意進制轉換,楊輝三角形計算等等。 字符串: 字符串其實就是一個數組(指針),在scanf的輸入列中是不需要在前面加“&”符號的,因為字符數組名本身即代表地址。值得注意的是字符串末尾的‘’,如果沒有的話,字符串很有可能會不正常的打印。另外就是字符串的定義和賦值問題了,筆者有一次的比較綜合的上機作業就是字符串打印老是亂碼,上上下下找了一圈問題,最后發現是因為
而不是
前者沒有說明指向哪兒,更沒有確定大小,導致了亂碼的錯誤,印象挺深刻的。 另外,字符串的賦值也是需要注意的,如果是用字符指針的話,既可以定義的時候賦初值,即
也可以在賦值語句中賦值,即
但如果是用字符數組的話,就只能在定義時整體賦初值,即char a[5]={"abcd"};而不能在賦值語句中整體賦值。 常用字符串函數列表如下,要會自己實現:
注:對字符串是不允許做==或!=的運算的,只能用字符串比較函數 指針: 指針可以說是C語言中最關鍵的地方了,其實這個“指針”的名字對于這個概念的理解是十分形象的。首先要知道,指針變量的值(即指針變量中存放的值)是指針(即地址)。指針變量定義形式中:基本類型 *指針變量名 中的“*”代表的是這是一個指向該基本類型的指針變量,而不是內容的意思。在以后使用的時候,如*ptr=a時,“*”才表示ptr所指向的地址里放的內容是a。 指針比較典型又簡單的一應用例子是兩數互換,看下面的程序,
這是不能實現a和b的數值互換的,實際上只是形參在這個函數中換來換去,對實參沒什么影響。現在,用指針類型的數據做為參數的話,更改如下:
這樣在swap中就把p1,p2 的內容給換了,即把a,b的值互換了。 指針可以執行增、減運算,結合++運算符的法則,我們可以看到:
指針和數組實際上幾乎是一樣的,數組名可以看成是一個常量指針,一維數組中ptr=&b[0]則下面的表示法是等價的: a[3]等價于*(a+3) ptr[3]等價于*(ptr+3) 下面看一個用指針來自己實現atoi(字符串型->整型)函數:
指向多維數組的指針變量也是一個比較廣泛的運用。例如數組a[3][4],a代表的實際是整個二維數組的首地址,即第0行的首地址,也就是一個指針變量。而a+1就不是簡單的在數值上加上1了,它代表的不是a[0][1],而是第1行的首地址,&a[1][0]。 指針變量常用的用途還有把指針作為參數傳遞給其他函數,即指向函數的指針。 看下面的幾行代碼:
總之,指針的應用是非常靈活和廣泛的,不是三言兩語能說完的,上面幾個小例子只是個引子,實際編程中,會逐漸發現運用指針所能帶來的便利和高效率。 文件:
注:以上用于文本文件的操作,如果是二進制文件就在上述字母后加“b”。 我們用文件最大的目的就是能讓數據保存下來。因此在要用文件中數據的時候,就是要把數據讀到一個結構(一般保存數據多用結構,便于管理)中去,再對結構進行操作即可。例如,文件aa.data中存儲的是30個學生的成績等信息,要遍歷這些信息,對其進行成績輸出、排序、查找等工作時,我們就把這些信息先讀入到一個結構數組中,再對這個數組進行操作。如下例:
鏈表: 鏈表是C語言中另外一個難點。牽扯到結點、動態分配空間等等。用結構作為鏈表的結點是非常適合的,例如:
其中next是指向自身所在結構類型的指針,這樣就可以把一個個結點相連,構成鏈表。 鏈表結構的一大優勢就是動態分配存儲,不會像數組一樣必須在定義時確定大小,造成不必要的浪費。用malloc和free函數即可實現開辟和釋放存儲單元。其中,malloc的參數多用sizeof運算符計算得到。 鏈表的基本操作有:正、反向建立鏈表;輸出鏈表;刪除鏈表中結點;在鏈表中插入結點等等,都是要熟練掌握的,初學者通過畫圖的方式能比較形象地理解建立、插入等實現的過程。
逆向建立:
用遞歸實現鏈表逆序輸出:
插入結點(已有升序的鏈表):
刪除結點:
? http://hi.baidu.com/zkheartboy/blog/item/9ee7d05c6d307e41fbf2c0bc.html | |||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
###
C語言面試題(zt) 2007-10-07 16:17
| ???????????C語言面試題 1、找錯 void test1() { ????char string[10]; ????char* str1="0123456789"; ???? strcpy(string, str1); } ???????這里string數組越界,因為字符串長度為10,還有一個結束符’\0’。所以總共有11個字符長度。string數組大小為10,這里越界了。 PS:使用strcpy函數的時候一定要注意前面目的數組的大小一定要大于后面字符串的大小,否則便是訪問越界。 void test2() { ????char string[10], str1[10]; ????for(i=0; i<10;i++) ???? { ??????? str1[i] ='a'; ???? } ???? strcpy(string, str1); } ???? 這里有一個一眼就能看出的問題,那就是變量i沒有定義,這在代碼編譯階段編譯器可以幫你發現,很容易搞定。然而很多問題是自己造成的漏洞,編譯器是幫不上什么忙的。這里最大的問題還是str1沒有結束符,因為strcpy的第二個參數應該是一個字符串常量。該函數就是利用判斷第二個參數的結束符來得到是否拷貝完畢。所以在for循環后面應加上str1p[9] = ‘\0’; PS:字符數組和字符串的最明顯的區別就是字符串會被默認的加上結束符’\0’。 void test3(char* str1) { ????char string[10]; ????if(strlen(str1)<=10) ???? { ??????? strcpy(string, str1); ???? } } ???????這里的問題仍是越界問題。strlen函數得到字符串除結束符外的長度。如果這里是<=10話,就很明顯越界了。 小結:上面的三個找錯的函數,主要是考查對字符串和字符數組的概念的掌握以及對strcpy函數和strlen函數的理解。 2、找錯 DSN get_SRM_no() { static int SRM_no; int I; for(I=0;I<MAX_SRM;I++) ???? { ??????? SRM_no %= MAX_SRM; ???????if(MY_SRM.state==IDLE) ??????? { ???????break; ??????? } ???? } ????if(I>=MAX_SRM) ???????return (NULL_SRM); ????else ???????return SRM_no; } 這里for循環的判斷語句是后來我加上的,估計在網上流傳的時候被人給弄丟了,根據對程序的分析,給補上了。估計錯誤應該不是這兒。 簡單的閱讀一下這個函數,可以大概的可以猜測出這個函數的功能是分配一個空閑的SRAM塊。方法:從上次分配的RAM塊后的RAM塊開始檢測SRAM每個RAM塊,看是否是IDLE狀態,如果是IDLE則返回當前的RAM塊的號SRM_no。如果所有的RAM塊都不是IDLE狀態,則意味著無法分配一個RAM給函數調用者,返回一個表示沒有RAM可分配的標志(NULL_SRM)。 經過上面的分析,則這里可以知道,這個函數的錯誤是for循環里面沒有給SRM_no這個變量累加1。 3、寫出程序運行結果 int sum(int a) { ????auto int c=0; ????static int b=3; ???? c+=1; ???? b+=2; ????return(a+b+c); } void main() { ????int I; ????int a=2; ????for(I=0;I<5;I++) ???? { ??????? printf("%d,", sum(a)); ???? } } 運行結果是:8,10,12,14,16, 在求和函數sum里面c是auto變量,根據auto變量特性知每次調用sum函數時變量c都會自動賦值為0。b是static變量,根據static變量特性知每次調用sum函數時變量b都會使用上次調用sum函數時b保存的值。 簡單的分析一下函數,可以知道,若傳入的參數不變,則每次調用sum函數返回的結果,都比上次多2。所以答案是:8,10,12,14,16, 4、func(1) = ? int func(int a) { ????int b; ????switch(a) ???? { ????????case 1: 30; ????????case 2: 20; ????????case 3: 16; ????????default: 0; ???? } ????return b; } 在 case 語句中可能忘記了對變量b賦值。如果改為下面的代碼: int func(int a) { ????int b; ????switch(a) ???? { ????????case 1:?????? b = 30; ????????case 2:?????? b = 20; ????????case 3:?????? b = 16; ????????default:????? b = 0; ???? } ????return b; } 因為case語句中漏掉了break語句,所以無論傳給函數的參數是多少,運行結果均為0。 5、a[q - p] = ? int a[3]; ???? a[0]=0; a[1]=1; a[2]=2; ????int *p, *q; ???? p=a; ???? q=&a[2]; ???? 很明顯:a[q - p] = a[2] = 2; 6、內存空間占用問題 定義 int **a[3][4], 則變量占有的內存空間為:16位系統24,32位編譯系統中是48。 PS:公式:3 * 4 * sizeof(int **) 。 7、程序編寫 ???? 編寫一個函數,要求輸入年月日時分秒,輸出該年月日時分秒的下一秒。如輸入2004年12月31日23時59分59秒,則輸出2005年1月1日0時0分0秒。 void ResetTheTime(int *year,int *month,int *date,int *hour,int *minute,int*second) { ????int dayOfMonth[12]={31,28,31,30,31,30,31,31,30,31,30,31}; ???? ????if( *year < 0??? || *month < 1 || *month > 12 || ??????? *date < 1??? || *date > 31 || *hour < 0??? || *hour > 23 || ??????? *minute < 0 ||*minute > 59|| *second <0 || *second >60 ) ???????return; ????if( *year%400 == 0 || *year%100 != 0 && *year%4 == 0 ) ??????? dayOfMonth[1] = 29; ????if(*second >= 60) ???? { ??????? *second = 0; ??????? *minute += 1; ???????if(*minute >= 60) ??????? { ??????????? *minute = 0; ??????????? *hour += 1; ???????????if(*hour >= 24) ??????????? { ?????????????? *hour = 0; ?????????????? *date += 1; ??????????????if(*date > dayOfMonth[*month-1]) ?????????????? { ?????????????????? *date = 1; ?????????????????? *month += 1; ??????????????????if(*month > 12) ?????????????????? { ????????????????????? *month=1; ?????????????????????? *year += 1; ?????????????????? } ?????????????? } ??????????? } ??????? } ???? } ????return; } |
轉載于:https://www.cnblogs.com/borcala/archive/2008/01/17/1042853.html
與50位技術專家面對面20年技術見證,附贈技術全景圖總結
- 上一篇: 3 CSS 高级语法
- 下一篇: 初学 Delphi 嵌入汇编[26] -