【C++学习笔记】三、C++的数据类型、存储以及基本运算

本文记录了C++中的数据类型以及基本运算,这部分是典型的每次记每次忘类型,所以烙印在此,以便用时随时查阅。

主要参考:http://www.runoob.com/cplusplus/cpp-data-types.html

1. C++中的数据类型

(1)基本类型

C++ 为程序员提供了种类丰富的内置数据类型和用户自定义的数据类型。下表列出了七种基本的 C++ 数据类型:

类型关键字
布尔型bool
字符型char
整型int
浮点型float
双浮点型double
无类型void
宽字符型wchar_t

一些基本类型可以使用一个或多个类型修饰符进行修饰:

  • signed
  • unsigned
  • short
  • long

下表显示了各种变量类型在内存中存储值时需要占用的内存,以及该类型的变量所能存储的最大值和最小值。

注意:不同系统会有所差异。

类型范围
char1 个字节-128 到 127 或者 0 到 255
unsigned char1 个字节0 到 255
signed char1 个字节-128 到 127
int4 个字节-2147483648 到 2147483647
unsigned int4 个字节0 到 4294967295
signed int4 个字节-2147483648 到 2147483647
short int2 个字节-32768 到 32767
unsigned short int2 个字节0 到 65,535
signed short int2 个字节-32768 到 32767
long int8 个字节-9,223,372,036,854,775,808 到 9,223,372,036,854,775,807
signed long int8 个字节-9,223,372,036,854,775,808 到 9,223,372,036,854,775,807
unsigned long int8 个字节0 to 18,446,744,073,709,551,615
float4 个字节+/- 3.4e +/- 38 (~7 个数字)
double8 个字节+/- 1.7e +/- 308 (~15 个数字)
long double16 个字节+/- 1.7e +/- 308 (~15 个数字)
wchar_t2 或 4 个字节1 个宽字符

从上表可得知,变量的大小会根据编译器和所使用的电脑而有所不同。

(2)存储方式

类型描述
bool存储值 true 或 false。
char通常是一个八位字节(一个字符)。这是一个整数类型。
int对机器而言,整数的最自然的大小。
float

单精度浮点值。单精度是这样的格式,1位符号,8位指数,23位小数。

double

双精度浮点值。双精度是1位符号,11位指数,52位小数。

void表示类型的缺失。
wchar_t宽字符类型。

整型数都按二进制的形式存储;

对浮点数的解释详见:https://blog.csdn.net/zl3090/article/details/84303694

2. 常量

常量包括数值型常量和字符型常量两大类。

常量是固定值,在程序执行期间不会改变。这些固定的值,又叫做字面量

常量可以是任何的基本数据类型,可分为整型数字、浮点数字、字符、字符串和布尔值。

常量就像是常规的变量,只不过常量的值在定义后不能进行修改。

(1)整数常量

整数常量可以是十进制、八进制或十六进制的常量。

a. 前缀指定基数:0x 或 0X 表示十六进制,0 表示八进制,不带前缀则默认表示十进制。

b. 后缀指定类型:后缀是 U 和 L 的组合,U 表示无符号整数(unsigned),L 表示长整数(long)。后缀可以是大写,也可以是小写,U 和 L 的顺序任意。

各种情况的示例如下:

85         // 十进制
0213       // 八进制 
0x4b       // 十六进制 
30         // 整数 
30u        // 无符号整数 
30l        // 长整数 
30ul       // 无符号长整数

(2)浮点数常量

浮点常量由整数部分、小数点、小数部分和指数部分组成。可以使用小数形式或者指数形式来表示浮点常量:

a. 当使用小数形式表示时,必须包含整数部分、小数部分,或同时包含两者;

b. 当使用指数形式表示时, 必须包含小数点、指数,或同时包含两者。带符号的指数是用 e 或 E 引入的。

示例如下:

3.14159       // 合法的 
314159E-5L    // 合法的 
510E          // 非法的:不完整的指数
210f          // 非法的:没有小数或指数
.e55          // 非法的:缺少整数或分数

在程序中,无论写成小数形式还是指数形式,在内存中都是以指数形式(浮点形式)储存的。

(3)布尔常量

布尔常量共有两个,它们都是标准的 C++ 关键字:

  • true 值代表真。
  • false 值代表假。

不应把 true 的值看成 1,把 false 的值看成 0。

