printf command in shell script

printf命令模仿C程序库(library)里的printf()库程序library routine)。它几乎复制了

该函数的所有功能。不过在Shell层级的版本上,会有些差异。由于printf的行为是由POSIX

标准所定义,因此使用printf的脚本比使用echo移植性好

如同echo命令,printf命令可以输出简单的字符串:

[root@master ~]#printf "Hello, Shell\n"
Hello, Shell
[root@master ~]#

你应该可以马上发现,最大的不同在于:printf不像echo那样会自动提供一个换行符号。你必须显式

地将换行符号指定成\n


printf命令的完整语法有两个部分:

printf format-string [arguments...]

第一部分为描述格式规格的字符串,用来描述输出的排列方式,最好为此字符串加上引号。此字符

串包含按字面显示的字符以及格式声明,格式声明时特殊的占位符,用来描述如何显示相应的参数。

第二部分是与格式声明相对应的参数列表,例如一系列的字符串或变量值。格式声明由两部分组成:

百分比符号(%)和指示符。最常用的格式指示符有两个%s用于字符串,而%d用于十进制整数


格式字符串中,一般字符会按字面显示。转义序列则像echo那样,解释后再输出成相应的字符。格

式声明以%符号开头,并以定义的字母集中的一个来结束,用来控制相应参数的输出。例如%s用于

字符串的输出:

[root@master ~]# printf "The first program always prints'%s,%d\n'" Hello Shell

输出结果为:

-bash: printf: Shell: invalid number
The first program always prints 'Hello,0

'[root@master ~]#

当尝试以%d的格式来显示字符串Shell时,会发现有警告,提示Shell为无效的数字,此时会打印出

默认值0;从这个试验中可以看出来:%s, %s两侧的单引号''并不是必须的。


printf的转义序列

序列                     说明

\a                        警告字符,通常为ASCIIBEL字符

\b                       后退

\c                       抑制(不显示)输出结果中任何结尾的换行字符(只在%b格式指示符控制下的

                         参数字符串中有效),而且,任何留在参数里的字符、任何接下来的参数以及

                         任何留在格式字符串中的字符,都被忽略

\f                      换页(formfeed

\n                    换行

\r                      回车(Carriage return

\t                      水平制表符

\v                    垂直制表符

\\                      一个字面上的反斜杠字符

\ddd                表示13位数八进制值的字符。仅在格式字符串中有效

\0ddd              表示13位的八进制值字符

 

默认情况下,转义序列只在格式字符串中会被特别对待,也就是说,如果转义序列出现在参数列表

的字符串中,将不会被解释

[root@master ~]#printf "a string, no processing:<%s>\n" "A\nB"

当你使用%b格式指示符时,printf会解释参数字符串里的转义序列:

[root@master ~]#printf "a string, no processing:<%b>\n" "A\nB"

无论时在格式字符串内还是在使用%b所打印的参数字符串里,大部分的转义序列都是被相同对待。

无论如何,\c\0ddd只有搭配%b使用才有效,而\ddd只有在格式字符串里才会被解释

 

现在来试试这些转义序列的效果

\a     警告字符,通常为ASCIIBEL字符

ASCII中的BEL代表的是beep,如果你在终端输入:

[root@master ~]#printf "test \a"
test [root@master ~]#

应该会听到一声beep。在我的电脑上前两天还能听到,今天测试时没有声音了,可能是系统的

设定有改变。


\b     后退

首先新建一个名为“sh_printf_b.sh”的文档,内容如下:

#! /bin/bash

printf "1";

sleep 1;

printf "\b";

sleep 1;

printf "2";

sleep 1;

printf "\b";

sleep 1;

printf "3";

sleep 1;

printf "\n"

其运行结果如下图所示:

结果如下图显示,不过每次只是出现一个数字,先是出现1,并且光标在1的右边;接着还是1,光标在1上面;

然后是2,光标在2的右边;接着还是2,光标在2的右边;然后是3,光标在3的右边;然后光标还是在3的右边跳动一下立马就退出了!


\c    抑制(不显示)输出结果中任何结尾的换行字符(只在%b格式指示符控制下的

      参数字符串中有效),而且,任何留在参数里的字符、任何接下来的参数以及

      任何留在格式字符串中的字符,都被忽略

\c要与%b配合使用,请参考下图,其中%b两侧的尖括号并不是必须的:

[root@master ~]# printf "123<%s>\n" "456" 
123<456>

[root@master ~]# printf "123%b\n" "\c456"   \c 抑制(不显示)输出结果中任何结尾的换行字符】
123[root@master ~]#

[root@master ~]# printf "123%b\n" "\n456" 【只在%b格式指示符控制下的参数字符串中有效】
123
456

[root@master ~]# printf "123%b\n" "456" "\c213" 【因为有两个参数,第二个没有说明就不处理】
123456
[root@master ~]#
[root@master ~]# printf "123%s%b\n" "\c456" 【后面如果还有参数则被忽略】
123\c456
[root@master ~]# printf "123%s%b\n" "\c456" "\c213" 【后面的那个"\c213"被忽略了】
123\c456[root@master ~]#

[root@master ~]# printf "123%s\n%b\n" "\c456" "\c213"【这里多了个\n效果就不一样了】
123\c456
[root@master ~]#


\f                      换页(formfeed

在我系统上的效果如下:

[root@master ~]# printf "123\f456\n"
123
   456
[root@master ~]#


\n                     换行  


\r                      回车(Carriage return

创建一个名为“sh_printf_r.sh”的文档,内容如下:

#! /bin/bash

printf "111111111";

sleep 1;

printf "\r";

printf "2";

sleep 1;

printf "2";

sleep 1;

printf "2";

sleep 1;

printf "2";

sleep 1;

printf "2";

sleep 1;

printf "2";

sleep 1;

printf "\r"

sleep 1;

printf "3";

sleep 1;

printf "3";

sleep 1;

printf "3";

sleep 1;

printf "\n"

其运行效果如下图所示:
每次只出现一行,并且一行出现9个数字,注意光标的移动,黄色的是光标每次执行所处的位置;
 

 

\t                      水平制表符

\v                     垂直制表符

\\                      一个字面上的反斜杠字符

[root@master ~]# printf "\\101\n"
A
[root@master ~]#

由此我们可以知道输出的是ASCII码

printf格式指示符

项目     说明

%b          相对应的参数被视为含有要被处理的转义序列之字符串

%c           ASCII字符。显示相对应参数的第一个字符

%d, %i    十进制整数

%e          浮点格式

%E          浮点格式

%f           浮点格式

%g          %e%f转换,看哪一个较短,则删除结尾的零

%G          %E%f转换,看哪一个较短,则删除结尾的零

%o           不带正负号的八进制值

%s           字符串

%u          不带正负号的十进制值

%x          不带正负号的十六进制值,使用af表示1015

%X          不带正负号的十六进制值,使用AF表示1015

%%         字面意义的%

根据POSIX标准:浮点格式%e%E%f%g%G是“不需要被支持”。

这是因为awk支持浮点预算,且有它自己的printf语句。这样Shell程序中

需要将浮点数值进行格式化的打印时,可使用小型的awk程序实现。然而,内建

bashksh93zsh中的printf命令都支持浮点格式


printf命令可用来指定输出字段的宽度以及进行对其操作。为实现此目的,接在%后

面的格式表达式可采用三个可选用的修饰符(modifier)以及前置的格式指示符(

format specifier):

     %flags width.precision format-specifier

输出字段的width为数字值。指定字段宽度时,字段的内容默认为向右对齐,如果你希望

文字向左靠,必须指定-标志。这样:“%-20s”会在一个有20个字符宽度的字段里,输出

一个向左对齐的字符串。如果字符串少于20个字符,则字段将以空白填满。下面的例子里,

|是输出,以表示字段的实际宽度。第一个例子为向右对齐文字:


下一个例子则为向左对齐文字:

precision修饰符是可选用的。对十进制或浮点数值而言,它可以控制数字出现于结果中的位数。
对于字符串值而言,它控制将要打印的字符串的最大字符数。具体的含义会因格式指示符而有不
同,如下所示:

精度的意义
转换                                        精度含义
%d,%i,%o,%u,%x,%X           要打印的最小位数。当值的位数小于此数字时,会在前面补零。
                                             默认精度(precision)为1
%e,%E                                要打印的最小位数。当值的位数小于此数字时,会在小数点后面
                                             补零,默认精度为6.精度为0时则表示不显示小数点
%f                                         小数点右边的位数
%g,%G                               有效位数(significant digit)的最大数目
%s                                        要打印字符的最大数目

下面是几个精度的例子:

对于转换指示符%b,%c与%s而言,相对应的参数都视为字符串。否则,它们会被解释为C语言的数字常数(
以0开头的为八进制,以0x或0X开头的为十六进制)。更进一步的说,如果参数的第一个字符为单引号或
双引号,则相对应的数值是字符串的第二个字符的ASCII值:
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值