如何通过键盘事件中断计算循环并返回参数输入阶段

发布时间 - 2025-12-29 00:00:00    点击率:

本文介绍在python控制台程序中,利用input()捕获特定关键词(如"reset"或"quit")实现循环中断与流程跳转的方法,无需额外依赖第三方键盘监听库,安全、简洁、兼容性强。

在您的辐射剂量计算脚本中,核心需求是:在持续输入 I1/I2 的 while True 计算循环中,随时通过键盘触发“返回上一级——重新选择 DTU 编号并重置参数”。但直接使用 keyboard 库监听按键(如 keyboard.is_pressed('esc'))在此类阻塞式 input() 场景下往往失效——因为 input() 会独占终端输入流,keyboard 无法同步捕获;且需管理员权限,在部分环境(如某些IDE、远程终端)中不可靠。

推荐方案:基于 input() 的语义化中断指令(零依赖、高兼容、易维护)

该方法不引入复杂异步逻辑,而是将“中断动作”转化为用户主动输入的可识别字符串命令(如 "reset"、"back"、"quit"),天然适配 input() 的阻塞特性,稳定可靠。

✅ 实现步骤(最小侵入式修改)

  1. 在外层 DTU 选择逻辑外包裹主循环
    将整个 DTU 选择 → 参数加载 → K_group 设置 → 计算循环 流程,置于一个 while True: 主循环内,便于统一控制流程跳转。

  2. 在 K_group 输入处支持 "quit" 退出整个程序

    # 替换原代码中:n = int(input("K_group = "))
    n_input = input("K_group = ").strip()
    if n_input.lower() in ["quit", "exit", "q"]:
        print("Выход из программы.")
        break  # 退出主循环,结束程序
    n = int(n_input)
  3. 在 I1 = 输入处支持 "reset" 返回 DTU 选择

    while True:
        i1_str = input("I1 = ").strip()
        if i1_str.lower() == "reset":
            print("→ Возврат к выбору ДТУ...")
            break  # 跳出内层计算循环,回到外层主循环开头(即重新选 DTU)
    
        # 后续处理 i1_str(逗号替换、转 float 等)...
        if ',' in i1_str:
            i1_str = i1_str.replace(',', '.')
        try:
            i_1 = float(i1_str)
        except ValueError:
            print(f"{Fore.RED}Ошибка: введите число для I1{Style.RESET_ALL}")
            continue
    
        # 同理处理 i2_str,并执行剂量计算与输出...
    ⚠️ 注意:break 在此处跳出的是内层 while True:(计算循环),控制权自然回到外层主循环开头,从而重新执行 dtu = int(input("Номер ДТУ = ")),完美实现“返回 #DTU_01(start)”。

✅ 完整关键结构示意(整合后逻辑)

# ... [json 加载、参数预读等初始化代码] ...

while True:  # ← 主循环:控制整体流程(DTU → K_group → 计算)
    print("\n" + "="*40)

    # DTU-01(start) —— 此处可被 reset 触发重新进入
    dtu_input = input("Номер ДТУ = ").strip()
    if dtu_input.lower() in ["quit", "exit"]:
        print("Программа завершена.")
        break

    try:
        dtu = int(dtu_input)
    except ValueError:
        print(f"{Fore.RED}Ошибка: введите число для номера ДТУ{Style.RESET_ALL}")
        continue

    # ... [根据 dtu 加载 a_m, b_m, k 等参数] ...

    # KdrDf(start)
    n_input = input("K_group = ").strip()
    if n_input.lower() in ["quit", "exit"]:
        break
    try:
        n = int(n_input)
        ks = k[n]
    except (ValueError, IndexError):
        print(f"{Fore.RED}Ошибка: некорректный K_group или индекс вне диапазона{Style.RESET_ALL}")
        continue

    k_dr = float(input('K_dreif = '))
    dfon = float(input("Dfon = "))

    # 内层计算循环(可被 reset 中断)
    while True:
        i1_str = input("I1 = ").strip()
        if i1_str.lower() == "reset":
            print("? Сброс параметров. Возврат к выбору ДТУ...")
            break  # ← 关键:跳出此循环,回到外层 while True 开头

        # 处理 I1(含逗号替换、类型转换、异常检查)
        i1_clean = i1_str.replace(',', '.')
        try:
            i_1 = float(i1_clean)
        except ValueError:
            print(f"{Fore.RED}Ошибка ввода I1: введите число{Style.RESET_ALL}")
            continue

        i2_str = input("I2 = ").strip()
        if i2_str.lower() == "reset":
            print("? Сброс параметров. Возврат к выбору ДТУ...")
            break

        i2_clean = i2_str.replace(',', '.')
        try:
            i_2 = float(i2_clean)
        except ValueError:
            print(f"{Fore.RED}Ошибка ввода I2: введите число{Style.RESET_ALL}")
            continue

        # 执行剂量计算与条件着色输出(保持原逻辑不变)
        i1 = i_1 * ks
        i2 = i_2 * ks
        d1 = (k_dr * i1 * a) + b
        d2 = (k_dr * i2 * a) + b
        D = ((d1 + d2) / 2) - dfon

        if 1.5 <= D < 15:
            a, b = a_m[1], b_m[1]
            d1 = (k_dr * i1 * a) + b
            d2 = (k_dr * i2 * a) + b
            D = ((d1 + d2) / 2) - dfon
            print(f"{Fore.YELLOW}{round(D, 3)}{Style.RESET_ALL}")
        elif D >= 15:
            a, b = a_m[2], b_m[2]
            d1 = (k_dr * i1 * a) + b
            d2 = (k_dr * i2 * a) + b
            D = ((d1 + d2) / 2) - dfon
            print(f"{Fore.RED}{round(D, 3)}{Style.RESET_ALL}")
        else:
            print(f"{Fore.GREEN}{round(D, 3)}{Style.RESET_ALL}")

