8

我正在尝试发布一个大视频(近 1 GB)。

我正在使用 FTP 将视频发送到服务器,但上传会在一段时间后停止。在服务器上,视频崩溃了,但我可以上传较小尺寸的视频。

我还使用 HTTP 将视频发送到服务器,作为 Base64 编码字符串发送,但编码时出现内存不足异常。

我尝试将视频作为文件上传,但没有成功。将大型视频上传到服务器的最佳方式是什么?

4

3 回答 3

10

使用 HTTP POST,并将内容发布为基于表单的文件上传(mime 类型:multipart/form-data)。该系统是用于发送表格和/或上传文件的网络标准。

使用 HTTP 分块发布模式,这样就不需要事先知道大小,您可以将任何文件分成小部分进行流式传输。您仍然需要在服务器上编写一些代码(例如在 PHP 中)以接受文件并执行所需的操作。

使用 HttpURLConnection 启动连接。然后使用我附加的类发送数据。它将创建适当的标头等,您将使用它作为 OutputStream 将原始数据写入其中,然后调用 close,就完成了。您可以覆盖其 onHandleResult 以处理生成的错误代码。

public class FormDataWriter extends FilterOutputStream{
   private final HttpURLConnection con;

   /**
    * @param formName name of form in which data are sent
    * @param fileName
    * @param fileSize   size of file, or -1 to use chunked encoding
    */
   FormDataWriter(HttpURLConnection con, String formName, String fileName, long fileSize) throws IOException{
      super(null);
      this.con = con;
      con.setDoOutput(true);

      String boundary = generateBoundary();
      con.setRequestProperty(HTTP.CONTENT_TYPE, "multipart/form-data; charset=UTF-8; boundary="+boundary);
      {
         StringBuilder sb = new StringBuilder();
         writePartHeader(boundary, formName, fileName==null ? null : "filename=\""+fileName+"\"",
               "application/octet-stream", sb);
         headerBytes = sb.toString().getBytes("UTF-8");

         sb = new StringBuilder();
         sb.append("\r\n");
         sb.append("--"+boundary+"--\r\n");
         footerBytes = sb.toString().getBytes();
      }
      if(fileSize!=-1) {
         fileSize += headerBytes.length + footerBytes.length;
         con.setFixedLengthStreamingMode((int)fileSize);
      }else
         con.setChunkedStreamingMode(0x4000);
      out = con.getOutputStream();
   }

   private byte[] headerBytes, footerBytes;

   private String generateBoundary() {
      StringBuilder sb = new StringBuilder();
      Random rand = new Random();
      int count = rand.nextInt(11) + 30;
      int N = 10+26+26;
      for(int i=0; i<count; i++) {
         int r = rand.nextInt(N);
         sb.append((char)(r<10 ? '0'+r : r<36 ? 'a'+r-10 : 'A'+r-36));
      }
      return sb.toString();
   }

   private void writePartHeader(String boundary, String name, String extraContentDispositions, String contentType, StringBuilder sb) {
      sb.append("--"+boundary+"\r\n");
      sb.append("Content-Disposition: form-data; charset=UTF-8; name=\""+name+"\"");
      if(extraContentDispositions!=null)
         sb.append("; ").append(extraContentDispositions);
      sb.append("\r\n");
      if(contentType!=null)
         sb.append("Content-Type: "+contentType+"\r\n");
      sb.append("\r\n");
   }

   @Override
   public void write(byte[] buffer, int offset, int length) throws IOException{
      if(headerBytes!=null) {
         out.write(headerBytes);
         headerBytes = null;
      }
      out.write(buffer, offset, length);
   }

   @Override
   public void close() throws IOException{
      flush();
      if(footerBytes!=null) {
         out.write(footerBytes);
         footerBytes = null;
      }
      super.close();
      int code = con.getResponseCode();
      onHandleResult(code);
   }
   protected void onHandleResult(int code) throws IOException{
      if(code!=200 && code!=201)
         throw new IOException("Upload error code: "+code);
   }
}
于 2013-07-03T13:50:54.550 回答
3

我猜它失败是因为大尺寸超时。

自从

小尺寸视频上传成功

, 我的建议是

  1. 将一个大文件拆分为几个小文件。
  2. 根据网络情况,一张一张或多张一起上传。
  3. 在服务器上加入所有部分(在所有成功上传之后)。
  4. 由于体积小,重新上传失败的部分将很容易。

只是一个理论。

这个网站可能会有所帮助。


添加 08.01.2013

好久不见了,不知道还需要这个。无论如何,我写了一些简单的代码来实现上面的理论,主要是因为兴趣。

  1. 将一个大文件拆分为几个小文件。将大文件读成几个小部分。

    ByteBuffer bb = ByteBuffer.allocate(partSize);
    int bytesRead = fc.read(bb);
    if (bytesRead == -1) {
        break;
    }
    byte[] bytes = bb.array();
    parts.put(new Part(createFileName(fileName, i), bytes));
    
  2. 根据网络情况,一张一张或多张一起上传。

    Part part = parts.take();
    if (part == Part.NULL) {
        parts.add(Part.NULL);// notify others to stop.
        break;
    } else {
        uploader.upload(part);
    }
    
  3. 在服务器上加入所有部分(在所有成功上传之后)。因为是通过HTTP,所以可以是任何语言,如Java、PHP、Python等。这里是一个java的例子。

    ...
    try (FileOutputStream dest = new FileOutputStream(destFile, true)) {
    
        FileChannel dc = dest.getChannel();// the final big file.
        for (long i = 0; i < count; i++) {
            File partFile = new File(destFileName + "." + i);// every small parts.
            if (!partFile.exists()) {
                break;
            }
            try (FileInputStream part = new FileInputStream(partFile)) {
                FileChannel pc = part.getChannel();
                pc.transferTo(0, pc.size(), dc);// combine.
            }
            partFile.delete();
        }
        statusCode = OK;// set ok at last.
    } catch (Exception e) {
        log.error("combine failed.", e);
    }
    

我把所有代码放在GitHub 上。并制作了一个Android示例

如果您仍然需要,请查看。

于 2013-07-09T08:57:59.963 回答
0

私有 HttpsURLConnection conn = null;

conn.setDoInput(true);

conn.setDoOutput(true);

conn.setUseCaches(false);

conn.setChunkedStreamingMode(1024);

于 2014-03-24T13:35:26.690 回答