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

如何使用Flutter开发执行操作系统shell命令的工具

简介
Flutter是一种由Google开发的移动应用程序开发框架,它允许开发人员使用单个代码库构建高性能、高质量的移动体验。而Android终端命令行工具则允许用户在Android手机上运行类似于Linux的操作系统命令。本文的目的是介绍如何在Flutter应用中开发一个Android终端命令行工具,包括终端命令行页面的布局设计、与Shell通信的基本原理、输入输出处理的基本技巧、终端样式和输出样式的可配置性,以及如何在具体应用中利用终端命令行工具来执行系统命令和与用户进行交互。
实现终端命令行页面
为了实现一个完整的终端命令行页面,我们可以使用Flutter提供的基础控制台模块,例如TextArea、TextField等。需要注意的是,在Android中Shell是另一个进程,我们需要使用Flutter提供的方法来与Shell进行通信。具体来说,我们可以通过创建一个Future或者Stream来与Shell进行通信。其中Future用于单个结果的返回,而Stream则可以返回多个结果。我们可以在Flutter使用Async函数和这些模块的onChanged()方法来处理输入和输出。以下是最简单的实现方式。
dart复制代码

class TerminalPage extends StatefulWidget {_TerminalPageState createState() => _TerminalPageState();
}class _TerminalPageState extends State<TerminalPage> {TextEditingController _controller;Future<Process> _process;Stream<String> _output;void initState() {super.initState();_controller = TextEditingController();_process = Process.start('/system/bin/sh', []);_controller.addListener(_handleInput);}void _handleInput() {_process.then((process) {process.stdin.writeln(_controller.text);_controller.clear();});}Widget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text("Terminal Command"),),body: Column(children: [Flexible(flex: 1,child: StreamBuilder<String>(stream: _output,builder: (BuildContext context, AsyncSnapshot<String> snapshot) {if (snapshot.hasData) {return Text(snapshot.data);} else {return CircularProgressIndicator();}},),),TextField(controller: _controller,decoration: InputDecoration(border: InputBorder.none, hintText: "Input terminal command"),)],),);}
}

终端样式和输出样式的设计
除了基本的终端命令行页面,我们也需要为终端和输出样式提供更多的可配置性。在Flutter中,我们可以使用TextStyle模块(例如,color、fontSize、fontFamily等)来控制终端和输出的样式。我们可以将TextStyle保存在Flutter的本地存储中,以便在应用中进行修改。以下是一个具体的样例:
dart复制代码

 class _TerminalPageState extends State<TerminalPage> {TextEditingController _controller;Future<Process> _process;Stream<String> _output;TextStyle _terminalStyle;TextStyle _outputStyle;void initState() {super.initState();_controller = TextEditingController();_process = Process.start('/system/bin/sh', []);_terminalStyle = TextStyle(color: Colors.white, backgroundColor: Colors.black, fontFamily: 'Monospace');_outputStyle = TextStyle(color: Colors.white, backgroundColor: Colors.black, fontSize: 16);_controller.addListener(_handleInput);}void _handleInput() {_process.then((process) {process.stdin.writeln(_controller.text);_controller.clear();});}Future<File> _getTerminalStyleFile() async {final directory = await getApplicationDocumentsDirectory();final file = File('${directory.path}/terminalStyle');if (file.existsSync()) {return file;} else {await file.create();await file.writeAsString('{"color": "#FFFFFF", "background": "#000000"}');return file;}}Future<File> _getOutputStyleFile() async {final directory = await getApplicationDocumentsDirectory();final file = File('${directory.path}/outputStyle');if (file.existsSync()) {return file;} else {await file.create();await file.writeAsString('{"color": "#FFFFFF", "background": "#000000", "size": 16}');return file;}}Future<void> _loadStyles() async {final terminalFile = await _getTerminalStyleFile();final terminalJson = await terminalFile.readAsString();final terminalStylesMap = jsonDecode(terminalJson);_terminalStyle = TextStyle(color: Color(int.parse(terminalStylesMap['color'].toString(), radix: 16)),backgroundColor: Color(int.parse(terminalStylesMap['background'].toString(), radix: 16)),fontFamily: 'Monospace',);final outputFile = await _getOutputStyleFile();final outputJson = await outputFile.readAsString();final outputStylesMap = jsonDecode(outputJson);_outputStyle = TextStyle(color: Color(int.parse(outputStylesMap['color'].toString(), radix: 16)),backgroundColor: Color(int.parse(outputStylesMap['background'].toString(), radix: 16)),fontSize: outputStylesMap['size'].toDouble(),);}Future<void> _saveTerminalStyle(String color, String background) async {final terminalFile = await _getTerminalStyleFile();final terminalStylesMap = {"color": color, "background": background};final terminalJson = jsonEncode(terminalStylesMap);await terminalFile.writeAsString(terminalJson);}Future<void> _saveOutputStyle(String color, String background, double size) async {final outputFile = await _getOutputStyleFile();final outputStylesMap = {"color": color, "background": background, "size": size.toInt()};final outputJson = jsonEncode(outputStylesMap);await outputFile.writeAsString(outputJson);}Widget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text("Terminal Command"),),body: FutureBuilder(future: _loadStyles(),builder: (BuildContext context, AsyncSnapshot<void> snapshot) {switch (snapshot.connectionState) {case ConnectionState.done:return Column(children: [Flexible(flex: 1,child: StreamBuilder<String>(stream: _output,builder: (BuildContext context, AsyncSnapshot<String> snapshot) {if (snapshot.hasData) {return Text(snapshot.data, style: _outputStyle);} else {return CircularProgressIndicator();}},),),TextField(style: _terminalStyle,controller: _controller,decoration: InputDecoration(border: InputBorder.none,hintText: "Input terminal command",hintStyle: _terminalStyle,),)],);default:return Center(child: CircularProgressIndicator());}},),);}}