(4)字符常量

字符常量是括在单引号中。如果常量以 L(仅当大写时)开头,则表示它是一个宽字符常量(例如 L'x'),此时它必须存储在 wchar_t 类型的变量中。否则,它就是一个窄字符常量(例如 'x'),此时它可以存储在 char 类型的简单变量中。

字符常量是以它的ASCII码存储在内存中的。

a. 普通字符常量:

单引号中放的一个字符就是普通字符常量,如'a'等,在内存中占一个字节,单引号中只能放一个字符,'AB'这种是不合法的,字符常量是区分大小写的,'A'和'a'是两个不同的字符常量。

b. 转义字符常量:

前边带斜杠的字符,表示特殊的含义,如下所示:

转义序列含义
\\\ 字符
\'' 字符
\"" 字符
\?? 字符
\a警报铃声
\b退格键
\f换页符
\n换行符
\r回车
\t水平制表符
\v垂直制表符
\ooo一到三位的八进制数
\xhh . . .一个或多个数字的十六进制数

表中需要说明的是,八进制字符\ooo表示一到三位八进制数组成的字符,ooo均为0-7组成的数字,可以是相同或者不同的数字;例如'\101'表示八进制ASCII码101代表的字符,对应十进制数65,也就是对应字母A;而十六进制字符\xhh表示一到二位十六进制字符组成的字符,x是十六进制数的标志,不能被替换,而hh均为0-F组成的数字,可以相同或者不同。

c. 字符串常量:

用双引号引起来的字符代表字符串常量,此处有别于python中不区分单双引号,c++中单引号只表示单个字符,而双引号表示字符串,需要注意的是,双引号中可以放单个字符,但是此时仍是一个字符串,如"a"与'a'不同,"a"是字符串常量,在内存中占两个字节(包括\0,下边注意中细说),而'a'是字符常量,在内存中占一个字符,也就是说char类型只能定义字符,而不能定义字符串,即:

char c; //定义字符变量,一个字节
c = 'a'; //正确
c = "a"; //错误,不能容纳字符串类型

注意:编译系统会在字符串最后自动加一个'\0'作为结束的标志,它不是字符串的一部分,但是会占据一个字节的内存,如字符串"abc\n",该字符串包含4个字符,分别是a,b,c,\n,编译系统遇到 \ 时,会把它认作转义字符的标志,会将它和它后边的字符一起作为一个转义字符,若与后边的字符不能构成合法的转义字符,则编译会报错;但是该字符串在内存中占据5个字节,是因为最后包含\0字符。若想将 \ 单独作为一个字符,上述字符串应该写作"abc\\n",此时的字符为a,b,c,\,n。

(5)定义常量(符号常量)

在 C++ 中,有两种简单的定义常量的方式:

  • 使用 #define 预处理器。
  • 使用 const 关键字。

实例如下:

#include <iostream>
using namespace std;
 
#define LENGTH 10   //注意,此处不是语句,没有分号
#define WIDTH  5
#define NEWLINE '\n'
 
int main()
{
 
   int area;  
   
   area = LENGTH * WIDTH;
   cout << area;
   cout << NEWLINE;
   return 0;
}

 

#include <iostream>
using namespace std;
 
int main()
{
   const int  LENGTH = 10; //使用 const 前缀声明指定类型的常量
   const int  WIDTH  = 5;
   const char NEWLINE = '\n';
   int area;  
   
   area = LENGTH * WIDTH;
   cout << area;
   cout << NEWLINE;
   return 0;
}

注意:符号常量虽然有名字,但不是变量,编译中只要遇到这个名字,都会被替换成它对应的数值。

#define和const的区别:

#define是定义的符号常量,也就是说它用一个符号代替了一个字符串(或常量),在编译时,系统会把所有的符号常量替换为指定的字符串,它没有类型,在内存中并没有以符号常量命名的存储单元;

而const确切来说定义的是常变量,它具有变量的特征,也就是说具有类型,在内存中存在着以它命名的存储单元,可以用sizeof测出其长度,它与一般变量的区别是变量值不允许修改,也就是只读变量。

 

3. 变量

变量由变量名变量值组成,其中,变量名代表内存中的一个存储单元,在编译中由系统分配一个地址;变量值代表相应存储单元中的数据值。

命名规则

a. 只能由数字、字母或下划线构成;

