第 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. 运算结果是以 原码 形式展现的。
音乐页