1. 程式人生 > >c語言中get()函式的原理及返回值

c語言中get()函式的原理及返回值

首先要記住的一句話就是Never use gets().
這是因為gets()函式不檢查目標陣列是否能夠容納輸入,而若想把一個字串讀到程式中,最先要做的事情就是預留儲存字串的空間。所以這很容易導致分配的空間不夠大而陣列越界,然而gets()函式並不檢查這個方面,所以導致的結果就是程式很容易出現漏洞,著名的“蠕蟲”病毒的原理就是用很長的資料覆蓋原有資料導致崩潰。所以對於重要的程式設計,永遠不要使用gets()!

1, gets()的引數是一個地址,因為要把從鍵盤輸入的值確定的放到某一塊記憶體中,所以需要指定它的地址,而通常使用gets(陣列名)這種方式來把輸入的字串傳入給定的陣列中。注意:這個陣列的大小一定要事先定義好!若不將陣列的大小定義好,就有可能在輸入的時候不知道把字串輸入到哪塊記憶體中去了,就有可能會導致對該記憶體中原始碼的覆蓋。

2,    gets()的第一個用法:直接用gets(array's name);這種方式時,由於不知道什麼時候才會到字串結尾,所以每當鍵入'\n', gets()函式都會自動讀取換行符前面的所有內容並且在末尾加上'\0',並且直接把這個字串返回給呼叫它的程式,然後gets()再進行讀取並且會把讀取到的'\n'丟棄,這樣下一次讀取就會在新的一行開始。
例1:    
        #include <stdio.h>
        #include <stdlib.h>
        #define MAX 81
        int main(void)
        {
            char name[MAX];
            printf("Hi, what's your name?\n");
            gets(name);
            printf("Nice name, %s\n", name);
            return 0;
        }

        /*
            Hi, what's your name?
            Herry potter
            Nice name, Herry potter

        */



3,  gets()的原型: char * gets(char * s)
                               {
                                          ...
                                       return s;
                               }
    所以由此可以看出gets()返回的是一個指向char型別資料的指標,而且這個指標與傳遞給他的是同一個指標。因此有如下程式:
例2:
        #include <stdio.h>
        #include <stdlib.h>
        #define MAX 81
        int main(void)
        {
            char name[MAX];
            char * ptr;

            printf("Hi,what's your name?\n");
            ptr = gets(name); // 這裡的ptr是char型別的指標,此時ptr指向的是name的首地址。
            printf("%s? Ah? %s!", name, ptr); // 此時輸出name和ptr指向的值,可以發現,他們的輸出結果都一樣。
            return 0;
        }

        /*
            Hi,what's your name?
            Herry
            Herry? Ah? Herry!
        */




4, 實際上gets()有兩種可能的返回值型別:

                                      1)當程式正常輸入字串時:返回讀入字串的地址,也就是字串存放的陣列的首地址;

                                      2)當程式出現錯誤或者遇到檔案結尾時:返回空指標NULL,注意不要弄混空指標和空字元('\0');
                                          所以可以很方便的用如下形式檢測錯誤:
                                                                          while(gets(name) != NULL)

注意:現在基本上不使用gets(),可以說它是一個已經被廢棄的函式,現在可以用scanf(), getchar(), fgets()來代替它。

相關推薦

c言中get()函式原理返回

首先要記住的一句話就是Never use gets(). 這是因為gets()函式不檢查目標陣列是否能夠容納輸入,而若想把一個字串讀到程式中,最先要做的事情就是預留儲存字串的空間。所以這很容易導致分配的空間不夠大而陣列越界,然而gets()函式並不檢查這個方面,所以導致的結

C言中static的作用C語言中使用靜態函式有何好處

在C語言中,static的字面意思很容易把我們匯入歧途,其實它的作用有三條,分別是: 一是隱藏功能,對於static修飾的函式和全域性變數而言 二是保持永續性功能,對於static修飾的區域性變數而言。 三是因為存放在靜態區,全域性和區域性的static修飾的變數,都預設初始化為0 下面我逐一給

語法:C言中printf函式返回

平時真不怎麼關注printf的返回值,一般是直接呼叫printf格式化輸出,今天做騰訊的筆試題發現了一個知識漏洞,特此記錄。 首先,題目是這樣的: int f(int a, int b, int c) { return 0; } int main()

c言中 gotoxy() 函式的使用

轉自  https://blog.csdn.net/radjedef/article/details/79028329 #include <stdio.h> #include <windows.h> void gotoxy(int x, int y) {

關於C言中printf函式“輸出歧視”的問題

目錄 關於C語言中printf函式“輸出歧視”的問題 問題描述 探索問題原因 另一種研究方法 問題結論 關於C語言中printf函式“輸出歧視”的問題 問題描述 昨天晚上被問到一個問題,為什麼在同一個printf函式中兩次輸出一個double型變

