有时候,咱们在使用Java作一些操做时,可能性能上并不能达到咱们满意的效果,就拿最近工做中的遇到的一个场景来讲,须要对大量的小文件进行合并成一个大文件。java

最开始的想法是使用Java作文件操做,遍历全部小文件而后往一个文件写(能够作成并发写),可是发现操做过程当中遇到个问题,写一千多个小文件在本机Windows下须要花费几十秒的时间,即便在Linux环境下高配置的机器也须要将近十秒,这明显对接口的响应时间产生重要影响。这块怎么优化下呢?apache

咱们都知道在Linux下能够进行大文件的分割和合并,分别采用split和cat命令,因而作了个实验,在Linux下对相同的一个1G文件进行切割成1000个小文件,而后对这一千多个小文件进行合并。效果是惊人的!!!居然瞬间就能合成完成了!这更加让我坚决了应该使用系统命令进行批量小文件进行合并的想法。并发

咱们这里封装一个类,用来调用系统命令,而后获得系统调用的返回结果。app

咱们先封装了一个返回结果类:框架

package com.majing.learning.fileupload.common.process;

public class ProcessResult {

private boolean success = false;

private String errorMessage;

private String outputMessage;

public boolean isSuccess() {

return success;

}

public void setSuccess(boolean success) {

this.success = success;

}

public String getErrorMessage() {

return errorMessage;

}

public void setErrorMessage(String errorMessage) {

this.errorMessage = errorMessage;

}

public String getOutputMessage() {

return outputMessage;

}

public void setOutputMessage(String outputMessage) {

this.outputMessage = outputMessage;

}

}

接着咱们给出封装的系统调用实现类:ide

package com.majing.learning.fileupload.common.process;

import java.io.IOException;

import java.util.concurrent.CountDownLatch;

import java.util.concurrent.ExecutorService;

import org.apache.commons.lang3.StringUtils;

public class CommandUtils {

public static ProcessResult runCmdTest(ExecutorService executorService, String command) throws IOException, InterruptedException {

StringBuilder queryInputResult = new StringBuilder();

StringBuilder queryErroInputResult = new StringBuilder();

ProcessResult processResult = new ProcessResult();

String[] cmd = { "/bin/sh", "-c", command};

Process pro = Runtime.getRuntime().exec(cmd);

CountDownLatch lock = new CountDownLatch(2);

executorService.submit(new ProcessCheckTask(queryInputResult, lock, pro.getInputStream()));

executorService.submit(new ProcessCheckTask(queryErroInputResult, lock, pro.getErrorStream()));

boolean done = false;

while (!done) {

lock.await();

done = true;

}

processResult.setOutputMessage(queryInputResult.toString());

processResult.setErrorMessage(queryErroInputResult.toString());

processResult.setSuccess(StringUtils.isBlank(processResult.getErrorMessage()));

return processResult;

}

}

其中ProcessCheckTask类以下:性能

package com.majing.learning.fileupload.common.process;

import java.io.BufferedReader;

import java.io.InputStream;

import java.io.InputStreamReader;

import java.util.concurrent.CountDownLatch;

import org.slf4j.Logger;

import org.slf4j.LoggerFactory;

import com.majing.learning.fileupload.common.ConstValues;

public class ProcessCheckTask implements Runnable {

private static Logger logger = LoggerFactory.getLogger(ProcessCheckTask.class);

/** 锁 */

private CountDownLatch lock;

/** 执行结果输入流 */

private InputStream inputStream;

/** 字符拼接 */

private StringBuilder queryInputResult;

public ProcessCheckTask(StringBuilder queryInputResult, CountDownLatch lock, InputStream inputStream) {

super();

this.lock = lock;

this.inputStream = inputStream;

this.queryInputResult = queryInputResult;

}

@Override

public void run() {

try {

BufferedReader bf = new BufferedReader(new InputStreamReader(inputStream));

String line = null;

while ((line = bf.readLine()) != null && line.length() > 0) {

queryInputResult.append(line).append("\n");

}

} catch (Exception e) {

logger.error(ConstValues.EXCEPTION_OCCURED, e);

} finally {

lock.countDown();

}

}

}

