C语言之有符号数和无符号数

我们知道,在C语言中存在无符号数和有符号数(一些高级语言如Java里面是没有无符号数的),但是对于计算机而言,其本身并不区别有符号数和无符号数,因为在计算机里面都是0或者1,但是在我们的实际使用中有时候需要使用有符号数来表示一个整数,因此我们规定,当最高位为1的时,表示为负数,最高位为0时,表示为正数。

1:有符号数和无符号数在数值上的区别。

有符号数的最高位用来表示符号,所以在最大的数值上,有符号数的最大值小于无符号数。以一个字节为例:

有符号数的取值范围为:-128 — 0 — 127

无符号数的取值范围为:0 — 255

2:正数和负数的转换

转换关系为:负数(正数) = 正数(负数)的补码 + 1;

例如:

5  = 0000 0101

-5 = 1111 1011

实际的计算:

最大值 - 当前值 +1;

0xFF -5 +1 = -5(1111 1011)

0xFF -(-5) +1 = 5(0000 0101)

3:正负数在计算机中的存储

在计算机中,并不存在所谓正负,具体看下面的代码

int main(void)
 {
        int x = -1;
         int i = 0;
      unsigned int ux = (unsigned)x;
        for(i = 0;i<32;i++)
         {
                 ux = ux >> i;
                if((ux & 0x01) == 0)
                         printf("%d = 1\r\n",i);
         }

ux = (unsigned) x;

printf("ux = %d \n",ux);

printf("ux = %u \n",ux);

}

运行结果为:

111111111(32个1)

ux = -1

ux = 4294967295

原因是,当我们将 -1 通过强制类型转换赋值给ux时,此时ux变量所对应的地址,所存放的值是-1,也就是0xFFFFFFFF,也就是说,从存储的角度上讲,-1和4294967295在计算机的存储值都是0xFFFFFFFF,关键是你按怎样的方式去解析,

ux = -1;此时我们是按%d也就是有符号整形的方式去解析这个存储空间所对应的值,所以得到的解析结果是-1;

ux = 4294967295,此时我们是按%u也就是无符号整形的方式去解析这个存储空间的值,所以得到的最高位就是数值位,而不是符号位。

解析过程如下,这里假设int类型为一个字节(4个字节也是一样的原理,只是数值更大而已)

                          255                             -1
 位    值      位     值
1 1 1 1 1
2 1 2 1 2
4 1 4 1 4
8 1 8 1 8
16 1 16 1 16
32 1 32 1 32
64 1 64 1 64
128 1 128 1 -128

所以

-1 = 1+2+4+8+16+32+64+(-128)

255 = 1+2+4+8+16+32+64+128

综上所述,计算机中的存储方式并不区分正负,关键在于程序员用什么方式去解析这块存储空间(地址)的值。

时间: 2024-10-12 12:11:31

C语言之有符号数和无符号数的相关文章

C语言中 有符号数、无符号数、整数溢出 (转)

[cpp] view plain copy print? #include<stdio.h> void main() { int l=-1; unsigned int c=135; printf("%u\n",l+c); } 这个的结果134,而不是我之前认为的很大的正数,实际上需要注意的是-1(0xffffffff)被提升为unsigned int后是一个差1就溢出的unsigned int,所以相加后结果是134.但是如果l=-10000,那么结果就真是一个很大的正数了

深入理解计算机系统(2.5)------C语言中的有符号数和无符号数以及扩展和截断数字

上一篇博客我们讲解了计算机中整数的表示,包括无符号编码和补码编码,以及它们之间的互相转换,个人觉得那是非常重要的知识要点.这篇博客我们将介绍C语言中的有符号数和无符号数以及扩展和截断数字. 1.C语言中的有符号数和无符号数 上一篇博客我们给出了C语言中在32位机器和64位机器中支持的整型类型数据,我们这里只给出32位机器上的: 尽管 C 语言标准没有指定有符号数要采用某种编码表示,但是几乎所有的机器都使用补码.通常大多数数字是默认有符号的,比如当声明一个像12345或者0xABC这样的常量的时候

