python c cmd_在Python cmd模块中处理CTRL-C

我发现了一些使用Ctrl-C实现所需行为的hacky方法.

设置use_rawinput = False并替换stdin

这个(或多或少……)粘在cmd.Cmd的公共接口上.不幸的是,它禁用了readline支持.

您可以将use_rawinput设置为false并传递一个不同的类文件对象来替换Cmd .__ init __()中的stdin.实际上,只在此对象上调用readline().因此,您可以为stdin创建一个包装器,捕获KeyboardInterrupt并执行您想要的行为:

class _Wrapper:

def __init__(self,fd):

self.fd = fd

def readline(self,*args):

try:

return self.fd.readline(*args)

except KeyboardInterrupt:

print()

return '\n'

class TestShell(cmd.Cmd):

def __init__(self):

super().__init__(stdin=_Wrapper(sys.stdin))

self.use_rawinput = False

self.prompt = '$'

def precmd(self,line):

if line == 'EOF':

return 'exit'

return line

def emptyline(self):

pass

def do_exit(self,line):

return True

TestShell().cmdloop()

当我在终端上运行它时,Ctrl-C显示^ C并切换到新行.

猴子补丁输入()

如果你想要input()的结果,除了你想要Ctrl-C的不同行为,一种方法是使用不同的函数而不是input():

def my_input(*args): # input() takes either no args or one non-keyword arg

try:

return input(*args)

except KeyboardInterrupt:

print('^C') # on my system,input() doesn't show the ^C

return '\n'

但是,如果您只是盲目地设置input = my_input,则会得到无限递归,因为my_input()将调用input(),现在它本身就是.但这是可以修复的,你可以在cmd模块中修补__builtins__字典,以便在Cmd.cmdloop()期间使用修改后的input()方法:

def input_swallowing_interrupt(_input):

def _input_swallowing_interrupt(*args):

try:

return _input(*args)

except KeyboardInterrupt:

print('^C')

return '\n'

return _input_swallowing_interrupt

class TestShell(cmd.Cmd):

def cmdloop(self,*args,**kwargs):

old_input_fn = cmd.__builtins__['input']

cmd.__builtins__['input'] = input_swallowing_interrupt(old_input_fn)

try:

super().cmdloop(*args,**kwargs)

finally:

cmd.__builtins__['input'] = old_input_fn

# ...

请注意,这会更改所有Cmd对象的input(),而不仅仅是TestShell对象.如果这是你不能接受的,你可以……

复制Cmd.cmdloop()源并进行修改

由于你是子类化它,你可以让cmdloop()做你想做的任何事情. “你想要的任何东西”可能包括复制Cmd.cmdloop()的部分并重写其他部分.通过调用另一个函数替换对input()的调用,或者在重写的cmdloop()中捕获并处理KeyboardInterrupt.

如果您害怕使用新版本的Python更改底层实现,您可以将整个cmd模块复制到一个新模块中,并更改您想要的内容.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值