上面是一个简单实现,可是可能会存在一个问题,那就是执行系统命令的时间若是自己比较长,若是不想一直等待到系统命令执行完,而是在一段时间没有返回就直接认为失败,因此须要增长过时时间的考虑。这里我借助于Future框架,将上面的调用系统命令的方法封装成一个Callable对象。优化

package com.majing.learning.fileupload.common.process;

import java.util.concurrent.Callable;

import java.util.concurrent.ExecutorService;

public class CommandTask implements Callable{

private ExecutorService executorService;

private String command;

public CommandTask(ExecutorService executorService, String command){

this.executorService = executorService;

this.command = command;

}

@Override

public ProcessResult call() throws Exception {

return CommandUtils.runCmdTest(executorService, command);

}

}

而后在上面的CommandUtils的基础上再封装一层变成CommandHelper,具体实现以下:ui

package com.majing.learning.fileupload.common.process;

import java.util.concurrent.ExecutorService;

import java.util.concurrent.Executors;

import java.util.concurrent.Future;

import java.util.concurrent.TimeUnit;

import org.slf4j.Logger;

import org.slf4j.LoggerFactory;

import com.majing.learning.fileupload.common.ConstValues;

public class CommandHelper {

private static Logger logger = LoggerFactory.getLogger(CommandHelper.class);

private static ExecutorService executorService=Executors.newFixedThreadPool(50);

private static long default_timeout = 8000;

public static ProcessResult process(String command){

return process(command, default_timeout, TimeUnit.MILLISECONDS);

}

public static ProcessResult process(String command, long timeout, TimeUnit unit){

CommandTask commandTask = new CommandTask(executorService, command);

Future processResult = executorService.submit(commandTask);

ProcessResult result = null;

try{

result = processResult.get(timeout, unit);

}catch(Exception e){

logger.error(ConstValues.EXCEPTION_OCCURED, e);

}

return result;

}

}

至此,咱们在须要调用系统命令时直接调用CommandHelper.process(command)就能够了,而后拿到返回结果ProcessResult。我也是本身作个记录,有须要的朋友能够直接拿去用。this

顺便说一句,采用封装的这个类在完成上面相同的任务时,时间都在相同的机器上,耗时从原来的10s瞬间减小至200ms之内,因而可知,在适当的场景调用系统命令是多么重要啊。

