前言

1. 数据类型详细介绍

2. 整形在内存中的存储:原码、反码、补码

3. 大小端字节序介绍及判断

4. 浮点型在内存中的存储解析

一、数据类型介绍

1.类型的基本归类

1.整形家族

char
unsigned char
signed char
short
unsigned short [int]
signed short [int]
int
unsigned int
signed int
long
unsigned long [int]
signed long [int]

2.浮点型家族

float
double

3.构造类型

数组类型
结构体类型 struct
枚举类型 enum
联合类型 union

4.指针类型

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

5.空类型

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

二、整型在内存中的存储

1.原码、反码、补码

计算机中的有符号数有三种表示方法,即原码、反码和补码。
三种表示方法均有符号位和数值位两部分,符号位都是用0表示“正”,用1表示“负”,而数值位
三种表示方法各不相同。
原码:
直接将二进制按照正负数的形式翻译成二进制就可以。
反码:
将原码的符号位不变,其他位依次按位取反就可以得到了。

代码如下:

#define _CRT_SECURE_NO_WARNINGS   1
#include<stdio.h>
int main()
{
	int a = 20;                //00000000 00000000 00000000 00010100  20的原码、反码、补码相同

	int b = -10;               //10000000 00000000 00000000 00001010  10的原码
    printf("%d\n", a + b);     //11111111 11111111 11111111 11110101  10的反码

	return 0;                  //11111111 11111111 11111111 11110110  10的补码
}                              //00000000 00000000 00000000 00010100  20的补码
                              //100000000 00000000 00000000 00001010  两者相加,因为int只有32位丢掉最前面的

2.内存中怎样存储


3.大小端字节序

1.大端小端的概念

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

2.大小端字节序的由来

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

3.判断系统是大端还是小端

代码如下:

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

4.整形存储的典型例题

代码如下:

#define _CRT_SECURE_NO_WARNINGS   1
#include<stdio.h>
int main()
{
	char a = -1;                              //10000000 00000000 00000000 000000001  -1的原码
	signed char b = -1;                       //11111111 11111111 11111111 111111110  -1的反码
	unsigned char c = -1;                     //11111111 11111111 11111111 111111111  -1的补码
	printf("a=%d,b=%d,c=%d", a, b, c);        //因为a是字符类型,所以发生截断 111111111
	return 0;                                 //%d打印十进制,所以发生整形提升,按符号位来提升
}                                             //11111111 11111111 11111111 11111111  -1的补码,所以原码为-1
                                              //对于unsigned char来说      11111111 中1不是符号位所以高位补0
                                              //00000000 00000000 00000000 11111111 所以为正数,原、反、补相同为255

代码如下:

int main()
{
	char a = -128;                           //10000000 00000000 00000000 10000000  128的原码
	printf("%u\n", a);                       //11111111 11111111 11111111 01111111  128的反码
}                                            //11111111 11111111 11111111 10000000  128的补码
                                             //因为a是字符类型,所以发生截断 10000000 a的值
                                             //%u是打印无符号整形,所以发成整形提升
                                             //11111111 11111111 11111111 10000000
                                             //%u是无符号整形,所以原码、反码、补码相同

代码如下:

int main()
{
	char a = 128;                           //00000000 00000000 00000000 10000000  128的原码 
	printf("%u\n", a);                      //01111111 11111111 11111111 01111111  128的反码
}                                           //01111111 11111111 11111111 10000000  128的补码
                                            //因为a是char类型,所以发生截断 10000000 -a
                                            //%u是打印无符号整形,所以发生整形提升,因为是char类型,所以高位是符号位
                                            //11111111 11111111 11111111 100000000 
                                            //%u是无符号整形,所以原码、反码、补码相同

代码如下:

int main()
{
	int i = -20;                            //10000000 00000000 00000000 00010100  -20的原码
	unsigned int j = 10;                    //11111111 11111111 11111111 11101011  -20的反码
	printf("%d\n", i + j);                  //11111111 11111111 11111111 11101100  -20的补码
}                                           //00000000 00000000 00000000 00001010  10的原码、反码、补码
                                            //11111111 11111111 11111111 11110110  相加为补码
                                            //11111111 11111111 11111111 11110101   反码
                                            //10000000 00000000 00000000 00001010   补码  为-10

代码如下:

#define _CRT_SECURE_NO_WARNINGS   1
#include<stdio.h>
#include<Windows.h>
int main()
{
	unsigned int i = 0;           //此时打印完9 8 7 6 5 4 3 2 1 0减减为-1是无符号在内存中补码是全部为1,i恒大于0
	for (i = 9; i >= 0; i--)
	{
		printf("%u \n", i);
		Sleep(1000);
	}
	return 0;
}

代码如下:

#define _CRT_SECURE_NO_WARNINGS   1
#include<stdio.h>
#include<Windows.h>
int main()
{
	unsigned int i = 0;           //此时打印完9 8 7 6 5 4 3 2 1 0减减为-1是无符号在内存中补码是全部为1,i恒大于0
	for (i = 9; i >= 0; i--)
	{
		printf("%u \n", i);
		Sleep(1000);
	}
	return 0;
}

代码如下:

unsigned char i = 0;
int main()    
{
	for (i = 0; i <= 255; i++)     //死循环的打印bit,因为unsigned char取值范围是0-255,所以for循环永远成立
	{    
		printf("bit\n");
	}
	return 0;
}

三、浮点型在内存中的存储

1.浮点型怎么存储

根据国际标准IEEE(电气和电子工程协会) 754,任意一个二进制浮点数V可以表示成下面的形式:
1.(-1)^S * M * 2^E
(-1)^s表示符号位,当s=0,V为正数;当s=1,V为负数。
2.M表示有效数字,大于等于1,小于2。
3.2^E表示指数位。
举例来说: 十进制的5.0,写成二进制是 101.0 ,相当于 1.01×2^2 。 那么,按照上面V的格式,可以得出s=0,M=1.01,E=2。
十进制的-5.0,写成二进制是 -101.0 ,相当于 -1.01×2^2 。那么,s=1,M=1.01,E=2。
IEEE 754规定: 对于32位的浮点数,最高的1位是符号位s,接着的8位是指数E,剩下的23位为有效数字M

2.浮点型存储例题

代码如下:

#define _CRT_SECURE_NO_WARNINGS   1
#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;
}

总结

以上就是今天要讲的内容,本文仅仅简单介绍了数据整形和浮点型在内存中的存储,了解这个更能令我们了解数据怎么在内存中存储。另外,如果上述有任何问题,请懂哥指教,不过没关系,主要是自己能坚持,更希望有一起学习的同学可以帮我指正,但是如果可以请温柔一点跟我讲,爱与和平是永远的主题,爱各位了。

到此这篇关于关于C语言中数据在内存中存储的文章就介绍到这了,更多相关C语言数据在内存的存储内容请搜索程序员的世界以前的文章或继续浏览下面的相关文章希望大家以后多多支持程序员的世界!

