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

C語言小知識之為什么要使用指針詳析

 更新時間:2021年10月15日 12:42:46   作者:嵌入式@hxydj  
指針是C語言中一個非常重要的概念,也是C語言的特色之一,使用指針可以對復(fù)雜數(shù)據(jù)進(jìn)行處理,能對計算機(jī)的內(nèi)存分配進(jìn)行控制,在函數(shù)調(diào)用中使用指針還可以返回多個值,這篇文章主要給大家介紹了C語言小知識之為什么要使用指針的相關(guān)資料,需要的朋友可以參考下

剛開始學(xué)習(xí)C語言的時候,感覺最難理解的就是指針,什么指針變量,變量指針,指向指針的變量,指向變量的指針?一堆概念,搞得人云里霧里的,今天不討論這些概念的問題,從最底層來分析C語言中為什么要使用指針,指針存在的意義又是什么呢?

首先從一個簡單的例子來看,寫一段代碼來交換x、y的值。

void main( void )
{
    u8 x = 10, y = 20;
    u8 temp;

    __asm( "sim" );                             //禁止中斷
    SysClkInit();
    delay_init( 16 );
    LED_GPIO_Init();
    Uart1_IO_Init();
    Uart1_Init( 9600 );
    ADC_GPIO_Init();
    __asm( "rim" );                             //開啟中斷
    while( 1 )
    {
        LED = ~LED;

        printf( "x = %d,y = %d\r\n", x, y );

        temp = x;
        x = y;
        y = temp;

        printf( "---> x = %d,y = %d \r\n\r\n\r\n", x, y );

        delay_ms( 200 );
    }
}

在STM8單片機(jī)中,交換x、y的值,并將值打印出來,打印結(jié)果如下:

image.png

通過第三個變量temp很輕松的就將x、y的值交換了。

但是為了程序的美觀性,不想再主程序中寫這么多的代碼,于是決定用一個函數(shù)在外部來實現(xiàn)這個功能。

void swap( u8 x, u8 y )
{
    u8 temp;
    temp = x;
    x = y;
    y = temp;
}

void main( void )
{
    u8 x = 10, y = 20;
    __asm( "sim" );                             //禁止中斷
    SysClkInit();
    delay_init( 16 );
    LED_GPIO_Init();
    Uart1_IO_Init();
    Uart1_Init( 9600 );
    ADC_GPIO_Init();
    __asm( "rim" );                             //開啟中斷
    while( 1 )
    {
        LED = ~LED;

        printf( "x = %d,y = %d\r\n", x, y );
        sawp( x, y );
        printf( "---> x = %d,y = %d \r\n\r\n\r\n", x, y );

        delay_ms( 200 );
    }
}

在主函數(shù)外面定義一個函數(shù),專門用來交換x,y的值。程序運行結(jié)果如下:

image.png

此時發(fā)現(xiàn)x和y的值并沒有交換,這是怎么回事?那么在交換函數(shù)內(nèi)部,也將x和y的值打印出來。修改代碼如下:

void swap( u8 x, u8 y )
{
    u8 temp;
    
    printf( "in: x = %d,y = %d\r\n", x, y );
    temp = x;
    x = y;
    y = temp;
    
    printf( "in: ---> x = %d,y = %d \r\n\r\n\r\n", x, y );
}

打印結(jié)果如下:

image.png

可以看到在交換函數(shù)內(nèi)部,x和y的值已經(jīng)交換了,但是在函數(shù)外部,x和y的值并沒有交換。這是為什么呢?單步調(diào)試直接觀察x和y在單片機(jī)內(nèi)部分存儲情況。

image.png

進(jìn)入主程序之后,首先觀察主函數(shù)內(nèi)的x和y在內(nèi)存中的存儲情況,x值為10,也就是16進(jìn)制的0x0A,在內(nèi)存中0x000009的位置存儲,y的值為20,也就是16進(jìn)制的0x14,在內(nèi)存中0x00000B,位置存儲。

接下里進(jìn)入到swap函數(shù)中。為了方便觀察,將swap函數(shù)內(nèi)部的x和y替換成了m和n。

image.png

可以看出在進(jìn)入子函數(shù)之后,m和n的地址和值,都和main函數(shù)中的x和y一樣。接下來交換m和n的值。

image.png