b. 第一个字符只能是字母或下划线;

c. 大小写字母对应着不同的变量;

d. 变量名不能与c++的关键字、系统函数名和类名相同;

e. 常用驼峰命名法(以小写字母开头,下一个单词首字母大写,如studentName);

f.  命名有长度限制,一般不超过32字符。

变量的定义

数据类型 变量名列表

示例:

float a,b,c,d;

C++对变量使用前先定义的好处

a. 凡是未事先定义的,不作为变量名,可以保证程序中变量名使用正确,避免变量名写错的情况;

b. 为每一变量指定确定的类型,从而分配对应的存储单元;

c. 为每一变量指定确定的类型,编译时可以检查运算是否合法,如%运算要求左右两边都是int型。

 

4.运算符

运算符是一种告诉编译器执行特定的数学或逻辑操作的符号。C++ 内置了丰富的运算符,并提供了以下类型的运算符:

  • 算术运算符
  • 关系运算符
  • 逻辑运算符
  • 位运算符
  • 赋值运算符
  • 杂项运算符

(1)算术运算符

下表显示了 C++ 支持的算术运算符。

假设变量 A 的值为 10,变量 B 的值为 20,则:

运算符描述实例
+把两个操作数相加A + B 将得到 30
-从第一个操作数中减去第二个操作数A - B 将得到 -10
*把两个操作数相乘A * B 将得到 200
/分子除以分母B / A 将得到 2
%取模运算符,整除后的余数B % A 将得到 0
++自增运算符,整数值增加 1A++ 将得到 11
--自减运算符,整数值减少 1A-- 将得到 9

此处需要说明的是:

a. 除法/运算,若两个数都是整数,则结果为整数,小数部分不管超没超过0.5,都会被舍去,负数也一样,原则是向0取整,也就是说取更靠近0的整数,如-5/3 = -1;

b. 取余%运算,要求左右两边都是整数,否则会报错;

c. +、-、*、/中若有一个数是float型的,则结果是double型的,因为运算时会对所有float型的数据按照double型处理。

d. 运算中char和short都会被先转换成int型,float都会被转成double型,其他两个不同类型运算的转换级别是:

int到unsiged到long到double,最终输出什么类型以两个数据中级别高的为准,并且转换并不经过中间步骤,是一步到位的。

e. 自增自减类型:

++i是在i使用前先将i的值加1,再使用;

i++是先使用原始值,再将i加1;

自减同理,示例:

//谭浩强书中习题2-7
#include <iostream>
using namespace std;
int main()
{
    int i,j,m,n;
    i = 8;
    j = 10;
    m = ++i + j++;
    n = (++i) + (j++) + m;
    cout << i << '\t' << j << '\t' << m << '\t' << n << '\t';
    return 0;
}

输出为10,12,19,41,m中自增运算有无括号效果相同。

(2)关系运算符

下表显示了 C++ 支持的关系运算符。

假设变量 A 的值为 10,变量 B 的值为 20,则:

运算符描述实例
==检查两个操作数的值是否相等,如果相等则条件为真。(A == B) 不为真。
!=检查两个操作数的值是否相等,如果不相等则条件为真。(A != B) 为真。
>检查左操作数的值是否大于右操作数的值,如果是则条件为真。(A > B) 不为真。
<检查左操作数的值是否小于右操作数的值,如果是则条件为真。(A < B) 为真。
>=检查左操作数的值是否大于或等于右操作数的值,如果是则条件为真。(A >= B) 不为真。
<=检查左操作数的值是否小于或等于右操作数的值,如果是则条件为真。(A <= B) 为真。

(3)逻辑运算符

下表显示了 C++ 支持的关系逻辑运算符。

假设变量 A 的值为 1,变量 B 的值为 0,则:

运算符描述实例
&&称为逻辑与运算符。如果两个操作数都非零,则条件为真。(A && B) 为假。
||称为逻辑或运算符。如果两个操作数中有任意一个非零,则条件为真。(A || B) 为真。
!称为逻辑非运算符。用来逆转操作数的逻辑状态。如果条件为真则逻辑非运算符将使其为假。!(A && B) 为真。

(4)位运算符