c言中rand()函式的用法筆記

最近在學著用c寫一些小程式,過程中遇到很多問題,在網上查很多大神的講解,以及查閱vs2010的幫助文件。在此做個筆記,以便今後查閱,備忘。感謝無私奉獻講解的大神們! 一、rand() rand()函式用來產生隨機數,但是,rand()的內部實現是用線性同餘法實現的,是偽隨機數,由於週期較長,

(C言中printf函式讀取的具體分析)

(C語言中printf函式讀取的具體分析) 不多說,直接上。printf函式將傳入的資料傳送到記憶體堆區(緩衝區),然後再根據前面的(格式說明符一個個讀取,這樣會造成錯誤) #include<stdio.h> #include<limits.h> #incl

C言中main函式引數使用

在C99標準中定義main函式兩種正確的寫法 int main(void); int main(int argc, char* argv[]); 常見的不標準寫法 void main() main()

C言中strlen函式的模擬實現n種方法

strlen 函式採用遞迴形式編寫 # include <stdio.h> # include <stdlib.h> int strlen (char* num) // 形參接受實參傳遞的陣列 {

C言中scanf函式輸入回車符的問題

 在用c語言編寫輸入語句的時候常用到scanf函式,初學者在剛用scanf函式輸入時,經常會遇到各種各樣的輸入錯誤,最重要的是一定要記住scanf函式的輸入格式,scanf函式裡包含了哪些東西,輸入的時候就必須有哪些東西,比如:scanf("%c%c%c"),那麼輸入的時

c言中printf函式列印char型變數0xFF,輸出為0xFFFFFFFF的解決方法

問題描述:在編寫udp程式解析資料的時候,發現接收的char型變數為0xFF,但是使用printf後列印為0xFFFFFFFF,除錯程式查詢原因後發現使用char時,對於printf來說其值為0xFFFFFFFF。原因是%x要求的是無符號整形變數,你傳入的是char型,這裡有

C言中函式指標陣列的用途:轉移表

計算器的實現: 方法一: #include <stdio.h> int add(int a, int b) { return a + b; } int sub(int a, int b)

C言中 main 函式的引數 argc&argv

argc、argv用命令列編譯時有用 argc:整數 srgv:二維陣列、指標的指標、指標陣列 例子 #include<stdio.h> #include<stdlib.h> int main(int argc, char** argv) { i

C言中assert函式的用法

  2)每個assert只檢驗一個條件,因為同時檢驗多個條件時,如果斷言失敗,無法直觀的判斷是哪個條件失敗   不好: assert(nOffset>=0 && nOffset+nSize<=m_nInfomationSize);   好:  assert(nOffset >

c言中getchar()函式一個常見Error

getchar() 從這個名字來看應該是 得到一個字元。 正好C語言裡面有一個char型別, 很容易就出現瞭如下程式: # include <stdio.h> int main (voi

c言中signal函式詳細解釋說明

對於 訊號處理函式 位於 <signal.h> 中.void ( *signal( int sig, void (* handler)( int ))) ( int );這個函式的宣告很是嚇人, 一看就難弄懂. 下面是解釋用法.一步一步解釋:int (*p)();這是一個函式指標, p所指向的函

C言中fopen函式用法詳解

fopen函式用來開啟一個檔案,其呼叫的一般形式為:檔案指標名=fopen(檔名,使用檔案方式); 其中,“檔案指標名”必須是被說明為FILE 型別的指標變數;“檔名”被開啟檔案的檔名,是字串常量或字串陣列,要求是全路徑;“使用檔案方式”是指檔案的型別和操作要求。 檔案使用方

C言中函式不宣告也能使用,但會出現warning: implicit declaration of function

偶然發現有很多自定義函式未經宣告卻能在主程式中被呼叫,這就奇怪了,連使用標準庫函式printf()都要包括標準輸入輸出標頭檔案<stdio.h>,何況是自定義函式?這個問題困擾了我很久。 今天通過實驗,基本明白了箇中原因。一、在VC6中, 1、檔案test1.

14.C言中time函式和localtime獲取系統時間和日期

C語言中time函式和localtime獲取系統時間和日期可以通過time()函式來獲得計算機系統當前的日曆時間(Calendar Time),處理日期時間的函式都是以本函式的返回值為基礎進行運算。1. time 函式 返回1970-1-1, 00:00:00以來經過的秒數 

關於C言中printf函式的引數執行順序

RT...看到有類似的題目,說是從右到左,, 藍後在VS2012中寫了如下的程式碼...但是執行出來的結果如下,, 這是為啥...  留一疑惑在這先. ok,這裡有了一個解釋: i++會建立一個拷貝 %d取值輸出,引數的傳遞是從右到左壓入棧,所以從右邊開始,i++,此時拷貝