【C语言进阶】数据的存储----整型篇

🍁 博客主页:江池俊的博客

💫收录专栏:C语言——探索高效编程的基石
💻 其他专栏:数据结构探索
💡代码仓库:江池俊的代码仓库
🎪 社区:GeekHub

🍁 如果觉得博主的文章还不错的话,请点赞👍收藏🌟 三连支持一下博主💞

🌴数据类型介绍

- 基本的内置类型:

1. char1个字节          ----->字符数据类型
2. short 2个字节       -----> 短整型
3. int 4个字节        ----->整形
4. long 4/8个字节        ----->长整型
5. long long8个字节     -----> 更长的整形
6. float4个字节        -----> 单精度浮点数
7. double8个字节          -----> 双精度浮点数
注意C语言中没有字符串类型
类型的意义:

  1. 内存占用和范围:不同的数据类型在内存中占用不同的字节数,并且能够表示的数值范围也不同。正确选择数据类型可以优化内存使用,同时确保数据不会超出所允许的范围,避免溢出和精度丢失的问题。
  2. 数据操作和运算规则:不同数据类型支持不同的操作,并且对于不同数据类型之间的运算有明确的规则。了解这些规则可以帮助开发者正确地处理数据,避免数据转换错误和意外的计算结果。
  3. 类型检查和数据表达:C语言是一种静态类型语言,编译器在编译时会对数据类型进行检查。正确定义数据类型可以帮助发现类型错误和潜在问题,增强代码的可靠性。此外,不同的数据类型也能够更直观地表达数据,使代码更易读和理解。

📌类型的基本归类:

  • 整形家族:
  1. char
    unsigned char:无符号字符型,范围为 0 到 255。
    signed char:一个字节大小的有符号整数,范围为 -128 到 127。
    注意:char是否为signed char,C语言标准并没有规定,取决于编译器。
    字符在内存中存储的是字符的ASCII码值,ASCII码值是整型,所以字符类型归类到整型家族
  2. short
    unsigned short [int]:无符号短整型,范围为 0 到 65,535。
    signed short [int]:两个字节大小的有符号整数,范围为 -32,768 到 32,767。
  3. int
    unsigned int:无符号整型,范围为 0 到 4,294,967,295。
    signed int:四个字节大小的有符号整数,范围为 -2,147,483,648 到 2,147,483,647。
  4. long
    unsigned long [int]:无符号长整型,范围取决于编译器,但至少和 unsigned int 类型的范围相同。
    signed long [int]:四个或八个字节大小的有符号整数,范围取决于编译器,但至少是 int 类型的范围。

以下以char的两种类型为例(其他类型类似),我们可以利用画圈的方法来方便我们的记忆,我个人让我这是一个非常好的方法(牛🖊)具体参照下方两幅图片:
sigend charl类型的范围unsigned char类型的范围

  • 浮点数家族:
  1. float::float是单精度浮点类型,通常占用4个字节(32位)。它可以表示大约6到7位有效数字,并且具有范围约为1.2E-38到3.4E+38的取值范围。在表示浮点数时,通常使用后缀f或F来标识float类型,例如:3.14f。
  2. double:double是双精度浮点类型,通常占用8个字节(64位)。它可以表示大约15到16位有效数字,并且具有更大的范围,约为2.3E-308到1.7E+308。在表示浮点数时,C语言通常默认为double类型,不需要显式指定后缀。例如:3.14表示的是double类型。
  • 构造类型:
  1. 数组类型
  2. 结构体类型 struct
  3. 枚举类型 enum
  4. 联合类型 union
  • 指针类型:

1. int *pi;
2. char *pc;
3. float* pf;
4. void* pv;(无具体类型的指针)

  • 空类型:

void 表示空类型(无类型)
通常应用于函数的返回类型、函数的参数、指针类型。

🌴整形在内存中的存储

我们之前讲过一个变量的创建是要在内存中开辟空间的。空间的大小是根据不同的类型而决定的。

