1.简单介绍一下
在一些特殊场景,两台服务器之间需要进行文件传输的时候,或许我们会想到FTP,这也是我们常见的实现方式,但是如果我们不能操作远程服务器,无法判断远程服务器是否安装了FTP呢,众所周知,FTP使用的前提时确定服务器上配置了FTP服务,并且正在运行FTP服务器软件,这是最大的前提,如果我们不知道,那么就不能贸然的使用该方式。
我的需求是:我只知道对方服务器的ip、端口、用户名、密码,文件地址。其他的我一概不知。
那么还有什么方式呢,SFTP?SCP?
SFTP和FTP不就少了一个S,应该也需要安装FTP吧?答案是:NO
SFTP:是一种基于SSH协议的网络协议,用于在网络上安全的传输协议,SFTP提供了一个安全的文件传输机制,允许用户在传输过程中加密数据,从而保护数据免受窃听和篡改。
注意:OpenSSH支持SFTP,也就是说只要服务器中有OpenSSH,SFTP就可以使用。
FTP:是一种用于在网络上进行文件传输的协议。它是基于文本的协议,允许用户上传、下载、删除和重命名文件。
注意:需要安装配置FTP服务到服务器。
SCP:是一种用于在服务器之间安全复制文件的网络协议。基于SSH协议,提供了一个加密的方法来传输文件,确保数据在传输过程中的安全性。SCP是一个命令行工具,通常用于远程服务器管理、文件备份和系统维护。
注意:我认为只要服务器可以执行linux命令,应该都可以实现。
2.实现方式
我的实现方式是基于Spring Boot创建完成的哈,大家如果出现了什么问题,得放到评论区,得重新看看具体是什么原因。
2.1 FTP方式
这个大家就参考网上详细的文章哈,主要我的需求不能使用这个方式😄,所以没有代码😜。
2.2 SFTP方式
第一个肯定就是导入依赖
<dependency> <groupId>com.jcraft</groupId> <artifactId>jsch</artifactId> <version>0.1.53</version> </dependency>
然后,咱们就得写一个工具类了
import com.jcraft.jsch.Channel; import com.jcraft.jsch.ChannelSftp; import com.jcraft.jsch.JSch; import com.jcraft.jsch.Session; import java.io.File; public class ScpFileTransferUtil { // 远程服务器的IP地址 private static final String IP = "192.xxx.xxx.xx"; // 远程服务器的SSH端口 private static final int PORT = xxxx; // 远程服务器的用户名 private static final String USER_NAME = "root"; // 远程服务器的密码 private static final String PASSWORD = "xxxxxx"; /** * 从远程服务器下载文件到本地 * * @param remotePath 远程文件路径 * @param localPath 本地目标目录 */ public static void scpFromRemote(String remotePath, String localPath) { try { JSch jsch = new JSch(); Session session = jsch.getSession(USER_NAME, IP, PORT); session.setPassword(PASSWORD); // 设置配置项 java.util.Properties config = new java.util.Properties(); config.put("StrictHostKeyChecking", "no"); session.setConfig(config); // 连接会话 System.out.println("Connecting to SSH server..."); session.connect(); // 打开 SFTP 通道 Channel channel = session.openChannel("sftp"); channel.connect(); ChannelSftp sftpChannel = (ChannelSftp) channel; // 解析远程文件路径中的文件名 String fileName = extractFileNameFromPath(remotePath); // 创建本地文件路径 File localFile = new File(localPath, fileName); // 创建本地目标目录 File directory = localFile.getParentFile(); if (!directory.exists()) { boolean mkdirResult = directory.mkdirs(); if (mkdirResult) { System.out.println("创建文件夹成功"); } else { System.out.println("创建文件夹失败"); throw new RuntimeException("创建文件夹失败!"); } } // 下载文件 System.out.println("Downloading file from remote path: " + remotePath); sftpChannel.get(remotePath, localFile.getAbsolutePath()); System.out.println("文件下载成功"); // 关闭通道和会话 channel.disconnect(); session.disconnect(); } catch (Exception e) { e.printStackTrace(); System.err.println("文件下载失败:" + e.getMessage()); } } /** * 从路径中提取文件名 * * @param path 完整路径 * @return 文件名 */ private static String extractFileNameFromPath(String path) { int lastSeparatorIndex = path.lastIndexOf(File.separator); return path.substring(lastSeparatorIndex + 1); } public static void main(String[] args) { System.out.println("Main method started."); //获取开始时间 long startTime = System.currentTimeMillis(); String remotePath = "/data/ceshiDown/eac207eeeb2095cc560f94b6e7e33f49_4.mp4"; // 远程文件路径 String localPath = "E:\\mnt\\data_process_net\\token"; // 本地目标目录 scpFromRemote(remotePath, localPath); System.out.println("Main method finished."); //获取结束时间 long endTime = System.currentTimeMillis(); System.out.println("程序运行时间:" + (endTime - startTime)/1000 + "s"); } }
这段代码应该不需要我讲解一下了吧,我相信兄弟们都可以看懂的,必要的注释我都加上啦。
搞定!
2.3 SCP方式
SCP的实现方式,有一些稍微的麻烦,需要写一个expect脚本,为什么使用脚本呢,说明一下,在我们使用scp命令在拷贝远程服务器的时候,我们中间需要输入密码。
但是我们在执行代码的时候,可以自动识别需要输入密码,然后自动化输入密码吗?答案是不能的。
所以这里介入expect脚本的作用就是识别哪里需要输入密码,然后自动化输入密码。
这里应该不需要安装任何的依赖的。
直接就是工具类:
import lombok.extern.slf4j.Slf4j; import java.io.BufferedReader; import java.io.File; import java.io.IOException; import java.io.InputStreamReader; @Slf4j public class ScpUtil { public static void scpFromRemoteUsingExpect(String expectScriptPath, String user, String host, int port, String remotePath, String localPath, String password) { Process process = null; try { // 确保目标目录存在 File localDir = new File(localPath); if (!localDir.exists()) { boolean created = localDir.mkdirs(); // 创建多级目录 if (!created) { throw new IOException("无法创建目标目录: " + localPath); } } // 构建 expect 脚本的命令行参数 String[] params = {user, host, Integer.toString(port), remotePath, localPath, password}; String command = expectScriptPath + " " + String.join(" ", params); // 执行 expect 脚本 process = Runtime.getRuntime().exec(command); // 读取命令的输出 BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream())); String line; while ((line = reader.readLine()) != null) { System.out.println(line); } // 等待命令执行完成 int exitCode = process.waitFor(); if (exitCode == 0) { log.info("文件下载成功"); } else { log.info("文件下载失败,退出代码:{}", exitCode); } } catch (Exception e) { e.printStackTrace(); log.info("文件下载失败:{}", e.getMessage()); }finally { // 确保资源被释放 if (process != null) { process.destroy(); } } } public static void scpDirectoryFromRemoteUsingExpect(String expectScriptPath, String user, String host, int port, String remotePath, String localPath, String password) { Process process = null; try { // 确保目标目录存在 File localDir = new File(localPath); if (!localDir.exists()) { boolean created = localDir.mkdirs(); // 创建多级目录 if (!created) { throw new IOException("无法创建目标目录: " + localPath); } } // 构建 expect 脚本的命令行参数,确保 remotePath 是一个目录 String[] params = {user, host, Integer.toString(port), remotePath + "/", localPath, password}; String command = expectScriptPath + " " + String.join(" ", params); // 执行 expect 脚本 process = Runtime.getRuntime().exec(command); // 读取命令的输出 BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream())); String line; while ((line = reader.readLine()) != null) { System.out.println(line); } // 等待命令执行完成 int exitCode = process.waitFor(); if (exitCode == 0) { System.out.println("文件夹下载成功"); } else { System.err.println("文件夹下载失败,退出代码:" + exitCode); } } catch (IOException e) { e.printStackTrace(); log.info("文件夹下载失败:{}", e.getMessage()); } catch (InterruptedException e) { Thread.currentThread().interrupt(); // 重新设置中断状态 e.printStackTrace(); log.info("文件夹下载被中断:{}", e.getMessage()); } finally { // 确保资源被释放 if (process != null) { try { process.getInputStream().close(); process.getOutputStream().close(); process.getErrorStream().close(); } catch (IOException e) { e.printStackTrace(); } process.destroy(); } } } public static void main(String[] args) { log.info("Main method started."); // 获取开始时间 long startTime = System.currentTimeMillis(); String expectScriptPath = "/script/scp_download_directory.exp"; // expect 脚本的路径 String remotePath = "/data/ceshiDown/"; String localPath = "/data/ceshi2"; ScpUtil.scpDirectoryFromRemoteUsingExpect(expectScriptPath, USER_NAME, IP, PORT, remotePath, localPath, PASSWORD); log.info("Main method finished."); // 获取结束时间 long endTime = System.currentTimeMillis(); System.out.println("程序运行时间:" + (endTime - startTime) / 1000 + "s"); } }
上面这个工具类呢,scpFromRemoteUsingExpect()方法是复制文件的,scpDirectoryFromRemoteUsingExpect()是复制文件夹的。注意一下哦。
最终要的来啦,脚本怎么写呢。
复制文件的脚本下载地址:scp复制远程服务器文件至本地服务器expect脚本
复制文件夹的脚本下载地址:scp复制远程服务器文件夹至本地服务器expect脚本
若是文件下载不了,千万别花钱下载,联系我,我发你。
这里我就不测试了,我是在服务器测试的,是行得通的。
完结!撒花!以上内容,若有任何问题,欢迎在评论区留言,在此,若有更好的解决方案的兄弟,也请留言,让我学习一下哈