Python中实现输入超时及如何通过变量获取变量名


Posted in Python onJanuary 18, 2020

背景介绍

开发中遇到了一个需求:程序运行到某处时需要用户确认, 但不能一直傻等, 后面的程序不能被一直阻塞, 需要有个超时限制, 也就是这个程序如果在一段时间后还没有得到用户输入就执行默认操作.

解决思路 ? 多线程法

我就想到了用多线程的方式, 开启一个子线程用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()[待改的变量的原名]的值就能通过指定变量名来修改变量了.

Python中实现输入超时及如何通过变量获取变量名

通过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中实现输入超时及如何通过变量获取变量的名字,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

Python 相关文章推荐
剖析Python的Tornado框架中session支持的实现代码
Aug 21 Python
Python中使用asyncio 封装文件读写
Sep 11 Python
玩转python selenium鼠标键盘操作(ActionChains)
Apr 12 Python
Python上下文管理器和with块详解
Sep 09 Python
Django跨域请求问题的解决方法示例
Jun 16 Python
Python基本数据结构之字典类型dict用法分析
Jun 08 Python
基于python实现语音录入识别代码实例
Jan 17 Python
python环境下安装opencv库的方法
Mar 05 Python
pygame实现飞机大战
Mar 11 Python
Python Opencv中用compareHist函数进行直方图比较对比图片
Apr 07 Python
服务器端jupyter notebook映射到本地浏览器的操作
Apr 14 Python
如何通过python计算圆周率PI
Nov 11 Python
Pytorch 计算误判率,计算准确率,计算召回率的例子
Jan 18 #Python
python:目标检测模型预测准确度计算方式(基于IoU)
Jan 18 #Python
Python实现计算长方形面积(带参数函数demo)
Jan 18 #Python
python实现的Iou与Giou代码
Jan 18 #Python
Python 简单计算要求形状面积的实例
Jan 18 #Python
python实现用类读取文件数据并计算矩形面积
Jan 18 #Python
python不使用for计算两组、多个矩形两两间的iou方式
Jan 18 #Python
You might like
用PHP开发GUI
2006/10/09 PHP
PHP SQLite类
2009/05/07 PHP
php自定义函数call_user_func和call_user_func_array详解
2011/07/14 PHP
php数组中删除元素的实现代码
2012/06/22 PHP
用C/C++扩展你的PHP 为你的php增加功能
2012/09/06 PHP
php分页示例分享
2014/04/30 PHP
smarty内置函数{loteral}、{ldelim}和{rdelim}用法实例
2015/01/22 PHP
RSA实现JS前端加密与PHP后端解密功能示例
2019/08/05 PHP
用 javascript 实现的点击复制代码
2007/03/24 Javascript
Javascript中Event属性搜集整理
2013/09/17 Javascript
JavaScript中实现异步编程模式的4种方法
2014/09/24 Javascript
浅谈jQuery事件绑定原理
2015/01/02 Javascript
如何实现chrome浏览器关闭页面时弹出“确定要离开此面吗?”
2015/03/05 Javascript
新手快速学习JavaScript免费教程资源汇总
2015/06/25 Javascript
JS与Ajax Get和Post在使用上的区别实例详解
2016/06/08 Javascript
jQuery Dialog 打开时自动聚焦的解决方法(两种方法)
2016/11/24 Javascript
在javascript中,null>=0 为真,null==0却为假,null的值详解
2017/02/22 Javascript
jQuery中用on绑定事件时需注意的事项
2017/03/19 Javascript
微信小程序自定义底部弹出框功能
2020/11/18 Javascript
详解python之简单主机批量管理工具
2017/01/27 Python
python3中获取文件当前绝对路径的两种方法
2018/04/26 Python
python隐藏类中属性的3种实现方法
2019/12/19 Python
python软件都是免费的吗
2020/06/18 Python
Python爬虫教程知识点总结
2020/10/19 Python
世界上最好的旅行夹克:BauBax
2018/12/23 全球购物
美国糖果店:Sugarfina
2019/02/21 全球购物
LivingSocial英国:英国本地优惠
2019/02/22 全球购物
俄罗斯建筑和装饰材料在线商店:Stroilandia
2020/07/25 全球购物
Swanson中国官网:美国斯旺森健康产品公司
2021/03/01 全球购物
strlen的几种不同实现方法
2013/05/31 面试题
教师的实习鉴定
2013/12/15 职场文书
小学生美德少年事迹材料
2014/08/24 职场文书
中学生检讨书1000字
2014/10/28 职场文书
2014年驻村干部工作总结
2014/11/17 职场文书
2015年教导处教学工作总结
2015/07/22 职场文书
关于EntityWrapper的in用法
2022/03/22 Java/Android