手把手教你做一个Shell命令窗口

2023-11-08 14:38

本文主要是介绍手把手教你做一个Shell命令窗口,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这是一个类似于win下面的cmd打开后的窗口,可以跨平台使用,可以在win和linux下面同时使用,主要功能如下:


首先我们需要把这些功能的目录写出来,通过写一个死循环,让其每次回车之后都可以保持同样的标题:如,/home/admin1>:

<span style="white-space:pre">	</span>String userPath = System.getProperty("user.home");
<span style="white-space:pre">		</span>Scanner sc = new Scanner(System.in);
// 死循环while (true) {System.out.println(userPath + ">:");String command = sc.nextLine().trim();// command是用户输入的命令,这种命令有的会改变userPath的值if ("exit".equals(command)) {// 退出程序,打断循环break;} else if ("help".equals(command)) {// 使用FileInputStream来读取 help.txt文件helpOp();} else if ("date".equals(command)) {dateOp();} else if (command != null && !"".equals(command)&& command.startsWith("dir")) {// command:dir显示userPath下的内容dirOp(userPath, command);} else if (command != null && !"".equals(command)&& command.startsWith("cat")) {catOp(command); // cat绝对路径} else if (command != null && !"".equals(command)&& command.startsWith("type")) {typeOp(command); // 绝对路径} else if (command != null && !"".equals(command)&& command.startsWith("md")) {mdOp(userPath, command); // 相对路径} else if (command != null && !"".equals(command)&& command.startsWith("ren")) {renOp(userPath, command); // 原文件的相对路径名 新文件名 ren/home/a/a.txt// /home/a/a/b.txt} else if (command != null && !"".equals(command)&& command.startsWith("rd")) {rdOp(userPath, command); // 目录相对路径名} else if (command != null && !"".equals(command)&& command.startsWith("del")) {delOp(userPath, command); // 文件相对路径名} else if (command != null && !"".equals(command)&& command.startsWith("copy")) {copyOp(userPath, command); // 文件相对路径 绝对路径} else if (command != null && !"".equals(command)&& command.startsWith("cut")) {cutOp(userPath, command); // 原文件的相对路径名 相对路径} else if (command != null && !"".equals(command)&& command.startsWith("tree")) {treeOp(userPath); // 输出当前目录下的所有文件 递归} else if (command != null && !"".equals(command)&& command.startsWith("cd")) {userPath = cdOp(userPath, command); // cd. cd.. cd/ cd 目录 有返回值的} else {System.out.println("找不到这条命令");}}

当然,为了做到跨平台使用,我们可以先定义一下平台:

public final static int OS_TYPE_LINUX = 1; // linux操作系统
public final static int OS_TYPE_WINDOWS = 2; // window操作系统

我们还需要拿到操作系统的名字,当然首先我们来看一下操作系统的详细信息,我们可以使用System.getProperties()来查看系统的所有信息,这个时候我们就会发现,user.home就是我们的系统名,所以我们要把这个拿来使用。

// 区分操作系统private static int getSystemInfo() {// Properties就是一个键值对// Properties p=System.getProperties(); //System类当中存有当前系统的所有信息// Set<Entry<Object,Object>> set=p.entrySet(); //entry:键值对 Set:集合// Iterator<Entry<Object,Object>> its=set.iterator(); //迭代器// while(its.hasNext()){ //使用迭代器取出集合中的第一个元素,hasNext()返回true// Entry<Object,Object> entry=its.next(); //取出// System.out.println(entry.getKey()+":"+entry.getValue());// }String osName = System.getProperty("os.name");if (osName.toLowerCase().indexOf("linux") >= 0) {return OS_TYPE_LINUX;} else if (osName.toLowerCase().indexOf("windows") >= 0) {return OS_TYPE_WINDOWS;} else {return -1;}}

如果你想查看一下你盘符的容量,我们可以这样做。

// 显示版权private static void showCopyRight() {int ostype = getSystemInfo();if (ostype == OS_TYPE_LINUX) {System.out.println("ubuntu linux zp");} else if (ostype == OS_TYPE_WINDOWS) {System.out.println("Micrsoft windows zp");System.out.println("Copyright by (2016-2028)");}System.out.println("当前系统的盘符:");File[] fs = File.listRoots();System.out.println("盘符名\t总大小\t剩余空间:");for (File file : fs) {System.out.println(file.getAbsolutePath() + "\t"+ getSizeInPrety(file.getTotalSpace()) + "\t"+ getSizeInPrety(file.getFreeSpace()));}}
哦!好吧,如果只是这样写,那么显示出来的都是k为大小的,所以我们还要判断一下,把大小显示为G,M,K,B等

