c语言打印字符的函数参数,C语言格式化打印函数vsnprintf()的实现

Linux内核的格式化打印函数是printk(),它与printf()函数是类似的,都是根据格式字符串把可变参数列表转化成字符序列,然后输出到控制台。

printf()是打印到标准输出stdout。

printk()是打印到控制台终端。在使用串口线连接嵌入式硬件时,就是打印到电脑的串口终端软件,例如minicom。

转化可变参数列表这一步,这两个函数是一样的,都是调用vsnprintf()函数。

区别是内核没法调用C库,只能另外写一个简单的实现。

vsnprintf()的实现,依靠的是C语言处理可变参数的类型valist,以及使用它的三个宏:vastart,vaarg,vaend。

它们都定义在头文件里。

3b57bf3424fda3034fe22b415c519efb.png

我在电脑上调试时,直接把siskavalist定义为了C库的valist,如上图。

5-10行注释掉的部分,是32位C语言的valist定义。

snprintf的代码就这么几行,使用vastart获取参数列表的开头,然后调用vsnprintf()打印出来,最后使用vaend。

932f5ec8a2ded64ee5a4c308f8d08160.png

对格式串的解析在vsnprintf()里,带n的printf系列函数可以标示缓冲区的大小,避免字符串溢出。

vsnprintf()的实现:

buf,缓冲区的地址。

size,缓冲区的大小。

fmt,格式串。

ap,可变参数列表,开始时指向它的第1个元素。

先把字符的计数设置为0,size -1是为了给末尾的'\0'留一个位置,然后遍历格式串fmt。

9c5bce339b449ea23632d00e876dc687.png

130-133,不是%则直接打印到缓冲区。

135-139,是%则查看下一个,如果也是%则打印到缓冲区,所以%%会打印%。

141-145,查看是否是十六进制的前缀。

147-151,查看是否是长整型的前缀。

153开始的switch语句是对格式参数的解析:

5db147e8cad42b1f99f538c1af2acb84.png

154,c表示打印1个字符,它是按照int存储在参数里的,所以vaarg的类型选int。

157-162,根据是否有前缀选择普通整型或长整型,有符号的。

163-168,同上,无符号的。

169-181,十六进制的整数,根据格式参数选择是否打印0x前缀,是否长整型。

183,p表示打印指针,其中空指针会打印null。

185-188,浮点数,全按double处理。

190,字符串,它的内容也是一个'\0'结尾的char*字符列表。

197,移动到格式串的下一个字符,继续判断while条件。

这时无论格式串到了末尾'\0',还是缓冲区只剩了最后1个'\0'的空间,都会退出while循环,避免缓冲区越界。

200行,填充结尾的'\0',返回转化的字符总数。

aaa64f331bb9abc37c60b60d3cbd608a.png

siskaulong2a()函数,是把无符号长整型转换为字符串的函数,普通的整型也用它转换,编译器会自动把unsigned int类型升级到unsigned long。

打印字符会改变当前缓冲区的字符计数,所以参数传了int* pn,即计数的指针。它既是输入参数,也是输出参数。

4c97ca0c5025f4c4ae345d9663bb064b.png

num %10先获取个位数,然后 num /10去掉个位数,下一次就是获取十位数,以此类推,直到为0。具体的字符要加上'0'。

这么打印出来的数字字符串是反着的,低位先被打印,所以19-23行的while再把它正过来。我们在第6行提前记录了这串字符的起始位置。

siskalong2a(),有符号的打印除了负数时要先打印1个负号之外,其他的与无符号的一样。

siskadouble2a(),浮点数都是有符号的,负数也要先打印1个负号,然后先取整数部分,再取小数部分,把它们都当整数打印,中间打印小数点。

小数部分这里用了6位有效数字。

d98aaed730b9ee20194c1a4492ba638a.png

siskahex2a(),十六进制的都按无符号处理,除了从10的余数变成16的余数之外,与unsigned long的区别只有67行,即大于9的从'a'开始显示,9以内的加上'0'显示。

x -10+ 'a',就是10-15要显示的字符,10对应'a',15对应'f'。

bbeb0c8523e82f7371414edc74f7e9e0.png

如果带前缀打印十六进制,就先打印0x,占2个字符的空间。

siskap2a(),指针都带0x前缀,按十六进制打印,空指针显示null。

siskastr2a(),字符串按原样打印。

6f2ad903880d9f4ccf7cacb3721943b1.png

main()函数,和测试结果。

0ef7331981f013df71265f4ba2710aa3.png

下图第2张是缓冲区不足时的打印,第1张是缓冲区1024字节的打印。

