-
10.2.4.3 例子3:网络应用层协议的开发
清华大学出版社《Java程序员,上班那点事儿》作者:钟声——第10章《高手有多高有多菜》部分节选。
大家也许都用过FTP上传下载工具,比如“LeapFTP”这个工具是一个很方便的FTP服务器上传下载工具,如图所示。这个工具很方便,输入用户名密码以后,就可以看到FTP服务器端的文件列表,便于进行上传与下载操作。
你是否试过自己用Java编写一个FTP的文件上传与下载应用程序?
Java也可以开发出这样的程序来,并不复杂,我们先看看,利用Java的FTP类工具包制作FTP应用程序的开发是怎么做的,请看如下程序:
import sun.net.*;
import sun.net.ftp.*;
public class FTP {
public static void main(String[] args) {
String server="192.168.0.12"; //输入的FTP服务器的IP地址
String user="useway"; //登录FTP服务器的用户名
String password="!@#$%abce"; //登录FTP服务器的用户名的口令
String path="/home/useway"; //FTP服务器上的路径
try {
FtpClient ftpClient=new FtpClient(); //创建FtpClient对象
ftpClient.openServer(server); //连接FTP服务器
ftpClient.login(user, password); //登录FTP服务器
if (path.length()!=0) ftpClient.cd(path);
TelnetInputStream is=ftpClient.list();
int c;
while ((c=is.read())!=-1) {
System.out.print((char)c);
}
is.close();
ftpClient.closeServer();//退出FTP服务器
}
catch(Exception ex){
}
}
}
如果你感兴趣的话,可以自己写一个这个程序,当本程序运行以后,我们看到如图所示的情况,列出了服务器端程序的目录内容。
这个程序是一个简单的得到FTP服务器端文件列表的程序,但不要误会,这个程序可称不上“网络应用层协议”程序的开发!
这个程序仅仅是利用“sun.net.*;”和“sun.net.ftp.*;”中的相关类进行的对FTP端的操作的,我们根本没有利用Java的Socket的在网络层面向FTP服务器端发送任何请求,而是通过Java提供的工具包,向服务器端发送的链接请求。
利用Java的FTP包来链接FTP服务器的好处在于我们不需要关心网络层面发送数据的具体细节,而只要调用相应的方法就行了。利用Java的FTP包来链接FTP服务器的缺点是使开发者不知道应用层协议收发的来龙去脉,搞不清楚其中原理,对底层数据的把握程度非常弱。
讲到这里有程序员会问:“那么FTP在网络层面和PC与服务器间是如何交互的呢?”,好,就给大家列出FTP协议交互过程。
请看下面的一段FTP协议交互的例子:
FTP服务器: 220 (vsFTPd 2.0.1)
FTP客户端: USER useway
FTP服务器: 331 Please specify the password.
FTP客户端: PASS !@#$%abce
FTP服务器: 230 Login successful.
FTP客户端: CWD /home/useway
FTP服务器: 250 Directory successfully changed.
FTP客户端: EPSV ALL
FTP服务器: 200 EPSV ALL ok.
FTP客户端: EPSV
FTP服务器: 229 Entering Extended Passive Mode (|||62501|)
FTP客户端: LIST
FTP服务器: 150 Here comes the directory listing.
FTP服务器: 226 Directory send OK.
FTP客户端: QUIT
FTP服务器: 221 Goodbye.
以上这段文字其实就是FTP服务器和FTP客户端之间相互交互的过程,它们之间传递信息的协议是TCP协议,互相发送的内容就是上面这段文字所写的内容。
我们下面逐步的去解释每一句话的含义:
FTP服务器: 220 (vsFTPd 2.0.1) |说明:链接成功
FTP客户端: USER useway |说明:输入用户名
FTP服务器: 331 Please specify the password. |说明:请输入密码
FTP客户端: PASS !@#$%abce |说明:输入密码
FTP服务器: 230 Login successful. |说明:登录成功
FTP客户端: CWD /home/useway |说明:切换目录
FTP服务器: 250 Directory successfully changed. |说明:目录切换成功
FTP客户端: EPSV ALL |说明:为EPSV被动链接方式
FTP服务器: 200 EPSV ALL ok. |说明:OK
FTP客户端: EPSV |说明:链接
FTP服务器: 229 Entering Extended Passive Mode (|||62501|) |说明:被动链接端口为62501
FTP客户端: LIST |说明:执行LIST显示文件列表
FTP服务器: 150 Here comes the directory listing. |说明:列表从62501端口被发送
FTP服务器: 226 Directory send OK. |说明:发送完成
FTP客户端: QUIT |说明:退出FTP
FTP服务器: 221 Goodbye. |说明:再见
有了以上文字的内容,我们不需要任何工具也可以得到FTP文件列表了,不信你跟着我一起做一遍。
第一步:首先打开CMD进入DOS命令行模式,键入:
telnet 192.168.0.1 21[回车]
说明:Telnet 到Ftp服务器的21端口。
执行该命令后,得到的结果如图所示。
大家发现什么问题了吗?
提示的内容正好就是,我们上面一段文字的第一句:220 (vsFTPd 2.0.1),这说明FTP服务器已经接受了我们的链接,已经可以进行下一步操作了。
第二步:将后面的一系列发送内容逐个键入:
USER useway[回车]
PASS !@#$%abce[回车]
CWD /home/useway[回车]
EPSV ALL[回车]
EPSV[回车]
得到的结果如图所示。
好,这回FTP服务器给出了一系列的回应,在最后给出了一个新的端口号"58143"。
第三步:再打开一个新的CMD窗口,键入:
telnet 192.168.0.1 58143[回车]
注意,这次Telnet请求链接服务器的端口号是“58143”,是FTP服务器给我们的一个链接端口。链接后,窗口为空白没有任何提示,如图所示。
第四步:回到第一个CMD窗口,键入:
LIST[回车]
第五步:这时候第二CMD窗口就接收到了文件列表:
第二个窗口接收到了文件列表如图所示。
第六步:退出操作
QUIT[回车]
执行完成后,失去与主机的链接,如图所示。
大家看到了吧,FTP协议就是这样的一个交互过程,利用系统自带的Telnet工具也可以完成FTP的这些基本命令的操作。如果,你想用Java的Socket完成以上操作就只需要一步一步的按照上述内容发送字符串给FTP服务器端就行了。
我们下面也给出例子代码:
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
public class FTPClient{
public static void main(String[] args) throws Exception{
Socket socket = new Socket("192.168.0.1",21);
InputStream is = socket.getInputStream();
OutputStream os = socket.getOutputStream();
//接收初始链接信息
byte[] buffer = new byte[100];
int length = is.read(buffer);
String s = new String(buffer, 0, length);
System.out.println(s);
//发送用户名
String str = "USER usewayn";
os.write(str.getBytes());
//得到返回值
length = is.read(buffer);
s = new String(buffer, 0, length);
System.out.println(s);
//发送密码
str = "PASS !@#$%abcdn";
os.write(str.getBytes());
//得到返回值
length = is.read(buffer);
s = new String(buffer, 0, length);
System.out.println(s);
//发送切换文件夹指令
str = "CWD /home/usewayn";
os.write(str.getBytes());
//得到返回值
length = is.read(buffer);
s = new String(buffer, 0, length);
System.out.println(s);
//设置模式
str = "EPSV ALLn";
os.write(str.getBytes());
//得到返回值
length = is.read(buffer);
s = new String(buffer, 0, length);
System.out.println(s);
//得到被动监听信息
str = "EPSVn";
os.write(str.getBytes());
//得到返回值
length = is.read(buffer);
s = new String(buffer, 0, length);
System.out.println(s);
//取得FTP被动监听的端口号
String portlist=s.substring(s.indexOf("(|||")+4,s.indexOf("|)"));
System.out.println(portlist);
//实例化ShowList线程类,链接FTP被动监听端口号
ShowList sl=new ShowList();
sl.port=Integer.parseInt(portlist);
sl.start();
//执行LIST命令
str = "LISTn";
os.write(str.getBytes());
//得到返回值
length = is.read(buffer);
s = new String(buffer, 0, length);
System.out.println(s);
//关闭链接
is.close();
os.close();
socket.close();
}
}
//得到被动链接信息类,这个类是多线程的
class ShowList extends Thread{
public int port=0;
public void run(){
try{
Socket socket = new Socket("192.168.0.1",this.port);
InputStream is = socket.getInputStream();
OutputStream os = socket.getOutputStream();
byte[] buffer = new byte[10000];
int length = is.read(buffer);
String s = new String(buffer, 0, length);
System.out.println(s);
//关闭链接
is.close();
os.close();
socket.close();
}
catch(Exception ex){
}
}
}
该程序运行后得到的运行结果如图所示,基本上和上面的运行效果相同吧,底层又如何,无非是将那些封装好的方法解开来运行,只要了解到了它们运行的规则,我们自己可以开发出一样的程序来。