运算符描述实例
&如果同时存在于两个操作数中,二进制 AND 运算符复制一位到结果中。(A & B) 将得到 12,即为 0000 1100
|如果存在于任一操作数中,二进制 OR 运算符复制一位到结果中。(A | B) 将得到 61,即为 0011 1101
^如果存在于其中一个操作数中但不同时存在于两个操作数中,二进制异或运算符复制一位到结果中。(A ^ B) 将得到 49,即为 0011 0001
~二进制补码运算符是一元运算符,具有"翻转"位效果,即0变成1,1变成0。(~A ) 将得到 -61,即为 1100 0011,一个有符号二进制数的补码形式。
<<二进制左移运算符。左操作数的值向左移动右操作数指定的位数。A << 2 将得到 240,即为 1111 0000
>>二进制右移运算符。左操作数的值向右移动右操作数指定的位数。A >> 2 将得到 15,即为 0000 1111

(5)赋值运算符

下表列出了 C++ 支持的赋值运算符:

运算符描述实例
=简单的赋值运算符,把右边操作数的值赋给左边操作数C = A + B 将把 A + B 的值赋给 C
+=加且赋值运算符,把右边操作数加上左边操作数的结果赋值给左边操作数C += A 相当于 C = C + A
-=减且赋值运算符,把左边操作数减去右边操作数的结果赋值给左边操作数C -= A 相当于 C = C - A
*=乘且赋值运算符,把右边操作数乘以左边操作数的结果赋值给左边操作数C *= A 相当于 C = C * A
/=除且赋值运算符,把左边操作数除以右边操作数的结果赋值给左边操作数C /= A 相当于 C = C / A
%=求模且赋值运算符,求两个操作数的模赋值给左边操作数C %= A 相当于 C = C % A
<<=左移且赋值运算符C <<= 2 等同于 C = C << 2
>>=右移且赋值运算符C >>= 2 等同于 C = C >> 2
&=按位与且赋值运算符C &= 2 等同于 C = C & 2
^=按位异或且赋值运算符C ^= 2 等同于 C = C ^ 2
|=按位或且赋值运算符C |= 2 等同于 C = C | 2

(6)杂项运算符

下表列出了 C++ 支持的其他一些重要的运算符。

运算符描述
sizeofsizeof 运算符返回变量的大小。例如,sizeof(a) 将返回 4,其中 a 是整数。
Condition ? X : Y条件运算符。如果 Condition 为真 ? 则值为 X : 否则值为 Y。
,逗号运算符会顺序执行一系列运算。整个逗号表达式的值是以逗号分隔的列表中的最后一个表达式的值。
.(点)和 ->(箭头)成员运算符用于引用类、结构和共用体的成员。
Cast强制转换运算符把一种数据类型转换为另一种数据类型。例如,int(2.2000) 将返回 2。
&指针运算符 & 返回变量的地址。例如 &a; 将给出变量的实际地址。
*指针运算符 * 指向一个变量。例如,*var; 将指向变量 var。

(7) C++ 中的运算符优先级

运算符的优先级确定表达式中项的组合。这会影响到一个表达式如何计算。某些运算符比其他运算符有更高的优先级,例如,乘除运算符具有比加减运算符更高的优先级。

例如 x = 7 + 3 * 2,在这里,x 被赋值为 13,而不是 20,因为运算符 * 具有比 + 更高的优先级,所以首先计算乘法 3*2,然后再加上 7。

下表将按运算符优先级从高到低列出各个运算符,具有较高优先级的运算符出现在表格的上面,具有较低优先级的运算符出现在表格的下面。在表达式中,较高优先级的运算符会优先被计算。

类别 运算符 结合性 
后缀 () [] -> . ++ - -  从左到右 
一元 + - ! ~ ++ - - (type)* & sizeof 从右到左 
乘除 * / % 从左到右 
加减 + - 从左到右 
移位 << >> 从左到右 
关系 < <= > >= 从左到右 
相等 == != 从左到右 
位与 AND 从左到右 
位异或 XOR 从左到右 
位或 OR 从左到右 
逻辑与 AND && 从左到右 
逻辑或 OR || 从左到右 
条件 ?: 从右到左 
赋值 = += -= *= /= %=>>= <<= &= ^= |= 从右到左 
逗号 从左到右 

两个比较典型的例子:

a的初值为12,

a*a = 2+3,结果是60,自右向左结合,先计算2+3;

a +=a-=a*=a,结果是0,自右向左结合,每一步a都变化了,始终是自己跟自己做运算。

 

  • 10
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值