打印short、long、long long和unsigned类型

 

        打印unsigned int类型的值,使用%u转换说明;打印long类型的值,使
用%ld转换说明。如果系统中int和long的大小相同,使用%d就行。但是,这
样的程序被移植到其他系统(int和long类型的大小不同)中会无法正常工
作。在x和o前面可以使用l前缀,%lx表示以十六进制格式打印long类型整
数,%lo表示以八进制格式打印long类型整数。注意,虽然C允许使用大写或
小写的常量后缀,但是在转换说明中只能用小写。

        C语言有多种printf()格式。对于short类型,可以使用h前缀。%hd表示以
十进制显示short类型的整数,%ho表示以八进制显示short类型的整数。h和l
前缀都可以和u一起使用,用于表示无符号类型。例如,%lu表示打印
unsigned long类型的值。程序清单3.4演示了一些例子。对于支持long long类
型的系统,%lld和%llu分别表示有符号和无符号类型。


#include <stdio.h>
int main(void)
{
unsigned int un = 3000000000; /* int为32位和short为16位的系统 */
short end = 200;
long big = 65537;
long long verybig = 12345678908642;
printf("un = %u and not %d\n", un, un);
printf("end = %hd and %d\n", end, end);
printf("big = %ld and not %hd\n", big, big);
printf("verybig= %lld and not %ld\n", verybig, verybig);
return 0;
}
在特定的系统中输出如下(输出的结果可能不同):
un = 3000000000 and not -1294967296
end = 200 and 200
big = 65537 and not 1
130
verybig= 12345678908642 and not 1942899938

        该例表明,使用错误的转换说明会得到意想不到的结果。第 1 行输出,
对于无符号变量 un,使用%d会生成负值!其原因是,无符号值 3000000000
和有符号值−129496296 在系统内存中的内部表示完全相同(详见第15
章)。因此,如果告诉printf()该数是无符号数,它打印一个值;如果告诉它
该数是有符号数,它将打印另一个值。在待打印的值大于有符号值的最大值
时,会发生这种情况。对于较小的正数(如96),有符号和无符号类型的存
储、显示都相同。

         第2行输出,对于short类型的变量end,在printf()中无论指定以short类型
(%hd)还是int类型(%d)打印,打印出来的值都相同。这是因为在给函
数传递参数时,C编译器把short类型的值自动转换成int类型的值。你可能会
提出疑问:为什么要进行转换?h修饰符有什么用?第1个问题的答案是,
int类型被认为是计算机处理整数类型时最高效的类型。因此,在short和int类
型的大小不同的计算机中,用int类型的参数传递速度更快。第2个问题的答
案是,使用h修饰符可以显示较大整数被截断成 short 类型值的情况。第 3 行
输出就演示了这种情况。把 65537 以二进制格式写成一个 32 位数是
00000000000000010000000000000001。使用%hd,printf()只会查看后 16
位,所以显示的值是 1。与此类似,输出的最后一行先显示了verybig的完整
值,然后由于使用了%ld,printf()只显示了储存在后32位的值。

在计算机科学中,大端模式(Big-Endian)是一种数据存储的方式,在这种方式下,高位字节会被存放在内存低地址处,低位字节则存放于高地址位置。对于一个 `unsigned long long` 类型变量转化为大端的过程,主要涉及位运算字节操作。 以下是将一个小端机器上的 `unsigned long long` 类型变量转化为大端的具体步骤: ### 步骤说明: 假设有一个名为 num 的 `unsigned long long` 变量 (64-bit),其初始值为小端格式。 ```c++ #include <cstdint> // 提供 uint8_t 其他无符号整数类型定义 #include <cstring> // std::memcpy uint64_t htonll(uint64_t hostnum) { static const int numbytes = sizeof(hostnum); uint64_t netnum = 0; for(int i=0; i<numbytes; ++i){ ((uint8_t*)&netnum)[numbytes - 1 - i] = ((uint8_t*)&hostnum)[i]; } return netnum; } ``` 上面这个函数实现了将主机序的 `unsigned long long` 转换成网络序(即大端)的功能。 ### 解释上述代码块的工作原理: 1. 首先创建了一个静态常量表示我们要处理的数字占用多少个字节。 2. 初始化返回的大端数值为零。 3. 使用for循环迭代每一个字节的位置,并通过指针访问原数的每一位并将其逆置填入新的数中去完成大小端转换过程。 如果当前系统已经是大端了,则不需要任何动作;如果是小端就需要按照以上方法来进行变换。 此外也可以利用标准库提供的宏如htonl等针对特定宽度数据类型的换端方法来简化部分工作负载,不过它们通常只适用于较小的数据类型比如long或者是short而没有直接提供对long long的支持。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值