IO输入输出

介绍

IO 实际就是输入流IN 输出 OUT
输出流分stderr(标准错误输出) 和stdout(标准输出),c语言里 对应的就是全局对象stdin stdout stderr
java里对应的就是System.in System.out System.err。
操作系统启动程序的时候我们可以指定程序的输入流和输出流以及错误流指向哪里。

System.err和System.out

标准错误输出流和标准输出流是有区别的。输出流是有缓冲的,只有遇到换行符或者结束的时候才会输出,而err是直接输出的。

shell或者cmd里的重定向

> 重定向输出流
2> 重定向错误输出流
< 重定向输入流

echo 1 > 1.txt
echo < 1.txt 

java子进程

通过Runtime.exec会产生一个子进程。子进程的输入流 对应的就是父进程的输出流 子进程的输入流 对于父进程就是输出流。

public static void ping(){
    try {
        Process process=Runtime.getRuntime().exec("ping www.baidu.com");
        InputStream in = process.getInputStream();
        BufferedReader br=new BufferedReader(new InputStreamReader(in,"gbk"));
        String line=null;
        while((line=br.readLine())!=null){
            System.out.println(line);
        }
    } catch (IOException e) {
        e.printStackTrace();
    }
}

process.getInputStream实际就是指向子进程的标准输出流
process.getErrorStream实际就是指向子进程的标准错误输出流
process.getOutputStream实际就是子进程的输入流

由于计算机默认编码是gbk所以 获取ping的标准输出指向了gbk的Reader
读取每一行的ping输出 打印到我们程序的输出 完成了最简单的ping功能

IO就像插座 我们把任意的IO对接起来

把cmd的IO和java的IO对接起来

/**
 *  对接java的输出流到cmd的输入流
 *  必须调用
 *  最近\r\n模拟回车
 */
public static void pipe(InputStream in,OutputStream out){
    try {
        BufferedReader bufferedReader=new BufferedReader(new InputStreamReader(in,"utf8"));
        String line="";
        while((line=bufferedReader.readLine())!=null){
            out.write(line.getBytes("gbk"));
            out.write("\r\n".getBytes("gbk"));
            out.flush();
        }
    } catch (UnsupportedEncodingException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }
}

/**
 *  对接cmd的输入流到java的输出流
 */
public static void pipe(InputStream in,String charset,OutputStream out,String outCharset) throws IOException {
    InputStreamReader br = new InputStreamReader(in, charset);
    char[] chunk=new char[2048];
    int len=0;
    StringBuilder stringBuilder=new StringBuilder();
    while((len=br.read(chunk,0,2048))!=-1){
        stringBuilder.append(chunk,0,len);
        System.out.write(stringBuilder.toString().getBytes(outCharset));
        System.out.flush();
        stringBuilder.delete(0,stringBuilder.length());
    }
}

/**
 *  启动三个线程 对接输入输出流和错误输出流
 */
public static void socket() throws IOException {
    Process process=Runtime.getRuntime().exec("cmd");
    new Thread(new Runnable() {
        @Override
        public void run() {
                pipe(System.in,process.getOutputStream());
        }
    }).start();
    new Thread(new Runnable() {
        @Override
        public void run() {
            try {
                pipe(process.getInputStream(),"gbk",System.out,"utf8");
            } catch (IOException e) {
                e.printStackTrace();
            }
            pipe(process.getInputStream(),System.out);
        }
    }).start();
    new Thread(new Runnable() {
        @Override
        public void run() {
            try {
                pipe(process.getErrorStream(),"gbk",System.err,"utf8");
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }).start();
}

同理我们可以把cmd的IO对接到socket的IO上 那么就可以实现远程访问cmd

socketIO和cmdIO对接

启动Server之后再启动Client可以看到可以使用cmd

public class CMDParent {

    /**
     * Sysytem.in 专用
     * @param out
     * @param charset
     */
    public static void SystemINpipe(OutputStream out,String charset){
        try {
            BufferedReader bufferedReader=new BufferedReader(new InputStreamReader(System.in,"utf8"));
            String line="";
            while((line=bufferedReader.readLine())!=null){
                out.write(line.getBytes(charset));
                out.write("\r\n".getBytes(charset));
                out.flush();
            }
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void pipe(InputStream in, String charset, OutputStream out, String outCharset) throws IOException {
        InputStreamReader br = new InputStreamReader(in, charset);
        char[] chunk=new char[2048];
        int len=0;
        StringBuilder stringBuilder=new StringBuilder();
        while((len=br.read(chunk,0,2048))!=-1){
            stringBuilder.append(chunk,0,len);
            System.out.write(stringBuilder.toString().getBytes(outCharset));
            System.out.flush();
            stringBuilder.delete(0,stringBuilder.length());
        }
    }
}

public class CMDServer extends CMDParent{

    private Socket client;

    public CMDServer(Socket client) {
        try {
            Process process=Runtime.getRuntime().exec("cmd");
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                        pipe(client.getInputStream(),"utf8",process.getOutputStream(),"gbk");
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }).start();
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                        pipe(process.getInputStream(),"gbk",client.getOutputStream(),"utf8");
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }).start();
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                        pipe(process.getErrorStream(),"gbk",client.getOutputStream(),"utf8");
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }).start();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String agrs[]) throws IOException {
        ServerSocket serverSocket=new ServerSocket(8091);
        Socket client;
        while((client=serverSocket.accept())!=null){
            new CMDServer(client);
        }
    }
}


public class CMDClient extends CMDParent {

    public static void main(String agrs[]) throws IOException {
        Socket socket=new Socket("localhost",8091);
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    pipe(socket.getInputStream(),"utf8",System.out,"utf8");
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }).start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                   SystemINpipe(socket.getOutputStream(),"utf8");
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }).start();
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342

推荐阅读更多精彩内容

  • 介绍 IO 实际就是输入流IN 输出 OUT输出流分stderr(标准错误输出) 和stdout(标准输出),c语...
    Charon_Pluto阅读 316评论 0 0
  • 介绍 IO 实际就是输入流IN 输出 OUT输出流分stderr(标准错误输出) 和stdout(标准输出),c语...
    代码界的扫地僧阅读 288评论 0 1
  • 一.java.io.File类 java.io.File类用于表示文件/目录 File只用于表示文件的信息(名称,...
    liangxifeng833阅读 750评论 0 1
  • Java中输入/输出的类存放在java.io包中 输入流类都是抽象类InputStream(字节输入流)或抽象类R...
    朱Simon阅读 620评论 0 0
  • 输入输出流。 InputStream与OutputStream 读与写 使用InputStream读取字节并转为字...
    BlackNeko阅读 913评论 0 0