当前位置: 首页 > news >正文

C#实现控制台交互式操作

前言

上一篇文章《C#实现控制台多区域输出》中,我们介绍了如何利用 Console 实现类似 Agent CLI 的多区域动态界面。

如果说多区域布局解决的是:

界面如何展示的问题

那么本文要讨论的则是另外一个问题:

用户如何与界面交互

相信体验过Claude Code、OpenCode、Hermes 等工具的同学,经常能够看到类似界面:

请选择模型:> GPT-4.1Claude SonnetGemini Pro

或者:

即将修改以下文件:Program.cs
appsettings.json是否继续?[ 是 ]  [ 否 ]

这些效果看起来已经完全不像传统的命令行程序。
相比与:

请输入数字编号:

这种交互方式显然更加直观,交互性更强。
本文就通过两个简单示例,分析这些终端交互效果背后的实现原理:

  • 列表菜单
  • 确认选择框

看看这些在 Agent CLI 中随处可见的交互组件,如果使用原生的控制台方式究竟是如何实现的。

传统CLI与现代CLI

很多传统的控制台程序都是这样工作的:

请选择操作:1.启动服务
2.停止服务
3.重启服务请输入编号:

用户输入:

2

程序继续执行。
这种交互的模式本质上是:

输入
↓
提交
↓
执行

而现代 CLI 更倾向于:

实时交互
↓
实时反馈
↓
状态更新

例如:

请选择操作:
> 启动服务停止服务重启服务

用户按下方向键时变成:

  启动服务
> 停止服务重启服务

整个过程不需要输入任何字符, 这也是各种 Agent CLI 非常喜欢采用的交互方式。

实现一个列表菜单

先来看一个最常见的场景。
例如系统运维工具:

请选择执行操作:
> 启动系统服务停止系统服务重启数据库清理系统缓存查看运行日志退出程序

用户通过↑ ↓随意切换选项, 通过Enter确认选择, 最终返回用户选中的菜单项。

菜单数据定义

首先准备菜单数据:

string[] options =
{"启动系统服务","停止系统服务","重启数据库","清理系统缓存","查看运行日志","退出程序"
};

然后调用菜单组件:

int selectedIndex = ShowSelectionMenu(options);

该方法最终会返回0 1 2 3对应用户选择的菜单项。

菜单核心实现

整个菜单真正核心的代码其实并不多:

static int ShowSelectionMenu(string[] options)
{Console.CursorVisible = false;int selectedIndex = 0;int startY = Console.CursorTop;while (true){DrawMenu(options,  selectedIndex, startY);ConsoleKeyInfo keyInfo  = Console.ReadKey(true);switch (keyInfo.Key){case ConsoleKey.UpArrow:selectedIndex--;if (selectedIndex < 0){selectedIndex = options.Length - 1;}break;case ConsoleKey.DownArrow:selectedIndex++;if (selectedIndex >= options.Length){selectedIndex = 0;}break;case ConsoleKey.Enter:Console.CursorVisible = true;return selectedIndex;}}
}

整个流程其实非常清晰:

绘制菜单
↓
等待按键
↓
修改状态
↓
重新绘制菜单

不断循环直到用户按下Enter

菜单绘制逻辑

菜单的绘制代码如下:

static void DrawMenu(string[] options, int selectedIndex, int startY)
{for (int i = 0; i < options.Length; i++){Console.SetCursorPosition(0, startY + i);Console.Write( new string(' ', Console.WindowWidth - 1));Console.SetCursorPosition(0,  startY + i);if (i == selectedIndex){Console.BackgroundColor = ConsoleColor.White;Console.ForegroundColor =  ConsoleColor.Black;Console.Write($"> {options[i]}");}else{Console.Write($"  {options[i]}");}Console.ResetColor();}
}

运行效果:
image

这里有一个非常重要的细节。

为什么每次都要重绘

很多人第一次写菜单时都会这样:

Console.Write(text);

然后发现界面开始出现残影。
例如:

> Restart Database

切换成:

> Stop

之后可能变成:

> Stopart Database

原因很简单:

Console不会自动清理旧的内容

因此每次刷新之前必须先清空区域:

Console.Write(new string(' ',  Console.WindowWidth - 1));

然后重新绘制, 这也是很多终端UI框架的核心思想:

状态变化
↓
区域重绘

而不是:

修改控件

实现确认选择框

菜单实现完成之后, 我们再来看另外一个非常常见的交互组件。
例如:

是否继续执行?[ 是 ]    [ 否 ]

这种效果在:

  • 文件删除
  • 权限授权
  • 配置确认
  • Agent工具调用

等需要确认的场景中很常见。

确认框核心实现

因为确认框本质上只有两个状态是 否, 因此实现反而更容易实现。
核心代码如下:

static bool ShowConfirmation(string message)
{Console.WriteLine(message);int selectedIndex = 0;int startY = Console.CursorTop;while (true){DrawConfirmation(selectedIndex, startY);ConsoleKeyInfo keyInfo = Console.ReadKey(true);switch (keyInfo.Key){case ConsoleKey.LeftArrow:selectedIndex--;break;case ConsoleKey.RightArrow:selectedIndex++;break;case ConsoleKey.Tab:selectedIndex++;break;case ConsoleKey.Enter:return selectedIndex == 0;}if (selectedIndex < 0){selectedIndex = 1;}if (selectedIndex > 1){selectedIndex = 0;}}
}

上面的方法最终返回true或者false

确认框绘制逻辑

对应的绘制代码:

static void DrawConfirmation(int selectedIndex, int startY)
{Console.SetCursorPosition(0,  startY);Console.Write(new string(' ',  Console.WindowWidth - 1));Console.SetCursorPosition( 0, startY);if (selectedIndex == 0){Highlight();}Console.Write("[ 是 ]");Console.ResetColor();Console.Write("    ");if (selectedIndex == 1){Highlight();}Console.Write("[ 否 ]");Console.ResetColor();
}

效果如下:
image

从实现角度来看,与菜单并没有本质区别。

菜单和确认框的共同点

看到这里会发现, 虽然菜单和确认框长得完全不同。
菜单:

> 启动服务停止服务

确认框:

[ 是 ] [ 否 ]

但背后的逻辑完全一致。

第一部分:状态

无论什么交互组件, 都需要维护状态。
例如:

int selectedIndex;

表示:

当前选中了什么

第二部分:渲染

根据状态绘制界面:

DrawMenu();

或者:

DrawConfirmation();

状态决定界面长什么样。

第三部分:输入

等待用户操作:

Console.ReadKey(true);

获取:

↑
↓
←
→
Tab
Enter

第四部分:更新状态

根据按键l来修改状态:

selectedIndex++;

或者:

selectedIndex--;

第五部分:重新渲染

状态发生变化之后:

重新绘制界面

整个过程可以抽象成:

初始化状态
↓
渲染界面
↓
等待输入
↓
更新状态
↓
重新渲染

其实绝大多数终端交互组件都遵循这一模式。

由于篇幅有限,本文只展示了核心实现。 有兴趣的同学可以自行查看完整示例:
https://github.com/softlgl/ConsoleMultiRegion

用这些能力实现一个Agent配置向导

掌握菜单和确认框之后, 其实已经能够实现很多 Agent CLI 中的交互界面。
例如:

请选择模型:
> GPT-4.1Claude SonnetGemini Pro

选择完成:

请选择运行模式:
> 自动模式手动模式

继续:

是否启用联网搜索?
[ 是 ]    [ 否 ]

最终:

配置完成

整个过程实际上只是:

菜单
+
确认框
+
状态管理

的组合。

总结

上一篇文章中,我们实现了 Console 的多区域动态布局, 而本文则进一步实现了控制台程序中的交互能力。
通过菜单和确认框两个简单示例,可以发现现代 CLI 的很多交互效果并没有想象中复杂。其核心无非是:

状态管理
↓
键盘输入
↓
区域重绘

所以很多看起来十分高级的交互界面,最终追溯到底层实现,其实都建立在这些最基础的能力之上。

👇欢迎扫码关注我的公众号👇
http://www.jsqmd.com/news/1028632/

相关文章:

  • 福州黄金回收新手变现指南,五家靠谱门店推荐不被压价 - 讯息早知道
  • AI 浏览器和网页 Agent 来了,未来上网会变成“下任务”吗?
  • 潮州高口碑黄金铂金回收白银回收实体老店排行 5 家靠谱门店电话地址全收录
  • 青岛店铺2026旧金回收,成交即刻转账到账 - 名奢变现站
  • 沈阳持证鉴定师现场验金,仪器双重核验无掺假 - 逸程
  • 魔兽争霸III免费优化完全指南:三步解决宽屏适配、地图加载和帧率锁定问题
  • 2026北京卖黄金前先看这张分级榜:S级只有一家,理由我帮你跑遍了全城 - 逸程
  • 北京个人社保代缴服务商深度评测:四大维度横向对比 - 奔跑123
  • 2026成都LV回收避坑实录,街边小店vs正规门店变现差距拆解 - 奢侈品回收评测
  • QQ机器人-Astrbot搭配NapCat框架插件文件发送问题
  • 当我们谈论 Agent 时,我们在谈论什么——从 Lilian Weng 的解剖学到自主 AI 的生理学
  • 乌鲁木齐房屋渗漏水检测维修、卫生间漏水免砸砖维修、漏水点精准检测、厨房漏水防水补漏、正规防水补漏公司、口碑榜TOP5靠谱推荐、本地人必选的防水维修公司 - 安佳防水
  • 2026年6月水利工程超声波泥水界面仪优选品牌TOP10:技术迭代下的精准监测与国产替代新格局 - 仪表品牌排行榜
  • 寄电动车怎么选物流?最便宜又安全的方法来了 - 快递物流资讯
  • AI模型能力发布机制解析:从Gated Release到可信部署
  • Simple Keyboard:极简主义Android输入法的技术哲学与实践
  • 法律RAG最危险的事情之一是“乱切 Chunk”
  • Rust 系统编程实战:从所有权模型到零成本抽象的工程落地
  • 2026 广州翡翠回收深度测评!五家正规门店横向对比,变现首选已敲定 - 禹竞
  • 2026 杭州黄金回收实体店测评,正规渠道白名单 - 禹竞
  • 2026北京黄金回收靠谱渠道有哪些?热门疑问一次性解答 - 逸程
  • 2026乐山井用潜水泵权威厂家评测:靠谱供应商盘点 - 优质品牌商家
  • 2026 天津黄金回收避坑全攻略,五大陷阱逐个拆解,教你稳妥卖金 - 讯息早知道
  • 微软推出企业级 AgenticRAG!四个工具助力RAG新范式落地
  • 跑遍三镇实地探店,整理武汉闲置黄金回收精选门店 - 讯息早知道
  • 西安回收翡翠门店推荐|2026西安翡翠回收商家阶梯排名,禹竞名奢汇稳居TOP1 - 名奢变现站
  • Rust 错误处理的黄金搭档:一个定义错误,一个传播错误
  • VideoDownloadHelper:免费网页视频下载终极指南
  • 操作系统入门实践:从Shell命令到脚本编程的课堂练习指南
  • 2026保姆级教程:图片更换背景底色全方法,手机电脑PS详细操作步骤