二进制有符号分别以%d和%u形式输出

CodeMoggy 2017-12-12 12:32:50
#include<stdio.h> 
int main()
{
signed char a=-1;
printf("%d %u",a,a);
return 0;
}

分别输出-1 4294967295这是为什么?
我知道char 型-1的补码是1111 1111,提升为整形是1111 1111 1111 1111 1111 1111 1111 1111即32个1,以无符号%u输出是2的32次方减一,但以%d输出为什么是-1,从二进制角度讲怎么回事?
...全文
627 9 打赏 收藏 转发到动态 举报
写回复
用AI写文章
9 条回复
切换为时间正序
请发表友善的回复…
发表回复
CodeMoggy 2017-12-12
  • 打赏
  • 举报
回复
我懂了,真正重要的是补码遇到%d输出时看符号位是否为1,为1就为负,就的把这个补码转成原码读取,就是转成1000 0000 0000 0000 0000 0000 0000 0001 ,即负一,而遇到%u就不看符号位,把符号位当作有效值直接读取,即直接读取1111 1111 1111 1111 1111 1111 1111 1111 ,终于懂了!
大米粥哥哥 2017-12-12
  • 打赏
  • 举报
回复
引用 5 楼 风继续吹yc的回复:
那1111 1111 1111 1111 1111 1111 1111 1111 也是补码,那为啥到%u就直接读取了?
百度 补码的概念 ........
赵4老师 2017-12-12
  • 打赏
  • 举报
回复
http://edu.csdn.net/course/detail/2344 C语言指针与汇编内存地址-一.代码要素
赵4老师 2017-12-12
  • 打赏
  • 举报
回复
电脑内存或文件内容或传输内容只是一个一维二进制字节数组及其对应的二进制地址; 人脑才将电脑内存或文件内容或传输内容中的这个一维二进制字节数组及其对应的二进制地址的某些部分看成是整数、有符号数/无符号数、浮点数、复数、英文字母、阿拉伯数字、中文/韩文/法文……字符/字符串、汇编指令、函数、函数参数、堆、栈、数组、指针、数组指针、指针数组、数组的数组、指针的指针、二维数组、字符点阵、字符笔画的坐标、黑白二值图片、灰度图片、彩色图片、录音、视频、指纹信息、身份证信息…… 仅供参考:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <locale.h>
int main() {
    int i,v;
    char bs[33];
    char b[33];
    char hs[9];
    char h[9];
    char s[4];
    char *e;

// 十进制整数转二进制串;
    i=1024;
    ltoa(i,b,2);
    sprintf(bs,"%032s",b);
    printf("i=%d,bs=%s\n",i,bs);
// 十进制整数转十六进制串;
    i=1024;
    ltoa(i,h,16);
    sprintf(hs,"%08s",h);
    printf("i=%d,hs=%s\n",i,hs);
// 十六进制字符串转成十进制数
    strcpy(hs,"00000400");
    sscanf(hs,"%x",&i);
    printf("hs=%s,i=%d\n",hs,i);
// 二进制字符串转化为十六进制字符串;
    strcpy(bs,"00000000000000000000010000000000");
    i=strtol(bs,&e,2);
    ltoa(i,h,16);
    sprintf(hs,"%08s",h);
    printf("bs=%s,hs=%s\n",bs,hs);
// 二进制字符串转化为十进制数;
    strcpy(bs,"00000000000000000000010000000000");
    i=strtol(bs,&e,2);
    printf("bs=%s,i=%d\n",bs,i);
// 十六进制字符串转成二进制串
    strcpy(hs,"00000400");
    sscanf(hs,"%x",&i);
    ltoa(i,b,2);
    sprintf(bs,"%032s",b);
    printf("hs=%s,bs=%s\n",hs,bs);
// ASC\GBK字符串转十六进制串
    strcpy(s,"a汉");
    i=0;
    while (1) {
        if (0==s[i]) break;
        sprintf(hs+i*2,"%02X",(unsigned char)s[i]);
        i++;
    }
    setlocale(LC_ALL,"chs");
    printf("s=%s,hs=%s\n",s,hs);
// 十六进制字符串转成汉字(GBK)及字符(ASC)
    strcpy(hs,"61BABA");
    i=0;
    while (1) {
        if (1!=sscanf(hs+i*2,"%2x",&v)) break;
        s[i]=(char)v;
        i++;
    }
    s[i]=0;
    printf("hs=%s,s=%s\n",hs,s);

    return 0;

}
//i=1024,bs=00000000000000000000010000000000
//i=1024,hs=00000400
//hs=00000400,i=1024
//bs=00000000000000000000010000000000,hs=00000400
//bs=00000000000000000000010000000000,i=1024
//hs=00000400,bs=00000000000000000000010000000000
//s=a汉,hs=61BABA
//hs=61BABA,s=a汉
CodeMoggy 2017-12-12
  • 打赏
  • 举报
回复
那1111 1111 1111 1111 1111 1111 1111 1111 也是补码,那为啥到%u就直接读取了?
paschen 2017-12-12
  • 打赏
  • 举报
回复
%d是按有符号输出,%u是按无符号输出,正如你写unsinged int a = -1; 此时a的值4294967295
大米粥哥哥 2017-12-12
  • 打赏
  • 举报
回复
引用 2 楼 风继续吹yc的回复:
但为什么是-1呢?既然最左边的1是符号位,那么右边31个1就是有效值,为啥就偏偏变成-1呢?
那是补码 要取反加一就是 1..00000...1 就是负一
CodeMoggy 2017-12-12
  • 打赏
  • 举报
回复
但为什么是-1呢?既然最左边的1是符号位,那么右边31个1就是有效值,为啥就偏偏变成-1呢?
自信男孩 2017-12-12
  • 打赏
  • 举报
回复
%u那么所有位的1都是有效值,最高位也不是符号位而是有效值。 %d输出,要看最高位符号位,最高位为1,则是负值。1111 1111 1111 1111 1111 1111 1111 1111的%d输出就是-1

69,374

社区成员

发帖
与我相关
我的任务
社区描述
C语言相关问题讨论
社区管理员
  • C语言
  • 花神庙码农
  • 架构师李肯
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