在Flutter应用中使用终端命令行工具的例子
在具体应用中,我们可以使用终端命令行工具来执行一些系统命令,例如ping、df、top等。同时,我们也可以利用终端界面来与用户进行交互,例如读取用户的输入并将其传递给后端服务器。以下是一个简单的应用案例:
dart复制代码

class PingPage extends StatelessWidget {final String ip;PingPage({Key key, this.ip}) : super(key: key);Future<Process> _getPingProcess() async {final process = await Process.start('/system/bin/ping', ['-c', '5', ip]);return process;}Widget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text("Ping $ip")),body: Container(child: FutureBuilder(future: _getPingProcess(),builder: (BuildContext context, AsyncSnapshot<Process> snapshot) {if (snapshot.hasData) {return Column(children: [Expanded(child: TerminalOutput(process: snapshot.data),),Container(margin: EdgeInsets.all(10),child: TextField(decoration: InputDecoration(border: OutlineInputBorder(),hintText: 'Input Ping Command',),),),Container(margin: EdgeInsets.all(10),child: ElevatedButton(onPressed: () {},child: Text("Send Command"),),)],);} else {return Center(child: CircularProgressIndicator());}},),),);}
}class TerminalOutput extends StatelessWidget {final Process process;TerminalOutput({Key key, this.process}) : super(key: key);Widget build(BuildContext context) {return StreamBuilder<String>(stream: process.stdout.transform(utf8.decoder),builder: (BuildContext context, AsyncSnapshot<String> snapshot) {if (snapshot.hasData) {return SingleChildScrollView(child: Text(snapshot.data,style: TextStyle(color: Colors.white,fontFamily: 'Monospace',backgroundColor: Colors.black,),),);} else {return Center(child: CircularProgressIndicator());}},);}
}

总结
本文介绍了在Flutter应用中开发Android终端命令行工具的方法。主要包括终端命令行页面的布局设计、与Shell通信的基本原理、输入输出处理的基本技巧、终端样式和输出样式的可配置性,以及如何在具体应用中利用终端命令行工具来执行系统命令和与用户进行交互。
本文提供了一个完整的演示案例,包括输出和输入的显示、终端命令行页面的开发、以及关于终端样式为可配置的,输出样式也为可配置的。希望这篇文章对于Flutter开发者有所帮助,能够为他们在实际开发中提供参考。
除了本文所介绍的方法外,还有其他方法可以在Flutter应用中实现终端命令行工具,例如使用Dart的Process类进行进程调用,或者将Flutter应用封装为Docker容器。这些方法也值得开发者进一步探索和试验。未来,随着移动应用越来越复杂,终端命令行工具的需求也会越来越大。因此,掌握Flutter中开发终端命令行工具的技能将会变得越来越重要。

链接:https://juejin.cn/post/7248827630865596474

