欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語言 用指針作為函數返回值詳解

 更新時間:2016年08月23日 15:17:51   投稿:lqh  
本文主要介紹C語言 用指針作為函數返回值,這里整理了相關資料及示例代碼,幫助大家學習理解此部分知識,有需要的同學可以參考下

C語言允許函數的返回值是一個指針(地址),我們將這樣的函數稱為指針函數。下面的例子定義了一個函數 strlong(),用來返回兩個字符串中較長的一個:

#include <stdio.h>
#include <string.h>
char *strlong(char *str1, char *str2){
  if(strlen(str1) >= strlen(str2)){
    return str1;
  }else{
    return str2;
  }
}
int main(){
  char str1[30], str2[30], *str;
  gets(str1);
  gets(str2);
  str = strlong(str1, str2);
  printf("Longer string: %s\n", str);
  return 0;
}

運行結果:

C Language↙
c.biancheng.net↙
Longer string: c.biancheng.net

用指針作為函數返回值時需要注意的一點是,函數運行結束后會銷毀在它內部定義的所有局部數據,包括局部變量、局部數組和形式參數,函數返回的指針請盡量不要指向這些數據,C語言沒有任何機制來保證這些數據會一直有效,它們在后續(xù)使用過程中可能會引發(fā)運行時錯誤。請看下面的例子:

#include <stdio.h>
int *func(){
  int n = 100;
  return &n;
}
int main(){
  int *p = func(), n;
  n = *p;
  printf("value = %d\n", n);
  return 0;
}

運行結果:

value = 100

n 是 func() 內部的局部變量,func() 返回了指向 n 的指針,根據上面的觀點,func() 運行結束后 n 將被銷毀,使用 *p 應該獲取不到 n 的值。但是從運行結果來看,我們的推理好像是錯誤的,func() 運行結束后 *p 依然可以獲取局部變量 n 的值,這個上面的觀點不是相悖嗎?

為了進一步看清問題的本質,不妨將上面的代碼稍作修改,在第9~10行之間增加一個函數調用,看看會有什么效果:

#include <stdio.h>
int *func(){
  int n = 100;
  return &n;
}
int main(){
  int *p = func(), n;
  printf("c.biancheng.net\n");
  n = *p;
  printf("value = %d\n", n);
  return 0;
}

運行結果:

c.biancheng.net
value = -2

可以看到,現在 p 指向的數據已經不是原來 n 的值了,它變成了一個毫無意義的甚至有些怪異的值。與前面的代碼相比,該段代碼僅僅是在 *p 之前增加了一個函數調用,這一細節(jié)的不同卻導致運行結果有天壤之別,究竟是為什么呢?

前面我們說函數運行結束后會銷毀所有的局部數據,這個觀點并沒錯,大部分C語言教材也都強調了這一點。但是,這里所謂的銷毀并不是將局部數據所占用的內存全部抹掉,而是程序放棄對它的使用權限,棄之不理,后面的代碼可以隨意使用這塊內存。對于上面的兩個例子,func() 運行結束后 n 的內存依然保持原樣,值還是 100,如果使用及時也能夠得到正確的數據,如果有其它函數被調用就會覆蓋這塊內存,得到的數據就失去了意義。

關于函數調用的原理以及函數如何占用內存的更多細節(jié),我們將在《C語言和內存》專題中深入探討,相信你必將有所頓悟,解開心中的謎團。

第一個例子在調用其他函數之前使用 *p 搶先獲得了 n 的值并將它保存起來,第二個例子顯然沒有抓住機會,有其他函數被調用后才使用 *p 獲取數據,這個時候已經晚了,內存已經被后來的函數覆蓋了,而覆蓋它的究竟是一份什么樣的數據我們無從推斷(一般是一個沒有意義甚至有些怪異的值)。

以上就是對 C語言指針作為函數返回值的資料整理,后續(xù)繼續(xù)補充相關資料,謝謝大家對本站的支持!

相關文章

  • 矩陣的行主序與列主序的分析

    矩陣的行主序與列主序的分析

    這篇文章主要介紹了矩陣的行主序與列主序的分析的相關資料,需要的朋友可以參考下
    2017-07-07
  • C++實現簡易反彈小球游戲的示例代碼

    C++實現簡易反彈小球游戲的示例代碼

    我們利用printf 函數實現一個在屏幕上彈跳的小球。彈跳的小球游戲比較簡單、容易入門,也是反彈球消磚塊、接金幣、臺球等很多游戲的基礎,感興趣的可以了解一下
    2022-10-10
  • c++自定義sort()函數的排序方法介紹

    c++自定義sort()函數的排序方法介紹

    這篇文章主要介紹了c++自定義sort()函數的排序方法介紹,文章通過圍繞主題展開詳細的內容戒殺,具有一定的參考價值,需要的小伙伴可以參考一下
    2022-09-09
  • C語言全面細致精講關鍵字的使用

    C語言全面細致精講關鍵字的使用

    關鍵字是C語言非常重要的一部分,熟練的掌握和使用關鍵字有助于我們更加熟悉了解C語言,同時C語言的關鍵字也是面試筆試中??嫉膬热?。C語言的關鍵字共有32個,但并不是每個關鍵字都有坑,本篇文章將通過理論聯系實際的方式為大家講解C語言中易混易錯以及常考的一些關鍵字
    2022-05-05
  • C語言超詳細講解猜數字游戲的實現

    C語言超詳細講解猜數字游戲的實現

    現在很多游戲都有抽獎抽卡的功能,其實這個就類似于猜數字,生成一個隨機數,然后你去猜,猜對了就得獎。猜到一定次數就會保底。要實現猜數字的小游戲,首先是要讓程序生成隨機數,這就要用到rand、srand和time這三個函數,其次要了解時間戳
    2022-07-07
  • 清除3389遠程登錄日志

    清除3389遠程登錄日志

    這篇文章主要介紹了清除3389遠程登錄日志示例,需要的朋友可以參考下
    2014-01-01
  • C++二叉樹的前序中序后序非遞歸實現方法詳細講解

    C++二叉樹的前序中序后序非遞歸實現方法詳細講解

    前序遍歷的順序是根、左、右。任何一顆樹都可以認為分為左路節(jié)點,左路節(jié)點的右子樹。先訪問左路節(jié)點,再來訪問左路節(jié)點的右子樹。把訪問左路節(jié)點的右子樹看成一個子問題,就可以完整遞歸訪問了
    2023-03-03
  • C++實現拼圖游戲代碼(graphics圖形庫)

    C++實現拼圖游戲代碼(graphics圖形庫)

    這篇文章主要為大家詳細介紹了C++實現拼圖游戲代碼,帶有graphics圖形庫,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-05-05
  • C++實現紅黑樹應用實例代碼

    C++實現紅黑樹應用實例代碼

    紅黑樹它一種特殊的二叉查找樹,這意味著它滿足二叉查找樹的特征,但是也有許多自己的特性,這篇文章主要給大家介紹了關于C++實現紅黑樹的相關資料,需要的朋友可以參考下
    2021-11-11
  • C++中const用法小結

    C++中const用法小結

    C++ const 允許指定一個語義約束,編譯器會強制實施這個約束,允許程序員告訴編譯器某值是保持不變的。如果在編程中確實有某個值保持不變,就應該明確使用const,這樣可以獲得編譯器的幫助。
    2016-04-04

最新評論