交換完成后發(fā)現(xiàn),內(nèi)存中0x000009位置的值和0x00000B位置的值亞發(fā)生了交換。然后退出子函數(shù),返回到main函數(shù)中。

image.png

這時候奇怪的事情發(fā)生了,剛才內(nèi)存中交換的值又變回來了?這是為什么呢?在這里就不得不說在C語言中關(guān)于局部變量的問題,局部變量在C語言中是沒有固定的存儲位置的,它是由系統(tǒng)的堆棧統(tǒng)一來管理的,當(dāng)進(jìn)入函數(shù)內(nèi)部時,系統(tǒng)就會給這些局部變量臨時分配一個存儲空間存儲它的值,當(dāng)程序要離開函數(shù)時,系統(tǒng)就會將局部變量的值保存在堆棧中,然后變量的存儲位置就被釋放了。當(dāng)程序進(jìn)入另一個函數(shù)中時,又會給這個函數(shù)內(nèi)部局部變量分配空間,這時候可能就會出現(xiàn)兩個函數(shù)中的局部變量都使用了同一個內(nèi)存空間。此時這個內(nèi)存空間的值改變后,并不影響上一個函數(shù)中局部變量的值,因為上一個函數(shù)中的局部變量值此時在堆棧中存放。當(dāng)程序要離開當(dāng)前的這個函數(shù)時,又會將當(dāng)前的局部變量值保存在堆棧中?;氐缴弦粋€函數(shù)后,又將堆棧中存儲的值恢復(fù)給變量,此時變量的地址又是臨時申請的,可能此刻申請的地址值還是和上一次一樣。但是并不代表這個地址就永遠(yuǎn)屬于這個局部變量。

這個就很類似于超市中的儲物柜,你要進(jìn)去超市買東西,先將東西存到一個柜子中,買完東西后,又將東西存儲物柜取了出來。然后隔了幾個小時,又要去這個超市買東西,又需要將東西存起來,但是此時存儲的柜子編號還是上上一次存儲時一樣。但是這并不能代表這個柜子的編號就是專屬于你的了。它只是儲物柜臨時分配給你的空間,當(dāng)你取出東西后這個空間就會被系統(tǒng)收回,如果你下一次還需要用,系統(tǒng)又會自動給你分配,但是這兩次分配的剛好是一個編號而已。

那么要如何解決這種變量交換的問題呢?有兩種方法,第一種就是直接將要交換的這個兩個變量定義為全局變量,讓它在程序運行的過程中獨占一個地址空間,這樣就不會有其他變量來使用這個位置了。但是這樣的話就會比較浪費內(nèi)存空間,只使用了一次,但是卻要永久的占用。第二種方法就是直接使用指針。

下面將代碼改成使用指針的方式。

void swap( u8 *m, u8 *n )
{
    u8 temp;   
    temp = *m;
    *m = *n;
    *n = temp;    
}
void main( void )
{
    u8 x = 10, y = 20;
    __asm( "sim" );                             //禁止中斷
    SysClkInit();
    delay_init( 16 );
    LED_GPIO_Init();
    Uart1_IO_Init();
    Uart1_Init( 9600 );
    ADC_GPIO_Init();
    __asm( "rim" );                             //開啟中斷
    while( 1 )
    {
        LED = ~LED;

        printf( "x = %d,y = %d\r\n", x, y );
        sawp( &x, &y );
        printf( "---> x = %d,y = %d \r\n\r\n\r\n", x, y );

        delay_ms( 200 );
    }
}

在向swap函數(shù)傳遞參數(shù)的時候需要使用&符號。打印輸出結(jié)果

image.png

此時x和y的值已經(jīng)成功交換了?,F(xiàn)在也將子函數(shù)內(nèi)部的交換情況打印一下。

void swap( u8 *m, u8 *n )
{
    u8 temp;
    
    printf( "in: m = %d,n = %d\r\n", m, n );
    temp = *m;
    *m = *n;
    *n = temp;
    
    printf( "in: ---> m = %d,n = %d \r\n\r\n\r\n", m, n );
}

image.png

從輸出的結(jié)果來看,怎么m和n的值一個時1021,一個時1020.這個值是什么呢?直接單步調(diào)試看。

image.png

此時main函數(shù)中x的地址變成了0x0003FD,y的地址變成了0x0003FC.接著進(jìn)入子函數(shù)。

image.png

