Python中实现输入超时及如何通过变量获取变量名
作者:佚之狗 发布时间:2021-02-17 03:17:48
背景介绍
开发中遇到了一个需求:程序运行到某处时需要用户确认, 但不能一直傻等, 后面的程序不能被一直阻塞, 需要有个超时限制, 也就是这个程序如果在一段时间后还没有得到用户输入就执行默认操作.
解决思路 – 多线程法
我就想到了用多线程的方式, 开启一个子线程用stdin(比如python的input函数)获取用户输入, 主线程里设置线程启动和超时.
创建线程
Python中使用多线程很方便, threading.Threaded(函数, 参数表)然后thread.start就好了. 只是有一点需要注意, 上面参数表必须是个元组, 也就是每个元素后面必须跟个逗号.
import threading
def anyFunction(aaa):
return str(aaa) #某种处理结果, 比如字符串
def manualInput(xxx):
data = input("请输入%s: " % xxx)
pass # 各种处理(比如数据转换什么的)
exampleThread = threading.Thread(target = manualInput, args = ( anyFunction("汪汪汪"), ), name = "喵喵喵")
exampleThread.start()
通过函数修改某个指定的(通过名字即字符串)变量的值
但这又出来一个问题, 如果不能使用全局变量, 该如何在另一个函数里修改其参数对应的内容呢? 这里的重点归结起来是"函数如何修改自身参数的内容".
于是我想到了一个骚透了的方法——改变量字典…… 因为python的变量是基于标签的. python中的变量大致可以理解成给内容贴上标签(每个标签对应一个变量名, 多个标签可能会引用同一个内容, 没被引用的内容就会被python释放), 每个标签都会有一个id(同时, 一个内存数据只要被引用那么自身也有个id). 示例:
print(id("喵喵喵"),"~~", id("喵喵喵"),"~~", id("喵喵喵"),"~~")
[Out]:
1392371317520 ~~ 1392371317520 ~~ 1392371317520 ~~
print(id("喵喵喵")); print(id("喵喵喵")); print(id("喵喵喵"))
[Out]:
1392371318576
1392371318000
1392371318288
python维护这些标签和内容的对应关系可以通过字典的方式来读取和修改, 改globals()[待改的变量的原名]的值就能通过指定变量名来修改变量了.
通过globals的字典修改变量
通过变量来获取变量的名字(字符串)
上面通过globals()[待改的变量的原名] = 新的内容的方式实现了修改变量的内容, 可是, 待改的变量的原名是个字符串, 怎么通过变量得到这个变量的名字呢?
一个思路是字典法.
把当前运行环境中的所有变量复制一份(浅拷贝和深拷贝效果都一样, 因为深浅拷贝前后都是相同的标签), 然后新建一个"标签id-变量名"的对照表字典, 利用字典赋值的特性, 遍历复制来的全局变量, 把id(变量值)作为key而变量名作为value, 即标签id-变量名字典[id(变量值)] = 变量名.
test = "some values"
变量A = "汪汪汪"
当前所有变量 = globals().copy()
print(当前所有变量)
[Out]:
{'__name__': '__main__',..., 'test': 'some values', '变量A': '汪汪汪'}
内容_变量名字符串对照表 = {}
for 变量名, 变量值 in 当前所有变量.items():
内容_变量名字符串对照表[id(变量值)] = 变量名
print(内容_变量名字符串对照表)
[Out]:
{2437914516272: '__name__',..., 2437948462576: 'test', 2437948432816:'变量A'}
这样一来就建立一个内容-变量名字符串的对照表, 又因为id(变量A) 和 id(变量A的值)是相等, 利用这个特性就能通过变量来取变量值了.
变量A的值 = 变量A
print(id(变量A的值)
[Out]:
2437948432816
内容_变量名字符串对照表[id(变量A的值)]
[Out]:
'变量A'
通过函数修改变量
上面这一堆头发就是为了动态、通用地修改变量, 封装成函数就能在任何地方调用和修改了.
def 一个实现变量修改函数(要改的变量, 提示语):
当前所有变量 = globals().copy()
变量id表 = {}
for 变量名, 变量值 in 当前所有变量.items():
变量id表[id(变量值)] = 变量名
待改的变量的原名 = 变量id表[id(要改的变量)]
新的内容 = str(input(提示语))
if len(新的内容) > 0 :
globals()[待改的变量的原名] = 新的内容
return 待改的变量的原名
tmp = "汪汪汪"
一个实现变量修改函数(tmp, "请输入新值: ")
[Out]:
请输入新值: 喵喵喵
'tmp'
print(tmp)
[Out]:
喵喵喵
总结(demo)[不想看中间过程的话可以直接看这]
import time, threading
# 这里的demo是为了通用化. 因为在一个线程中再嵌套另个线程的话, 嵌套的线程获取不到所有变量
class ThreadWithReturn(threading.Thread):
def __init__( self, target = None, args = () ):
super(ThreadWithReturn,self).__init__()
self.func = target
self.args = args
def run(self):
self.result = self.func(*self.args)
def getResult(self):
try:
return self.result
except Exception as errInfo:
print("遇到错误: ", errInfo)
return None
def 一个实现变量修改函数(要改的变量, 提示语):
当前所有变量 = globals().copy()
变量id表 = {}
for 变量名, 变量值 in 当前所有变量.items():
变量id表[id(变量值)] = 变量名
try:
待改的变量的原名 = 变量id表[id(要改的变量)]
except KeyError:
print("***debug: 在不同的线程中运行, 获取不到出入变量的名字")
待改的变量的原名 = None
新的内容 = str(input(提示语))
if len(新的内容) > 0 :
if 待改的变量的原名 != None:
globals()[待改的变量的原名] = 新的内容
else:
新的内容 = None
return [待改的变量的原名, 新的内容]
def Gexit():
exitConfirm = "u"
waitForConirm = ThreadWithReturn( target = 一个实现变量修改函数, args = (exitConfirm, "收到了退出信号, 默认30秒后退出, 是否现在退出呢? (Y/n) 请输入: ",) )
waitForConirm.start()
waitForConirm.join(30)
try:
exitConfirm = waitForConirm.getResult()[1]
print("***debug, got:", exitConfirm)
except Exception as errInfo:
print("***debug:", errInfo)
exitConfirm = "u"
if exitConfirm == "u":
print("等待超时, 开始退出流程...")
exitConfirm = "Ytt"
if exitConfirm == "Ytt" or exitConfirm == "Y":
if exitConfirm == "Y":
print("确认退出, 开始退出流程...")
pass # 这里放程序退出逻辑
if exitConfirm == "n":
print("取消退出, 继续运行...")
pass # 这里放继续运行的逻辑
return 0
Thread_waitForExit = threading.Thread(target = Gexit, args = ())
Thread_waitForExit.start()
Thread_waitForExit.join(45)
总结
以上所述是小编给大家介绍的Python中实现输入超时及如何通过变量获取变量的名字网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!
来源:https://blog.csdn.net/qq_39707057/article/details/104002512


猜你喜欢
- 本文实例为大家分享了基于神经卷积网络的人脸识别,供大家参考,具体内容如下1.人脸识别整体设计方案客_服交互流程图:2.服务端代码展示sk =
- 一、MySQL中如何表示当前时间?其实,表达方式还是蛮多的,汇总如下:CURRENT_TIMESTAMPCURRENT_TIMESTAMP(
- ref:被用来给元素或子组件注册引用信息,引用信息将会注册在父组件的$refs对象上。如果在普通的DOM元素上使用,那么指向的就是普通的DO
- 按照CSS规范,浮动元素(floats)会被移出文档流,不会影响到块状盒子的布局而只会影响内联盒子(通常是文本)的排列。因此当其高度超出包含
- 导入注意点:使用phpmyadmin或navicat之类的工具的导入功能还是会相当慢,可以直接使用mysql进行导入导入命令如下:mysql
- 前言最近在学习微信小程序,在学习的过程中,有很多好玩的 API,经常点外卖的同学可能在小程序中遇到过,比如:某团、某了么都是有显示当前用户位
- 公用表表达式简介:公用表表达式 (CTE) 可以认为是在单个 SELECT、INSERT、UPDATE、DELETE 或 CREATE VI
- 使用python里的indexnums = [1, 2, 3, 4, 5, 6, 1, 9]print nums.index(max(num
- 前言go mod tidy的作用是把项目所需要的依赖添加到go.mod,并删除go.mod中,没有被项目使用的依赖。Tidy makes s
- 内容摘要:近年来,视频点播VOD(Video on Demand)的名字在媒体上出现得越来越多。VOD技术使人们可以根据
- 一.Pytorch虚拟环境简介Torch是一个用于深度学习的=数学计算库,而Pytorch则是一个基于Torch的Python机器学习库,可
- 问题你想定义一个函数或者方法,它的一个或多个参数是可选的并且有一个默认值。解决方案定义一个有可选参数的函数是非常简单的,直接在函数定义中给参
- 本文实例讲述了python定时执行指定函数的方法。分享给大家供大家参考。具体实现方法如下:# time a function using t
- golang作为一热门的兼顾性能 效率的热门语言,相信很多人都知道,在编程语言排行榜上一直都是很亮眼,作为一门强类型语言,二进制位的操作肯定
- 桥接模式Bridge Pattern是什么桥接模式是一种结构型模式,它将抽象部分与实现部分分离开来,使它们可以独立地变化。在桥接模式中,我们
- Microsoft office Access Database Engine 错误 '80040e14' Insert I
- 1.QLineEditQLineEdit.text()#输出str类型2.QCheckBoxQCheckBox.checkState()#状
- 本文将介绍如何使用公司运行服务器进行开发调试,以及使用远程服务器python解释器,整理了对应的配置流程。进入配置页面Pycharm菜单栏,
- 关于 *args与**args的用法*args 和 **kwargs主要用于函数定义,你可以将不定数量的参数传递给某个函数。*args*ar
- 首先 跳过权限表模式启动MySQL:mysqld --skip-grant-tables &从现在开始,你将踏入第一个坑