C语言中无符号数和有符号数之间的运算

C语言中有符号数和无符号数进行运算(包括逻辑运算和算术运算)默认会将有符号数看成无符号数进行运算,其中算术运算默认返回无符号数,逻辑运算当然是返回0或1了。

unsigned int和int进行运算

直接看例子来说明问题吧

#include <iostream>

using namespace std;

int main()

{

         int a = -;

         unsigned int b = ;

         if(a > b)

                  cout<<"负数竟然大于正数了!\n";

         return ;

}

输出结果为:

这是因为a和b进行比较的时候,编译器将有符号数a看成了无符号数,然后再和b进行比较,在内存中(32位)

a : 11111111 11111111 11111111 11111111

b : 00000000 00000000 00000000 00010000

看成无符号数,自然是a>b。

#include <iostream>

using namespace std;

int main()

{

         int a = -;

         unsigned int b = ;

         cout<<a + b<<endl;

         int c = -;

         unsigned int d = ;

         cout<<c + d<<endl;

         return ;

}

输出结果为:

可以看到a+b的结果貌似比较正常,但是c+d和我们想象的好像不太一样。其实4294967295就是11111111 1111111 11111111 1111111就是-1在内存中的形式,看成无符号数就是这个结果啦。所以unsigned int和int做运算会将int看成unsigned int,而且结果也是unsigned int。

unsigned char和char进行运算

看一个颠覆上面逻辑的例子:

#include <iostream>

using namespace std;

int main()

{

         char a = -;

         unsigned char b = ;

         if(a > b)

                  cout<<"负数大于正数了!\n";

         cout<<a+b<<endl;

         return ;

}

输出结果:

如果按照上面unsigned int和int进行运算的逻辑,这里unsigned char和char进行运算,那应该是a要看成无符号数,所以a的值比较大呀,而且a+b的结果应该是-2对应的无符号数,也就是254才对呀?

之所以会出现上面的结果是因为,C语言中比int小的整型(包括short 、unsigned short 、 unsigned char和char)在运算中都要转换成int然后进行运算,至于为什么选择转换为int,应该是从效率上考虑的,因为通常情况下int的长度被定义为机器处理效率最高的长度,比如32位机上,一次处理4个字节效率是最高的,所以虽然short(我机器上占2个字节)更节省内存,但是在运算中的效率,是int更高。所以上面,无论是逻辑运算a>b还是算术运算a+b中a和b都默认转换成了int,所以算术运算的结果也是带符号的。

但是还需要注意一个问题就是转换成int类型的时候,高位补齐的问题。

如果是unsigned的类型转换成int类型,高位补0.

如果是signed的类型转换成int类型,如果原来最高位是1则补1,如果是0则补0。

比如:

#include <iostream>

using namespace std;

int main()

{

         char a = -;

         unsigned char b = ;

         char c = ;

         cout<<a+b<<endl;

         cout<<a+c<<endl;

         return ;

}

其中char c = 255,在内存中为11111111,最高位是1,转换成int时高位补1,也就是11111111 11111111 11111111 11111111,就是-1

注意:转换成unsigned int类型时,也是一样的。

比如:

#include <iostream>

using namespace std;

int main()

{

         char a = ;

         unsigned int b = a;

         cout<<b<<endl;

         char c = ;

         unsigned int d = c;

         cout<<d<<endl;

         return ;

}

a在内存中为11111111,最高位是1,转换时高位补1,也就是

11111111 11111111 11111111 11111111,也就是无符号数2^32

c在内存中为00001111,最高位是0,转换高位补0,也就是

00000000 00000000 00000000 00001111,还是15

#include <iostream>

using namespace std;

int main()

{

         unsigned char a = ;

         int b = a;

         cout<<b<<endl;

         unsigned char c = ;

         int d = c;

         cout<<d<<endl;

         return ;

}

对于unsigned char在转换成int或unsigned int时,无论最高位是0还是1,都补0。

对于unsigned short和short进行运算,和char和unsigned char一样,都是要先转换成int,然后再进行运算。

转换时高位补齐的方法也和unsigned char、char一样。

如果你觉得对你有用,请赞一个吧

