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,希望能帮助你在工作中锦上添花。

最后依然用阿里新六脉神剑中的三脉送给大家:今天最好的表现是明天最低的要求;此时此刻非我莫属;认真生活快乐工作!

原文地址:https://www.cnblogs.com/socoool/p/12629766.html

时间: 2024-11-01 20:47:44

Java 配 Shell 等于美酒加咖啡的相关文章

Java调用Shell命令和脚本

1.介绍 有时候我们在Linux中运行Java程序时,需要调用一些Shell命令和脚本.而Runtime.getRuntime().exec()方法给我们提供了这个功能,而且Runtime.getRuntime()给我们提供了以下几种exec()方法: Process exec(String command) 在单独的进程中执行指定的字符串命令. Process exec(String[] cmdarray) 在单独的进程中执行指定命令和变量. Process exec(String[] cmd

java调用phantomjs采集ajax加载生成的网页

java调用phantomjs采集ajax加载生成的网页 日前有采集需求,当我把所有的对应页面的链接都拿到手,准备开始根据链接去采集(写爬虫爬取)对应的终端页的时候,发觉用程序获取到的数据根本没有对应的内容,可是我的浏览器看到的内容明明是有的,于是浏览器查看源代码也发觉没有,此时想起该网页应该是ajax加载的.不知道ajax的小朋友可以去学下web开发啦. 采集ajax生成的内容手段不外乎两种.一种是通过http观察加载页面时候的请求,然后我们模仿该请求去得到对应的内容,第二种则是模仿浏览器行为

JAVA调用Shell脚本

在实际项目中,Java有时候需要调用C写出来的东西,除了JNI以外,我认为一种比较好的方法是JAVA调用Shell.先把C写出来的make成可执行文件,然后再写一个shell脚本执行该可执行文件,最后是JAVA调用该shell脚本. JAVA调用很简单,例子如下: 首先是shell脚本 [plain] view plain copy print? #!/bin/sh echo Begin word cluster /home/felven/word2vec/word2vec -train /ho

java类中属性的加载顺序,以及内存分配情况介绍

看下面例子及说明: /** 假如有外部类调用了该类,代码为:new StaticTest(); 那么下面是类属性的加载顺序 */ public class StaticTest{ public int dsd=2;//第3 //多个对象会有多次分配内存 public awds() { int sdsfsd=2;//第4 //多个对象会有多次分配内存 } //静态代码块 static{ System.out.println("静态代码块正在加载...");//第1 } public st

shell脚本自动加黑恶意攻击IP

shell脚本自动加黑恶意攻击IP 系统环境:Centos 6.5 X64 如果我们对所有用户开放了SSH 22端口,那么我们就可以在/var/log/secure文件里查看,这里面全是恶意攻击的IP ,那么我们又该如何拒绝这些IP在下次攻击时直接把他拉黑,封掉呢? 或者这个IP再试图登陆4次或7次我就把他拒绝了,把他这个IP永久的封掉呢?这个时候我们就可以用这下面这个脚本来实现. [[email protected] ssh]# vi /etc/ssh/blocksship #!/bin/ba

(转)java类到底是如何加载并初始化的?

Java虚拟机如何把编译好的.class文件加载到虚拟机里面?加载之后如何初始化类?静态类变量和实例类变量的初始化过程是否相同,分别是如何初始化的呢?这篇文章就 是解决上面3个问题的. 若有不正之处,请多多谅解并欢迎各位能够给予批评指正,提前谢谢各位了. 1. Java虚拟机加载.class过程 虚拟机把Class文件加载到内存,然后进行校验,解析和初始化,最终形成java类型,这就是虚拟机的类加载机制.加载,验证,准备,初始化这5个阶段的顺序是确定的, 类的加载过程,必须按照这种顺序开始.这些

JAVA调用SHELL事例

以往一直都是crontab+shell调用java程序,最近需要反过来,使用java调用shell程序,实现定时管理,今天总结一下. 基础内容: java的java.lang.Runtime类提供了exec静态方法,可以执行本地脚本 程序事例: package study; import java.io.InputStreamReader; import java.io.BufferedReader; import java.io.FileOutputStream; import java.io

Java执行Shell脚本“No such file or directory” (win-&gt;Linux)异常的可能原因

转自:http://blog.csdn.net/zlpdaisy/article/details/6134314 用Runtime.getRuntime().exec()方法执行Linux的一个Shell脚本时,报 Cannot run program "./script/abc.sh": java.io.IOException: error=2, No such file or directory] java.io.IOException: Cannot run program &q

java中类与对象的加载顺序

首先了解一下Java虚拟机初始化的原理. JVM通过加装.连接和初始化一个Java类型,使该类型可以被正在运行的Java程序所使用.类型的生命周期如下图所示: 装载和连接必须在初始化之前就要完成. 类初始化阶段,主要是为类变量赋予正确的初始值.这里的"正确"初始值指的是程序员希望这个类变量所具备的起始值.一个正确的初始值是通过类变量初始化语句或者静态初始化语句给出的.初始化一个类包含两个步骤: 1) 如果类存在直接超类的话,且直接超类还没有被初始化,就先初始化直接超类. 2) 如果类存