首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > 编程 >

Java 小事例:通过 Socket 发送和接收文件

2012-12-21 
Java 小例子:通过 Socket 发送和接收文件这是一个简单的包含发送端和接收端的例子。发送端向接收端发送文件

Java 小例子:通过 Socket 发送和接收文件

这是一个简单的包含发送端和接收端的例子。发送端向接收端发送文件名和文件内容,接收端将收到的文件保存在磁盘上。接收端可以同时接收多个发送端传来的文件,但没有处理文件同名的情况。

?

这个例子中设计了一个简单的协议。发送的内容是这样的:

文件名长度(4字节)—文件名—文件内容长度(4字节)—文件内容。

?

接收端也按照这个结构进行解析。建议先看 Client 类,再看 Server 类。

import java.io.*;  import java.net.ServerSocket;  import java.net.Socket;     /**  * 简单的文件发送与接收示例  */  public class FileTrasmission {         //程序入口      public static void main(String[] args) throws Exception {          int port = 7788;          new Server(port, "c:\\save\").start();          new Client().sendFile("127.0.0.1", port, "c:\\迷失在康熙末年.txt");      }  }     /**  * 接收端。可同时接收多个发送端发来的文件。但如果发来的文件是同名的话那就乱了。  */  class Server {         private int listenPort;         private String savePath;         /**      * 构造方法      *      * @param listenPort 侦听端口      * @param savePath   接收的文件要保存的路径      *      * @throws IOException 如果创建保存路径失败      */      Server(int listenPort, String savePath) throws IOException {          this.listenPort = listenPort;          this.savePath = savePath;             File file = new File(savePath);          if (!file.exists() && !file.mkdirs()) {              throw new IOException("无法创建文件夹 " + savePath);          }      }         // 开始侦听      public void start() {          new ListenThread().start();      }         // 网上抄来的,将字节转成 int。b 长度不得小于 4,且只会取前 4 位。      public static int b2i(byte[] b) {          int value = 0;          for (int i = 0; i < 4; i++) {              int shift = (4 - 1 - i) * 8;              value += (b[i] & 0x000000FF) << shift;          }          return value;      }            /**      * 侦听线程      */      private class ListenThread extends Thread {             @Override          public void run() {              try {                  ServerSocket server = new ServerSocket(listenPort);                     // 开始循环                  while (true) {                      Socket socket = server.accept();                      new HandleThread(socket).start();                  }              } catch (IOException e) {                  e.printStackTrace();              }          }      }         /**      * 读取流并保存文件的线程      */      private class HandleThread extends Thread {             private Socket socket;             private HandleThread(Socket socket) {              this.socket = socket;          }             @Override          public void run() {              try {                  InputStream is = socket.getInputStream();                  readAndSave(is);              } catch (IOException e) {                  e.printStackTrace();              } finally {                  try {                      socket.close();                  } catch (IOException e) {                      // nothing to do                  }              }          }             // 从流中读取内容并保存          private void readAndSave(InputStream is) throws IOException {              String filename = getFileName(is);              int file_len = readInteger(is);              System.out.println("接收文件:" + filename + ",长度:" + file_len);                 readAndSave0(is, savePath + filename, file_len);                 System.out.println("文件保存成功(" + file_len + "字节)。");          }             private void readAndSave0(InputStream is, String path, int file_len) throws IOException {              FileOutputStream os = getFileOS(path);              readAndWrite(is, os, file_len);              os.close();          }             // 边读边写,直到读取 size 个字节          private void readAndWrite(InputStream is, FileOutputStream os, int size) throws IOException {              byte[] buffer = new byte[4096];              int count = 0;              while (count < size) {                  int n = is.read(buffer);                  // 这里没有考虑 n = -1 的情况                  os.write(buffer, 0, n);                  count += n;              }          }             // 读取文件名          private String getFileName(InputStream is) throws IOException {              int name_len = readInteger(is);              byte[] result = new byte[name_len];              is.read(result);              return new String(result);          }             // 读取一个数字          private int readInteger(InputStream is) throws IOException {              byte[] bytes = new byte[4];              is.read(bytes);              return b2i(bytes);          }             // 创建文件并返回输出流          private FileOutputStream getFileOS(String path) throws IOException {              File file = new File(path);              if (!file.exists()) {                  file.createNewFile();              }                 return new FileOutputStream(file);          }      }  }     /**  * 发送端  */  class Client {         // 网上抄来的,将 int 转成字节      public static byte[] i2b(int i) {          return new byte[]{                  (byte) ((i >> 24) & 0xFF),                  (byte) ((i >> 16) & 0xFF),                  (byte) ((i >> 8) & 0xFF),                  (byte) (i & 0xFF)          };      }         /**      * 发送文件。文件大小不能大于 {@link Integer#MAX_VALUE}      *      * @param hostname 接收端主机名或 IP 地址      * @param port     接收端端口号      * @param filepath 文件路径      *      * @throws IOException 如果读取文件或发送失败      */      public void sendFile(String hostname, int port, String filepath) throws IOException {          File file = new File(filepath);          FileInputStream is = new FileInputStream(filepath);             Socket socket = new Socket(hostname, port);          OutputStream os = socket.getOutputStream();             try {              int length = (int) file.length();              System.out.println("发送文件:" + file.getName() + ",长度:" + length);                 // 发送文件名和文件内容              writeFileName(file, os);              writeFileContent(is, os, length);          } finally {              os.close();              is.close();          }      }         // 输出文件内容      private void writeFileContent(InputStream is, OutputStream os, int length) throws IOException {          // 输出文件长度          os.write(i2b(length));             // 输出文件内容          byte[] buffer = new byte[4096];          int size;          while ((size = is.read(buffer)) != -1) {              os.write(buffer, 0, size);          }      }         // 输出文件名      private void writeFileName(File file, OutputStream os) throws IOException {          byte[] fn_bytes = file.getName().getBytes();             os.write(i2b(fn_bytes.length));         // 输出文件名长度          os.write(fn_bytes);    // 输出文件名      }  }  

?

热点排行