C语言基础(5)-有符号数、无符号数、printf、大小端对齐

1.有符号数和无符号数 有符号数就是最高位为符号位,0代表正数,1代表负数 无符号数最高位不是符号位,而就是数的一部分而已. 1011 1111 0000 1111 1111 0000 1011 1010,如果当有符号数看待,那么他是一个负数的补码,如果当一个无符号数看待,他就是一个正数的原码 无符号数最小为0,不可能是负数 定义一个无符号的int unsigned int a; //使用unsigned int 定义了一个无符号的int变量,名字叫a short,long,long long,

有符号数和无符号数

摘抄自:http://www.cnblogs.com/glacierh/archive/2013/07/16/3194658.html 1.      补码 在计算机中无符号数用原码表示,有符号数用补码表示.w位补码表示的值为: 最高位 也称符号位,1表示负数,0表示正数,符号位为0时,和无符号数的表示是相同的,以下是4位补码的示例: 0101 = -0*23 + 1*22 + 0*21 + 1*20 = 5 1101 = -1*23 + 1*22 + 0*21 + 1*20 = -3 w位的补

原码、反码、补码、有符号数和无符号数运算

原码 原码就是符号位加上真值的绝对值, 即用第一位表示符号, 其余位表示值. 比如如果是8位二进制: [+1]原 = 0000 0001 [-1]原 = 1000 0001 第一位是符号位. 因为第一位是符号位, 所以8位二进制数的取值范围就是: [1111 1111 , 0111 1111] 即 [-127 , 127] 原码是人脑最容易理解和计算的表示方式. 反码 反码的表示方法是: 正数的反码是其本身 负数的反码是在其原码的基础上, 符号位不变,其余各个位取反. [+1] = [00000

有符号数与无符号数比较的坑

前言 在c/c++ 的项目编译时经常会遇到 “comp.c:59:42: warning: comparison between signed and unsigned integer expressions [-Wsign-compare]” 这种错误.作为一个”合格的程序员“ 对这种编译告警,通常的处理是忽略,毕竟大家一致的观点是:只有“warning”不算问题! 下面给出一个小case: #include <stdio.h> int sum_elements(int a[], unsig

有符号数的加减法 和无符号数的加减法,和,系统是如何识别有符号数和无符号数的

一.有符号数的加减法 1.符号数与无符号数的人为规定性: 一个数,是有符号数还是无符号数都是人为规定的.进行二进制运算时用无符号数或是补码运算时,结果都是正确的. 10000100+00001110 若规定为无符号数,即 132+146=146D . 若规定为符号数,则为-124+14=-110,而[-110]补=10010010.解释:10000100是 -124的补码,0001110是14的补码,在机器中运算后得出的结果是[-110]的补码.机器中的有符号数的运算一般就是补码的运算. 2.补

有符号数和无符号数负数(转)

有符号数和无符号数负数 理解有符号数和无符号数负数在计算机中如何表示呢? 这一点,你可能听过两种不同的回答. 一种是教科书,它会告诉你:计算机用"补码"表示负数.可是有关"补码"的概念一说就得一节课,这一些我们需要在第6章中用一章的篇幅讲2进制的一切.再者,用"补码"表示负数,其实一种公式,公式的作用在于告诉你,想得问题的答案,应该如何计算.却并没有告诉你为什么用这个公式就可以和答案? 另一种是一些程序员告诉你的:用二进制数的最高位表示符号,最高

FPGA中的有符号数和无符号数的运算

在FPGA设计中,所有的算数运算符都是按照无符号数进行的.如果要完成有符号数计算,对于加.减操作通过补码处理即可用无符号加法完成.对于乘法操作,无符号数直接采用“*”运算符,有符号数运算可通过定义输出为 signed 来处理. 需要注意的是,尽量不要使有符号数与无符号数进行混合计算.因为只要有一个无符号数的运算单元,整个算式将被将成无符号数进行计算.   总之,正数和负数处理时都是按照补码的形式处理,具体究竟把这些补码理解为符号型还是无符号型,这就要看reg signed这样的声明了.如果声明了