相关文章:

如何使用Flutter开发执行操作系统shell命令的工具

简介 Flutter是一种由Google开发的移动应用程序开发框架&#xff0c;它允许开发人员使用单个代码库构建高性能、高质量的移动体验。而Android终端命令行工具则允许用户在Android手机上运行类似于Linux的操作系统命令。本文的目的是介绍如何在Flutter应用中开发一个Android终端命…...

西山居 游戏研发工程师实习生 面经

西山居实习面经 面试时长&#xff1a;26min&#xff08;两个面试官交替问&#xff09; 1、自我介绍 2、你平常怎么学习的 3、你实习接受加班么 4、说一下Unity的生命周期&#xff0c;Start和Awake哪里不同 5、Unity中Update与FixedUpdate的区别&#xff0c;怎么设置Fixed…...

YOLOv8训练自己的数据集+改进方法复现

yolov8已经出来好几个月了&#xff0c;并且yolov8从刚开始出来之后的小版本也升级好几次&#xff0c;总体变化不大&#xff0c;个别文件存放位置发生了变化&#xff0c;以下以最新版本的YOLOv8来详细学习和使用YOLOv8完成一次目标检测。 一、环境按照 深度学习环境搭建就不再…...

尚硅谷kafka3.0.0

目录 &#x1f483;概述 ⛹定义 ​编辑⛹消息队列 &#x1f938;‍♂️消息队列应用场景 ​编辑&#x1f938;‍♂️两种模式&#xff1a;点对点、发布订阅 ​编辑⛹基本概念 &#x1f483;Kafka安装 ⛹ zookeeper安装 ⛹集群规划 ​编辑⛹流程 ⛹原神启动 &#x1f938;‍♂️…...

【Andriod】Appium的不同版本(Appium GUI、Appium Desktop、Appium Server )的安装教程

文章目录 前言一.Appium GUI二.Appium Desktop三.Appium Server 命令行版本1.安装node.js2.安装Appium Server 前言 Appium 安装提供两2方式&#xff1a;桌面版和命令行版。其中桌面版又分为 Appium GUI 和 Appium Desktop。 建议&#xff1a;使用Appium Desktop 一.Appium …...

leetcode:面试题 17.04. 消失的数字(找单身狗/排序/公式)

一、题目&#xff1a; 函数原型&#xff1a;int missingNumber(int* nums, int numsSize) 二、思路&#xff1a; 思路1 利用“找单身狗”的思路&#xff08;n^n0&#xff1b;0^nn&#xff09;&#xff0c;数组中有0-n的数字&#xff0c;但缺失了一个数字x。将这些数字按位异或0…...

基于SpringBoot的时间管理系统

基于SpringBoot的时间管理系统的设计与实现~ 开发语言&#xff1a;Java数据库&#xff1a;MySQL技术&#xff1a;SpringBootMyBatis工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 登录界面 管理员界面 用户界面 摘要 基于Spring Boot的时间管理系统是一款功能丰富…...

centos搭建elastic集群

1、环境可以在同一台集群上搭建elastic&#xff0c;也可以在三台机器上搭建&#xff0c;这次演示的是在同一台机器搭建机器。 2、下载elastic &#xff1a;https://www.elastic.co/cn/downloads/past-releases#elasticsearch 2、​​​​​​ tar -zxvf elasticsearch-xxx-版…...

CUDA学习笔记(九)Dynamic Parallelism

本篇博文转载于https://www.cnblogs.com/1024incn/tag/CUDA/&#xff0c;仅用于学习。 Dynamic Parallelism 到目前为止&#xff0c;所有kernel都是在host端调用&#xff0c;CUDA Dynamic Parallelism允许GPU kernel在device端创建调用。Dynamic Parallelism使递归更容易实现…...

周记之马上要答辩了

“ 要变得温柔和强大&#xff0c;就算哪天突然孤身一人&#xff0c;也能平静地活下去&#xff0c;不至于崩溃。” 10.16 今天提前写完了一篇六级阅读&#xff0c;积累了一些词组&#xff1a; speak out against 公然反对&#xff0c;印象最深刻的就这个&#xff1b; 先了解…...

git简介和指令

