http://blog.csdn.net/wumenglu1018/article/details/54233826
FTP的主動(dòng)傳輸和被動(dòng)傳輸
主動(dòng)模式:
- 客戶端從端口N向服務(wù)器21端口發(fā)起連接.
- 客戶端再N+1端口監(jiān)聽(tīng)服務(wù)器的連接.
-
服務(wù)器從端口20 向客戶端 N+1 端口發(fā)起連接.
image.png
被動(dòng)模式:
- 客戶端從端口N向服務(wù)器21端口發(fā)起連接.
- 服務(wù)端在X端口(X>1024)監(jiān)聽(tīng)客戶端的連接.
-
客戶端從端口N+1 向服務(wù)端 X 端口發(fā)起連接.
image.png
主動(dòng)FTP:
命令連接:客戶端 大于1024端口 -> 服務(wù)器 21端口
數(shù)據(jù)連接:客戶端 大于1024端口 <- 服務(wù)器20端口
被動(dòng)FTP:
命令連接:客戶端 大于1024端口 -> 服務(wù)器 21端口
數(shù)據(jù)連接:客戶端 大于1024端口 -> 服務(wù)器 大于1024端口
簡(jiǎn)要總結(jié):
- 主動(dòng)FTP對(duì)FTP服務(wù)器的管理有利,但對(duì)客戶端的管理不利。因?yàn)镕TP服務(wù)器企圖與客戶端的高位隨機(jī)端口建立連接,而這個(gè)端口很有可能被客戶端的防火墻阻塞掉。
- 被動(dòng)FTP對(duì)FTP客戶端的管理有利,但對(duì)服務(wù)器端的管理不利。因?yàn)榭蛻舳艘c服務(wù)器端建立兩個(gè)連接,其中一個(gè)連到一個(gè)高位隨機(jī)端口,而這個(gè)端口很有可能被服務(wù)器端的防火墻阻塞掉。
辦法:
- 既然FTP服務(wù)器的管理員需要他們的服務(wù)器有最多的客戶連接,那么必須得支持被動(dòng)FTP。我們可以通過(guò)為FTP服務(wù)器指定一個(gè)有限的端口范圍來(lái)減小服務(wù)器高位端口的暴露。這樣,不在這個(gè)范圍的任何端口會(huì)被服務(wù)器的防火墻阻塞。雖然這沒(méi)有消除所有針對(duì)服務(wù)器的危險(xiǎn),但它大大減少了危險(xiǎn).
例:
Class FTPParam : FTP所需參數(shù)
import lombok.Builder;
/**
* ftp配置屬性
*/
@Builder
public class FTPParam {
private String host;
private int port;
private String username;
private String password;
private String uploadPath;
public String getHost() {
return host;
}
public void setHost(String host) {
this.host = host;
}
public int getPort() {
return port;
}
public void setPort(int port) {
this.port = port;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
public String getUploadPath() {
return uploadPath;
}
public void setUploadPath(String uploadPath) {
this.uploadPath = uploadPath;
}
/* (non-Javadoc)
* @see java.lang.Object#toString()
*/
@Override
public String toString() {
StringBuilder builder = new StringBuilder();
builder.append("FTPParam [host=");
builder.append(host);
builder.append(", port=");
builder.append(port);
builder.append(", username=");
builder.append(username);
builder.append(", password=");
builder.append(password);
builder.append(", uploadPath=");
builder.append(uploadPath);
builder.append("]");
return builder.toString();
}
}
Class FTPUtil: ftp工具類(lèi)
public class FTPUtil {
public final static Logger LOG = LoggerFactory.getLogger(FTPUtil.class);
/**
* 發(fā)送文件到FTP
*
* @param fileLocalPath 本地文件路徑(包含文件名 eg: //var//temp//test)
* @param fileName 遠(yuǎn)程服務(wù)器文件名
* @param param ftp配置
* @return
*/
public static boolean downLoadFileToLocal(String fileLocalPath, String fileName, FTPParam param) {
LOG.info("開(kāi)始下載文件至本地,param:{}",param);
//狀態(tài)
boolean result = false;
FTPClient ftpClient = new FTPClient();
FileOutputStream fout = null;
try {
ftpClient.connect(param.getHost(), param.getPort());
//判斷ftp連接
if(!FTPReply.isPositiveCompletion(ftpClient.getReplyCode())) {
LOG.error("ftp connect fail,host:{}",param.getHost());
ftpClient.disconnect();
return result;
}
LOG.info("ftpClient.connect success.");
//判斷ftp登陸
if(!ftpClient.login(param.getUsername(), param.getPassword())) {
LOG.error("ftp login fail,username:{}",param.getHost());
ftpClient.logout();
return result;
}
//設(shè)置文件類(lèi)型 :二進(jìn)制
ftpClient.setFileType(FTPClient.BINARY_FILE_TYPE);
//采用被動(dòng)模式傳輸
ftpClient.enterLocalPassiveMode();
//設(shè)置路徑
LOG.info("ftpClient.connect 設(shè)置路徑:{}",param.getUploadPath());
boolean change = ftpClient.changeWorkingDirectory(param.getUploadPath());
if(change){
fout = new FileOutputStream(fileLocalPath);
result = ftpClient.retrieveFile(fileName, fout);
LOG.info("---FTP下載完成---");
}
else {
LOG.info("ftpClient.connect 設(shè)置路徑失?。簕}", param.getUploadPath());
}
} catch (Exception e) {
LOG.error("下載文件出錯(cuò),報(bào)錯(cuò):",e);
} finally {
try {
if(fout != null) {
fout.close();
}
if(ftpClient.isConnected()) {
ftpClient.logout();
ftpClient.disconnect();
}
} catch (IOException e) {
LOG.error("關(guān)閉FTP或字節(jié)流出錯(cuò),報(bào)錯(cuò):",e);
}
}
return result;
}
/**
* 發(fā)送文件到FTP
*
* @param fileLocalPath 本地文件路徑(包含文件名 eg: //var//temp//test)
* @param fileName 文件名(上傳到遠(yuǎn)程服務(wù)器的文件名)
* @param param ftp配置
* @return
*/
public static boolean upLoadFileToFTP(String fileLocalPath, String fileName, FTPParam param) {
LOG.info("開(kāi)始發(fā)送文件至FTP,param:{}",param);
//狀態(tài)
boolean fal = false;
FTPClient ftpClient = new FTPClient();
FileInputStream fis = null;
File file = null;
try {
ftpClient.connect(param.getHost(), param.getPort());
//判斷ftp連接
if(!FTPReply.isPositiveCompletion(ftpClient.getReplyCode())) {
LOG.error("ftp connect fail,host:{}",param.getHost());
ftpClient.disconnect();
return fal;
}
LOG.info("ftpClient.connect success.");
//判斷ftp登陸
if(!ftpClient.login(param.getUsername(), param.getPassword())) {
LOG.error("ftp login fail,username:{}",param.getHost());
ftpClient.logout();
return fal;
}
//設(shè)置文件類(lèi)型 :二進(jìn)制
ftpClient.setFileType(FTPClient.BINARY_FILE_TYPE);
file = new File(fileLocalPath);
fis = new FileInputStream(file);
//采用被動(dòng)模式傳輸
ftpClient.enterLocalPassiveMode();
//設(shè)置上傳路徑
boolean change = ftpClient.changeWorkingDirectory(param.getUploadPath());
LOG.info("ftpClient.connect 設(shè)置上傳路徑:{}",param.getUploadPath());
if(change) {
fal = ftpClient.storeFile(fileName, fis);
LOG.info("---FTP上傳完成---");
}else{
LOG.info("ftpClient.connect 設(shè)置上傳路徑失?。簕}",param.getUploadPath());
}
} catch (Exception e) {
LOG.error("上傳FTP出錯(cuò),報(bào)錯(cuò):",e);
} finally {
try {
if(fis != null) {
fis.close();
}
fis = null;
file = null;
if(ftpClient.isConnected()) {
ftpClient.logout();
ftpClient.disconnect();
}
} catch (IOException e) {
LOG.error("關(guān)閉FTP或字節(jié)流出錯(cuò),報(bào)錯(cuò):",e);
}
}
return fal;
}

