printf输出格式

article/2025/8/23 0:48:50

1.printf()简介

printf()是C语言标准库函数,用于将格式化后的字符串输出到标准输出。标准输出,即标准输出文件,对应终端的屏幕。printf()申明于头文件stdio.h。

函数原型:

int printf ( const char * format, ... );
  • 1

返回值:
正确返回输出的字符总数,错误返回负值。与此同时,输入输出流错误标志将被置值,可由指示器函数ferror(FILE *stream)来检查输入输出流的错误标志,如果ferror()返回一个非零值,表示出错。

调用格式:
printf()函数的调用格式为:printf("格式化字符串",输出表列)

格式化字符串包含三种对象,分别为:
(1)字符串常量;
(2)格式控制字符串;
(3)转义字符。
字符串常量原样输出,在显示中起提示作用。输出表列中给出了各个输出项,要求格式控制字符串和各输出项在数量和类型上应该一一对应。其中格式控制字符串是以%开头的字符串,在%后面跟有各种格式控制符,以说明输出数据的类型、宽度、精度等。

2.格式控制字符串详解

printf的格式控制字符串组成如下:

%[flags][width][.prec][length]type
  • 1

分别为:

%[标志][最小宽度][.精度][类型长度]类型。
  • 1

2.1类型(type)

首先说明类型,因为类型是格式控制字符串的重中之重,是必不可少的组成部分,其它的选项都是可选的。type用于规定输出数据的类型,含义如下:

字符对应数据类型含义示例
d/iint输出十进制有符号32bits整数,i是老式写法printf("%i",123);输出123
ounsigned int无符号8进制(octal)整数(不输出前缀0)printf("0%o",123);输出0173
uunsigned int无符号10进制整数printf("%u",123);输出123
x/Xunsigned int无符号16进制整数,x对应的是abcdef,X对应的是ABCDEF(不输出前缀0x)printf("0x%x 0x%X",123,123);输出0x7b 0x7B
f/lffloat(double)单精度浮点数用f,双精度浮点数用lf(printf可混用,但scanf不能混用)printf("%.9f %.9lf",0.000000123,0.000000123);输出0.000000123 0.000000123。注意指定精度,否则printf默认精确到小数点后六位
Ffloat(double)与f格式相同,只不过 infinity 和 nan 输出为大写形式。例如printf("%f %F %f %F\n",INFINITY,INFINITY,NAN,NAN);输出结果为inf INF nan NAN
e/Efloat(double)科学计数法,使用指数(Exponent)表示浮点数,此处"e"的大小写代表在输出时“e”的大小写printf("%e %E",0.000000123,0.000000123);输出1.230000e-07 1.230000E-07
gfloat(double)根据数值的长度,选择以最短的方式输出,%f或%eprintf("%g %g",0.000000123,0.123);输出1.23e-07 0.123
Gfloat(double)根据数值的长度,选择以最短的方式输出,%f或%Eprintf("%G %G",0.000000123,0.123);输出1.23E-07 0.123
cchar字符型。可以把输入的数字按照ASCII码相应转换为对应的字符printf("%c\n",64)输出A
schar*字符串。输出字符串中的字符直至字符串中的空字符(字符串以空字符’\0‘结尾)printf("%s","测试test");输出:测试test
Swchar_t*宽字符串。输出字符串中的字符直至字符串中的空字符(宽字符串以两个空字符’\0‘结尾)setlocale(LC_ALL,"zh_CN.UTF-8");
wchar_t wtest[]=L"测试Test";
printf("%S\n",wtest);
输出:测试test
pvoid*以16进制形式输出指针printf("0x%p","lvlv");输出:0x000000013FF73350
nint*什么也不输出。%n对应的参数是一个指向signed int的指针,在此之前输出的字符数将存储到指针所指的位置int num=0;
printf("lvlv%n",&num);
printf("num:%d",num);
输出:lvlvnum:4
%字符%输出字符‘%’(百分号)本身printf("%%");输出:%
m打印errno值对应的出错内容printf("%m\n");
a/Afloat(double)十六进制p计数法输出浮点数,a为小写,A为大写printf("%a %A",15.15,15.15);输出:0x1.e4ccccccccccdp+3 0X1.E4CCCCCCCCCCDP+3

注意:
(1)使用printf输出宽字符时,需要使用setlocale指定本地化信息并同时指明当前代码的编码方式。除了使用%S,还可以使用%ls。
(2)%a和%A是C99引入的格式化类型,采用十六进制p计数法输出浮点数。p计数法类似E科学计数法,但不同。数以0x开头,然后是16进制浮点数部分,接着是p后面是以 2为底的阶码。以上面输出的15.15为例,推算输出结果。15.15转换成二进制为1111.00 1001 1001 1001 1001 ...,因为二进制表示数值的离散特点,计算机对于小数有时是不能精确表示的,比如0.5可以精确表示为 0.120.120. 1 2 0.120.12 0.1_2 0.120.120.120.12,而0.15却不能精确表示。将15.15对应的二进制右移三位,为1.1110 0100 1100 1100 1100 ...转换对应的十六进制就是0x1.e4ccccccccccd,注意舍入时向高位进了1位。由于右移三位,所以二进制阶码就是3。最后的结果就是0x1.e4ccccccccccdp+3。

(3)格式控制字符串除了指明输出的数据类型,还可以包含一些其它的可选的格式说明,依序有 flags, width, .precision and length。下面一一讲解。

2.2标志(flags)

flags规定输出样式,取值和含义如下:

字符名称说明
-减号结果左对齐,右边填空格。默认是右对齐,左边填空格。
+加号输出符号(正号或负号)
space空格输出值为正时加上空格,为负时加上负号
#井号type是o、x、X时,增加前缀0、0x、0X。
type是a、A、e、E、f、g、G时,一定使用小数点。默认的,如果使用.0控制不输出小数部分,则不输出小数点。
type是g、G时,尾部的0保留。
0数字零将输出的前面补上0,直到占满指定列宽为止(不可以搭配使用“-”)

示例:

printf("%5d\n",1000); 				//默认右对齐,左边补空格
printf("%-5d\n",1000); 				//左对齐,右边补空格printf("%+d %+d\n",1000,-1000);		//输出正负号printf("% d % d\n",1000,-1000);		//正号用空格替代,负号输出printf("%x %#x\n",1000,1000);		//输出0xprintf("%.0f %#.0f\n",1000.0,1000.0)//当小数点后不输出值时依然输出小数点printf("%g %#g\n",1000.0,1000.0);	//保留小数点后后的0printf("%05d\n",1000);				//前面补0

输出结果为:
这里写图片描述

2.3输出最小宽度(width)

用十进制整数来表示输出的最少位数。若实际位数多于指定的宽度,则按实际位数输出,若实际位数少于定义的宽度则补以空格或0。width的可能取值如下:

width描述示例
数值十进制整数printf("%06d",1000);输出:001000
*星号。不显示指明输出最小宽度,而是以星号代替,在printf的输出参数列表中给出printf("%0*d",6,1000);输出:001000

2.4精度(.precision)

精度格式符以“.”开头,后跟十进制整数。可取值如下:

.precision描述
.数值十进制整数。
(1)对于整型(d,i,o,u,x,X),precision表示输出的最小的数字个数,不足补前导零,超过不截断。
(2)对于浮点型(a, A, e, E, f ),precision表示小数点后数值位数,默认为六位,不足补后置0,超过则截断。
(3)对于类型说明符g或G,表示可输出的最大有效数字。
(4)对于字符串(s),precision表示最大可输出字符数,不足正常输出,超过则截断。
precision不显示指定,则默认为0
.*以星号代替数值,类似于width中的*,在输出参数列表中指定精度。

示例:

printf("%.8d\n",1000);			//不足指定宽度补前导0,效果等同于%08d
printf("%.8f\n",1000.123456789);//超过精度,截断
printf("%.8f\n",1000.123456);	//不足精度,补后置0
printf("%.8g\n",1000.123456);	//最大有效数字为8位
printf("%.8s\n",“abcdefghij”);	//超过指定长度截断
  • 1
  • 2
  • 3
  • 4
  • 5

输出结果:

00001000
1000.12345679
1000.12345600
1000.1235
abcdefgh
  • 1
  • 2
  • 3
  • 4
  • 5

注意: 在对浮点数和整数截断时,存在四舍五入。

2.5类型长度(length)

类型长度指明待输出数据的长度。因为相同类型可以有不同的长度,比如整型有 16bits 的 short int,32bits的int,也有 64bits 的 long int,浮点型有32bits的单精度 float 和 64bits 的双精度 double。为了指明同一类型的不同长度,于是乎,类型长度(length)应运而生,成为格式控制字符串的一部分。

因为Markdown表格不支持单元格合并,背景颜色等样式,所以直接引用printf.C++ reference的表格。
这里写图片描述

注意: 黄色背景行标识的类型长度说明符和相应的数据类型是C99引入的。

示例代码:

printf("%hhd\n",'A');				//输出有符号char
printf("%hhu\n",'A'+128);			//输出无符号char
printf("%hd\n",32767);				//输出有符号短整型short int
printf("%hu\n",65535);				//输出无符号短整型unsigned short int
printf("%ld\n",0x7fffffffffffffff);	//输出有符号长整型long int
printf("%lu\n",0xffffffffffffffff);	//输出有符号长整型unsigned long int
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

输出结果:

65
193
32767
65535
9223372036854775807
18446744073709551615
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

注意:
long int到底是32bits还是64bits跟生成的程序是32bits还是64bits一一对应,如果使用g++编译程序的话,可通过-m32-m64选项分别生成32bits和64bits的程序。因本人测试代码编译生成的是64bits的程序,所以long int也就是64btis。

3.转义字符

转义字符在字符串中会被自动转换为相应操作命令。printf()使用的常见转义字符如下:

转义字符意义
\a警报(响铃)符
\b回退符
\f换页符
\n换行符
\r回车符
\t横向制表符
\v纵向制表符
\\反斜杠
"双引号

4.关于 printf 缓冲

在 printf 的实现中,在调用 write 之前先写入 IO 缓冲区,这是一个用户空间的缓冲。系统调用是软中断,频繁调用,需要频繁陷入内核态,这样的效率不是很高,而 printf 实际是向用户空间的 IO 缓冲写,在满足条件的情况下才会调用 write 系统调用,减少IO次数,提高效率。

printf 在 glibc 中默认为行缓冲,遇到以下几种情况会刷新缓冲区,输出内容:
(1)缓冲区填满;
(2)写入的字符中有换行符\n或回车符\r
(3)调用fflush手动刷新缓冲区;
(4)调用scanf要从输入缓冲区中读取数据时,也会将输出缓冲区内的数据刷新。

可使用setbuf(stdout,NULL)关闭行缓冲,或者setbuf(stdout,uBuff)设置新的缓冲区,uBuff为自己指定的缓冲区。也可以使用setvbuf(stdout,NULL,_IOFBF,0);来改变标准输出为全缓冲。全缓冲与行缓冲的区别在于遇到换行符不刷新缓冲区。

printf 在 VC++ 中默认关闭缓冲区,输出时会及时的输到屏幕 [ 3 ] [ 3 ] [ 3 ] [3][3] ^{[3]} [3][3][3][3]。如果显示开启缓冲区,只能设置全缓冲。因为微软闭源,所以无法研究 printf 函数的实现源码。

Linux 和 Windows 下的缓冲区管理可见:C的全缓冲、行缓冲和无缓冲。

5.printf与wprintf不能同时使用

该小结写在2018年1月15日。两年后的今日,在网上苦苦搜索寻求答案,终于解决了之前的疑惑。

在输出宽字符串时,发现将printf和wprintf同时使用时,则后使用的函数没有输出。这里建议不要同时使用printf和wprintf,以免发生错误。

printf和wprintf不能同时输出宽字符串的示例代码如下:

#include <stdio.h>
#include <wchar.h>
#include <locale.h>

int main(int argc,char* argv[])
{
char test[]=“测试Test”;
setlocale(LC_ALL,“zh_CN.UTF-8”);
wchar_t wtest[]=L"0mK0mUTest";
printf(“printf:%S\n”,wtest); //语句1:可正常输出"测试Test"
wprintf(L"wprintf:%S\n",wtest); //语句2:无任何内容输出
}

上面的代码中语句1和语句二不能同时存在,否则只能正常输出第一个。也不知道在Windows平台是否也存在这种问题,有兴趣的读者可以尝试一下。关于原因,GNU官方文档中有明确说明不能同时使用printf与wprintf,参见The GNU C Library Section 12.6 Streams in Internationalized Applications,内容如下:

It is important to never mix the use of wide and not wide operations on a stream. There are no diagnostics issued. The application behavior will simply be strange or the application will simply crash. 

这里是因为输出流在被创建时,不存在流定向,一旦使用了printf(多字节流)或wprintf(宽字符流)后,就被设置为对应的流定向,且无法更改。可以使用如下函数获取当前输出流的流定向。

//
//@param:stream:文件流;mode:取值可以>0,=0或<0
//@ret:<0:流已被设置为多字节流定向;=0:流尚未被设置;>0:流已被设置为宽字符流定向
//
int fwide (FILE* stream, int mode);

//获取当前标准输出流定向
int ret=fwide(stdout,0);

通过fwide可以设置当前流定向,前提是未有任何的I/O操作,也就是当前流尚未被设置任何流定向。顺带吐槽一下,不知为何标准库函数fwide为何实现的如此受限。具体操作如下:

//设置标准输出流定向为多字节流定向
fwide(stdout,-1);

//设置标准输出流定向为宽字符流定向
fwide(stdout,1);

既然GNU C存在这个问题,那该如何解决呢?这里有两种办法:
(1)统一使用一种函数。
例如:

wprintf(L"%s","a\n");
wprintf(L"b\n");

printf("a\n");
printf("%ls\n",L"b");
(2)使用freopen清空流定向。
//重新打开标准输出流,清空流定向
FILE* pFile=freopen("/dev/tty","w",stdout);
wprintf(L"wide freopen succeeded\n");

//重新打开标准输出流,清空流定向
pFile=freopen("/dev/tty",“w”,stdout);
printf(“narrow freopen succeeded\n”);

上面可以让printf与wprintf同时使用。

补充
printf("%I64d\n",1111111111111)或printf(“lld”,111111111111)输出长整型数据
printf("%f",a)输出float或者double,printf的%f说明符的确既可以输出float型又可以输出double型。 根据"默认参数提升"规则(在printf这样的函数的可变参数列表中 ,不论作用域内有没有原型,都适用这一规则)float型会被提升为double型。因此printf()只会看到双精度数。对于scanf,情况就完全不同了,它接受指针,这里没有类似的类型提升。(通过指针)向float存储和向double存储大不一样,因此,scanf区别%f和%lf。

http://chatgpt.dhexx.cn/article/Q3jR4h5Z.shtml

相关文章

【Python笔记】SciPy的统计模块:scipy.stats

【Python笔记】NumPy数组 【DA】数据可视化matplotlib 【Python笔记】pandas常用函数图码总结 SciPy的统计模块是scipy.stats&#xff0c;其中有一个类是连续分布的实现&#xff0c;一个类是离散分布的实现。此外&#xff0c;该模块中还有很多用于统计检验的函数。 # 导入包 f…

使用scipy.signal函数进行信号滤波

目录 1、scipy.signal.filtfilt()函数介绍2、滤波器构造函数(巴特沃斯滤波器)3、如何进行高通、低通、带通、带阻滤波 1、scipy.signal.filtfilt()函数介绍 在信号的滤波过程中&#xff0c;因为scipy.signal.filtfilt()函数可以方便快捷得实现常见的多种滤波功能&#xff0c;所…

SciPy简单应用

SciPy简单应用 SciPy是在NumPy的基础上增加了大量用于数学计算&#xff0c;科学计算以及工程计算的模块&#xff0c;包括线性代数&#xff0c;常微分方程求解&#xff0c;信号处理&#xff0c;图像处理于稀疏矩阵等。参考文档 目录 SciPy简单应用文件输入/输出&#xff1a;scip…

使用scipy来进行曲线拟合

导读 曲线拟合的应用在生活中随处可见&#xff0c;不知道大家是否还记得物理实验中的自由落体运动中下降高度与时间关系之间的探究&#xff0c;在初速度为0的情况下&#xff0c;我们想要探究下降高度与时间的关系。 我们当时采用的方法是通过设置不同的下降时间来记录下降的高…

SciPy 优化

章节 SciPy 介绍SciPy 安装SciPy 基础功能SciPy 特殊函数SciPy k均值聚类SciPy 常量SciPy fftpack(傅里叶变换)SciPy 积分SciPy 插值SciPy 输入输出SciPy 线性代数SciPy 图像处理SciPy 优化SciPy 信号处理SciPy 统计 优化是指在某些约束条件下&#xff0c;求解目标函数最优解的…

Python scipy拟合分布

scipy 拟合分布文档&#xff1a;https://docs.scipy.org/doc/scipy/reference/tutorial/stats.html#fitting-distributions 代码&#xff1a; import numpy as np from scipy import statsnumber np.random.normal(10, 5, 4000) # 生成均值为10&#xff0c;方差为5的正态分布…

scipy求极值代码

1.求解一元函数极值 1.1 导包&#xff1a; from scipy.optimize import fmin这个函数主要用于求某个点附近的极小值。相应的&#xff0c;如果要求某个点附近的最大值&#xff0c;可以使用↓ from scipy.optimize import fmax1.2 定义求导函数 def f(x):return x(x-1) #retu…

scipy.interpolate插值

python SciPy库依赖于NumPy&#xff0c;提供了便捷且快速的N维数组操作。 可以实现插值&#xff0c;积分&#xff0c;优化&#xff0c;图像处理&#xff0c;特殊函数等等操作。 参考官方文档&#xff1a; Interpolation (scipy.interpolate) — SciPy v1.7.1 Manualhttps:/…

Scipy系列目录

Python科学计算和数据分析库系列目录 Scipy简介 Scipy是基于Numpy的科学计算工具库&#xff0c;方便、易于使用、专为科学和工程设计&#xff0c;是一个用于数学、科学、工程领域的常用软件包。 Scipy提供了许多用户友好和高效的高阶方法&#xff0c;如插值&#xff0c;积分&…

scipy笔记:FFT

数学笔记&#xff1b;离散傅里叶变化 DFT_UQI-LIUWJ的博客-CSDN博客 数学笔记&#xff1a;FFT&#xff08;快速傅里叶变换&#xff09;_快速傅里叶变换矩阵_UQI-LIUWJ的博客-CSDN博客 【个人理解&#xff1a;FFT是DFT的一种优化&#xff0c;DFT需要N个谱域信号来表示N个时域信…

Scipy简介

Scipy简介 Scipy依赖于NumpyScipy包含的功能&#xff1a;最优化、线性代数、积分、插值、拟合、特殊函数、快速傅里叶变换、信号处理、图像处理、常微分方程求解器等应用场景:Scipy是高端科学计算工具包&#xff0c;用于数学、科学、工程学等领域Scipy由一些特定功能的子模块组…

系统工程--011详细设计 伪码 程序流程图 PAD图 N-S图 判断表和判断树

详细设计 三种控制结构&#xff1a;顺序、选择、循环 一个程序的代码块仅仅通过顺序、选择和循环&#xff0c;3种基本控制结构进行连接&#xff0c;并每个代码块只有一个入口和一个出口伪码 程序流程图 PAD图 N-S图 判断表和判断树

程序流程图、N-S图、PAD图

在需求分阶段经常使用3种方法去剖析我们所面对的业务。 程序流程图 任何复杂的程序图都应由5种基本控制结构组成或嵌套而成。 盒图&#xff08;N-S图&#xff09; Nassi和Scheiderman提出了一种符合结构化程序设计原则的图形描述工具&#xff0c;叫作盒图&#xff0c;也叫做N…

PAD图初认识

程序流程图&N-S图&PAD图 程序流程图 任何复杂的程序图都应由5种基本控制结构组成或嵌套而成。 盒图&#xff08;N-S图&#xff09; Nassi和Scheiderman提出了一种符合结构化程序设计原则的图形描述工具&#xff0c;叫作盒图&#xff0c;也叫做N-S图。任何一个N-S图&a…

python numpy 图片 pad 参数详解

python numpy 图片 pad 参数详解 ‘constant’, ‘edge’, ‘linear_ramp’, ‘maximum’, ‘mean’, ‘median’, ‘minimum’, ‘reflect’, ‘symmetric’, ‘wrap’, ‘empty’ 光看文档不太好理解&#xff0c;因为在网上找不到详细的例子&#xff0c;所以我就自己动手做了…

E-R图、N-S图、PAD图、程序流程图

E-R图&#xff1a; E-R图也称实体-联系图(Entity Relationship Diagram)&#xff0c;提供了表示实体类型、属性和联系的方法&#xff0c;用来描述现实世界的概念模型。 矩形框&#xff1a;表示实体&#xff0c;在框中记入实体名。 菱形框&#xff1a;表示联系&#xff0c;在框…

np.pad()用于卷积网络中对图片进行填充

有一张RGB的图像&#xff0c;我们要在这个图像的周围加上填充元素&#xff0c;使得这个图像不会再卷积操作后导致边缘信息丢失和图像尺寸的减小。 为此&#xff0c;我们需要padding操作&#xff0c;numpy库中对这个进行了封装numpy.pad()函数&#xff1a; 对一个一维数组来说…

numpy.pad对图片进行填充

一、接口 pad(array, pad_width, mode, **kwargs) 其中&#xff0c;第一个参数是输入数组&#xff1b; 第二个参数是需要pad的值&#xff0c;参数输入方式为&#xff1a;&#xff08;(before_1, after_1), … (before_N, after_N)&#xff09;&#xff0c;其中(before_1, af…