java调用linux设备,使用Java调用Linux系统命令相关推荐

  1. linux设备和驱动注册,Linux驱动第五篇-----驱动注册和生成设备节点

    加载驱动的指令是:insmod xx.ko 查看驱动的指令是: lsmod 卸载驱动的指令是:rmmod xx 在include/linux/platform_device.h这个文件中定义了平台驱动 ...

  2. Linux设备驱动01:Linux设备驱动概述

    目录 1. 设备驱动的作用 2. 有无操作系统时的设备驱动 2.1 无操作系统 2.1.1 硬件.驱动和应用程序的关系 2.1.2 单任务软件典型架构 2.2 有操作系统 2.2.1 硬件.驱动.操作 ...

  3. linux 设备驱动阻塞,深入浅出:Linux设备驱动中的阻塞和非阻塞I/O

    今天写的是Linux设备驱动中的阻塞和非阻塞I/0,何谓阻塞与非阻塞I/O?简单来说就是对I/O操作的两种不同的方式,驱动程序可以灵活的支持用户空间对设备的这两种访问方式. 一.基本概念: 阻塞操作 ...

  4. linux设备端口编写,在Linux上开发HDMI端口

    如何在不允许操作系统自动配置显示输出的情况下,从应用程序专门驱动HDMI输出怎么可能? 例如,使用标准DVI/VGA作为主显示器,但使用设备文件将Mplayer视频输出发送到HDMI. 这是一个很难通 ...

  5. linux设备文件的分类,Linux设备模型组件-类设备-设备类及subsystem

    Linux设备模型一.sysfs文件系统: sysfs文件系统是Linux2.6内核引入的,它被看成是与proc.devfs和devpty等同类别的文件系统,sysfs文件系统也是一个虚拟文件系统,它 ...

  6. linux设备驱动总结,《Linux设备驱动开发详解(第3版)》海量更新总结

    本博实时更新<Linux设备驱动开发详解(第3版)>的最新进展. 2015.2.26 几乎完成初稿. [F]是修正或升级:[N]是新增知识点:[D]是删除的内容 第1章 <Linux ...

  7. linux设备驱动程序调试方法,Linux设备驱动程序学习(2)-调试技术

    Linux设备驱动程序学习(2)-调试技术 Linux设备驱动程序学习(2)-调试技术 今天进入<Linux设备驱动程序(第3版)>第四章调试技术的学习. 一.内核中的调试支持 在前面已经 ...

  8. 嵌入式linux设备驱动程序是,嵌入式Linux设备驱动开发之:按键驱动程序实例-嵌入式系统-与非网...

    11.6  按键驱动程序实例 11.6.1  按键工作原理 LED和蜂鸣器是最简单的GPIO的应用,都不需要任何外部输入或控制.按键同样使用GPIO接口,但按键本身需要外部的输入,即在驱动程序中要处理 ...

  9. Linux设备之我是usb,linux那些事儿之我是usb

    linux那些事儿之我是usb,复旦大学教授肖林甫先生给学生们解说的linux操作系统的一些硬件驱动开发的事儿. 内核说明: 我是U盘 说的是2.6.10的内核 我是Sysfs 说的是2.6.10的内 ...

最新文章

  1. pytorch 维度练习
  2. linux用户密码策略求图,Linux用户密码策略
  3. python最长回文子串leetcode_Python版LeetCode5. 最长回文子串
  4. PHP 5.3以上版本新增的一些方法
  5. org.apache.ibatis.binding.BindingException: Type interface com.kuang.dao.UserDao is not known to the
  6. delimiter在mysql中的作用_细细研究MySql中delimiter起到的作用_MySQL
  7. 电机的入门之路系列4--PWM控制直流电机
  8. gin context和官方context_[系列文章] Gin框架 - 安装和路由配置
  9. 占据语音入口?苹果或将在2020年WWDC上推出SiriOS
  10. 破解日记 WiFi密码跑字典暴力破解 尽可能通俗易懂的版本
  11. 计算并输出九九乘法表用c语言流程图,输出九九乘法表 (C语言代码)
  12. iPhone 4 Cydia使用教程!精选Cydia源!cydia怎么添加源!Cydia源使用方法!越狱后使用cydia全攻略!...
  13. 原生小程序 引入vant-weapp 组件库 和 WeUI组件库
  14. php 图片后加参数缩图,智能的PHP缩图类
  15. Web3js 小笔记
  16. java arraylist 求和_java-ArrayList中类对象属性的总和
  17. Vue+iView身份证、统一社会信用编码校验
  18. 斑马打印机Zebra 105SL每次开机就出纸
  19. 神经网络学习笔记(一):全连接层的作用是什么?
  20. 小学就会背的乘法表,还藏着这么多秘密?

热门文章

  1. [Linux学习笔记] Linux常用命令 - 文件搜索命令
  2. 你以为PHP那么好自定义升级?
  3. 网络监控工具--ntop
  4. Ubuntu 10.10用LibreOffice替换OpenOffice
  5. 点击调试时提示MFC不包含调试信息
  6. DB pivot unpivot
  7. Python isdigit() isalnum()
  8. R语言-数据清洗-缺失值处理
  9. Shell编程 之 while循环 和 until循环 和 总结
  10. Expression.Blend.4 Chapter 图片和视频的使用