C語言深入探索數據類型的存儲

數據類型介紹

首先,對於我們C語言中的數據類型,大傢應該都有一個清晰的認識吧!如果不記得也沒有關系哦~ 在這裡來跟著小劉同學回顧一下吧!

關於數據類型,我們在前面已經學習過瞭一些內置數據類型,以及它們所占的內存空間的大小,例如:

char         //字符數據類型
int          //整型
short        //短整型
long         //長整型
long long    //更長的整型
float        //單精度浮點型
double       //雙精度浮點型

那,在這裡,我們需要思考一個問題:在C語言中存在字符串類型嘛?

這個問題的答案需要大傢自己去思考喲~(當然,如果不知道的話可以私信小劉同學喲~

類型的意義:1.使用這個類型開辟的內存空間的大小,也就是說類型決定瞭我們可以使用的空間的大小。2.如何看待內存空間的視角。

類型的基本歸納

首先,當然是我們經常使用的整型類型傢族啦!

整型傢族

char //字符數據類型
    unsigned char //無符號字符型
    signed char //有符號字符型
short //短整型
     unsigned short[int] //無符號短整型
     signed short[int] //有符號短整型
int //整型
   unsigned int //無符號整型
   signed int //有符號整型
long //長整型
    unsigned long[int] //無符號長整型
    signed long[int] //有符號長整型

當然,整型傢族當然是少不瞭我們的long long int 型的,但是由於更長長整型的分類與整型傢族其他成員的分類是相似的,所以在這裡我們不加贅述。

浮點數傢族

float
double

構造類型

>數組類型
>結構體類型 struct
>枚舉類型 enum
>聯合類型 union

指針類型

int * pi;
char * pc;
float * pf;
void * pv; 

空類型

void表示空類型(無類型)

空類型通常應用於函數的返回類型,函數的參數,指針類型。

整型在內存中的存儲

首先,我們在之前的文章中談到過,變量的創建是要在內存中開辟一部分空間的。而開辟的空間的大小是要根據不同的類型而決定的。那麼,下面就跟著小劉同學一起來學習一下數據在它所開辟的內存中究竟是如何進行存儲的吧!

我們先來舉一個例子吧:

int a=10;
int b=-10;

我們知道,因為a為int型,所以為a分配瞭4個字節的空間,那麼,在內存又該如何存儲吶?

要想瞭解這一點,我們先來明白幾個概念:

原碼,反碼,補碼

在計算機中的整數存在3種2進制的表示方式:原碼、反碼、補碼。

這三種方法均存在符號位和數值位這兩個部分。符號位使用0表示'正',使用1表示'負'。而數值位正數的原碼、反碼、補碼都相同。而負整數的三種表示方法都不相同。

原碼

直接將數值按照正負數的形式轉換成二進制的形式就可以得到原碼。

反碼

將原碼的符號位不變,其他位依次求反就可以得到反碼。

補碼

在反碼的基礎上+1就可以得到補碼。

對於整形來說,數據存放在內存中其實是以補碼的形式存放的。(相信學過數電的小夥伴一定知道這是為什麼。那麼其他不知道所以然的小夥伴請自行百度詢問哦。

在vs裡面我們可以看到a和b在內存中的地址,但是,這樣看是不是有點不太對?

按照我們上面所講的內容,列出a和b在內存中存儲的二進制形式再根據二進制轉為十六進制的規則,a和b在內存中的存儲應該是下圖所示的形式,那麼為什麼我們在VS的調試中會出現不一樣的結果吶?

int a=10;
//原碼:0000 0000 0000 0000 0000 0000 0000 1010
//反碼:0000 0000 0000 0000 0000 0000 0000 1010
//補碼:0000 0000 0000 0000 0000 0000 0000 1010
// 00 00 00 0a
int b=-10;
//原碼:1000 0000 0000 0000 0000 0000 0000 1010
//反碼:1111 1111 1111 1111 1111 1111 1111 0101
//補碼:1111 1111 1111 1111 1111 1111 1111 1010 
//ff ff ff fa

這個時候,我們來再瞭解一個概念——大小端。

大小端

大小端指的是兩種不同的存儲方式。那麼下面我們來詳細瞭解一下什麼是大小端:

大端(存儲)模式:是指數據的低位數保存在內存的高地址中,而數據的高位,保存在內存的低地址中。

小端(存儲)模式:是指數據的低位數保存在內存的低地址中,而數據的高位,保存在內存的高地址中。

當然,大端存儲和小端存儲是根據編譯器的不同來決定的。

這樣子隻看概念大傢是否感到十分疑惑吶?下面,小劉同學給大傢畫圖模擬一下大端存儲和小端存儲的基本情況吧!

這樣子,大傢是不是對大端存儲和小端存儲有瞭一個基本的概念瞭解吶?

那麼,大傢現在是不是非常好奇自己目前使用的編譯器究竟使用瞭那種存儲方式吶?

下面請跟著小劉同學來判斷一下自己所使用的編譯器究竟是哪種存儲方式吧!

我們首先定義一個變量:int a=1;

int a=1;
//原碼:0000 0000 0000 0000 0000 0000 0000 0001
//反碼:0000 0000 0000 0000 0000 0000 0000 0001
//補碼:0000 0000 0000 0000 0000 0000 0000 0001
//如果是大端存儲,返回的第一個字節一定為00 ——> 0
//如果是小端存儲,返回的第一個字節一定為01 ——> 1

完整代碼如下:

#include<stdio.h>
int chek_sys()
{
	int i = 1;
	return (*(char*)&i);
}
int main()
{
	int ret = chek_sys();
	if (ret == 1)
	{
		printf("小端\n");
	}
	else 
	{
		printf("大端\n");
	}
	return 0;
}

浮點數在內存中的存儲

浮點數存儲的規則

首先,我們來舉一個例子

#include<stdio.h>
int main()
{
	int n = 9;
	float* pFloat = (float*)&n;
	printf("n的值為:%d\n", n);
	printf("*pFloat的值為:%f\n", *pFloat);
	*pFloat = 9.0;
	printf("num的值為:%d\n", n);
	printf("*pFloat的值為:%f\n", *pFloat);
	return 0;
}

以上代碼執行的結果為:

它這個結果好像和我們想象的結果不太一樣哎。這是為什麼吶?在這裡就不得不提一下浮點數在內存中的存儲規則瞭。

根據國際標準IEEE(電氣和電子工程協會) 754,任意一個二進制浮點數V可以表示成下面的形式:

  • (-1)^S * M * 2E
  • (-1)S表示符號位,當S=0,V為正數;當S=1,V為負數。
  • M表示有效數字,大於等於1,小於2。
  • 2^E表示指數位。

看到這裡,大傢是不是有很多的疑問和不理解?別急,下面小劉同學來為大傢講解清楚。

要想理解上面的內容,我們先來舉一個例子吧!

例如:9.5f

小數點後面的權重是2的-1次方,-2次方這樣子依次類推。

當然,並不是所有的浮點數都可以表示出來的,有的浮點數是無法表示出來的,小數點後面的數無法精確表示,可能會出現精度缺失。

IEEE 754對有效數字M和指數E,還有一些特別規定。

前面說過, 1≤M<2 ,也就是說,M可以寫成 1.xxxxxx 的形式,其中xxxxxx表示小數部分。 IEEE 754規定,在計算機內部保存M時,默認這個數的第一位總是1,因此可以被舍去,隻保存後面的xxxxxx部分。比如保存1.01的時候,隻保存01,等到讀取的時候,再把第一位的1加上去。這樣做的目的,是節省1位有效數字。以32位浮點數為例,留給M隻有23位,將第一位的1舍去以後,等於可以保存24位有效數字。

至於指數E,情況就比較復雜。

首先,E為一個無符號整數(unsigned int) 這意味著,如果E為8位,它的取值范圍為0~255;如果E為11位,它的取值范圍為0~2047。但是,我們知道,科學計數法中的E是可以出現負數的,所以IEEE 754規定,存入內存時E的真實值必須再加上一個中間數,對於8位的E,這個中間數是127;對於11位的E,這個中間 數是1023。比如,2^10的E是10,所以保存成32位浮點數時,必須保存成10+127=137,即 10001001。

在我們已經知道如何存儲浮點數後,我們再來學習一下如何將浮點數從內存中取出。

指數E從內存中取出還可以再分成三種情況:

E不全為0或不全為1

這時,浮點數就采用下面的規則表示,即指數E的計算值減去127(或1023),得到真實值,再將有效數字M前加上第一位的1。 比如: 0.5(1/2)的二進制形式為0.1,由於規定正數部分必須為1,即將小數點右移1位,則為 1.0*2^(-1),其階碼為-1+127=126,表示為01111110,而尾數1.0去掉整數部分為0,補齊0到23位00000000000000000000000,則其二進制表示形式為:

//0 01111110 00000000000000000000000

E全為0

這時,浮點數的指數E等於1-127(或者1-1023)即為真實值, 有效數字M不再加上第一位的1,而是還原為0.xxxxxx的小數。這樣做是為瞭表示±0,以及接近於0的很小的數字。

E全為1

這時,如果有效數字M全為0,表示±無窮大(正負取決於符號位s)。

以上及是數據存儲的內容。另外,關於數據存儲方面的一些補充和講解

到此這篇關於C語言深入探索數據類型的存儲 的文章就介紹到這瞭,更多相關C語言數據存儲 內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: