dc采用逆波兰式计算表达式,计算过程是对栈的手动维护。逆波兰式看起来很别扭,但跑起来当然是很有效率的。
乱七八糟的东西就不说了,直接切入script正题

负号在dc里用_表示而非-,因为-是二元运算符“减号”

打印类命令:
p            :打印栈顶元素并换行
n            : 打印栈顶元素并将其弹出栈,完毕后不换行
P            : putchar ( int(栈顶元素) % 256) 并弹栈顶,不换行
f            : 从栈顶至栈底打印栈中所有值,每个一行

算数运算符:
+            : 依次弹出w1与w2,将w2+w1压栈。精度为结果值精度
-            : 依次弹出w1与w2,将w2-w1压栈
*            : 依次弹出w1与w2,将w2*w1压栈。精度为结果值精度与precision中较大值
/            : 依次弹出w1与w2,将w2/w1压栈。精度为precision
%            : 依次弹出w1与w2,将w2-w2/w1*w1压栈
~            : 依次弹出w1与w2,依次将w2/w1与w2%w1压栈
^            : 依次弹出w1与w2,将w2^((int)w1)压栈。精度为w2精度与precision中较大值
|            : 依次弹出w1 w2与w3,将 w3 ^ ((int)w2) (mod w1) 压栈。w1 w3 需为整数
v            : 弹出w1,将sqrt(v)压栈。精度为precision

栈操作:
c            : 清空栈
d            : 将栈顶元素复制并压栈
r            : 交换栈顶两元素 XXX

寄存器操作:
dc提供至少256个寄存器,分别用256个字符表示。一般用'a'..'z',不够可以用123ABC!@#,再不够就用\x00..\xff
除了主栈外,每个寄存器都是独立的栈,寄存器栈顶的值被称为这个寄存器的值
sr           : 弹出主栈顶元素w1,若r栈为空,则将w1压入r栈,否则将r栈顶元素改为w1
lr           : 将r寄存器的值压入主栈
Sr           : 弹出主栈顶元素w1,压入r栈
Lr           : 弹出r栈顶元素w1并将其压入主栈

参数:
dc有三个参数:输入进制 输出进制与precision(精度)。precision永远被作为10进制来看。
i            : 弹出栈顶w1并将其作为输入进制
o            : 弹出栈顶w1并将其作为输出进制
k            : 弹出栈顶w1并将其作为precision
I            : 将输入进制压栈
O            : 将输出进制压栈
K            : 将precision压栈

字符串:
dc寄存器可以存放字符串,字符串可以被打印或被作为dc命令去执行
[characters] : 将字符串characters压栈。characters中可以包含任何字符,除了[]需两两匹配
a            : 若栈顶w1是数字,则将栈顶修改为(char)((int)(w1)%256),否则将w1首字母入栈
x            : 若栈顶w1是数字,则什么都不做,否则将其弹出并将w1作为命令交给dc执行
>r           : 依次弹出w1与w2,若w1>w2则执行r寄存器中的命令
!>r          : 依次弹出w1与w2,若w1<=w2则执行r寄存器中的命令
<r           : 依次弹出w1与w2,若w1<w2则执行r寄存器中的命令
!<r          : 依次弹出w1与w2,若w1>=w2则执行r寄存器中的命令
=r           : 依次弹出w1与w2,若w1==w2则执行r寄存器中的命令
!=r          : 依次弹出w1与w2,若w1!=w2则执行r寄存器中的命令
?            : 从标准输入读入一条命令并将其交给dc执行
q            : 终止当前dc与当前dc的调用dc
Q            : 弹出栈顶w1,终止w1层dc。当w1大于当前dc层数时会报错,因此Q不会使主dc退出

状态查询:
Z            : 弹出栈顶w1,将w1长度压栈
X            : 弹出栈顶w1,若w1为数字,则将w1小数位数压栈,否则将0压栈
z            : 将当前栈深度压栈

其他:
!            : 将!之后的部分作为命令交给sh执行。若命令以'>' '='开头,则需在!后加上空格
#            : 将#之后的部分作为注释
:r           : 依次弹出w1与w2,r[w1]=w2
;r           : 弹出栈顶w1,将r[w1]压栈

真正的精髓好像是通过x与opr进行的流程控制,等着再看吧
Logo

更多推荐