开发环境:Jdk 1.8
引入第三方库:commons-net-2.2.jar(针对第一种方法)
一、基于第三方库FtpClient的FTP服务器数据传输
由于是基于第三方库,所以这里基本上没有太多要说明的东西。就是导入第三方库再调用即可,调用过程从下面的代码可以参见。为了便于文章的完整性,这也是给出其程序结构图吧。
图-1 基于FtpClient的FTP网络文件传输图
所需要
commons.net-1.4.1.jar
jar包已保存到百度网盘ftptest中。或者http://pan.baidu.com/s/1hq5p7NI
/**
* ftp链接常量
*
*/
public class Ftp {
private String ipAddr;//ip地址
private Integer port;//端口号
private String userName;//用户名
private String pwd;//密码
private String path;//aaa路径
public String getIpAddr() {
return ipAddr;
}
public void setIpAddr(String ipAddr) {
this.ipAddr = ipAddr;
}
public Integer getPort() {
return port;
}
public void setPort(Integer port) {
this.port = port;
}
public String getUserName() {
return userName;
}
public void setUserName(String userName) {
this.userName = userName;
}
public String getPwd() {
return pwd;
}
public void setPwd(String pwd) {
this.pwd = pwd;
}
public String getPath() {
return path;
}
public void setPath(String path) {
this.path = path;
}
}
import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; import java.io.OutputStream; import org.apache.commons.net.ftp.FTPClient; import org.apache.commons.net.ftp.FTPFile; import org.apache.commons.net.ftp.FTPReply; import org.apache.log4j.Logger; public class FtpUtil { private static Logger logger=Logger.getLogger(FtpUtil.class); private static FTPClient ftp; /** * 获取ftp连接 * @param f * @return * @throws Exception */ public static boolean connectFtp(Ftp f) throws Exception{ ftp=new FTPClient(); boolean flag=false; int reply; if (f.getPort()==null) { ftp.connect(f.getIpAddr(),21); }else{ ftp.connect(f.getIpAddr(),f.getPort()); } ftp.login(f.getUserName(), f.getPwd()); ftp.setFileType(FTPClient.BINARY_FILE_TYPE); reply = ftp.getReplyCode(); if (!FTPReply.isPositiveCompletion(reply)) { ftp.disconnect(); return flag; } ftp.changeWorkingDirectory(f.getPath()); flag = true; return flag; } /** * 关闭ftp连接 */ public static void closeFtp(){ if (ftp!=null && ftp.isConnected()) { try { ftp.logout(); ftp.disconnect(); } catch (IOException e) { e.printStackTrace(); } } } /** * ftp上传文件 * @param f * @throws Exception */ public static void upload(File f) throws Exception{ if (f.isDirectory()) { ftp.makeDirectory(f.getName()); ftp.changeWorkingDirectory(f.getName()); String[] files=f.list(); for(String fstr : files){ File file1=new File(f.getPath()+"/"+fstr); if (file1.isDirectory()) { upload(file1); ftp.changeToParentDirectory(); }else{ File file2=new File(f.getPath()+"/"+fstr); FileInputStream input=new FileInputStream(file2); ftp.storeFile(file2.getName(),input); input.close(); } } }else{ File file2=new File(f.getPath()); FileInputStream input=new FileInputStream(file2); ftp.storeFile(file2.getName(),input); input.close(); } } /** * 下载链接配置 * @param f * @param localBaseDir 本地目录 * @param remoteBaseDir 远程目录 * @throws Exception */ public static void startDown(Ftp f,String localBaseDir,String remoteBaseDir ) throws Exception{ if (FtpUtil.connectFtp(f)) { try { FTPFile[] files = null; boolean changedir = ftp.changeWorkingDirectory(remoteBaseDir); if (changedir) { ftp.setControlEncoding("GBK"); files = ftp.listFiles(); for (int i = 0; i < files.length; i++) { try{ downloadFile(files[i], localBaseDir, remoteBaseDir); }catch(Exception e){ logger.error(e); logger.error("<"+files[i].getName()+">下载失败"); } } } } catch (Exception e) { logger.error(e); logger.error("下载过程中出现异常"); } }else{ logger.error("链接失败!"); } } /** * * 下载FTP文件 * 当你需要下载FTP文件的时候,调用此方法 * 根据<b>获取的文件名,本地地址,远程地址</b>进行下载 * * @param ftpFile * @param relativeLocalPath * @param relativeRemotePath */ private static void downloadFile(FTPFile ftpFile, String relativeLocalPath,String relativeRemotePath) { if (ftpFile.isFile()) { if (ftpFile.getName().indexOf("?") == -1) { OutputStream outputStream = null; try { File locaFile= new File(relativeLocalPath+ ftpFile.getName()); //判断文件是否存在,存在则返回 if(locaFile.exists()){ return; }else{ outputStream = new FileOutputStream(relativeLocalPath+ ftpFile.getName()); ftp.retrieveFile(ftpFile.getName(), outputStream); outputStream.flush(); outputStream.close(); } } catch (Exception e) { logger.error(e); } finally { try { if (outputStream != null){ outputStream.close(); } } catch (IOException e) { logger.error("输出文件流异常"); } } } } else { String newlocalRelatePath = relativeLocalPath + ftpFile.getName(); String newRemote = new String(relativeRemotePath+ ftpFile.getName().toString()); File fl = new File(newlocalRelatePath); if (!fl.exists()) { fl.mkdirs(); } try { newlocalRelatePath = newlocalRelatePath + '/'; newRemote = newRemote + "/"; String currentWorkDir = ftpFile.getName().toString(); boolean changedir = ftp.changeWorkingDirectory(currentWorkDir); if (changedir) { FTPFile[] files = null; files = ftp.listFiles(); for (int i = 0; i < files.length; i++) { downloadFile(files[i], newlocalRelatePath, newRemote); } } if (changedir){ ftp.changeToParentDirectory(); } } catch (Exception e) { logger.error(e); } } } public static void main(String[] args) throws Exception{ Ftp f=new Ftp(); f.setIpAddr("1111"); f.setUserName("root"); f.setPwd("111111"); FtpUtil.connectFtp(f); File file = new File("F:/test/com/test/Testng.java"); FtpUtil.upload(file);//把文件上传在ftp上 FtpUtil.startDown(f, "e:/", "/xxtest");//下载ftp文件测试 System.out.println("ok"); } }
1.FTP的连接及登录
- public static FtpClient connectFTP(String url, int port, String username, String password) {
- //创建ftp
- FtpClient ftp = null;
- try {
- //创建地址
- SocketAddress addr = new InetSocketAddress(url, port);
- //连接
- ftp = FtpClient.create();
- ftp.connect(addr);
- //登陆
- ftp.login(username, password.toCharArray());
- ftp.setBinaryType();
- } catch (FtpProtocolException e) {
- e.printStackTrace();
- } catch (IOException e) {
- e.printStackTrace();
- }
- return ftp;
- }
2.上传文件到FTP服务器
- public static void upload(String localFile, String ftpFile, FtpClient ftp) {
- OutputStream os = null;
- FileInputStream fis = null;
- try {
- // 将ftp文件加入输出流中。输出到ftp上
- os = ftp.putFileStream(ftpFile);
- File file = new File(localFile);
- // 创建一个缓冲区
- fis = new FileInputStream(file);
- byte[] bytes = new byte[1024];
- int c;
- while((c = fis.read(bytes)) != -1){
- os.write(bytes, 0, c);
- }
- System.out.println("upload success!!");
- } catch (FtpProtocolException e) {
- e.printStackTrace();
- } catch (IOException e) {
- e.printStackTrace();
- } finally {
- try {
- if(os!=null) {
- os.close();
- }
- if(fis!=null) {
- fis.close();
- }
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- }
3.从FTP服务器下载文件
- public static void download(String localFile, String ftpFile, FtpClient ftp) {
- InputStream is = null;
- FileOutputStream fos = null;
- try {
- // 获取ftp上的文件
- is = ftp.getFileStream(ftpFile);
- File file = new File(localFile);
- byte[] bytes = new byte[1024];
- int i;
- fos = new FileOutputStream(file);
- while((i = is.read(bytes)) != -1){
- fos.write(bytes, 0, i);
- }
- System.out.println("download success!!");
- } catch (FtpProtocolException e) {
- e.printStackTrace();
- } catch (IOException e) {
- e.printStackTrace();
- } finally {
- try {
- if(fos!=null) {
- fos.close();
- }
- if(is!=null){
- is.close();
- }
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- }
二、基于Socket的FTP服务器数据传输
其实上面的基于第三方包FtpClient的方法中,原理层也是基于Socket来进行通信的。所以,我们当然也可以使用Socket直接来写这个FtpClient的代码。下面给出基于Socket通信的结构构架图。这里有一点需要大家注意一下,我们的FTP协议中有两个端口(20和21)。通常情况下,我们的21号端口就是平时大家口口相传的是FTP服务器的端口号,不过其实它只是FTP服务器中的命令端口号。它是负责传送命令给FTP,一些操作如“登录”、“改变目录”、“删除文件”,依靠这个连接发送命令就可完成。而对于20号端口号(也有可能是其它的一些端口号),对于有数据传输的操作,主要是显示目录列表,上传、下载文件,我们需要依靠另一个Socket来完成。
所以在下面的结构图中,我们可以看到我们有重新获得端口号的过程,正是这个原因。
图-2 基于Socket的FTP网络文件传输图
1.FTP连接
- public void connectFtp() {
- try {
- mFtpClient = new Socket(Config.FTP.HOST_IP, Config.FTP.HOST_PORT);
- mReader = new BufferedReader(new InputStreamReader(mFtpClient.getInputStream()));
- mWriter = new BufferedWriter(new OutputStreamWriter(mFtpClient.getOutputStream()));
- sendCommand("USER " + Config.FTP.FTP_USERNAME);
- sendCommand("PASS " + Config.FTP.FTP_PASSWD);
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
2.向FTP服务器发送命令
- private void sendCommand(String command) throws IOException {
- if (Tools.StringTools.isEmpty(command)) {
- return;
- }
- if (mFtpClient == null) {
- return;
- }
- mWriter.write(command + " ");
- mWriter.flush();
- }
3.向FTP服务器上传文件
- public void uploadFile(String localPath, String ftpPath) throws IOException {
- // 进入被动模式
- sendCommand("PASV");
- // 获得ip和端口
- String response = readNewMessage();
- String[] ipPort = getIPPort(response);
- String ip = ipPort[0];
- int port = Integer.parseInt(ipPort[1]);
- // 建立数据端口的连接
- Socket dataSocket = new Socket(ip, port);
- sendCommand("STOR " + ftpPath);
- // 上传文件前的准备
- File localFile = new File(localPath);
- OutputStream outputStream = dataSocket.getOutputStream();
- FileInputStream fileInputStream = new FileInputStream(localFile);
- // 上传文件
- int offset;
- byte[] bytes = new byte[1024];
- while ((offset = fileInputStream.read(bytes)) != -1) {
- outputStream.write(bytes, 0, offset);
- }
- System.out.println("upload success!!");
- // 上传文件后的善后工作
- outputStream.close();
- fileInputStream.close();
- dataSocket.close();
- }
4.从FTP服务器下载文件
- public void downloadFile(String localPath, String ftpPath) throws IOException {
- // 进入被动模式
- sendCommand("PASV");
- // 获得ip和端口
- String response = readNewMessage();
- String[] ipPort = getIPPort(response);
- String ip = ipPort[0];
- int port = Integer.parseInt(ipPort[1]);
- // 建立数据端口的连接
- Socket dataSocket = new Socket(ip, port);
- sendCommand("RETR " + ftpPath);
- // 下载文件前的准备
- File localFile = new File(localPath);
- InputStream inputStream = dataSocket.getInputStream();
- FileOutputStream fileOutputStream = new FileOutputStream(localFile);
- // 下载文件
- int offset;
- byte[] bytes = new byte[1024];
- while ((offset = inputStream.read(bytes)) != -1) {
- fileOutputStream.write(bytes, 0, offset);
- }
- System.out.println("download success!!");
- // 下载文件后的善后工作
- inputStream.close();
- fileOutputStream.close();
- dataSocket.close();
- }
5.断开FTP服务器连接
- public void disconnectFtp() {
- if (mFtpClient == null) {
- return;
- }
- if (!mFtpClient.isConnected()) {
- return;
- }
- try {
- mFtpClient.close();
- } catch (IOException e) {
- e.printStackTrace();
- }
- }