那接下来我们谈谈数据在所开辟内存中到底是如何存储的?

比如:

int a = 20;
int b = -10;

我们知道为 a 分配四个字节的空间。
那如何存储?
下来了解下面的概念:

📌原码、反码、补码

计算机中的整数有三种2进制表示方法,即原码、反码和补码。
三种表示方法均有符号位和数值位两部分,符号位都是用0表示“正”,用1表示“负”,数值位
正数的原、反、补码都相同。
负整数的三种表示方法各不相同。

在我上一篇文章中也有涉及到,具体请移步【C语言】操作符—详解

  • 原码
    直接将数值按照正负数的形式翻译成二进制就可以得到原码。
  • 反码
    将原码的符号位不变,其他位依次按位取反就可以得到反码。
  • 补码
    反码+1就得到补码。

注意:补码+1得到的是原码。

对于整形来说:数据存放内存中其实存放的是补码。
为什么呢?

在计算机系统中,数值一律用补码来表示和存储。原因在于,使用补码,可以将符号位和数值域统一处理;
同时,加法和减法也可以统一处理(CPU只有加法器)此外,补码与原码相互转换,其运算过程 是相同的,
不需要额外的硬件电路。

我们看看在内存中的存储:
注意:这里是各个变量对应补码的16进制表示形式
在这里插入图片描述
我们可以看到对于a和b分别存储的是补码。但是我们发现顺序有点不对劲。
这是因为数据的存储被分为大小端两种情况。

📌大小端介绍

1. 什么是大端小端:

- 大端(存储)模式,是指数据的低位保存在内存的高地址中,而数据的高位,保存在内存的低地址中。
- 小端(存储)模式,是指数据的低位保存在内存的低地址中,而数据的高位,保存在内存的高地址中。

2.为什么有大端和小端:

为什么会有大小端模式之分呢?这是因为在计算机系统中,我们是以字节为单位的,每个地址单元
都对应着一个字节,一个字节为8 bit。但是在C语言中除了8 bit的char之外,还有16 bit的short
型,32 bit的long型(要看具体的编译器),另外,对于位数大于8位的处理器,例如16位或者32
位的处理器,由于寄存器宽度大于一个字节,那么必然存在着一个如何将多个字节安排的问题。因
此就导致了大端存储模式和小端存储模式。
例如:一个 16bit 的 short 型 x ,在内存中的地址为 0x0010 , x 的值为 0x1122 ,那么 0x11 为
高字节, 0x22 为低字节。对于大端模式,就将 0x11 放在低地址中,即 0x0010 中, 0x22 放在高
地址中,即 0x0011 中。小端模式,刚好相反。我们常用的 X86 结构是小端模式,而 KEIL C51 则
为大端模式。很多的ARM,DSP都为小端模式。有些ARM处理器还可以由硬件来选择是大端模式
还是小端模式。

🌴练习

📌1.百度2015年系统工程师笔试题:

题目描述:请简述大端字节序和小端字节序的概念,设计一个小程序来判断当前机器的字节序。(10分)

思路
因为1的二进制序列的16进制表示形式为00 00 00 01,所以如果机器是小端字节序存储形式,那么整型 1 在内存中存储地址由低到高依次为 01 00 00 00;如果机器是大端字节序存储形式,那么在整型 1 在内存中存储地址由低到高依次为 00 00 00 01,这时我们可以利用字符(cha)型指针访问此地址存储的值,因为char型指针一次只能访问一个字节,所以如果是小端,则访问的值为01,如果是打端,则访问的值是00。

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

📌2.练习2