关于C语言中数据在内存中的存储详解的更多相关文章

  1. C语言递归之汉诺塔和青蛙跳台阶问题

    递归就是一个函数执行过程中调用自己,在c语言中有很多关于递归的经典问题,例如:斐波那契数列问题、汉诺塔问题等,在研究递归问题时我们要注意三点:1.递归的结束条件2.递归在每次进行过程中,都得离条件越来越近3.相邻两次递归调用之间的关联关系汉诺塔问题:有三根杆子A, B, C。A杆上有N个(N >......

  2. C++ 控制台弹出文件管理对话框案例

    在控制台程序中打开文件管理对话框,效果图如下所示:在需要弹出对话框的地方插入以下代码://打开文件管理窗口TCHAR szBuffer[MAX_PATH] = { 0 };OPENFILENAME file = { 0 };file.hwndOwner = NULL;file.lStructSize......

  3. C语言的进制转换及算法实现教程

    1、其他进制转十进制1.1、二进制转十进制转换规程: 从最低位开始,将每个位上的数提取出来,乘以2的(位数-1)次方,然后求和,例如:二进制 1011 = 1*2^0 + 1*2^1 + 0*2^2 + 1*2^3 = 1 + 2 + 0 + 8 = 111.2、八制转十进制转换规则: 从最低位开始......

  4. C/C++内存对齐详解

    1、什么是内存对齐 还是用一个例子带出这个问题,看下面的小程序,理论上,32位系统下,int占4byte,char占一个byte,那么将它们放到一个结构体中应该占4+1=5byte;但是实际上,通过运行程序得到的结果是8 byte,这就是内存对齐所导致的。 //32位系统 #inclu......

  5. C语言中sprintf()函数的用法

    sprintf函数的用法1、该函数包含在stdio.h的头文件中。2、sprintf和平时我们常用的printf函数的功能很相似。sprintf函数打印到字符串中,而printf函数打印输出到屏幕上。sprintf函数在我们完成其他数据类型转换成字符串类型的操作中应用广泛。3、sprintf函数的格......

  6. std::async的使用总结

    C++98标准中并没有线程库的存在,直到C++11中才终于提供了多线程的标准库,提供了管理线程、保护共享数据、线程间同步操作、原子操作等类。多线程库对应的头文件是#include ,类名为std::thread。然而线程毕竟是比较贴近系统的东西,使用起来仍然不是很方便,特别是线程同步及获取线程运行结......

  7. 关于C++中构造函数的常见疑问

    基本概念我们已经知道在定义一个对象时,该对象会根据你传入的参数来调用类中对应的构造函数。同时,在释放这个对象时,会调用类中的析构函数。其中,构造函数有三种,分别是默认构造函数,有参构造函数和拷贝构造函数。在类中,如果我们没有自行定义任何的构造函数,编译器会为我们提供两种构造函数(默认构造函数和拷贝构......

  8. C语言位运算解决只出现一次的数字

    解题所需要的C语言基础知识hello!从现在开始就进入本题解的正式内容了。首先给大家用图解的方式介绍3个C语言位运算的基本操作符 & | ^这些知识对下面的解题都非常重要,一定要熟练掌握,不然等会会有一种“我在哪,我是谁我在干什么”的感觉。只出现一次的数字I题目描述只出现一次的数字给定一个非......

  9. 虚函数表-C++多态的实现原理解析

    参考:http://c.biancheng.net/view/267.html1、说明我们都知道多态指的是父类的指针在运行中指向子类,那么它的实现原理是什么呢?答案是虚函数表在 关于virtual 一文中,我们详细了解了C++多态的使用方式,我们知道没有 virtual 关键子就没法使用多态2、虚函......

  10. std::async的使用总结

    C++98标准中并没有线程库的存在,直到C++11中才终于提供了多线程的标准库,提供了管理线程、保护共享数据、线程间同步操作、原子操作等类。多线程库对应的头文件是#include ,类名为std::thread。然而线程毕竟是比较贴近系统的东西,使用起来仍然不是很方便,特别是线程同步及获取线程运行结......

随机推荐

  1. pytorch实现线性回归以及多元回归

    本文实例为大家分享了pytorch实现线性回归以及多元回归的具体代码,供大家参考,具体内容如下最近在学习pytorch,现在把学习的代码放在这里,下面是github链接直接附上github代码# 实现一个线性回归# 所有的层结构和损失函数都来自于 torch.nn# torch.optim 是一个实......

  2. 理解C#中的 async await

    前言一个老掉牙的话题,园子里的相关优秀文章已经有很多了,我写这篇文章完全是想以自己的思维方式来谈一谈自己的理解。(PS:文中涉及到了大量反编译源码,需要静下心来细细品味)从简单开始为了更容易理解这个问题,我们举一个简单的例子:用异步的方式在控制台上分两步输出“Hello World!”,我这边使用的......

  3. 【函数分享】每日PHP函数分享

    str_pad() 使用另一个字符串填充字符串为指定长度 。string str_pad ( string $input, int $pad_length[, string $pad_string=" "[, int $pad_type=STR_PAD_RIGHT]])参数描......

  4. .NET 5 程序高级调试-WinDbg

    上周和大家分享了.NET 5开源工作流框架elsa,程序跑起来后,想看一下后台线程的执行情况。抓了个进程Dump后,使用WinDbg调试,加载SOS调试器扩展,结果无法正常使用了:0:000> .loadby sos clrUnable to find module 'clr'这引起了个人的兴......

  5. 翻译 - ASP.NET Core 基本知识 - Web 主机 (Web Host)

    翻译自 https://docs.microsoft.com/en-us/aspnet/core/fundamentals/host/web-host?view=aspnetcore-5.0ASP.NET Core 应用程序配置和启动一个 Host。Host 负责应用程序的启动和生命周期的管理。至少......

  6. Android事件分发机制四:学了事件分发有什么用?

    前言Android事件分发机制已经来到第四篇了,在前三篇中:Android事件分发机制一:事件是如何到达activity的? : 从window机制出发分析了事件分发的整体流程,以及事件分发的真正起点Android事件分发机制二:viewGroup与view对事件的处理 : 源码分析了viewGro......

  7. Java中的CPU占用高和内存占用高的问题排查

    下面通过模拟实例分析排查Java应用程序CPU和内存占用过高的过程。如果是Java面试,这2个问题在面试过程中出现的概率很高,所以我打算在这里好好总结一下。1、Java CPU过高的问题排查举个例子,如下:package com.classloading;public class Test {sta......

  8. Vue启动index.html如何被渲染的

    第一步:cmd进入项目文件里,运行npm run serve 启动项目 这里说明启动端口号是8080第二步:往页面输入:localhost:8081二、解析渲染步骤 先看整体框架样式和index.html:<!DOCTYPE html><html lang......

  9. php中PHPUnit框架实例用法

    本期给大家带来的框架是一个单元测试内容,本身不是很难的东西,大家觉得困难,可能是把自己框在了技术上,事实上,并非如此,我们没有设置好一个项目,不一定是自己的技术不达标,更多可能也许是对一个项目时间的安排的衡量和考虑并不全面,下面就大家遇到的一些问题,给大家编写以下内容,供大家参考。安装:compos......

  10. Python中的内部对象

    目录简介内置函数内置常量内置类型逻辑值检测逻辑值的布尔运算比较运算数字类型整数类型的位运算整数类型的附加方法浮点类型的附加方法迭代器序列类型集合类型映射类型字典视图对象内置异常简介Python中内置了很多非常有用的对象,本文将会介绍Python中的内置函数,内置常量,内置类型和内置异常。内置函数Py......