git是一个开源的的分布式版本控制系统&#xff0c;用于高效的管理各种大小项目和文件 用途&#xff1a;防止代码丢失&#xff0c;做备份 项目的版本管理和控制&#xff0c;可以通过设置节点进行跳转 建立各自的开发环境分支&#xff0c;互不影响&#xff0c;方便合并 在多终端开…...

alibaba.fastjson的使用(五)-- Json数组字符串 ==》 JSONArray

目录 1. 使用到的方法 2. 实例演示 1. 使用到的方法 static JSONArray parseArray(String text) 2. 实例演示 /*** 将Json数组字符串转JsonArray*/@Testpublic void test5() {String jsonArrStr = "[{\"name\":\"郭靖\",\"age\":35},{\…...

ts json的中boolean布尔值或者int数字都是字符串,转成对象对应类型

没啥好写的再水一篇 json中都是字符串&#xff0c;转换一下就好&#xff0c;简单来说就是转换一次不行&#xff0c;再转换换一次&#xff0c;整体转换不够&#xff0c;细分的再转换一次 这是vue中 ts写法 ,我这里是拿对象做对比&#xff0c;不好字符和对象做对比&#xff0c;…...

【OpenGL】七、混合

混合 文章目录 混合混合公式glBlendFunc&#xff08;混合函数&#xff09;glBlendFuncSeparate渲染半透明纹理 参考链接 混合(Blending)通常是实现物体透明度(Transparency)的一种技术 简而言之&#xff1a;混合就是如何将输出颜色和目标缓冲区颜色结合起来。 混合公式 C_fina…...

JVM——堆内存调优(Jprofiler使用)Jprofile下载和安装很容易,故没有记录,如有需要,在评论区留言)

堆内存调优 当遇到OOM时&#xff0c;可以进行调参 1、尝试扩大堆内存看结果 2、分析内存&#xff0c;看哪个地方出现了问题&#xff08;专业工具&#xff09; 调整初始分配内存为1024M&#xff0c;调整最大分配内存为1024M&#xff0c;打印GC细节&#xff08;如何添加JVM操…...

Android cmdline-tools 版本与其最小JDK关系

关键词&#xff1a;Android cmdline-tools 历史版本、Android cmdline-tools 最小JDK版本、JDK 对应 major version、JDK LTS 信息 由于 JDK8 是一个常用的、较低的版本&#xff0c;因此只需要关注 JDK8 及以上版本的运行情况。 cmdline-tools 版本和最低 JDK 最终结论&…...

基于ARM+FPGA+AD的多通道精密数据采集仪方案

XM 系列具备了数据采集仪应具备的“操作简单、便于携带、满足各种测量需求”等功能的产品。具有超小、超轻量的手掌大小尺寸&#xff0c;支持8 种测量模块&#xff0c;还可进行最多576 Ch的多通道测量。另外&#xff0c;支持省配线系统&#xff0c;可大幅削减配线工时。使用时不…...

【JAVA学习笔记】43 - 枚举类

项目代码 https://github.com/yinhai1114/Java_Learning_Code/tree/main/IDEA_Chapter11/src/com/yinhai/enum_ 〇、创建时自动填入版权 作者等信息 如何在每个文件创建的时候打入自己的信息以及版权呢 菜单栏-File-setting-Editor-File and Code Templaters -Includes-输入信…...

Springcloud介绍

1.基本介绍 Spring Cloud是一系列框架的有序集合。它利用Spring Boot的开发便利性巧妙地简化了分布式系统基础设施的开发&#xff0c;如服务发现注册、配置中心、消息总线、负载均衡、断路器、数据监控等&#xff0c;都可以用Spring Boot的开发风格做到一键启动和部署。Spring …...

LK光流法和LK金字塔光流法(含python和c++代码示例)

0 引言 本文主要记录LK光流算法及LK金字塔光流算法的详细原理,最后还调用OpenCV中的cv2.calcOpticalFlowPyrLK()函数实现LK金字塔光流算法,其中第3部分是python语言实现版本,第4部分是c++语言实现版本。 1 LK光流算法 1.1 简述 LK光流法是一种计算图像序列中物体运动的光…...

RTX4090D性能实测:OpenClaw调用Qwen3-32B镜像的token消耗优化

RTX4090D性能实测&#xff1a;OpenClaw调用Qwen3-32B镜像的token消耗优化 1. 测试背景与设备环境 去年底入手RTX4090D显卡后&#xff0c;我一直想验证它在本地大模型推理场景的实际表现。最近在星图平台发现预置Qwen3-32B模型的优化镜像&#xff0c;正好配合OpenClaw做自动化…...