//下面程序输出什么?
#include <stdio.h>
int main()
{char a= -1;//10000000000000000000000000000001 -a的原码//11111111111111111111111111111110//11111111111111111111111111111111 -a的补码//11111111 - asigned char b=-1;//11111111 - bunsigned char c=-1;//11111111 - cprintf("a=%d,b=%d,c=%d",a,b,c);//结果为a=-1 b=-1 c=255//%d 是以10进制的形式打印有符号的整数//打印时a,b为有符号的char,所以打印时将第一个1看成符号位,在左边补上24个1//所以打印时a,b又变成11111111111111111111111111111111,但这是补码,//打印的数字需要为原码对应的值,所以转换为原码为 //10000000000000000000000000000001,对应的十进制为-1//而c为无符号的char,所以打印时不分符号位,左边补上24个0//即为00000000000000000000000011111111,//因为无符号为正数,所以原码补码相同,对应的十进制为255 return 0;
}

📌3.练习3

#include <stdio.h>
int main()
{char a = -128;//10000000000000000000000010000000 -a的原码//11111111111111111111111101111111//11111111111111111111111110000000 -a的补码//10000000 - 内存中存储的a//打印时需要整型提升,此时看a的类型,因为a的类型为有符号的char,//所以将内存中a的首位看成符号位,为1则左边全补上1;为0,则全补0//即:11111111111111111111111110000000 ---将它看成无符号的数打印,//则它是一个很大的数,值为:4294967168printf("%u\n",a);//结果为:4294967168//%u 是10进制的形式,打印无符号的整数//%d 是10进制的形式,打印有符号的整数return 0;
}
#include <stdio.h>
int main()
{char a = 128;//00000000000000000000000010000000 -a的原码,反码,补码//10000000 -内存中存储的a//11111111111111111111111110000000 - a整型提升后的值printf("%u\n",a);//结果为:4294967168//%u 是10进制的形式,打印无符号的整数//%d 是10进制的形式,打印有符号的整数return 0;
}

📌4.练习4

int i= -20;
//10000000000000000000000000010100 -i的原码
//11111111111111111111111111101011
//11111111111111111111111111101100 -i的补码
unsigned  int  j = 10;
//   00000000000000000000000000001010 -j的原码,反码,补码
// + 11111111111111111111111111101100
// = 11111111111111111111111111110110 -i+j的补码
//   10000000000000000000000000001001 -i+j的反码
//	 10000000000000000000000000001010 -i+j的原码,对应十进制值为:-10
printf("%d\n", i+j); //结果为:-10
//按照补码的形式进行运算,最后格式化成为有符号整数

📌5.练习5

unsigned int i;//因为是无符号的整型,所以i一直大于等于于0
for(i = 9; i >= 0; i--)
{printf("%u\n",i);
}//所以此代码会进入死循环

如图说示,i=9开始,逆时针走,一直在0~65535这个范围内。
unsigned int类型的范围

📌6.练习6