這時可以看出m的值為0x03FD,n的值為0x03FC.*m的值為0x0A也是就10,*0x14也就是20.
接下來開始交換值。

image.png

可以看出m和n的值沒變,但是m和n的值交換了。接下來回到主函數(shù)中。

image.png

此時主函數(shù)中x和y的值也交換了。

那剛才串口打印出來的1021和1020是什么呢?1021的十六進(jìn)制是0x03FD,1020的十六進(jìn)制是0x03FC.也就是說剛才打印的m的值和x的地址一樣,n的值和y的地址一樣。

那為什么m和n會變成x和y地址,*m 和 *n又會變成 x 和 y 的值。這里就要說指針的本質(zhì)了。在存儲器內(nèi)部,它是不認(rèn)識什么變量和指針的,對于存儲空間來說,它只有地址和值。也就是說在什么地址處,存儲什么值。對于普通的變量來說,變量的名稱就會被編譯器編譯成地址,也就是說x和y就是它自己地址的別名。x和y的值就是地址中對應(yīng)的值。

當(dāng)操作普通變量x和y的時候,系統(tǒng)默認(rèn)操作的就是它的值。而指針剛好和它相反,指針默認(rèn)是把地址作為它的值,當(dāng)操作指針的時候,默認(rèn)操作的就是地址。為了將普通變量和指針進(jìn)行區(qū)分,那么如果要使用指針的時候,就需要給它貼一個標(biāo)簽,告訴系統(tǒng),我這個是特殊變量,它是直接操作地址的,不是操作值的。

所以在定義指針的時候給變量前面加一個*號,就表示告訴系統(tǒng),我這個是特殊的。比如定義了一個int *m。就表示告訴系統(tǒng),當(dāng)我默認(rèn)操作m的時候,你就給我它的地址,而不要給我它的值。當(dāng)需要取值的時候就需要添加上標(biāo)簽 *m,告訴系統(tǒng)我現(xiàn)在要取的值,不是地址,這是特殊情況,不要把默認(rèn)的地址給我。

當(dāng)要將普通變量傳遞給指針時,因為直接操作變量默認(rèn)就是普通變量的值,而指針存貯的是地址,所以當(dāng)普通變量和指針傳遞數(shù)據(jù)的時候,也要給普通變量添加一個標(biāo)簽 & ,這個符號就告訴系統(tǒng),我現(xiàn)在不要默認(rèn)的值,我要的是特殊情況的地址。

所以將x和y傳遞給指針的時候,前面要加&符號。

swap( &x, &y );對應(yīng)的就是 swap( u8 *m, u8 *n );

剛才上面不是說了嗎,指針默認(rèn)的是地址,加上*號就是值了。那么這樣直接傳遞過去不就是相當(dāng)于 *m = &x 了嗎?

由于這個子函數(shù)是定義和傳值在一起操作了,省略了一步,標(biāo)準(zhǔn)操作應(yīng)該是。

int *m;

m=&x;

先定義一個指針,然后將普通變量的特殊情況,也就是取普通變量的地址,傳遞給指針的默認(rèn)情況。這樣m的默認(rèn)情況下就代表的值x的地址,而x的值就是*m。

如果定義變量和給變量賦值在一條語句時,上面的代碼就可以簡寫為

int *m = &x;

所以上面的函數(shù) swap( &x, &y ); 給指針傳遞值的時候,指針的定義和賦值是在一條語句完成的, swap( u8 *m, u8 *n ); 這是一種常用的簡寫形式。

在swap函數(shù)內(nèi)部操作 *m 也就相當(dāng)于直接操作的是x,操作 *n 就是直接操作的y,所以交換 *m 和 *n的值,就相當(dāng)于交換x和y的值。

在沒有使用指針時通過子函數(shù)交換,此時傳遞的是變量的值,相當(dāng)于把變量的值拷貝了一份,給了子程序。

而有了指針之后,相當(dāng)于將變量的地址直接給了子函數(shù)。相當(dāng)于給變量x和y又起了一個別名。操作別名的時候,也就相當(dāng)于直接操作的是x。

由此可見,指針只是為了方便編寫程序而設(shè)置的一種給變量起別名的方法,也就是相當(dāng)于給自己柜子配了了一把鑰匙。只要別人有這個鑰匙,也就可以打開你的柜子。所以指針在使用的時候會有危險性。

