printf格式化输出、大端小端原理、struct结构体对齐、memcpy函数实现原理

1.printf格式化输出

(参考链接:https://baike.baidu.com/item/printf%28%29/402521?fr=aladdin

printf()函数是式样化输出函数, 一般用于向准则输出设备按规定式样输出消息。正在编写步骤时经常会用到此函数。printf()函数的挪用式样为: printf("<式样化字符串>",<参数表>);

函数printf从右到左压栈,然后将先读取放到栈底,最后读取的放在栈顶,处理时候是从栈顶开始的,所以我们看见的结果是,从右边开始处理的;

函数的原型为:int printf(const char *format, ...);

printf()函数的调用格式为:printf("<格式化字符串>", <参量表>);

其中格式化字符串包括两部分内容: 一部分是正常字符, 这些字符将按原样输出;另一部分是格式化规定字符, 以"%"开始,,后跟一个或几个规定字符,用来确定输出内容格式。参量表是需要输出的一系列参数, 其个数必须与格式化字符串所说明的输出参数个数一样多, 各参数之间用","分开, 且顺序一一对应, 否则将会出现意想不到的错误。

规定符

%d 十进制有符号整数

%u 十进制无符号整数

%f 浮点数

%s 字符串

%c 单个字符

%p 指针的值        C++ 也需要格式化输出才能输出指针的值:static_cast<const void *>(pszStr)

%e 指数形式的浮点数

%x, %X 无符号以十六进制表示的整数

%o 无符号以八进制表示的整数

%g 把输出的值按照%e或者%f类型中输出长度较小的方式输出

%p 输出地址符

%lu 32位无符号整数

%llu 64位无符号整数

说明

(1). 可以在"%"和字母之间插进数字表示最大场宽

例如: %3d 表示输出3位整型数, 不够3位右对齐

%9.2f 表示输出场宽为9的浮点数, 其中小数位为2, 整数位为7,小数点占一位, 不够9位右对齐。

%8s 表示输出8个字符的字符串, 不够8个字符右对齐。如果字符串的长度、或整型数位数超过说明的场宽, 将按其实际长度输出。但对浮点数, 若整数部分位数超过了说明的整数位宽度, 将按实际整数位输出;若小数部分位数超过了说明的小数位宽度, 则按说明的宽度以四舍五入输出

另外, 若想在输出值前加一些0, 就应在场宽项前加个0。例如: %04d 表示在输出一个小于4位的数值时, 将在前面补0使其总宽度为4位。

如果用浮点数表示字符整型量的输出格式, 小数点后的数字代表最大宽度,小数点前的数字代表最小宽度。例如: %6.9s 表示显示一个长度不小于6不大于9的字符串。若大于9, 则第9个字符以后的内容将被删除。

(2). 可以在"%"和字母之间加小写字母l, 表示输出的是长型数。

例如: %ld 表示输出long整数,%lf 表示输出double浮点数

(3). 可以控制输出左对齐或右对齐, 即在"%"和字母之间加入一个"-" 号可

说明输出为左对齐, 否则为右对齐。

例如: %-7d 表示输出7位整数左对齐

         %-10s 表示输出10个字符左对齐

2. 一些特殊规定字符

作用

\n 换行

\f 清屏并换页

\r 回车

\t Tab符

\xhh 表示一个ASCII码用16进表示,

其中hh是1到2个16进制数

下面是Turbo C2.0上编写的简例:

#include<stdio.h>
#include<string.h>
int main(void)
{
    char c,s[20],*p;
    int a=1234,*i;
    float f=3.141592653589;
    double x=0.12345678987654321;
    p="Howdoyoudo";
    strcpy(s,"Hello,Comrade");
    *i=12;
    c='\x41';
    printf("a=%d\n",a);/*结果输出十进制整数a=1234*/
    printf("a=%6d\n",a);/*结果输出6位十进制数a=1234*/
    printf("a=%06d\n",a);/*结果输出6位十进制数a=001234*/
    printf("a=%2d\n",a);/*a超过2位,按实际值输出a=1234*/
    printf("*i=%4d\n",*i);/*输出4位十进制整数*i=12*/
    printf("*i=%-4d\n",*i);/*输出左对齐4位十进制整数*i=12*/
    printf("i=%p\n",i);/*输出地址i=06E4*/
    printf("f=%f\n",f);/*输出浮点数f=3.141593*/
    printf("f=%6.4f\n",f);/*输出6位其中小数点后4位的浮点数
    f=3.1416*/
    printf("x=%lf\n",x);/*输出长浮点数x=0.123457*/
    printf("x=%18.16lf\n",x);/*输出18位其中小数点后16位的长浮点
    数x=0.1234567898765432*/
    printf("c=%c\n",c);/*输出字符c=A*/
    printf("c=%x\n",c);/*输出字符的ASCII码值c=41*/
    printf("s[]=%s\n",s);/*输出数组字符串s[]=Hello,Comrade*/
    printf("s[]=%6.9s\n",s);/*输出最多9个字符的字符串s[]=Hello,
    Co*/
    printf("s=%p\n",s);/*输出数组字符串首字符地址s=FFBE*/
    printf("*p=%s\n",p);/*输出指针字符串p=Howdoyoudo*/
    printf("p=%p\n",p);/*输出指针的值p=0194*/
    return 0;
}

输出结果为:

 

2.大端小端问题

无论是大端模式还是小段模式,都是从物理内存低地址的底地址开始存放数据!

大端模式:把数据的低地址放在内存的低地址,逐渐将数据的高地址放到内存的高地址

小端模式:把数据的高地址放在内存的低地址,逐渐将数据的低地址放到内存的高地址

 

3.struct结构体对齐

题目:

在一个64位的操作系统中定义如下结构体:

1

2

3

4

5

6

struct st_task

{

  uint16_t id;

  uint32_t value;

  uint64_t timestamp;

};

同时定义fool函数如下:

1

2

3

4

5

6

7

void fool()

{

  st_task task = {};

  uint64_t a = 0x00010001;

  memcpy(&task,&a,sizeof(uint64_t));

  printf("%11u,%11u,%11u",task.id,task.value,task.timestamp);

}

上述fool函数()程序的执行结果为:

答案:1,0,0

分析一下原因:

首先struct是存在内存对齐一说的,具体的对齐方式对于结构的各个成员,第一个成员位于偏移为0的位置,以后每个数据的偏移量必须是Min(编译器被指定的对齐字节数,该数据成员的自身长度)的倍数。GCC中,Min(系统默认(4), 数据成员自身长度(char/short/int/double))。

在数据成员完各自对齐之后,结构体(或联合体)本身也要进行对齐,对齐将按照编译器被指定的对齐字节数和结构体(或联合体)最大数据成员长度中,比较小的那个进行。

结合题目画图看看:其中绿色代表id,中间阴影区域代表为对齐接下来的int型而补零的两个字节,蓝色区域代表value,因为接下来的变量是64位,而前面恰好是64位,可以使后面的内容进行对齐,橙色代表value,所以struct结构体的内存区域分布及对其方式如图所示。求其sizeof,所占字节数即为16B。

 

4.memcpy函数的实现原理

 

5.strcpy函数的实现原理

 

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值