lark 发表于 2015-7-23 19:06:00

补码 ,打印结果 和位域相关的

插入代码:(问题在注释里面)
#include<stdio.h>
#include<stdlib.h>

int main(void){
        int i=-3;
        char str;
       
        struct bin{
        char nCHAR:1;       
        }use;

        while(i<=3){
                use.nCHAR=i;
                itoa(use.nCHAR,str,2);
                printf("use.nCHAR=%2d ,i=%2d Bin=%s\n",use.nCHAR,i,str);
                i++;
        }
        return 0;
}
//问题:为什么(有符号数,且限制只有符号,1个bit为 1时,编译器会把补码 二进制全部打印出来?



时间:2013年7月23日19:06:54;

ryxcaixia 发表于 2015-7-23 19:06:01

template<typename T>
void PrintComplement(T num)
{
        for (int i = sizeof(T)*8 - 1; i >= 0; i--)
                printf("%d", (num >> i) & 0x01);
}

#include<stdio.h>
#include<stdlib.h>

int main(void){
        int i=-3;
        char str;

        struct bin{
                char nCHAR:1;      
        }use;

        while(i<=3){
                use.nCHAR=i;
                itoa(use.nCHAR,str,2);
                printf("use.nCHAR=%2d ,i=%2d Bin=%s\n",use.nCHAR,i,str);

                PrintComplement(use.nCHAR);
                printf("\n");

                i++;
        }
        return 0;
}

编译器没那么智能 能自动打印出补码
你给他指定啥 就是啥
比如 指定%d, 那么编译器认为这个东西就是一个有符号的整形, 直接打印, 如果你制定一个字符a, 让他以%d打印出来 他直接打印一个97出来
同理 你给一个int a=97, 然后printf("%c", a), 他直接打印出一个字符a来, 占位符只会打印你指定的形式 编译器不会把补码打印出来

这个函数就是按位取, 把一个变量的内存中的每一位(补码)都取出来 然后打印出来

lwmheaton 发表于 2015-7-23 20:25:20

不懂~~~~帮顶·~~~:lol:

lark 发表于 2015-7-25 17:07:23

本帖最后由 lark 于 2015-7-25 17:11 编辑

ryxcaixia 发表于 2015-7-24 10:48
编译器没那么智能 能自动打印出补码
你给他指定啥 就是啥
比如 指定%d, 那么编译器认为这个东西就是 ...

这段代码不太懂,请问是什么意思?
我的意思是说,可以帮我加个注释吗?我理解下;
void PrintComplement(T num)
{
      for (int i = sizeof(T)*8 - 1; i >= 0; i--)
                printf("%d", (num >> i) & 0x01);
}

ryxcaixia 发表于 2015-7-25 22:38:27

本帖最后由 ryxcaixia 于 2015-7-25 22:49 编辑

lark 发表于 2015-7-25 17:07
这段代码不太懂,请问是什么意思?
我的意思是说,可以帮我加个注释吗?我理解下;

// 先不考虑模板函数
// 函数功能: 打印一个整数型的补码(支持负数)
void OutputComplement(int num)
{
        for (int i = 31; i >= 0; i--) // 一个整数是32位, 自然要循环32次, 当然更重要的是从最低位开始打印.
        {
                int tmp = num >> i; // 我们想要从他的最低位开始打印, 为了获得最低位, 要将这个32位的整数右移31位, >> 这是里位移命令
                                                        // 然后第二次循环就会获得他的第2位..依次类推
                tmp &= 1; // 获得到当前位后 与 1进行与操作, 取得当前位的真实数值, 因为 0 & 1 == 0, 1 & 1 == 1(如果想强制置1, 那么进行的都是或操作, 之前楼主不是经常弄int:9这样么 编译器的汇编代码就是将最低9位全部与1进行与操作)
                printf ("%d", tmp); // 然后打印出来当前的最低位

                if (i % 4 == 0)
                        printf (" ");
        }

        printf("\n");
}

#include <stdio.h>

int main()
{
        OutputComplement(10); // 这里可以看到 正 10 和 负 10 的真实补码
        OutputComplement(-10);

        return 0;
}

ryxcaixia 发表于 2015-7-25 22:43:10

至于模板函数, 可以自动识别变量的类型进行绑定 更智能
楼主如果不清楚模板函数 看看c++primer的 第十六章 模板与泛型编程
简单来说 模板是通用的泛型编程 他会根据变量的类型自动实例化同类参数
举个例子

void PrintIntComplement(T num)
{
        for (int i = 32 - 1; i >= 0; i--)
                printf("%d", (num >> i) & 0x01);
}

void PrintCharComplement(T num)
{
        for (int i = 8 - 1; i >= 0; i--)
                printf("%d", (num >> i) & 0x01);
}

同样是打印补码 仅仅是某一处改变 我就要 写两个不同的函数 很麻烦
后来提出了一个叫元编程的东西 就是编译器自动帮助你“写代码”

这样我就可以把多个功能一样的函数 写在一起 不用仅仅因为某一处(多处)改变而分开写多个函数

weisuo 发表于 2015-7-26 11:58:19

不懂 帮忙顶

cqj9006 发表于 2015-7-26 21:30:57

不懂~~~~帮顶·

lark 发表于 2015-7-27 16:16:27

weisuo 发表于 2015-7-26 11:58
不懂 帮忙顶

Thanks:lol:

lark 发表于 2015-7-27 16:17:03

cqj9006 发表于 2015-7-26 21:30
不懂~~~~帮顶·

Thanks:handshake

lark 发表于 2015-7-27 16:18:10

lwmheaton 发表于 2015-7-23 20:25
不懂~~~~帮顶·~~~

:lol:

lark 发表于 2015-7-27 16:19:45

ryxcaixia 发表于 2015-7-25 22:43
至于模板函数, 可以自动识别变量的类型进行绑定 更智能
楼主如果不清楚模板函数 看看c++primer的 第十六 ...

非常感谢:handshake
页: [1]
查看完整版本: 补码 ,打印结果 和位域相关的