int main()
{char a[1000];int i;for(i=0; i<1000; i++){a[i] = -1-i;//这里也可以使用画圈的方法,数组中的值从-1开始一直到0,如下://-1 -2 -3 -4 -5 .. -128 127 .. 6 5 4 3 2 1 0  //-1~-128一个有:128个数,127~1一共有:127个数,所以共有:128 + 127 = 255}printf("%d",strlen(a));//strlen 统计的是\0之前出现的字符的个数//'\0' - 对应的ASCII码值:0 return 0;
}

📌7.练习7

#include <stdio.h>
unsigned char i = 0;
//无符号的插入取值范围在0~255之间
int main()
{for(i = 0;i<=255;i++)//i的值从0到255,然后再从255到0,一直这样循环{printf("hello world\n");//陷入死循环}return 0;
}

🔥今天的分享就到这里,如果觉得博主的文章还不错的话,请👍三连支持一下博主哦🤞
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/21896.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Liunx环境下git的详细使用(gitee版)

Liunx环境下git的详细使用&#xff08;gitee版&#xff09; 1.git是什么2.git操作2.1在gitee创建一个仓库2.2.gitignore2.3.git 3.git三板斧3.1add3.2 commit3.3push 4.git其他命令4.1查看当前仓库状态4.2查看提交日志4.3修改git里面文件名称4.4删除文件4.5修改远端仓库内容 1.…

postgresql表膨胀处理之pgcompacttable部署及使用

环境&#xff1a; 1&#xff09;redhat-release&#xff1a;CentOS Linux release 7.6.1810 (Core) 2&#xff09;database version&#xff1a;postgresql 14.6 一、添加pgstattuple pgcompacttable工具使用过程中需要依赖pgstattuple&#xff0c;因此需先添加pgstattuple…

【SEO基础】百度权重是什么意思及网站关键词应该怎么选?

百度权重是什么意思及网站关键词应该怎么选&#xff1f; 正文共&#xff1a;3253字 20图 预计阅读时间&#xff1a;9分钟 ​ 1.什么是网站权重&#xff1f; 这段时间和一些朋友聊到网站权重以及关键词&#xff0c;发现蛮多人对于这两个概念的认知还是存在一些错误的&#xf…

数组的使用(逆序、冒泡)

内存连续数据类型相同从0开始索引 找出数组中的最大值 #include <iostream> #include <stdlib.h> //随机数所在文件 using namespace std;int main() {int arr[5]{104,134,145,129,89};//初始化没有填的为0 int max0;for(int i0;i<5;i){if(arr[i]>max){ma…

203. 移除链表元素

203. 移除链表元素 题目方法1递归方法2迭代 题目 给你一个链表的头节点 head 和一个整数 val &#xff0c;请你删除链表中所有满足 Node.val val 的节点&#xff0c;并返回 新的头节点 。 方法1递归 class Solution { public:ListNode* removeElements(ListNode* head, in…

RabbitMQ快速入门

文章目录 1、RabbitMQ的概述1.1、什么是消息队列&#xff1f;1.2、为什么要使用消息队列&#xff1f;1.3、RabbitMQ的特点&#xff1a; 2、RabbitMQ的安装2.1 下载与安装2.2 常用命令 3、RabbitMQ消息发送和接受3.1 消息发送和接受机制3.2 AMQP的消息路由3.3 Exchange(交换机)的…

Datax 数据同步-使用总结(一)

1&#xff0c;实时同步&#xff1f; datax 通常做离线数据同步使用。 目前能想到的方案 利用 linux 的定时任务时间戳的方式做增量同步。 2&#xff0c;同步速度快不快&#xff1f; 单表同步速度还是挺快的 但是如果遇到复杂的 sql 查询&#xff0c;其同步效率&#xff0c…

中国信通院发布《高质量数字化转型产品及服务全景图(2023)》

2023年7月27日&#xff0c;由中国信息通信研究院主办的2023数字生态发展大会暨中国信通院铸基计划年中会议在北京成功召开。 本次大会发布了中国信通院《高质量数字化转型产品及服务全景图&#xff08;2023&#xff09;》&#xff0c;中新赛克海睿思受邀出席本次大会并成功入选…

viewerjs 如何新增下载图片功能(npm包补丁)

文章目录 先实现正常的效果实现下载图片改变viewerjs的build函数源码改变之后&#xff0c;执行npm i 之后node_modules源码又变回了原样 1、viwerjs所有功能都很完善&#xff0c;但唯独缺少了图片的下载 2、需求&#xff1a;在用viwerjs旋转图片后&#xff0c;可以直接下载旋转…

Dockerfile构建Tomcat镜像

准备apache包和jdk并解压 [rootlocalhost tomcat]# ll 总用量 196728 -rw-r--r--. 1 root root 9690027 7月 17 2020 apache-tomcat-8.5.40.tar.gz -rw-r--r--. 1 root root 674 8月 2 20:19 Dockerfile -rw-r--r--. 1 root root 191753373 7月 17 2020 jdk-8u191-…

DAY3,C高级(shell中的变量、数组、算术运算、分支结构)

1.整理思维导图&#xff1b; 2.判断家目录下&#xff0c;普通文件的个数和目录文件的个数&#xff1b; 1 #!/bin/bash2 arr1(ls -la ~/ | cut -d r -f 1 | grep -w -)3 arr2(ls -la ~/ | cut -d r -f 1 | grep -w d)4 echo "普通文件个数&#xff1a;${#arr1[*]}"5 e…

android Android Studio Giraffe | 2022.3.1 版本Lombok不兼容 解决方案

android Android Studio Giraffe | 2022.3.1 版本Lombok不兼容 解决方案 1.查看当前的android studio 版本 Android Studio Giraffe | 2022.3.1 Build #AI-223.8836.35.2231.10406996, built on June 29, 2023 2.打开 idea 官网下载页面 idea下载历史版本 找到对应的版本编号…

docker-compose --version报错

在部署docker-compose后&#xff0c;查看版本时有如下报错: 解决方法: 解决方法&#xff1a; 直接在release中下载对应的linux发行版【docker-compose-linux-x86_64】 https://github.com/docker/compose/releases/tag/v2.18.1 下载完后将软件上传至 Linux的【/usr/local/bin】…

el-table点击表格某一行添加到URL参数,访问带参URL加载表格内容并滚动到选中行位置 [Vue3] [Element-plus 2.3]

写在最前 需求&#xff1a;有个表格列出了一些行数据&#xff0c;每个行数据点击后会加载出对应的详细数据&#xff0c;想要在点击了某一行后&#xff0c;能够将该点击反应到URL中&#xff0c;这样我复制这个URL发给其他人&#xff0c;他们打开时也能看到同样的行数据。 url会根…

【Leetcode】二叉树的最近公共祖先,二叉搜索树转换成排好序的双向链表,前序遍历与中序遍历构造二叉树

一.二叉树的最近公共祖先 链接 二叉树的最近公共祖先 题目再现 『Ⅰ』思路一&#xff1a;转换成相交链表问题 观察上图&#xff0c;节点1和节点4的最近公共祖先是3&#xff0c;这是不是很像相交链表的问题&#xff0c;关于相交链表&#xff0c;曾经我在另一篇文章里写到过&a…

面试热题100(二叉树的右视图)

给定一个二叉树的 根节点 root&#xff0c;想象自己站在它的右侧&#xff0c;按照从顶部到底部的顺序&#xff0c;返回从右侧所能看到的节点值。 树这类问题用的最多的就是递归&#xff0c;因为树具有天然的递归结构&#xff1a; 我们来分析一下题目&#xff0c;给定一棵树根结…

pytorch中torch.einsum函数的详细计算过程图解

第一次见到 rel_h torch.einsum(“bhwc,hkc->bhwk”, r_q, Rh)这行代码时&#xff0c;属实是懵了&#xff0c;网上找了很多博主的介绍&#xff0c;但都没有详细的说明函数内部的计算过程&#xff0c;看得我是一头雾水&#xff0c;只知道计算结果的维度是如何变化的&#xf…

SpringCloud之微服务API网关Gateway介绍

文章目录 1 微服务API网关Gateway1.1 网关简介1.2 Spring Cloud Gateway介绍1.3 Gateway特性1.4 Gateway核心概念1.4.1 路由1.4.1.1 定义1.4.1.2 动态路由 1.4.2 断言1.4.2.1 默认断言1.4.2.2 自定义Predicate 1.4.3 过滤器1.4.3.1 默认过滤器1.4.3.2 自定义Filter&#xff08;…

C++ 用vector创建数组对象

C标准库提供了被封装的动态数组——vector&#xff0c;而且这种被封装的数组可以具有各种类型&#xff0c;这就使我们免去了一些重复性工作。 vector不是一类&#xff0c;而是一个类模板。 1. vector定义动态数组的形式为 vector<元素类型>数组对象名(数组长度);尖括号…

【C语言初阶】指针篇—下

目录 4. 指针运算4.1 指针-整数4.2 指针-指针4.3 指针的关系运算 5. 指针和数组6. 二级指针7. 指针数组 C语言初阶—指针上 点击跳转 4. 指针运算 指针 整数指针-指针指针的关系运算 4.1 指针整数 #define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h>int main() {in…