如果系統(tǒng)中有關(guān)鍵數(shù)據(jù),那么如果這個數(shù)據(jù)用指針傳遞給了外部函數(shù),那么當(dāng)外部函數(shù)修改數(shù)據(jù)的時候,系統(tǒng)就會存在風(fēng)險。有可能外部函數(shù)修改了一個值,而這個值是非法的,自己的系統(tǒng)就奔潰了。所以在使用指針的時候一定要注意安全性問題。

通過上面的例子,相信對指針就有了更深層次的理解了。它是為了方便操作變量而設(shè)置的特殊情況,是被貼了標(biāo)簽的變量。至于什么指針變量,變量指針,那都是起的名字而已,搞不清楚這些概念也不用去糾結(jié),只要在使用的時候,知道如何使用就行了。

總結(jié)

到此這篇關(guān)于C語言小知識之為什么要使用指針的文章就介紹到這了,更多相關(guān)C語言使用指針內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • c++學(xué)習(xí)之構(gòu)造函數(shù)

    c++學(xué)習(xí)之構(gòu)造函數(shù)

    類多么重要我就不多說了,只講講學(xué)習(xí),因為個人認(rèn)為類的學(xué)習(xí)無論從概念的理解還是實際代碼的編寫相對其他C兼容向的代碼都是比較有難度的, 對于以前學(xué)C 的人來說這才是真正的新概念和內(nèi)容,STL其實還比較好理解,不就是一個更大的函數(shù)庫和代碼可以使用嘛。
    2015-06-06
  • 基于C語言實現(xiàn)五子棋游戲

    基于C語言實現(xiàn)五子棋游戲

    這篇文章主要為大家詳細(xì)介紹了基于C語言實現(xiàn)五子棋游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-11-11
  • C++處理圖存儲的方式分享

    C++處理圖存儲的方式分享

    這篇文章主要介紹了C++處理圖存儲的方式分享,文章圍繞鄰接矩陣、鄰接表、鏈?zhǔn)角跋虻闹黝}展開詳細(xì)內(nèi)容,具有一定的參考價值,需要的小伙伴可以參考一下
    2022-03-03
  • C語言標(biāo)準(zhǔn)庫<math.h>和<setjmp.h>的實現(xiàn)

    C語言標(biāo)準(zhǔn)庫<math.h>和<setjmp.h>的實現(xiàn)

    本文主要介紹了C語言標(biāo)準(zhǔn)庫<math.h>和<setjmp.h>的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-11-11
  • C與匯編混合編程的實現(xiàn)示例

    C與匯編混合編程的實現(xiàn)示例

    本文主要介紹了C與匯編混合編程的實現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-07-07
  • C++類和對象到底是什么

    C++類和對象到底是什么

    C++ 是一門面向?qū)ο蟮木幊陶Z言,理解 C++,首先要理解類(Class)和對象(Object)這兩個概念。下面和小編一起來學(xué)習(xí)吧
    2021-09-09
  • 一波C語言字符數(shù)組實用技巧集錦

    一波C語言字符數(shù)組實用技巧集錦

    這篇文章主要介紹了一波C語言字符數(shù)組實用技巧集錦,包括許多字符的轉(zhuǎn)換與提取等基本操作示例,需要的朋友可以參考下
    2016-04-04
  • C++中函數(shù)重載詳解

    C++中函數(shù)重載詳解

    大家好,本篇文章主要講的是C++中函數(shù)重載詳解,感興趣的同學(xué)趕快來看一看吧,對你有幫助的話記得收藏一下
    2022-02-02
  • 基于C++的拼多多算法在線筆試題示例

    基于C++的拼多多算法在線筆試題示例

    這篇文章主要介紹了基于C++的拼多多算法在線筆試題,列舉了四個拼多多的算法筆試題,包括分治法、大數(shù)相乘、貪心算法以及迷宮問題,需要的朋友可以參考下
    2017-08-08
  • 使用C語言遞歸與非遞歸實現(xiàn)字符串反轉(zhuǎn)函數(shù)char *reverse(char *str)的方法

    使用C語言遞歸與非遞歸實現(xiàn)字符串反轉(zhuǎn)函數(shù)char *reverse(char *str)的方法

    本篇文章是對使用C語言遞歸與非遞歸實現(xiàn)字符串反轉(zhuǎn)函數(shù)char *reverse(char *str)進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05

最新評論