- 浏览: 102986 次
- 性别:
- 来自: 北京
文章分类
- 全部博客 (99)
- 经济 (1)
- dwr (2)
- 测试 (0)
- java (29)
- resin (1)
- oracle (3)
- 感悟 (1)
- jvm (15)
- mina2 (5)
- j2se (12)
- linux (28)
- protobuf (1)
- tcp/ip (0)
- jdbc (0)
- 数据库 (4)
- 游戏 (0)
- 技术文档 (1)
- nosql (2)
- 算法 (2)
- apache (2)
- mysql (1)
- hashcode (1)
- spring (2)
- quartz (5)
- netcat (2)
- 分页 (1)
- 正则 (0)
- shell (1)
- lsof (1)
- nginx (1)
- git (1)
最新评论
-
fys124974704:
你试下将第三条写成以下这样,你会发现你的结论不对:select ...
ORACLE分页SQL语句 -
ikon:
两个乘数没有转成integer,而是当成字符串;BigInte ...
计算任意2个正整数的乘积 -
kidding87:
效率不是很高,思路没有问题,但是你的两个乘数输入都都转为Int ...
计算任意2个正整数的乘积 -
k1280000:
------------------------同意!
学习之道
前些天使用Java调用外部程序的时候,发现线程会堵塞在waitfor()方法。 如果直接在Shell中调用这个程序,程序会很快结束,不会僵死。 为什么会堵塞呢,原因是当调用exec(cmd)后,JVM会启动一个子进程,该进程会与JVM进程建立3个管道连接,标准输入,标准输出和标准错误流。假设该程序不断在向标准输出流和标准错误流写数据,而JVM不读取,数据会暂时缓冲在Linux的缓冲区,缓冲区满后该程序将无法继续写数据,会僵死,所以Java程序就会僵死在waitfor(),永远无法结束。 解决办法就是增加两个线程,一个线程负责读标准输出流,另一个负责读标准错误流,这样子数据就不会积压在缓冲区,程序就能够顺利运行。 查看源代码后,还发现一个潜在的问题。但程序执行到exec的时候,JVM会使用管道,占有3个文件句柄,但程序运行结束后,这三个句柄并不会自动关闭,这样最终会导致java.io.IOException: Too many open files。所以就算外部程序的没有输出,也必须关闭句柄: 我们发觉当调用close()方法后,JVM并不会立即回收句柄,具体的回收时间不确定。另外如果不调用close(),句柄也会被回收,也可能发生“Too many open files”的错误。根据这篇文章,不同的垃圾收集器会选择不同的回收策略。所以最好还是要关闭。 总结
调用方法如下:
1.如果外部程序有大量输出,需要启动额外的线程来读取标准输出和标准错误流
2.必须关闭三个句柄
import java.io.Closeable; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; public class ProcessUtil { private static final int DEFAULT_BUFFER_SIZE = 1024; private static Log log = LogFactory.getLog(ProcessUtil.class); private InputStream in; private OutputStream out; private OutputStream err; public ProcessUtil(OutputStream out, OutputStream err, InputStream in) { if (out == null) { out = new NullOutputStream(); } if (err == null) { err = new NullOutputStream(); } this.out = out; this.err = err; this.in = in; } public int process(String cmd, String[] envp, File dir) throws IOException, InterruptedException { Process p = Runtime.getRuntime().exec(cmd, envp, dir); return process(p); } public int process(String[] cmdarray, String[] envp, File dir) throws IOException, InterruptedException { Process p = Runtime.getRuntime().exec(cmdarray, envp, dir); return process(p); } private int process(Process p) throws IOException, InterruptedException { try { OutputStream pin = p.getOutputStream(); StreamGobbler outg = new StreamGobbler(p.getInputStream(), out); StreamGobbler errg = new StreamGobbler(p.getErrorStream(), err); outg.start(); errg.start(); if (in != null) { byte[] inBuf = new byte[DEFAULT_BUFFER_SIZE]; int inN = 0; while (-1 != (inN = in.read(inBuf))) { pin.write(inBuf, 0, inN); } pin.flush(); } int code = p.waitFor(); outg.join(); errg.join(); return code; } finally { closeQuietly(p.getOutputStream()); closeQuietly(p.getInputStream()); closeQuietly(p.getErrorStream()); } } private void closeQuietly(Closeable closeable) { try { if (closeable != null) closeable.close(); } catch (Exception e) { log.warn("close error", e); } } } class StreamGobbler implements Runnable { private static final int DEFAULT_BUFFER_SIZE = 1024; private static Log log = LogFactory.getLog(StreamGobbler.class); private InputStream is; private OutputStream os; private Thread thread; public StreamGobbler(InputStream is, OutputStream os) { this.is = is; this.os = os; } public void start() { thread = new Thread(this); thread.setDaemon(true); thread.start(); } public void run() { try { byte[] buf = new byte[DEFAULT_BUFFER_SIZE]; int n = 0; while (-1 != (n = is.read(buf))) { os.write(buf, 0, n); } os.flush(); } catch (Exception ex) { log.warn("stream error", ex); } } public void join() throws InterruptedException { thread.join(); } } class NullOutputStream extends OutputStream { public static final NullOutputStream NULL_OUTPUT_STREAM = new NullOutputStream(); @Override public void write(byte[] b, int off, int len) { } @Override public void write(int b) { } @Override public void write(byte[] b) throws IOException { } }
发表评论
-
获取堆栈转储信息
2012-09-12 10:21 0咨询个关于获得java进程堆栈信息的问题 ... -
ulimit 放开用户限制 生成core文件
2012-09-12 10:04 1389ulimit -a 用来显示当前的各种用户进程限制。 ... -
coredump使用方法
2012-09-12 10:02 1523Linux系统下程序出错时会自动生成一个core文件,该 ... -
一个java crash的故障分析过程
2012-09-12 09:55 834一个应用在周五出现java进程消失,没有任何日志。先查看 ... -
Linux 系统设置 ulimit 以及 Core文件的生成
2012-09-12 09:55 1147查看限制情况 ulimit -a 可以看到如下信息 ... -
线上游戏的gc情况
2012-07-18 15:47 846resin jvm参数配置如下: ... -
JVM调优的几种策略(好)
2012-07-18 14:59 849JVM参数调优是一个很 ... -
Java GC 日志详解
2012-07-18 14:51 776ava GC日志可以通过 +PrintGCDetails ... -
JVM GC日志时间问题
2012-07-18 14:22 1054几乎所有的资料上说到打印JVM GC log的时候都会 ... -
jstat应用
2012-06-27 17:33 741用以判断JVM是否存在内存问题呢?如何判断JVM垃圾回收 ... -
Quartz定时任务学习(一)简单任务
2012-04-27 12:30 846学习quartz首先了解三个概念: 调度器:负责调度作 ... -
介绍Quartz
2012-04-27 12:22 1416介绍Quartz Quartz是一个开源的任务调度 ... -
理解Quartz触发器(1)
2012-04-27 11:37 870Quartz中一个Job往往是 ... -
Quartz中SimpleTrigger的探讨
2012-04-27 11:34 7111.来写一个每隔10秒启动一次任务的例子. import j ... -
关于hashcode 里面 使用31 系数的问题
2012-03-27 09:16 811首先我们来了解一下h ... -
理解Java对象序列化
2012-02-15 09:38 697关于Java序列化的文章早 ... -
认识Arrays(一)打印
2012-02-03 16:52 575Arrays提供了一组操作array的静态方法。 一、基本类 ... -
apache.commons.io 笔记1
2012-01-19 17:13 1068看看,常见的东西都有了,如查询盘的剩余空间,文件夹大小, ... -
commons-io 自动加载配置文件
2012-01-19 16:55 754org.apache.commons.io.monitor.F ... -
oracle java数据类型对应表
2011-12-05 13:37 0[img]http://dl.iteye.com/upload ...
相关推荐
主要介绍了Java基于Runtime调用外部程序出现阻塞的解决方法,是一个非常实用的技巧,需要的朋友可以参考下
Java数据压缩与传输实例,可以学习一下实例化套按字、得到文件输入流、压缩输入流、文件输出流、实例化缓冲区、写入数据到文件、关闭输入流、关闭套接字关闭输出流、输出错误信息等Java编程小技巧。 Java数组倒置...
Java数据压缩与传输实例,可以学习一下实例化套按字、得到文件输入流、压缩输入流、文件输出流、实例化缓冲区、写入数据到文件、关闭输入流、关闭套接字关闭输出流、输出错误信息等Java编程小技巧。 Java数组倒置...
13.16.3 用Java 1.1 AWT制作窗口和程序片 13.16.4 再探早期示例 13.16.5 动态绑定事件 13.16.6 将商业逻辑与UI逻辑区分开 13.16.7 推荐编码方法 13.17 Java 1.1 UI API 13.17.1 桌面颜色 13.17.2 打印 13.17.3 剪贴...
13.16.3 用Java 1.1 AWT制作窗口和程序片 13.16.4 再探早期示例 13.16.5 动态绑定事件 13.16.6 将商业逻辑与UI逻辑区分开 13.16.7 推荐编码方法 13.17 Java 1.1 UI API 13.17.1 桌面颜色 13.17.2 打印 13.17.3 剪贴...
13.16.3 用Java 1.1 AWT制作窗口和程序片 13.16.4 再探早期示例 13.16.5 动态绑定事件 13.16.6 将商业逻辑与UI逻辑区分开 13.16.7 推荐编码方法 13.17 Java 1.1 UI API 13.17.1 桌面颜色 13.17.2 打印 13.17.3 剪贴...
第1章 让自己的第一个Java程序跑起来 2 教学视频:19分钟 1.1 想要用Java改变这个世界吗? 2 1.1.1 Java有什么优势? 2 1.1.2 Java在哪儿? 3 1.2 准备好开始Java之旅 3 1.2.1 下载JDK 4 1.2.2 安装JDK 5 ...
学生提问:当我们使用编译C程序时,不仅需要指定存放目标文件的位置,也需要指定目标文件的文件名,这里使用javac编译Java程序时怎么不需要指定目标文件的文件名呢? 13 1.5.3 运行Java程序 14 1.5.4 根据...
第1章 让自己的第一个Java程序跑起来 2 教学视频:19分钟 1.1 想要用Java改变这个世界吗? 2 1.1.1 Java有什么优势? 2 1.1.2 Java在哪儿? 3 1.2 准备好开始Java之旅 3 1.2.1 下载JDK 4 1.2.2 安装JDK 5 ...
13.16.3 用Java 1.1 AWT制作窗口和程序片 13.16.4 再探早期示例 13.16.5 动态绑定事件 13.16.6 将商业逻辑与UI逻辑区分开 13.16.7 推荐编码方法 13.17 Java 1.1 UI API 13.17.1 桌面颜色 13.17.2 打印 13.17.3 剪贴...
13.16.3 用Java 1.1 AWT制作窗口和程序片 13.16.4 再探早期示例 13.16.5 动态绑定事件 13.16.6 将商业逻辑与UI逻辑区分开 13.16.7 推荐编码方法 13.17 Java 1.1 UI API 13.17.1 桌面颜色 13.17.2 打印 13.17.3 剪贴...
13.16.3 用Java 1.1 AWT制作窗口和程序片 13.16.4 再探早期示例 13.16.5 动态绑定事件 13.16.6 将商业逻辑与UI逻辑区分开 13.16.7 推荐编码方法 13.17.1 桌面颜色 13.17.2 打印 13.17.3 剪贴板 13.18 可视编程和 ...
13.16.3 用Java 1.1 AWT制作窗口和程序片 13.16.4 再探早期示例 13.16.5 动态绑定事件 13.16.6 将商业逻辑与UI逻辑区分开 13.16.7 推荐编码方法 13.17 Java 1.1 UI API 13.17.1 桌面颜色 13.17.2 打印 13.17.3 剪贴...
13.16.3 用Java 1.1 AWT制作窗口和程序片 13.16.4 再探早期示例 13.16.5 动态绑定事件 13.16.6 将商业逻辑与UI逻辑区分开 13.16.7 推荐编码方法 13.17 Java 1.1 UI API 13.17.1 桌面颜色 13.17.2 打印 13.17...
28 1.3 程序开发过程 29 1.4 编码规范 29 1.5 HelloWorld:第一个Java程序 30 1.5.1 编写程序代码 30 1.5.2 编译程序代码并运行 30 1.5.3 注意事项 31 1.6 使用Eclipse集成开发工具开发 32 1.7 综合练习 32 1.8 小结...
13.16.3 用Java 1.1 AWT制作窗口和程序片 13.16.4 再探早期示例 13.16.5 动态绑定事件 13.16.6 将商业逻辑与UI逻辑区分开 13.16.7 推荐编码方法 13.17 Java 1.1 UI API 13.17.1 桌面颜色 13.17.2 打印 13.17.3 剪贴...
增加了大量处理各种常见JVM问题的技巧和最佳实践;增加了若干与生产环境相结合的实战案例;对第1版中的错误和不足之处的修正;等等。 第2版不仅技术更新、内容更丰富,而且实战性更强。全书共分为五大部分,围绕...