运行环境:CentOS release 5.8 (Final)

#include<stdio.h>
#include<iostream>
using namespace std;
int main()
{
unsigned short u = ;
unsigned int n = ; //Style left:
// u = u - 11;
// n = u; //Style right:
n = u - ; cout<<"n=="<<n<<endl;
return ;
}

运行结果:

Style left :

$ ./a.out

n==65535

Style right :

$ ./a.out

n==4294967295

下面是分别对这两种方式的汇编:

unsigned 变量先减(sub),后赋值给n                                                                      unsigned 变量先扩展,后进行减运算,最后赋值给非 unsigned 变量

注:movzx,汇编语言数据传送指令MOV的变体。无符号扩展,并传送。

movzx 主要作用是,把一个变量进行编译器扩展,放到寄存器eax

1个二进制位称为1个bit(比特位)                   1 bit

8个bit称为1个Byte(字节)                              1 Byte == 8 bit

2个Byte就是1个 Word(机器字)                    1 Word == 2 Byte

2个Word就是1个 DWord(机器双字)           1 DWord == 2 Word

WORD 表示2个字节,即 sizeof(unsigned short)。unsigned short 表示的范围:[0, 65536)

DWORD 表示4个字节,sizeof(unsigned int)。unsigned int 表示的范围:[0, 4294967296)

(一)左侧的方式:先运算后赋值分两步进

//Style left:
u = u - ;
n = u;

对应的汇编为:

sub      WORD PTR [ ebp-0xc ], 0xb
movzx DOWRD PTR [ ebp-0x8 ], eax

【解释】:

因为第一步是对 unsigned 变量 u 进行自减,所以此时不需要进行编译器扩展

10 – 11 得到值为 -1,即 0xffff (unsigned short 能表示的最大范围就是4个f)。

第二步涉及到了赋值操作,

编译器要会赋值之前,先对赋值操作符=右侧的表达式进行编译器扩展:

将 0xffff 扩展为 0x0000ffff,然后接受赋值的变量是一个 DWORD,

根据 DWORD 的长度对扩展的值进行截取,DWORD 长度刚好能接受 0x0000ffff,

转换成 10 进制即为 65535

【结论】:

编译器先进行计算后,按 WORD 进行扩展,所以得到的是 WORD 范围的最大值(65535)

(二)右侧的方式:运算和赋值合为一步进行

//Style right:
n = u - ;

对应的汇编为:

movzx eax, WORD PTR [ ebp-0xc ]
sub eax, 0xb

由于赋值符右侧的表示范围和传进去的变量不是一种类型,编译器先对赋值符=右侧的值进行扩展,

10 – 11 == -1,编译器扩展为 0x0000ffffffff(只能扩展到DWORD?)

右侧的 DWORD 接收后,转换为 10 进制即为 4294967295

【结论】:

编译器先进行计算后,按 DWORD 进行扩展,所以得到的是 DWORD 范围的最大值(4294967295)