C语言中无符号数和有符号数之间的运算的更多相关文章

  1. matlab和FPGA中无符号数和有符号数的转化(转)

    在FPGA 设计过程中经常会遇到关于数表示之间的转化问题,最常见的是无符号数和有符号数之间的转化问题.(1)在FPGA设计过程中,能够很直接的看出数字的位宽,但经常以无符号数的形式输出,在后继的处理中 ...

  2. c语言中为什么左移不分符号数无符号数,而右移分呢??

    因为在C语言标准中,只规定了无符号数的移位操作是采用逻辑移位(即左移.右移都是使用的逻辑左移和逻辑右移).而对于有符号数,其左移操作还是逻辑左移,但右移操作是采用逻辑右移还是算术右移就取决于机器了!( ...

  3. C语言-无符号数与有符号数不为人知的秘密

    一.无符号数与有符号数 1.计算机中的符号位 数据类型的最高位用于标识数据的符号 -最高位为1,表明这个数为负数 -最高位为0,表明这个数为正数 #include <stdio.h> in ...

  4. 论C语言中二级指针和二维数组之间的区别

    刚开始学习C语言的时候,觉得一个数组可以定义一个一级指针去访问,想当然的就觉得可以定义一个二级指针去访问二维数组.很显然这是错误的. 我们来看看C语言的数组在内存中的存储方式. 实际上C语言中的数组, ...

  5. C语言中函数和指针的參数传递

    近期写二叉树的数据结构实验.想用一个没有返回值的函数来创建一个树,发现这个树就是建立不起来,那么我就用这个样例讨论一下c语言中指针作为形參的函数中传递中隐藏的东西. 大家知道C++中有引用的概念,两个 ...

  6. 浅谈C#语言中的各种数据类型,与数据类型之间的转换

    什么是数据类型? 数据类型,百度百科是这样解释的:数据类型在数据结构中的定义是一个值的集合以及定义在这个值集上的一组操作.这样的解释对于一个初学者来说未必太过于深奥. 简单点说,数据类型就是不同长度的 ...

  7. C语言中倒序输出你输入的数。

    int n; scanf("%d",&n); while(n>0) { printf("%d",n%10); n/=10;  //其实就是n的自除 ...

  8. R语言算法 ▪ 计算随意输入的两数之间的区域和

    sumfu<-function(a,b,n=){ if(a<b){ for(i in a:b){n=n+i} }else for(i in b:a){n=n+i} return <- ...

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

    #include<stdio.h> void main() { int l=-1; unsigned int c=135; printf("%u\n",l+c); } ...

随机推荐

  1. 浅谈js中如何动态添加表头/表列/表格内容

    我想很多童鞋用js动态向表格中添加数据很熟悉,而且也觉得非常简单!是的,对于写页面的童鞋来说,最喜欢写查询的页面了,动态向表格绑定数据.用for循环就可以轻松搞定. 如果我们的业务需求有所变化,可能我 ...

  2. JS的事件多次触发,只执行最后一次

    有时候我们在JS编程的过程中经常遇到一个问题就是事件频繁高速被触发,利用计时器来控制频率又会丢弃掉有用的事件,我们只是想让程序执行最后一次的事件,那么就可以用如下方法解决问题 //写在事件外边,防止被 ...

  3. SQL Server数据库基础笔记

    启动和停止SQL Server服务三种形式 后台启动服务 计算机->右键->管理->服务和应用程序->服务->sql server(MSSQLSERVER) SQL Se ...

  4. webService 客户端调用及异常信息First Element must contain the local name, Envelope , but found definitions

    报错:“First Element must contain the local name, Envelope , but found definitions”: 原因:EndpointReferen ...

  5. 关于浏览器解析html全过程详解

    本人web前端菜鸟一枚,第一次在这里发博客梳理知识,知识都是从各地方查阅引用以及自己的理解得来,有什么错误的地方欢迎指正. DOM文档通常加载的步骤: 1.解析HTML结构. 2.加载外部脚本和样式表 ...

  6. MFC对话框中显示背景图片

    在MFC对话框中显示图片,四个步骤. 1.首先得在VC6.0或者VS2008(其他版本也是一样)中导入GDI文件.(网上下载:) GDI含义是图形设备接口,主要任务是负责系统与绘图程序之间的信息交换, ...

  7. Android App 压力测试 monkeyrunner

    Android App 压力测试 第一部分 背景 1. 为什么要开展压力测试? 2. 什么时候开展压力测试?第二部分 理论 1. 手工测试场景 2. 自动测试创建 3. Monkey工具 4. ADB ...

  8. Nginx也应用场景小结

    Nginx简介    Nginx一是一款轻量级的.高性能的HTTP和反向代理服务器, 具有很高的稳定性和支持热部署.模块扩展也很容易.当遇到访问的峰值,或者有人恶意发起慢速连接时,也很可能会导致服务器 ...

  9. javascript 实现字符串反转的两种方法

    第一种方法:利用数组方法 //先split将字串变成单字数组,然后reverse()反转,然后将数组拼接回字串 var str = "abcdef"; str.split(&quo ...

  10. 原创:路由配置实践 两个局域网主机的互连 VM linux

    又开始齐天大圣讲课的时间了 我相信网络是每个运维人员和开发人员必不可少要接触的   今天我们要讲的是在VM虚拟机中 我们三台虚拟机划分两个局域网 实现不同局域网的互联 也就是下面图中的AC通过B主机的 ...