别再死记硬背了!用Codesys可视化玩转按钮和指示灯:5个工业场景实战案例拆解(含配方管理思路)

Codesys可视化实战&#xff1a;5个工业场景下的按钮与指示灯高阶应用 在工业自动化领域&#xff0c;人机界面(HMI)的设计直接影响操作效率和系统可靠性。传统PLC编程往往过于关注功能实现而忽视交互体验&#xff0c;导致许多工业现场的操作面板充斥着杂乱无章的按钮和难以理解的…...

在WSL中一键部署Phi-4-mini-reasoning:Windows开发者的Linux模型推理环境搭建

在WSL中一键部署Phi-4-mini-reasoning&#xff1a;Windows开发者的Linux模型推理环境搭建 1. 为什么选择WSL部署Phi-4-mini-reasoning 对于习惯Windows环境的开发者来说&#xff0c;直接在本地运行Linux环境下的AI模型往往是个挑战。Windows Subsystem for Linux (WSL) 提供了…...

OpenClaw+千问3.5-9B资料整理术:自动归类学术PDF与笔记

OpenClaw千问3.5-9B资料整理术&#xff1a;自动归类学术PDF与笔记 1. 为什么需要自动化资料整理 作为一名长期与学术文献打交道的科研工作者&#xff0c;我电脑里的PDF文件数量已经突破四位数。每次下载新论文时&#xff0c;手动重命名文件、记录关键信息、整理到对应文件夹的…...

3.30~4.5补题

牛客周赛Round 137 A.小苯的时钟显示牛客周赛Round 137 B.小苯的输入法牛客周赛Round 137 C.小苯的观景路线牛客周赛Round 137 D.小苯的序列涂色牛客周赛Round 137 E.小苯的凝聚区间SMU 2026 Spring 天梯赛6&#xff08;补题&#xff09;7-1 1-1 输出金字塔图案SMU 2…...

WebGL/Three.js性能优化实战:你的3D模型为什么卡?从理解栅格化与渲染管线开始

WebGL/Three.js性能优化实战&#xff1a;从栅格化原理到渲染管线调优 当你用Three.js加载一个精致的3D模型时&#xff0c;是否遇到过页面突然卡顿、风扇狂转的情况&#xff1f;这背后往往与浏览器如何将矢量图形转换为屏幕像素的过程密切相关。今天我们就从栅格化的底层原理出发…...

Linux文件系统探秘:当你删除一个文件时,inode位图究竟发生了什么变化?

Linux文件系统探秘&#xff1a;当你删除一个文件时&#xff0c;inode位图究竟发生了什么变化&#xff1f; 在Linux系统中&#xff0c;删除文件看似是一个简单的操作&#xff0c;但背后却隐藏着一系列精密的元数据操作。对于系统开发者和运维人员而言&#xff0c;理解这一过程不…...

告别打印乱码与错位:手把手教你配置SAP Smartforms的CNSAPWIN打印机格式

告别打印乱码与错位&#xff1a;手把手教你配置SAP Smartforms的CNSAPWIN打印机格式 在SAP系统的日常使用中&#xff0c;打印问题是最令人头疼却又无法回避的挑战之一。想象一下&#xff0c;当你精心设计的发票Smartforms报表终于完成&#xff0c;却在打印时发现内容被截断、错…...

别再只会用na.omit删数据了!R语言缺失值处理保姆级教程:从均值填补到随机森林实战

R语言缺失值处理实战&#xff1a;从基础填补到随机森林的完整指南 第一次拿到带有缺失值的数据集时&#xff0c;大多数人的本能反应是直接删除那些不完整的记录。这种简单粗暴的做法看似省事&#xff0c;却可能让你的分析结果偏离真实情况。想象一下&#xff0c;你正在分析一组…...

SecGPT-14B API保护:防止OpenClaw任务过度消耗模型资源

SecGPT-14B API保护&#xff1a;防止OpenClaw任务过度消耗模型资源 1. 为什么需要API保护机制 上周我在本地部署了SecGPT-14B模型&#xff0c;并尝试通过OpenClaw实现自动化安全报告生成。凌晨3点突然收到服务器告警——模型服务因资源耗尽崩溃了。检查日志发现&#xff0c;O…...