Java 配 Shell 等于美酒加咖啡
化学中我们得知「氢气加氧气在点燃的情况下会生成水」。
生活中我们得知「良辰加美景的情况下会得到千金春宵一刻」。
技术上又何尝不是如此呢?先假设一个场景:BOSS 让你实现一个服务监控的指挥室,能看到每个服务器的磁盘剩余空间,能看到。。。能看到。。。
其实讲真,实现思路有很多,但是不管黑猫白猫能抓住老鼠都是好猫,今天我们尝试用 Java 与 Shell 搭配一下,看看是否会产生惊奇的反应。
1.
首先通过 JDK 源码,品一品 Runtime 这杯美酒。
上图是摘取 JDK 中 Runtime 的部分源码,主要分成 4 大代码段来粗略认识她。
第一块代码段,可以看出 Runtime 构造私有化,提供了静态属性,并提前创建对象实例,并提供获取实例的静态方法,这不就是单例设计模式的使用么,当有面试官再问设计模式,拿去狂喷。
第二块代码段,主要是 addShutdownHook() 方法,添加关闭的钩子,说的直白点,其实允许研发人员插入一段在 JVM 关闭时执行的代码。例如在搭建服务框架时,面对需要完成优雅停服,打扫战场,释放资源等等,诸如此类的场景下都很有用。其中在 Tomcat、Jetty 等容器中都可以看到 shutdownHook 的身影。
Runtime runtime = Runtime.getRuntime();
runtime.addShutdownHook(new Thread() {
@Override
public void run() {
System.out.println("打扫战场,释放资源,完成优雅停服");
}
});
System.out.println("服务启动完成");
代码运行效果如下。
服务启动完成
打扫战场,释放资源,完成优雅停服
第三块代码段,主要展现 JDK 针对 Runtime 提供的系列 exec 重载方法,这个是本次分享的重点,重头戏最后再说。
第四块代码段,主要是 Runtime 提供的一些获取系统信息的 API,直接抛代码,拿去用就行了。
Runtime runtime = Runtime.getRuntime();
System.out.println(String.format("JVM可用本机CPU内核数 %d", runtime.availableProcessors()));
//默认为系统的1/4
System.out.println(String.format("最大可用内存空间 %d M", runtime.maxMemory() / 1024 / 1024));
//默认为系统的1/64
System.out.println(String.format("可用内存空间 %d M", runtime.totalMemory() / 1024 / 1024));
System.out.println(String.format("空闲内存空间 %d M", runtime.freeMemory() / 1024 / 1024));
代码运行输出如下,其实真实环境中不妨用模板引擎 FreeMarker 渲染,然后通过邮件告警,实现的逼格高一些。
2.
在详细说 Runtime.exec() 这个重头戏之前,再品一品 df 这款咖啡。
Linux df 命令,用于显示目前在系统上的磁盘使用情况统计,主要用于查看磁盘的分区,磁盘已使用的空间,剩余的空间。
命令如下:
df [选项]... [FILE]...
常用选项如下:
3.
Runtime 美酒加 Shell 咖啡会发生什么呢?
重头戏开始,回到 Runtime 的源码,我们看到 exec() 系列方法会帮我们启动一个 Process 进程,那不妨把 df -h 命令传入进去一探究竟。
public class Foo {
public static void main(String[] args) throws Exception {
//df命令用于查看磁盘的分区,磁盘已使用的空间,剩余的空间
//df -h以合适的单位来显示信息
System.out.println(exec("df -h"));
}
private static String exec(String command) throws Exception {
String[] cmd = {"/bin/sh", "-c", command};
StringBuilder out = new StringBuilder();
BufferedReader reader = null;
InputStream in = null;
try {
Process process = Runtime.getRuntime().exec(cmd);
in = process.getInputStream();
reader = new BufferedReader(new InputStreamReader(in));
String line;
while ((line = reader.readLine()) != null) {
out.append(line + "\n");
}
process.waitFor();
} finally {
if (reader != null) {
reader.close();
}
}
return out.toString();
}
}
代码中会发现调用了 process 的 waitFor() 方法,此方法作用会导致当前线程等待,一直要等到由该 Process 对象表示的进程终止,其实也就是等待把 exec 里面启动的 Process 中的所有事都干完(生产上出问题的大多出在这儿),代码运行效果如下。
效果确实可以,那么这么一来,想监控统计什么功能,不妨直接把命令交给 Java 程序去执行即可。
4.
如果关注一猿小讲的伙伴应该清楚,在《Runtime 与 ProcessBuilder 的主要区别是啥呢?
其实 Runtime.exec() 方法设计,可接受一个单独的字符串,这个字符串是通过空格来分隔可执行命令程序和参数的;当然也可以接受字符串数组参数。
如上图所示,ProcessBuilder 的方法入参是一个List<String>或者多个字符串。
相同点是 ProcessBuilder.start() 和 Runtime.exec() 方法都被用来创建一个操作系统进程(执行命令行操作)。
5.
好了,几分钟的简单分享,主要让你接触并真正认识一下 Runtime,希望能帮助你在工作中锦上添花。
最后依然用阿里新六脉神剑中的三脉送给大家:今天最好的表现是明天最低的要求;此时此刻非我莫属;认真生活快乐工作!