private static String getSizeInPrety(long size) {if (size / 1024 / 1024 / 1024 / 1024 > 0) {return size / 1024 / 1024 / 1024 / 1024 + "T";} else if (size / 1024 / 1024 / 1024 > 0) {return size / 1024 / 1024 / 1024 + "G";} else if (size / 1024 / 1024 > 0) {return size / 1024 / 1024 + "M";} else if (size / 1024 > 0) {return size / 1024 + "K";} else {return size + "B";}}

好了,言归正传,我们开始写以上要实现的14条命令。

1、先来写help好了,这样我们就可以看一下帮助了(在不知道有哪些命令的情况下),既然想写help,那么我们就先把help.txt这个文档准备好,然后通过 使用FileInputStream来读取 help.txt文件。最后把这个显示出来就可以了。

private static void helpOp() throws IOException {// 使用FileInputStream来读取 help.txt文件// 通过test1的字节码类,找到它的字节码加载器,这个加载器从bin目录开始扫描,查找help.txt文件,再自动以流的方式加载它InputStream fis = test1.class.getClassLoader().getResourceAsStream("help.txt");// File filename=new// File(System.getProperty("java.class.path")+File.separator+"help.txt");// FileInputStream fis = new FileInputStream(filename);// 第三步:操作!byte[] buff = new byte[1024];int len = -1;// 定义缓冲区while ((len = fis.read(buff, 0, buff.length)) != -1) {String s = new String(buff, 0, len, "gbk");System.out.println(s);}// 第四步:关闭资源(字符流必须关闭资源,因为它中间有缓冲区!对于字节流可以不用关闭,但是还是建议写上,习惯!)fis.close();}

2、然后我们写一下退出吧!纳尼,退出还需要写么,根本不需要好吧,我在前面已经写了,直接break就好了啊,你说你是不是傻 偷笑!你自己看我的第一段代码!

3、既然如此,那我们就来写一下date吧,这个可以用来查看系统时间。用SimpleDateFormat来格式化一个时间。

	private static void dateOp() {// TODO Auto-generated method stubDate d = new Date();SimpleDateFormat sd = new SimpleDateFormat("yyyy-M-d HH:mm:ss E");String s = sd.format(d);// 这个方法继承于SimpleDateFormat的父类DateFormat类!System.out.println(s);}

4、那我们再来看一下dir命令吧!dir是可以查看所有的文件和目录的哦,注意不要查看路径太深的文件夹,不然的话,嘿嘿,会循环调用很久哒!这里我们可以统计一下总共有多少个文件和目录,然后还可以判断一下文件的权限rwx,当然,还可以计算一下大小啦,那么我们就可以调用前面说过的getSizeInPrety()方法来格式化一下G,M,K等的大小。
private static void dirOp(String userPath, String command) {String[] contents = command.split(" ");if (contents.length == 2) {// 如果长度为2,就显示userPath下的内容userPath = contents[1];}// y用file类来完成取到这个目录的信息File f = new File(userPath);File[] fs = f.listFiles();int totalFile = 0;int totalDir = 0;long totalFileSize = 0;if (fs != null && fs.length > 0) {for (File file : fs) {long time = file.lastModified();Date d = new Date();SimpleDateFormat sd = new SimpleDateFormat("yyyy-M-d HH:mm");String timeString = sd.format(time);// 权限String read = file.canRead() ? "r" : "-";String write = file.canWrite() ? "w" : "-";String execute = file.canExecute() ? "x" : "-";// 文件还是目录String fileorDir = file.isFile() ? "\t" : "<dir>";// 取大小long fileSize = file.isFile() ? file.length() : 0;String fileSizeString = "\t";if (file.isFile()) {fileSizeString = getSizeInPrety(fileSize);totalFile++;totalFileSize += fileSize;} else {totalDir++;}System.out.println(timeString + "\t\t" + read + write + execute+ "\t" + fileorDir + "\t" + fileSizeString+ file.getName());}}System.out.println("总共有:" + totalFile + "个文件," + totalDir + "个目录");}

5、那cat命令也类似,

private static void catOp(String command) {// 解析command中的文件String[] contents = command.split(" ");String filePath = null;if (contents.length == 2) {// 如果长度为2,就显示userPath下的内容filePath = contents[1];}// 用file类来完成取到这个目录的信息File f = new File(filePath);if (f.exists() == false || f.isFile() == false) {System.out.println(f.getName() + "不是一个有效文件,无法读取");return;}FileReader fr = null;try {fr = new FileReader(f);BufferedReader br = new BufferedReader(fr);String line = null;int num = 0;while ((line = br.readLine()) != null) {num++;System.out.println(num + "\t" + line);}} catch (Exception e) {e.printStackTrace();} finally {try {fr.close();} catch (IOException e) {e.printStackTrace();}}}

6、type的话呢,好像也差不多。

private static void typeOp(String command) throws IOException {// TODO Auto-generated method stub// 使用BuffedInputStream来完成// 解析command中的文件String[] strs = command.split(" ");if (strs == null || strs.length != 2) {System.out.println(command + "格式,标准格式:type 文件的绝对路径,请确认后重新输入");return;}File file = new File(strs[1]);if (file.exists() == false) {System.out.println(file.getAbsolutePath() + "文件不存在");return;}if (file.isFile() == false) {System.out.println(file.getAbsolutePath() + "不是一个可以读取的文件");return;}InputStream iis = null;try {iis = new BufferedInputStream(new FileInputStream(file));byte[] bs = new byte[1024];int length = -1;while ((length = iis.read(bs, 0, bs.length)) != -1) {String s = new String(bs, 0, length, "gbk");System.out.println(s);}} catch (Exception e) {// TODO Auto-generated catch blocke.printStackTrace();} finally {try {iis.close();} catch (IOException e) {e.printStackTrace();}}}
7、现在,我们就可以来看一下如何创建目录的吧!这种的话如果不会一定要多看一些API文档了。

	private static void mdOp(String userPath, String command) {// 在userPath下面创建一个目录 md/a/b/c md/a// File类中的mkdirs()String[] strs = command.split(" ");if (strs == null || strs.length != 2) {System.out.println(command + "格式,标准格式:md  文件的相对路径,请确认后重新输入");return;}File f = new File(userPath, strs[1]);if (f.exists()) {System.out.println("文件已存在,无需创建");return;}// 创建目录System.out.println(f.mkdir() ? "创建目录成功" : "创建目录失败");System.out.println();<span style="font-family: Arial, Helvetica, sans-serif;">}</span>
8、重命名一下文件。主要是调用一下renameTo()方法。

private static void renOp(String userPath, String command) {// TODO Auto-generated method stub// File类的renameTo()String[] strs = command.split(" ");// 读取command格式是否正确if (strs == null || strs.length != 3) {System.out.println(command+ "格式,标准格式:ren  原文件的相对路径    新文件相对路径名,请确认后重新输入");return;}// 创建一个file表示原文件 new File(userPath,新文件相对路径)File old = new File(userPath, strs[1]);// 判断原文件是否存在,不存在则返回if (old.exists() == false) {System.out.println(old.getAbsolutePath() + "不存在,请确认后输入");return;}// 判断新文件是否存在,若存在,则返回File newfile = new File(userPath, strs[2]);if (newfile.exists()) {System.out.println(newfile.getAbsolutePath() + "已存在,不能重名");return;}// 将原文件改名为新文件old.renameTo(newfile);System.out.println("重命名成功");System.out.println();}

9、删除目录,使用rd命令,当然,这种方法我个人是感觉不太好的,因为假如这个目录下面不是空目录就不能删除。


private static void rdOp(String userPath, String command) {// TODO Auto-generated method stub// 删目录String[] strs = command.split(" ");// 读取command格式是否正确if (strs == null || strs.length != 2) {System.out.println(command + "格式,标准格式:rd   目录的相对路径,请确认后重新输入");return;}// 创建一个文件 new File(userPath,strs[1]);File f = new File(userPath, strs[1]);// 判断是否存在,是否为目录if (f.exists() == false) {System.out.println(f.getAbsolutePath() + "不存在,无法删除");return;}// 判断这个目录是否为空目录,空目录才可以删除File[] fs = f.listFiles();if (fs != null && fs.length > 0) {System.out.println(f.getAbsolutePath() + "不是空目录,不能删除");return;}// 若是空目录,则删除boolean result = f.delete();// 显示删除成功System.out.println(result ? "删除" + f.getName() + "成功" : "删除"+ f.getName() + "失败");<span style="font-family: Arial, Helvetica, sans-serif;">}</span>
所以我们可以来看一下方法2,就是可以循环把整个目录一起删除掉,总之,慎用这条命令,一下小心你可能会删除掉有用下信息哦!

<span style="white-space:pre">		private static void rdOp(String userPath, String command) {</span>String[] contents = command.split(" "); String filePath = null; if(contents.length == 2) { // 如果长度为2,就显示userPath下的内容 filePath =contents[1]; } File dir = new File(filePath);// 定义文件路径deleteFile(dir);}private static void deleteFile(File dir) { // TODO Auto-generated methodFile[] subFiles = dir.listFiles(); for (File file : subFiles) { if(file.isFile()) { file.delete(); System.out.println("已成功删除文件"); } else {deleteFile(file); } } dir.delete(); System.out.println("已成功删除文件夹"); }


10、del就是用来删除文件的。主要就是调用了delete()方法。

private static void delOp(String userPath, String command) {// TODO Auto-generated method stub// 删除文件String[] strs = command.split(" ");// 读取command格式是否正确if (strs == null || strs.length != 2) {System.out.println(command + "格式,标准格式:del   文件的相对路径,请确认后重新输入");return;}// 创建一个文件 new File(userPath,strs[1]);File f = new File(userPath, strs[1]); // 定义文件路径// 判断是否存在,是否为目录if (f.exists() == false) {System.out.println("文件不存在");return;}if (f.isFile() == false) {System.out.println("不是文件,不能删除");return;}// 判断是否文件,是则删除boolean result = f.delete();// 显示删除成功System.out.println(result ? "删除" + f.getName() + "成功" : "删除"+ f.getName() + "失败");<span style="font-family: Arial, Helvetica, sans-serif;">}</span>

11、tree命令,用来按层级打印输出路径。

private static void treeOp(String userPath) {tree(userPath, 1);}//递归调用private static void tree(String path, int level) {StringBuffer sb = new StringBuffer();for (int i = 0; i < level; i++) {sb.append("      ");}File f = new File(path);File[] fs = f.listFiles();System.out.println(sb.toString() + f.getName());if (fs != null && fs.length > 0) {for (File file : fs) {if (file.isFile()) {System.out.println(sb.toString() + "     " + file.getName());} else {tree(file.getAbsolutePath(), level + 1);}}}}

12、copy,还有一个磨人的小妖精,就是这个命令啦!虽然就是用了一个 FileOutputStream来输入输出文本文件内容而已。

private static void copyOp(String userPath, String command) {String[] strs = command.split(" ");// 读取command格式是否正确if (strs == null || strs.length != 3) {System.out.println(command+ "格式,标准格式:copy   原文件的相对路径  新文件的绝对路径名,请确认后重新输入");return;}// 创建一个file表示原文件 new File(userPath,新文件相对路径)File old = new File(userPath, strs[1]);// 判断原文件是否存在,不存在则返回if (old.exists() == false) {System.out.println(old.getAbsolutePath() + "不存在,请确认后输入");return;}// 判断新文件是否存在,若存在,则返回File newfile = new File(userPath, strs[2]);if (newfile.exists()) {System.out.println(newfile.getAbsolutePath() + "已存在,不能复制");return;}// 判断原文件和新文件是否重复if (old.equals(newfile)) {System.out.println("目录文件夹是源文件夹的子文件夹");} else {copy(old, newfile);}}private static void copy(File old, File newfile) {// TODO Auto-generated method stubtry (InputStream is = new FileInputStream(old);OutputStream os = new FileOutputStream(newfile);) {byte[] b = new byte[1024];int len;while ((len = is.read(b)) != -1) {os.write(b);}} catch (IOException e) {e.printStackTrace();}System.out.println("复制成功");}

13、那么既然,我们前面写了copy和delete,就可以综合使用这两个来写一下cut命令咯!

	private static void cutOp(String userPath, String command) {// TODO Auto-generated method stubcopyOp(userPath, command);     //String[] strs = command.split(" ");File old = new File(userPath, strs[1]);old.delete();System.out.println("剪切成功");}

14、最后就只剩下cd命令了,就是需要每次可以使用cd. cd.. cd/ cd 来切换目录。

private static String cdOp(String userPath, String command) {// TODO Auto-generated method stubif (command.equals("cd")) {return userPath;}if (command.equals("cd .")) {return userPath;}String[] strs = command.split(" ");if (strs == null || strs.length != 2) {System.out.println("命令格式错误,正确格式是:cd 路径表示法");return userPath;}File file = new File(userPath); // 当前路径if (command.equals("cd ..")) {if(file.getParent()!=null){   //判断父路径是否为空return file.getParent();}}if (command.equals("cd /")) {return getRoot(userPath); }File f = new File(strs[1]);if (f.exists() && f.isDirectory()) {return f.getAbsolutePath();}System.out.println(strs[1] + "路径异常");return userPath;}private static String getRoot(String userPath) {int system_type = getSystemInfo();if (system_type == OS_TYPE_LINUX) {return "/";} else {int last = userPath.indexOf(":\\" + 2);String path = userPath.substring(0, last);return path;}}
以上就基本实现了一个shell命令窗口的功能了,还有其他的功能,你也来挑战一下吧!

总结:本文主要介绍了shell命令窗口的基本命令的书写,采用java来重写一些基本命令,对文件输入输出流的使用由了进一步的使用,综合了各种循环,判断,各种流的使用,以及API的部分常用语法的使用。

源码地址:点击打开链接https://github.com/sdksdk0/Shell。

这篇关于手把手教你做一个Shell命令窗口的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/370527

相关文章

Linux使用nohup命令在后台运行脚本

《Linux使用nohup命令在后台运行脚本》在Linux或类Unix系统中,后台运行脚本是一项非常实用的技能,尤其适用于需要长时间运行的任务或服务,本文我们来看看如何使用nohup命令在后台... 目录nohup 命令简介基本用法输出重定向& 符号的作用后台进程的特点注意事项实际应用场景长时间运行的任务服

Redis的Hash类型及相关命令小结

《Redis的Hash类型及相关命令小结》edisHash是一种数据结构,用于存储字段和值的映射关系,本文就来介绍一下Redis的Hash类型及相关命令小结,具有一定的参考价值,感兴趣的可以了解一下... 目录HSETHGETHEXISTSHDELHKEYSHVALSHGETALLHMGETHLENHSET

手把手教你idea中创建一个javaweb(webapp)项目详细图文教程

《手把手教你idea中创建一个javaweb(webapp)项目详细图文教程》:本文主要介绍如何使用IntelliJIDEA创建一个Maven项目,并配置Tomcat服务器进行运行,过程包括创建... 1.启动idea2.创建项目模板点击项目-新建项目-选择maven,显示如下页面输入项目名称,选择

如何使用 Bash 脚本中的time命令来统计命令执行时间(中英双语)

《如何使用Bash脚本中的time命令来统计命令执行时间(中英双语)》本文介绍了如何在Bash脚本中使用`time`命令来测量命令执行时间,包括`real`、`user`和`sys`三个时间指标,... 使用 Bash 脚本中的 time 命令来统计命令执行时间在日常的开发和运维过程中,性能监控和优化是不

bat脚本启动git bash窗口,并执行命令方式

《bat脚本启动gitbash窗口,并执行命令方式》本文介绍了如何在Windows服务器上使用cmd启动jar包时出现乱码的问题,并提供了解决方法——使用GitBash窗口启动并设置编码,通过编写s... 目录一、简介二、使用说明2.1 start.BAT脚本2.2 参数说明2.3 效果总结一、简介某些情

基于Redis有序集合实现滑动窗口限流的步骤

《基于Redis有序集合实现滑动窗口限流的步骤》滑动窗口算法是一种基于时间窗口的限流算法,通过动态地滑动窗口,可以动态调整限流的速率,Redis有序集合可以用来实现滑动窗口限流,本文介绍基于Redis... 滑动窗口算法是一种基于时间窗口的限流算法,它将时间划分为若干个固定大小的窗口,每个窗口内记录了该时间

PHP执行php.exe -v命令报错的解决方案

《PHP执行php.exe-v命令报错的解决方案》:本文主要介绍PHP执行php.exe-v命令报错的解决方案,文中通过图文讲解的非常详细,对大家的学习或工作有一定的帮助,需要的朋友可以参考下... 目录执行phpandroid.exe -v命令报错解决方案执行php.exe -v命令报错-PHP War

CentOS系统使用yum命令报错问题及解决

《CentOS系统使用yum命令报错问题及解决》文章主要讲述了在CentOS系统中使用yum命令时遇到的错误,并提供了个人解决方法,希望对大家有所帮助,并鼓励大家支持脚本之家... 目录Centos系统使用yum命令报错找到文件替换源文件为总结CentOS系统使用yum命令报错http://www.cppc

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

30常用 Maven 命令

Maven 是一个强大的项目管理和构建工具,它广泛用于 Java 项目的依赖管理、构建流程和插件集成。Maven 的命令行工具提供了大量的命令来帮助开发人员管理项目的生命周期、依赖和插件。以下是 常用 Maven 命令的使用场景及其详细解释。 1. mvn clean 使用场景:清理项目的生成目录,通常用于删除项目中自动生成的文件(如 target/ 目录)。共性规律:清理操作