ChannelShell和ChannelExec的区别

ChannelShell、ChannelExec、ChannelSftp,前两类用于执行命令(命令可以是shell语句,也可以是python xxx.py),后一种是用于上传下载文件。

ChannelShell和ChannelExec的区别: 前者是交互式的,在channel.connect()之前,需要获取outputStream和inputStream,然后outputstream发送命令,从instream中读取命令的结果(注意,发送命令之后,读取命令之前要等待一会儿,一般需要写个循环判断,每秒读一次,根据实际情况设置xx秒超时退出),但是有个坏处是,它执行就像登陆到vm上打印的信息一样,无论执行命令后的任何信息,它都会通过instream返回到客户端来,而你可能仅仅需要命令执行后的结果;于是就有了后者,非交互的,一次通道执行一条命令(当然如果你组合的好,也可以多条,反正就是一个字符串,一次交互,偷偷的告诉你,这一个python脚本,下发的命令去执行这个脚本,可以做好多好多事情哦),好处是,它返回的信息非常干净,只返回标准输出,标准错误是不返回的,这时你可以利用python的print,正确时你print正确信息,错误时print错误信息,客户端都能获取到结果啦(因为print是标准输出)。
使用步骤:

1、new一个JSch对象;
2、从JSch对象中获取Session,用于连接,并设置连接信息(根据SSH的连接原理,有两种方式,一是用户名+密码,二是用户名+privatekey+passphrase,第二种方式要在第1步设置,即jsch.addIdentity(xxx));
3、使用session对象调用opnChannel("xxx")打开通信信道,并连接;
4、后面就是不同的channel,不同的操作啦。

在用jsch中的ChannelShell时,遇到问题:
①这个方法会返回命令提示符;
比如我要执行下面几个命令:

ChannelShell channel = (ChannelShell) session.openChannel("shell");
        channel.connect();
        InputStream inputStream = channel.getInputStream();
        OutputStream outputStream = channel.getOutputStream();
        String cmd = "ls \n\r";
        outputStream.write(cmd.getBytes());
        String cmd2 = "cd /home/jenkins/workspace/ggservice \n\r";
        outputStream.write(cmd2.getBytes());
        String cmd3 = "pwd \n\r";
        outputStream.write(cmd3.getBytes());
        outputStream.flush();
BufferedReader in = new BufferedReader(new InputStreamReader(inputStream));
        String msg = null;
        while((msg = in.readLine())!=null){
            System.out.println(msg);
        }
        in.close();

结果连[root@master01 ~]这样的命令提示符和输入的命令都出来,我其实是不需要这个,我要的只是结果。

②由于使用BufferedReader的readLine()方法,结果会产生阻塞。

BufferedReader in = new BufferedReader(new InputStreamReader(inputStream)); String msg = null;
while((msg = in.readLine())!=null){//当所有的命令都执行完毕后,就会产生阻塞
    System.out.println(msg);
}
in.close();

为什么会这样呢?
因为我们建立的是shell管道,并且我们又使用readLine方法,当命令全部执行完毕后,远程端并不知道执行完毕,还在等待接受数据,所以呢reandLine就一直阻塞在那里。
即便你换成read方法还是一样的,因为shell管道本身就是交互模式的。要想停止,有两种方式:
①人为的发送一个exit命令,告诉程序本次交互结束啦
②使用字节流中的available方法,来获取数据的总大小,然后循环去读。

①代码:

InputStream inputStream = channel.getInputStream();//从远程端到达的所有数据都能从这个流中读取到
    OutputStream outputStream = channel.getOutputStream();//写入该流的所有数据都将发送到远程端。
    //使用PrintWriter流的目的就是为了使用println这个方法
    //好处就是不需要每次手动给字符串加\n
    PrintWriter printWriter = new PrintWriter(outputStream);

    String cmd = "ls";
    printWriter.println(cmd);
    String cmd2 = "cd /home/jenkins/workspace/ggservice";
    printWriter.println(cmd2);
    String cmd3 = "ls";
    printWriter.println(cmd3);
    printWriter.println("exit");//加上个就是为了,结束本次交互
    printWriter.flush();
BufferedReader in = new BufferedReader(new InputStreamReader(inputStream));
String msg = null;

    while((msg = in.readLine())!=null){
        System.out.println(msg);
    }
    in.close();

ChannelShell channel = (ChannelShell) session.openChannel("shell");
channel.connect();
//从远程端到达的所有数据都能从这个流中读取到
InputStream in = channel.getInputStream();
//写入该流的所有数据都将发送到远程端。
OutputStream outputStream = channel.getOutputStream();
byte[] tmp=new byte[1024];
while(true){
  while(in.available()>0){
    int i=in.read(tmp, 0, 1024);
    if(i<0)break;
    System.out.print(new String(tmp, 0, i));
  }
  if(channel.isClosed()){
    if(in.available()>0) continue;
    System.out.println("exit-status: "+channel.getExitStatus());
    break;
  }
}

这样就不会阻塞啦

最后我就去查ChannelShell和ChannelExec区别
ChannelShell

对于ChannelShell,以输入流的形式,提供命令和输入这些命令,这就像在本地计算机上使用交互式shell
(它通常用于:交互式使用)
ChannelExec

对于ChannelExec,在调用connect()方法之前这个命令提供了setCommand()方法,
并且这些命令作为输入将以输入流的形式被发送出去。
(通常,你只能有调用setCommand()方法一次,多次调用只有最后一次生效),
但是你可以使用普通shell的分隔符(&,&&,|,||,; , n, 复合命令)来提供多个命令。
这就像在你本机上执行一个shell脚本一样(当然,如果一个命令本身就是个交互式shell,这样就像ChannelShell)

你可能感兴趣的:(JAVA)