关于 unsigned 型变量在计算过程中发生的事情的更多相关文章

  1. 从输入 URL 到浏览器接收的过程中发生了什么事情

    从输入 URL 到浏览器接收的过程中发生了什么事情? 原文:http://www.codeceo.com/article/url-cpu-broswer.html 从触屏到 CPU  首先是「输入 U ...

  2. 【转】SQL Server -- 已成功与服务器建立连接,但是在登录过程中发生错误

    SQL Server -- 已成功与服务器建立连接,但是在登录过程中发生错误 最近在VS2013上连接远程数据库时,突然连接不上,在跑MSTest下跑的时候,QTAgent32 crash.换成IIS ...

  3. Sql server2012连接Sql server 2008时出现的问题:已成功与服务器建立连接,但在登陆过程中发生错误。(provider:SSL Provider,error:0-接收到的消息异常,或格式不正确。)

    以前连接是正常的,就这两天连不上了.(没有耐心的直接看末尾解决办法) 错误消息如下: 1.尝试读取或写入受保护的内存.这通常指示其他内存已损坏.(System.Data) 2.已成功与服务器建立连接, ...

  4. 从输入 URL 到浏览器接收的过程中发生了什么事情?

    从输入 URL 到浏览器接收的过程中发生了什么事情? What really happens when you navigate to a URL 上面两篇文章都解读的很好,值得阅读. 接下来在总结一 ...

  5. 已成功与服务器建立连接,但是在登录过程中发生错误。 (provider: SSL Provider, error: 0 - 接收到的消息异常,或格式不正确。)

    之前做好的asp.net部署后,发现 访问数据库时: 异常:已捕获: "已成功与服务器建立连接,但是在登录过程中发生错误. (provider: SSL Provider, error: 0 ...

  6. vs 或 Sql server2012连接Sql server时出现的问题:已成功与服务器建立连接,但在登陆过程中发生错误

    以前连接是正常的,就这两天连不上了.(没有耐心的直接看末尾解决办法) 错误消息如下: 1.尝试读取或写入受保护的内存.这通常指示其他内存已损坏.(System.Data) 2.已成功与服务器建立连接, ...

  7. SQL Server(解决问题)已成功与服务器建立连接,但是在登录过程中发生错误。(provider: Shared Memory Provider, error:0 - 管道的另一端上无任何进程。

    http://blog.csdn.net/github_35160620/article/details/52676416 如果你在使用新创建的 SQL Server 用户名和密码 对数据库进行连接的 ...

  8. CASE:DB shutdown/open 过程中发生异常导致JOB不能自动执行

    CASE:DB shutdown/open 过程中发生异常导致JOB不能自动执行 现象: 一个DB中的所有JOB在3月25日之后就不再自动运行,查询DBA_JOBS,发现LAST_DATE定格在3月2 ...

  9. sql server 2008启动时:已成功与服务器建立连接,但是在登录过程中发生错误。(provider:命名管道提供程序,error:0-管道的另一端上无任何进程。)(Microsoft SQL Server,错误:233) 然后再连接:错误:18456

    问题:sql server 2008启动时:已成功与服务器建立连接,但是在登录过程中发生错误.(provider:命名管道提供程序,error:0-管道的另一端上无任何进程.)(Microsoft S ...

随机推荐

  1. phpstorm 设置Utf8编码

    点击file 再点击setting 找到file Encoding gbk改成utf-8就ok了

  2. JS存取Cookie值

    一:存Cookie //存Cookie document.cookie = "id=" + escape(value); 二:取Cookie //提取Cookie值 functio ...

  3. centos7 安装redis 开机启动

    redis 下载 https://redis.io/download wget http://download.redis.io/releases/redis-3.2.6.tar.gz 解压缩 .ta ...

  4. java 重载规则

    首先看Java重载的规则: 1.必须具有不同的参数列表: 2.可以有不同的返回类型,只要参数列表不同就可以: 3.可以有不同的访问修饰符: 4.可以抛出不同的异常: 5.方法能够在一个类中或者在一个子 ...

  5. 数据挖掘算法(一)C4.5

    统计了14天的气象数据D(指标包括outlook,temperature,humidity,windy),并已知这些天气是否打球(play).如果给出新一天的气象指标数据:sunny,cool,hig ...

  6. MySQL使用用户变量更新分组排序

    第一个需求是根据A字段进行排序,排序结果更新到B字段 简单搜索之后,很快得到答案 http://dev.mysql.com/doc/refman/5.7/en/update.html ; ) ORDE ...

  7. oracle 配置监听 windows下 oracle 12c

    1, 在命令行窗口中 输入 hostname 查看主机名 2 查看本机ip地址 ,输入 ipconfig 本地连接 下面的 IPv4 地址 192.168.1.1053,测试连接 输入 ping lo ...

  8. Android百度地图附加搜索和公交路线方案搜索

    合肥程序员群:49313181.    合肥实名程序员群:128131462 (不愿透露姓名和信息者勿加入) Q  Q:408365330     E-Mail:egojit@qq.com 综述: 今 ...

  9. 课堂作业二 PAT1025 反转链表

    MyGitHub 终于~奔溃了无数次后,看到这个结果 ,感动得不要不要的::>_<:: 题目在这里 题目简述:该题可大致分为 输入链表 -> 链表节点反转 -> 两个步骤 输入 ...

  10. SQL Server数据库备份:通过Windows批处理命令执行

    通过Windows批处理命令执行SQL Server数据库备份 建立mybackup.bat ,输入以下内容直接运行该脚本,即可开始自动备份数据库也可把该脚本加入windows任务计划里执行. --- ...