7a55ec7131eff7e30173b4a186610d42.png

9f36c941a3dd780e46a8eeb7c1185081.png

Linux使用bochs模拟BIOS读磁盘

先调用这个函数把数据转化到缓冲区里,然后通过串口线打印出来,就是printk()。

如果通过标准输出stdout打印出来,就是printf()。

如果通过FILE* fp 文件句柄打印出来,就是fprintf()。

还可以继续添加格式字符,让它支持更多的数据类型。

但在linux内核里,实际上连浮点数都尽量不用,支持有符号和无符号的整数以及字符串,基本就够用了。

想了解更多精彩内容,快来关注闲聊代码

PS:在32位的堆栈传参模式下,格式串const char* fmt后面就是参数列表,所以只要取格式串的地址&fmt,加上4字节就是下一个参数的地址,然后根据格式串里%之后的类型字符依次打印就行。

32位是按4字节对齐,char、short这种不到4字节的类型也是转化为4字节压栈,double、long long这种按8字节压栈。

64位是用寄存器传前6个参数,多于6个的按堆栈传参,而且还是整数与浮点数分开传,整数使用rdi、rsi、rdx、rcx、r8、r9,浮点数使用xmm0、xmm1、xmm2,一直到xmm7。

如果参数是printf("%d,%f\n",1,2.71)这样,rdi是格式串,rsi是整数1,xmm0是浮点数2.71。

如果自己实现vastart,vaarg的话,需要让printf()函数先调用自己实现的printf(),这样才能自己控制寄存器参数的存放顺序,然后在printf()里在调用vsnprintf()。

否则,只能依赖gcc提供的valist,vastart,vaarg,vaend,因为寄存器参数在这种情况下怎么保存,是编译器的权限范围。

而寄存器参数的保存方式,则关系到valist的实现。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/555772.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用SpringMVC参数传递时,解决get请求时中文乱码的问题

问题描述: 使用SpringMVC参数传递时, 遇到get请求中文信息时,页面应答会显示中文乱码. 解决办法: 一, 我们需要把request.getParameter(“参数名”)获取到的字符串先用ISO-8859-1编码成字节流,然后再将其用utf-8解码成字符流,代码如下: String str new String(…

get请求URL传中文参数后台接收乱码的解决之道

现在需求是在浏览器里面直接发请求,浏览器发请求,默认发的是get请求,然后带参数,参数包含中文汉字。麻烦就在这里,参数里面的汉字,在后台接收的时候,是乱码。 这么简单的传递参数的方式&#xf…

c语言 dec2oct函数,【Excel函数】DEC2OCT函数 - 曹海峰个人博客

本文介绍Microsoft Excel中DEC2OCT函数的语法和用法。DEC2OCT函数适用于:Excel2003、Excel2007、Excel2010、Excel2013、Excel2016。函数说明DEC2OCT函数的主要作用是将十进制数转换为八进制数。如果该函数不可用,并返回错误值 #NAME?,请安装…

如何解决springboot参数传中文乱码

如何解决springboot参数传中文乱码 前言 本文案例来自业务部门的一个业务场景。他们的业务场景是他们部门研发了一个微服务上下文透传组件,其透传原理也挺简单的,就是通过springboot拦截器把请求参数塞进threadlocal,然后下游通过threadloc…

android原生调用nextjs方法,详解使用Next.js构建服务端渲染应用

next.js简介最近在学React.js,React官方推荐使用next.js框架作为构建服务端渲染的网站,所以今天来研究一下next.js的使用。next.js作为一款轻量级的应用框架,主要用于构建静态网站和后端渲染网站。框架特点使用后端渲染自动进行代码分割(code…

SpringBoot2基础-请求参数处理和原理

SpringBoot2基础-请求参数处理和原理 tags: Spring Boot2021尚硅谷雷丰阳 categories: 静态文件配置静态文件配置原理欢迎页和自定义 FaviconRest表单请求原理请求映射原理各种参数使用和原理 文章目录SpringBoot2基础-请求参数处理和原理第一节 [SpringMVC](https://so.cs…

Spring MVC 执行过程原理(请求映射原理、参数处理原理、返回值处理器)

Spring MVC 执行过程分析 文章目录Spring MVC 执行过程分析请求映射原理适配器Adapter执行目标方法参数处理器解析器HandlerMethodArgumentResolverRequestBody测试RequestParam测试RequestParamMethodArgumentResolver分析执行目标方法体返回值处理器解析器目标方法执行完成处…

推箱子android源代码,android自定义view实现推箱子小游戏

本文实例为大家分享了android推箱子游戏的具体实现代码,供大家参考,具体内容如下自定义view:package com.jisai.materialdesigndemo.tuixiangzhi;import android.content.Context;import android.graphics.Bitmap;import android.graphics.Bi…

android没有捂脸表情,微信新emoji表情安卓机怎么没有?微信新emoji表情安卓机没有怎么回事?...

在最新版ios版微信中不少用户在聊天时发现了新emoji表情哟,但是安卓机最新版即没有发现,那么微信新emoji表情安卓机怎么没有?微信新emoji表情安卓机没有怎么回事?下面小编就给大家具体介绍下。iOS版微信总共加入了9款新表情&#…

springboot源码分析: 请求方式+请求映射原理+获取参数原理

注解相关 AliasFor:.在同个注解中为同一个功能定义两个名称不一样的属性,那么这两个属性彼此互为别名 RequestMapping注解里面的代码 AliasFor("path")String[] value() default {};AliasFor("value")String[] path() default {};G…

android 或者vide的高度和宽度,关于Android中videoView.setVideoPath(“PATH”)的问题!!!急!!...

满意答案yfdsan31972015.06.03采纳率:52% 等级:9已帮助:364人path 是获取的路径,如果你把视频文件夹写在raw文件夹下 ,/*** raw文件夹下的文件处理工具类** */public class RawFileUtils {private RawFileUtils( ){…

吃透Java IO:字节流、字符流、缓冲流

文章目录前言1 初识Java IO1.1 IO流分类1.2 案例实操2 IO流对象2.1 File类2.2 字节流2.3 字符流2.4 序列化3 IO流方法3.1 字节流方法3.2 字符流方法4 附加内容4.1 位、字节、字符4.2 IO流效率对比4.3 NIO前言 有人曾问fastjson的作者(阿里技术专家高铁)&…

android 侧滑删除功能,200行代码让你在Android中完美实现iOS版侧滑删除效果

使用几个月的IOS之后,发现IOS中侧滑删除俺就大家好,自己开始学习Android已经差不多半年了吧,前前后后看了不少的博客获益匪浅。渐渐的随着技术的提升,慢慢感觉网上其它的一些功能的实现又不是那么完美,今天就给大家带来一篇在Andr…

java IO体系的学习总结

1.Java Io流的概念,分类,类图。 1.1 Java Io流的概念 ? ? java的io是实现输入和输出的基础,可以方便的实现数据的输入和输出操作。在java中把不同的输入/输出源(键盘,文件,网络连接等)抽象表…

android10获取imei,Android 10 root用户获取imei

IMEI(International Mobile Equipment Identity)是国际移动设备识别码的缩写,由15-17位数字组成,与手机是一一对应的关系。无论刷机还是恢复出厂设置,该设备标识码都不会改变,所以在广告和流量统计等方面特别好用,备受…

JAVA基础之HttpServletResponse响应

JAVA基础之HttpServletResponse响应 用户在客户端输入网址(虚拟路径)时,开始发送一个HTTP请求(请求行、请求头、请求体)至服务器。服务器内的Tomcat引擎会解析请求的地址,去找XML文件,然后根据…

Logger之Logger.getLogger(CLass)

之前一直在使用System.out.println()来调试.但是用这种方式开发项目部署到生产环境,会因为众多的控制台输出降低应用的性能.这时候Log4J就成为可平衡开发和部署应用的利器了. 在项目中使用Log4J并不是一件困难的事情,简单粗暴的方式就是在每个类A中声明一个Logger私有属性 pri…

订阅号助手android,微信订阅号助手app

微信订阅号助手app属于微信官方推出的软件,可以把我们的手机当做公众号的平台,让你直接用手机来实现公众号的各种工作、互动内容,微信订阅号助手app不用担心使用一些第三方软件被封号了,非常实用可靠。【应用介绍】订阅号助手是一…

使用LoggerFactory.getLogger(xxx.class)方法在控制台打印日志信息

使用System.out.println()来调试.但是用这种方式开发项目部署到生产环境,会因为众多的控制台输出降低应用的性能.这时候Log4J就成为可平衡开发和部署应用的利器了. 使用指定的类XXX初始化日志对象,方便在日志输出的时候,可以打印出日志信息所属的类。 …

html自动增加vbs代码,vbs脚本病毒代码大全编步骤四个

应用程序通过ActiveX的属性HTMLPageTextWithTags(主页不嵌有HTML代码时用属性HTMLPageTextWithOutTags)读页面文本交互模式;允许显示用户提示和脚本错误2.ntlm.vbs***************ntlm.vbsby黑嘿黑***************dimwshsetwshCreateObject("WScript.Shell&quo…