第 2 節

數據類型與數據存放

0瀏覽次數0訪問次數--跳出率--平均停留

數據類型

整型

作用:整型變量表示的是==整數類型==的數據

C++中能夠表示整型的類型有以下幾種方式,區別在於所佔內存空間不同

數據類型佔用空間取值範圍
short(短整型)2字節(-2^15 ~ 2^15-1)
int(整型)4字節(-2^31 ~ 2^31-1)
long(長整形)Linux 64位通常爲8字節,32位環境通常爲4字節由平臺位數決定
long long(長長整形)8字節(-2^63 ~ 2^63-1)

sizeof關鍵字

**作用:**利用sizeof關鍵字可以==統計數據類型所佔內存大小==

語法: sizeof( 数据类型 / 变量)

示例:

int main() {
    // 程序从 main 函数开始执行,下面的语句会按顺序运行。

    cout << "short 类型所占内存空间为: " << sizeof(short) << endl;

    cout << "int 类型所占内存空间为: " << sizeof(int) << endl;

    cout << "long 类型所占内存空间为: " << sizeof(long) << endl;

    cout << "long long 类型所占内存空间为: " << sizeof(long long) << endl;


    // 返回 0 表示程序正常结束。
    return 0;
}

運行/觀察結果: 運行後會按輸出語句打印對應內容,變量值可結合初始化、賦值和函數調用順序推導。

整型結論:==short < int <= long <= long long==

實型(浮點型)

作用:用於==表示小數==

浮點型變量分爲兩種:

  1. 單精度float
  2. 雙精度double

兩者的區別在於表示的有效數字範圍不同。

數據類型佔用空間有效數字範圍
float4字節7位有效數字
double8字節15~16位有效數字

示例:

int main() {

    float f1 = 3.14f;
    double d1 = 3.14;

    cout << f1 << endl;
    cout << d1<< endl;

    cout << "float  sizeof = " << sizeof(f1) << endl;
    cout << "double sizeof = " << sizeof(d1) << endl;

    //科学计数法
    float f2 = 3e2; // 3 * 10 ^ 2 
    cout << "f2 = " << f2 << endl;

    float f3 = 3e-2;  // 3 * 0.1 ^ 2
    cout << "f3 = " << f3 << endl;


    return 0;
}

運行/觀察結果: 運行後會按輸出語句打印對應內容,變量值可結合初始化、賦值和函數調用順序推導。

字符型

**作用:**字符型變量用於顯示單個字符

語法:char ch = 'a';

注意1:在顯示字符型變量時,用單引號將字符括起來,不要用雙引號

注意2:單引號內只能有一個字符,不可以是字符串

  • C和C++中字符型變量只佔用==1個字節==。
  • 字符型變量並不是把字符本身放到內存中存儲,而是將對應的ASCII編碼放入到存儲單元

示例:

int main() {
    
    char ch = 'a';
    cout << ch << endl;
    cout << sizeof(char) << endl;

    //ch = "abcde"; //错误,不可以用双引号
    //ch = 'abcde'; //错误,单引号内只能引用一个字符

    cout << (int)ch << endl;  //查看字符a对应的ASCII码
    ch = 97; //可以直接用ASCII给字符型变量赋值
    cout << ch << endl;


    return 0;
}

運行/觀察結果: 運行後會按輸出語句打印對應內容,變量值可結合初始化、賦值和函數調用順序推導。

ASCII碼錶格:

ASCII控制字符ASCII字符ASCII字符ASCII字符
0NUT32(space)64@96
1SOH33!65A97a
2STX34"66B98b
3ETX35#67C99c
4EOT36$68D100d
5ENQ37%69E101e
6ACK38&70F102f
7BEL39,71G103g
8BS40(72H104h
9HT41)73I105i
10LF42*74J106j
11VT43+75K107k
12FF44,76L108l
13CR45-77M109m
14SO46.78N110n
15SI47/79O111o
16DLE48080P112p
17DCI49181Q113q
18DC250282R114r
19DC351383S115s
20DC452484T116t
21NAK53585U117u
22SYN54686V118v
23TB55787W119w
24CAN56888X120x
25EM57989Y121y
26SUB58:90Z122z
27ESC59;91[123{
28FS60<92/124|
29GS61=93]125}
30RS62>94^126`
31US63?95_127DEL

ASCII 碼大致由以下兩部分組成:

  • ASCII 非打印控制字符: ASCII 表上的數字 0-31 分配給了控制字符,用於控制像打印機等一些外圍設備。
  • ASCII 打印字符:數字 32-126 分配給了能在鍵盤上找到的字符,當查看或打印文檔時就會出現。

轉義字符

**作用:**用於表示一些==不能顯示出來的ASCII字符==

現階段我們常用的轉義字符有: \n \\ \t

轉義字符含義ASCII碼值(十進制)
\a警報007
\b退格(BS) ,將當前位置移到前一列008
\f換頁(FF),將當前位置移到下頁開頭012
\n換行(LF) ,將當前位置移到下一行開頭010
\r回車(CR) ,將當前位置移到本行開頭013
\t水平製表(HT) (跳到下一個TAB位置)009
\v垂直製表(VT)011
\\代表一個反斜線字符""092
'代表一個單引號(撇號)字符039
"代表一個雙引號字符034
?代表一個問號063
\0數字0000
\ddd8進制轉義字符,d範圍0~73位8進制
\xhh16進制轉義字符,h範圍09,af,A~F3位16進制

示例:

int main() {
    // 程序从 main 函数开始执行,下面的语句会按顺序运行。
    
    
    cout << "\\" << endl;
    cout << "\tHello" << endl;
    cout << "\n" << endl;


    // 返回 0 表示程序正常结束。
    return 0;
}

運行/觀察結果: 運行後會按輸出語句打印對應內容,變量值可結合初始化、賦值和函數調用順序推導。

字符串型

作用:用於表示一串字符

兩種風格

  1. C風格字符串char 变量名[] = "字符串值"
    示例:
    int main() {
    
        char str1[] = "hello world";
        cout << str1 << endl;
        
    
        return 0;
    }
    

注意:C風格的字符串要用雙引號括起來

  1. C++風格字符串string 变量名 = "字符串值"
    示例:
    int main() {
    
        string str = "hello world";
        cout << str << endl;
        
    
        return 0;
    }
    

注意:C++風格字符串,需要加入頭文件==#include<string>==

布爾類型 bool

**作用:**布爾數據類型代表真或假的值

bool類型只有兩個值:

  • true --- 真(本質是1)
  • false --- 假(本質是0)

bool類型佔==1個字節==大小

示例:

int main() {

    bool flag = true;
    cout << flag << endl; // 1

    flag = false;
    cout << flag << endl; // 0

    cout << "size of bool = " << sizeof(bool) << endl; //1
    

    return 0;
}

運行/觀察結果: 運行後會按輸出語句打印對應內容,變量值可結合初始化、賦值和函數調用順序推導。

類型別名 typedef

C/C++ 提供了 typedef 關鍵字,您可以使用它來爲類型取一個新的名字。下面的實例爲單字節數字定義了一個術語 BYTE

typedef unsigned char byte;
typedef unsigned char uint8_t;
typedef float fp32;
typedef double fp64;

運行/觀察結果: 這段偏語法定義,通常需要配合調用代碼一起編譯,重點看定義方式和使用位置。

數據單位、二進制與補碼

  1. 數據單位及其轉換
    1. 1字節byte = 8比特bit = 8位二進制
    2. 數據是以 二進制 的形式存在電腦內存中的
    3. 進制轉換
      1. 在C語言中,0b開頭的數據代表2進制,0x開頭的數代表16進制。因爲2進制較長,所以我們在代碼中,一般把2進制轉化成16進制進行表示。
    4. 數據的原碼
      1. 無符號(unsigned)的數據類型: 無符號數據所佔的所有2進制都表示數據的大小,假設佔8位2進制,那麼該數最大是255(其對應的2進制:1111 1111),最小是0(對應2進制是0000 0000)。
      2. 有符號(signed)的數據類型:有符號的數據其最高位二進制是符號位,符號位是0則是正數,符號位是1則爲負數。假設佔8位2進制,去掉1位2進制,還剩7位2進制來表示數據的大小。那麼其最大是127(其2進制是0111 1111),最小是-128(其2進制是1000 0000,-127的是1111 1111),這個-128其實是用了-0的二進制表示,因爲咱們有0(0000 0000)了,所以-0(理論是1000 0000)就沒有啥意義了,所以我們規定把-0的二進制表示成-128。
    5. 數據的命名 :
      1. 由於unsigned char,int等數據類型名稱無法突出 數據的二進制 有無符號,佔多少二進制,所以我們用typedef給其起新的別名(比如 新別名 uint8_t,int32_t)
        1. uint8_t 中的u代表unsigned無符號,8代表佔8位二進制,也就是1個字節。
        2. int32_t前面不帶u,所以是有符號的,佔32位二進制,所以也就是佔4個字節,所以是int類型的別名
        3. 因爲char 佔 1 字節,8 比特,8 位二進制,而且有符號,所以我們叫他int8_t
        4. 因爲unsigned short int 無符號,佔 2字節,16比特,16位2進制,所以叫uint16_t,其他的類似。
        5. float佔4字節,32位二進制,所以叫fp32
        6. double佔8字節,64位二進制,所以叫fp64
      2. 具體的代碼(在C++中自帶一部分別名,可以不寫一部分別名,但建議寫):
        1. typedef unsigned char uint8_t;
        2. typedef short int int16_t;
        3. typedef unsigned int uint32_t;
        4. typedef float fp32;

  2. 數據解析:
    1. 應用場景介紹:一般傳感器會一直給我們的單片機、工控機發送數據,比如說發送我們機器人所走的路程(假設單位是毫米,是個短整型數)。
    2. 數據處理目標: 比如獲取路程(短整型,2個字節,16位2進制)
    3. 傳感器傳數據: 一般的通信來說,傳感器會一個字節一個字節的給單片機和工控機不斷髮數據,一般一個佔n字節的物理量數據,要分成n個字節,也就是n個變量來發,又因爲數據以2進制的形式存儲,一般先發最高的8位二進制。
    4. 單片機、工控機接受到的數據: 比如獲取機器人的路程,該數據佔2字節,所以傳感器要分成兩個變量來發送。先接收到的變量我們叫DH(數據的高8位2進制),後接收到的數據我們叫DL(數據的低8位2進制)。
    5. 數據處理思路: 我們現在擁有的是兩個普通的8位二進制變量(uint8_t類型的),我們想要獲得帶符號的16位2進制的數據,機器人的路程(int16_t類型的)。
      1. 位操作 : 假如說,DH是0x9D(對應2進制是1001 1101),DL是0x57(對應2進制是0101 0111)。那麼我們想要的16位數據就是把DH當16位中的最高的8位,DL當最低的8位,也就是我們想要的數據DATA(其二進制是 1001 1101 0101 0111)。我們想實現這個效果,就需要位操作,把DH向左移8位,讓他變成1001 1101 0000 0000,然後再把左移後的DH 與 DL進行按位或運算,也就是1001 1101 0000 0000 與 0000 0000 0101 0111進行按位或,最後就得出來DATA是1001 1101 0101 0111了。
      2. 強轉類型: 我們上面已經得出來data的2進制了,再進行強轉化成有符號的數,也就是把最高位中的1變成符號位,具體代碼是int16_t DATA = (int16_t) ((uint16_t)DH << 8 | DL ),這樣就處理成功了。
  3. 數據在內存中的二進制表示方式
    1. 計算機存儲數據的前置知識
      1. 計算機底層存儲數據時使用的是二進制數字,但是計算機在存儲一個數字時並不是直接存儲該數字對應的二進制數字,而是存儲該數字對應二進制數字的 補碼
      2. 機器碼:一個數在計算機的存儲形式是二進制數,我們稱這些二進制數爲機器數,機器數是有符號,在計算機中用機器數的次最高位的左側存放符號位,0表示正數,1表示負數
      3. 真值:因爲機器數帶有符號位,所以機器數的形式值不等於其真實表示的值(真值),以機器數1000 0001爲例,其真正表示的值(首位爲符號位)爲-1,而形式值(首位就是代表1)爲129; 因此將帶符號的機器數的真正表示的值稱爲機器數的真值
    2. 原碼、反碼、補碼:
      1. 原碼 的表示與機器數真值表示的一樣,即用第一位表示符號,其餘位表示數值。也就是

          正數:就是它對應的二進制數。 負數:將絕對值對應的二進制最左邊位變爲1。
      【+1】= 原:[ 0000 0001 ]
      【-1】= 原:[ 1000 0001 ]
      
      1. 反碼:

          正數 : 和原碼相同。 負數 : 在其原碼的基礎上,符號位不變,其餘各位取反。
      【+1】= 原: [ 0000 0001 ] = 反:[ 0000 0001 ]
      【-1】= 原: [ 1000 0001 ] = 反:[ 1111 1110 ]
      
      1. 補碼

          正數 : 補碼是其原碼本身。 負數 : 補碼是在其原碼的基礎上,符號位不變,其餘各位取反後加1(即在反碼的基礎上加1)
      【+1】= 原: [ 0000 0001 ] = 反:[ 0000 0001 ] = 补:[ 0000 0001 ]
      【-1】= 原: [ 1000 0001 ] = 反:[ 1111 1110 ] = 补:[ 1111 1111 ]
      
    3. 數據在計算機中的存儲形式
      1. 計算機實際只存儲補碼,所以原碼轉換爲補碼的過程,也可以理解爲數據存儲到計算機內存中的過程:

      1. 正數:在原、反、補碼中,正數的表示是一模一樣的
      2. 負數:負數的表示是不相同的,所以對於負數的補碼來說,我們是不能直接用進制轉換將其轉換爲十進制數值的,因爲這樣是得不到計算機真正存儲的十進制數的,所以應該將其轉換爲原碼後,再將轉換得到的原碼進行進制轉換爲十進制數(機器數包含符號位)
      3. 爲什麼要誕生原碼、反碼和補碼?
        1. 原因:對於只有正數的加減運算來說,用原碼計算是沒有任何問題的,但是當既有正數,也有負數的時候,計算機無法判斷最高位是否是符號位。
    4. 總結:
      1. 二進制的最高位是符號位:0表示正數,1表示負數
      2. 正數的原碼反碼補碼都一樣,三碼合一。
      3. 負數
        1. 負數的反碼 = 它的原碼符號位不變,其它位取反。
        2. 負數的補碼 = 它的反碼 + 1, 負數的反碼 = 負數的補碼 - 1 。
      4. 0 的反碼、補碼都是 0 。
      5. 在計算機運算的時候都是以 “補碼” 的方式來運算的。
      6. 運算結果是以 原碼 形式展現的。
音乐页