Java 配 Shell 等于美酒加咖啡的更多相关文章
- linux下的shell运算(加、减、乘、除)
linux下的shell运算(加.减.乘.除) 标签: linuxshell运算加减乘除 2014-03-12 16:25 15127人阅读 评论(0) 收藏 举报 分类: linux(17) ((i ...
- linux下的shell运算(加、减、乘、除
linux下的shell运算(加.减.乘.除 摘自:https://blog.csdn.net/hxpjava1/article/details/80719112 2018年06月17日 16:03: ...
- [转载]JAVA调用Shell脚本
FROM:http://blog.csdn.net/jj12345jj198999/article/details/11891701 在实际项目中,JAVA有时候需要调用C写出来的东西,除了JNI以外 ...
- java调用shell脚本小demo
复制指定文件cpp.sh: [root@localhost soft]# vim cpp.sh#!/bin/bash name="$1"\cp /home/soft/test/${ ...
- 时隔两年最近再次折腾opensuse 的一些笔记 - opensuse linux java service shell
时隔两年最近再次折腾opensuse 的一些笔记 - opensuse linux java service shell opensuse 一些常用命令: service xxx start/s ...
- java调用phantomjs采集ajax加载生成的网页
java调用phantomjs采集ajax加载生成的网页 日前有采集需求,当我把所有的对应页面的链接都拿到手,准备开始根据链接去采集(写爬虫爬取)对应的终端页的时候,发觉用程序获取到的数据根本没有对应 ...
- java类到底是如何加载并初始化的?
Java虚拟机如何把编译好的.class文件加载到虚拟机里面?加载之后如何初始化类?静态类变量和实例类变量的初始化过程是否相同,分别是如何初始化的呢?这篇文章就 是解决上面3个问题的. 若有不正之处, ...
- 深入java虚拟机学习 -- 类的加载机制(续)
昨晚写 深入java虚拟机学习 -- 类的加载机制 都到1点半了,由于第二天还要工作,没有将上篇文章中的demo讲解写出来,今天抽时间补上昨晚的例子讲解. 这里我先把昨天的两份代码贴过来,重新看下: ...
- 深入java虚拟机学习 -- 类的加载机制(三)
类的初始化时机 在上篇文章中讲到了类的六种主动使用方式,反射是其中的一种(Class.forName("com.jack.test")),这里需要注意一点:当调用ClasLoade ...
随机推荐
- 必备技能六、Vue框架引入JS库的正确姿势
在Vue.js应用中,可能需要引入Lodash,Moment,Axios,Async等非常好用的JavaScript库.当项目变得复杂庞大,通常会将代码进行模块化拆分.可能还需要跑在不同的环境下,比如 ...
- Vuex的理解以及它的辅助函数
理解:vue中的“单向数据流”,这里借用官网的图示: Vue是单向数据流,v-model只是语法糖而已.单向数据流就是:数据总是[向下传递]从父级组件传递给子组件,只能单向绑定.子组件内部不能直接修改 ...
- JS中面向对象中的继承(常用写法)---核心部分
1.基本概念 子类继承父类,但是不能影响父类.包括1.混合继承(构造函数+原型) 2.ES6新增class的继承. 接下来介绍,面向对象中继承的两种常用写法.即混合继承(构造函数+原型)和class继 ...
- ASP.NETMVC中js非空验证实例
页面代码 @using (Html.BeginForm("Edit", "Home", FormMethod.Post, new { @Id = "f ...
- 《JavaScript 模式》读书笔记(2)— 基本技巧1
这篇文章的主要内容,介绍了一些js编程中的基本技巧,其实这些技巧,大家在开发的过程中,或多或少都在使用,或者已经可以熟练的应用于自己的代码或项目中了.那么,这篇文章,就一起来回顾下这些“基本技巧”. ...
- CSS--transform相关属性实现2d到3d的具体变化
先放上一段我写的相关代码(可能有一定冗杂代码,请见谅) <!DOCTYPE html> <html lang="en"> <head> ...
- pytest、tox、Jenkins实现python接口自动化持续集成
pytest介绍 pytest是一款强大的python测试工具,可以胜任各种级别的软件测试工作,可以自动查找测试用并执行,并且有丰富的基础库,可以大幅度提高用户编写测试用例的效率,具备可扩展性,用户自 ...
- numpy——>数组拼接np.concatenate
语法:np.concatenate((a1, a2, ...), axis=0) 1.默认是 axis = 0,也就是说对0轴(行方向)的数组对象,进行其垂直方向(axis=1)的拼接(即数据整行整行 ...
- 大型Java进阶专题(四) 设计模式之工厂模式
前言 今天开始我们专题的第三课了,开始对设计模式进行讲解,本章节介绍:了解设计模式的由来,介绍设计模式能帮我们解决那些问题以及剖析工厂模式的历史由来及应用场景.本章节参考资料书籍<Sprin ...
- iview mock main.js
main.js // 实际打包时应该不引入mock /* eslint-disable */ // if (process.env.NODE_ENV !== 'production') require ...