✅ 优势与注意事项

  • 零风险兼容性:仅用标准库 input(),无需 keyboard 权限/后台线程,适用于 PyCharm、VS Code、Linux/macOS Terminal、Windows CMD/PowerShell。
  • 用户体验友好:用户明确知道输入 "reset" 即可返回,比盲按 Esc/ Ctrl+C 更直观、不易误操作。
  • 健壮性增强:已加入 try/except 处理非法数字输入,避免程序崩溃。
  • 扩展提示:可在 input() 提示符中直接注明快捷指令,例如:
    I1 = (введите число или 'reset' для сброса) →
    进一步降低用户学习成本。

通过这一设计,您既满足了“键盘中断并返回”的核心交互需求,又保持了代码的清晰性、可维护性与跨平台稳定性。


# linux  # python  # js  # json  # windows  # mac  # macos  # win  # pycharm  # vs code 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 黑客如何通过漏洞一步步攻陷网站服务器?  Laravel如何使用Blade组件和插槽?(Component代码示例)  php静态变量怎么调试_php静态变量作用域调试技巧【解答】  如何快速使用云服务器搭建个人网站?  如何快速搭建FTP站点实现文件共享?  英语简历制作免费网站推荐,如何将简历翻译成英文?  uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址  html5的keygen标签为什么废弃_替代方案说明【解答】  如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?  如何正确下载安装西数主机建站助手?  如何用狗爹虚拟主机快速搭建网站?  javascript基本数据类型及类型检测常用方法小结  专业商城网站制作公司有哪些,pi商城官网是哪个?  如何获取免费开源的自助建站系统源码?  厦门模型网站设计制作公司,厦门航空飞机模型掉色怎么办?  Laravel如何实现模型的全局作用域?(Global Scope示例)  EditPlus中的正则表达式 实战(2)  制作网站软件推荐手机版,如何制作属于自己的手机网站app应用?  网站建设要注意的标准 促进网站用户好感度!  弹幕视频网站制作教程下载,弹幕视频网站是什么意思?  个人网站制作流程图片大全,个人网站如何注销?  如何获取上海专业网站定制建站电话?  Windows10电脑怎么设置虚拟光驱_Win10右键装载ISO镜像文件  北京的网站制作公司有哪些,哪个视频网站最好?  Laravel怎么配置.env环境变量_Laravel生产环境敏感数据保护与读取【方法】  如何破解联通资金短缺导致的基站建设难题?  Laravel Debugbar怎么安装_Laravel调试工具栏配置指南  Laravel如何配置Horizon来管理队列?(安装和使用)  公司门户网站制作公司有哪些,怎样使用wordpress制作一个企业网站?  高防服务器租用首荐平台,企业级优惠套餐快速部署  如何用低价快速搭建高质量网站?  Laravel模型事件有哪些_Laravel Model Event生命周期详解  无锡营销型网站制作公司,无锡网选车牌流程?  Laravel如何实现文件上传和存储?(本地与S3配置)  Laravel路由怎么定义_Laravel核心路由系统完全入门指南  简单实现Android文件上传  如何用VPS主机快速搭建个人网站?  如何在腾讯云免费申请建站?  潮流网站制作头像软件下载,适合母子的网名有哪些?  JavaScript数据类型有哪些_如何准确判断一个变量的类型  如何在局域网内绑定自建网站域名?  在线制作视频的网站有哪些,电脑如何制作视频短片?  Swift中switch语句区间和元组模式匹配  百度浏览器如何管理插件 百度浏览器插件管理方法  Python自然语言搜索引擎项目教程_倒排索引查询优化案例  如何为不同团队 ID 动态生成多个独立按钮  网页制作模板网站推荐,网页设计海报之类的素材哪里好?  桂林网站制作公司有哪些,桂林马拉松怎么报名?  Python文件流缓冲机制_IO性能解析【教程】  Laravel如何实现全文搜索功能?(Scout和Algolia示例)