Android实现上传文件到服务器实例详解_Android

本实例实现每隔5秒上传一次,通过服务器端获取手机上传过来的文件信息并做相应处理;采用Android+Struts2技术。

一、Android端实现文件上传
1)、新建一个Android项目命名为androidUpload,目录结构如下:

2)、新建FormFile类,用来封装文件信息

package com.ljq.utils;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.InputStream;

/**
 * 上传文件
 */
public class FormFile {
  /* 上传文件的数据 */
  private byte[] data;
  private InputStream inStream;
  private File file;
  /* 文件名称 */
  private String filname;
  /* 请求参数名称*/
  private String parameterName;
  /* 内容类型 */
  private String contentType = "application/octet-stream";

  public FormFile(String filname, byte[] data, String parameterName, String contentType) {
    this.data = data;
    this.filname = filname;
    this.parameterName = parameterName;
    if(contentType!=null) this.contentType = contentType;
  }

  public FormFile(String filname, File file, String parameterName, String contentType) {
    this.filname = filname;
    this.parameterName = parameterName;
    this.file = file;
    try {
      this.inStream = new FileInputStream(file);
    } catch (FileNotFoundException e) {
      e.printStackTrace();
    }
    if(contentType!=null) this.contentType = contentType;
  }

  public File getFile() {
    return file;
  }

  public InputStream getInStream() {
    return inStream;
  }

  public byte[] getData() {
    return data;
  }

  public String getFilname() {
    return filname;
  }

  public void setFilname(String filname) {
    this.filname = filname;
  }

  public String getParameterName() {
    return parameterName;
  }

  public void setParameterName(String parameterName) {
    this.parameterName = parameterName;
  }

  public String getContentType() {
    return contentType;
  }

  public void setContentType(String contentType) {
    this.contentType = contentType;
  }

}

3)、新建SocketHttpRequester类,封装上传文件到服务器代码

package com.ljq.utils;

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.Socket;
import java.net.URL;
import java.util.Map;

/**
 * 上传文件到服务器
 *
 * @author Administrator
 *
 */
public class SocketHttpRequester {
  /**
   * 直接通过HTTP协议提交数据到服务器,实现如下面表单提交功能:
   *  <FORM METHOD=POST ACTION="http://192.168.1.101:8083/upload/servlet/UploadServlet" enctype="multipart/form-data">
      <INPUT TYPE="text" NAME="name">
      <INPUT TYPE="text" NAME="id">
      <input type="file" name="imagefile"/>
      <input type="file" name="zip"/>
     </FORM>
   * @param path 上传路径(注:避免使用localhost或127.0.0.1这样的路径测试,因为它会指向手机模拟器,你可以使用http://www.iteye.cn或http://192.168.1.101:8083这样的路径测试)
   * @param params 请求参数 key为参数名,value为参数值
   * @param file 上传文件
   */
  public static boolean post(String path, Map<String, String> params, FormFile[] files) throws Exception{
    final String BOUNDARY = "---------------------------7da2137580612"; //数据分隔线
    final String endline = "--" + BOUNDARY + "--\r\n";//数据结束标志

    int fileDataLength = 0;
    for(FormFile uploadFile : files){//得到文件类型数据的总长度
      StringBuilder fileExplain = new StringBuilder();
       fileExplain.append("--");
       fileExplain.append(BOUNDARY);
       fileExplain.append("\r\n");
       fileExplain.append("Content-Disposition: form-data;name=\""+ uploadFile.getParameterName()+"\";filename=\""+ uploadFile.getFilname() + "\"\r\n");
       fileExplain.append("Content-Type: "+ uploadFile.getContentType()+"\r\n\r\n");
       fileExplain.append("\r\n");
       fileDataLength += fileExplain.length();
      if(uploadFile.getInStream()!=null){
        fileDataLength += uploadFile.getFile().length();
       }else{
         fileDataLength += uploadFile.getData().length;
       }
    }
    StringBuilder textEntity = new StringBuilder();
    for (Map.Entry<String, String> entry : params.entrySet()) {//构造文本类型参数的实体数据
      textEntity.append("--");
      textEntity.append(BOUNDARY);
      textEntity.append("\r\n");
      textEntity.append("Content-Disposition: form-data; name=\""+ entry.getKey() + "\"\r\n\r\n");
      textEntity.append(entry.getValue());
      textEntity.append("\r\n");
    }
    //计算传输给服务器的实体数据总长度
    int dataLength = textEntity.toString().getBytes().length + fileDataLength + endline.getBytes().length;

    URL url = new URL(path);
    int port = url.getPort()==-1 ? 80 : url.getPort();
    Socket socket = new Socket(InetAddress.getByName(url.getHost()), port);
    OutputStream outStream = socket.getOutputStream();
    //下面完成HTTP请求头的发送
    String requestmethod = "POST "+ url.getPath()+" HTTP/1.1\r\n";
    outStream.write(requestmethod.getBytes());
    String accept = "Accept: image/gif, image/jpeg, image/pjpeg, image/pjpeg, application/x-shockwave-flash, application/xaml+xml, application/vnd.ms-xpsdocument, application/x-ms-xbap, application/x-ms-application, application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, */*\r\n";
    outStream.write(accept.getBytes());
    String language = "Accept-Language: zh-CN\r\n";
    outStream.write(language.getBytes());
    String contenttype = "Content-Type: multipart/form-data; boundary="+ BOUNDARY+ "\r\n";
    outStream.write(contenttype.getBytes());
    String contentlength = "Content-Length: "+ dataLength + "\r\n";
    outStream.write(contentlength.getBytes());
    String alive = "Connection: Keep-Alive\r\n";
    outStream.write(alive.getBytes());
    String host = "Host: "+ url.getHost() +":"+ port +"\r\n";
    outStream.write(host.getBytes());
    //写完HTTP请求头后根据HTTP协议再写一个回车换行
    outStream.write("\r\n".getBytes());
    //把所有文本类型的实体数据发送出来
    outStream.write(textEntity.toString().getBytes());
    //把所有文件类型的实体数据发送出来
    for(FormFile uploadFile : files){
      StringBuilder fileEntity = new StringBuilder();
       fileEntity.append("--");
       fileEntity.append(BOUNDARY);
       fileEntity.append("\r\n");
       fileEntity.append("Content-Disposition: form-data;name=\""+ uploadFile.getParameterName()+"\";filename=\""+ uploadFile.getFilname() + "\"\r\n");
       fileEntity.append("Content-Type: "+ uploadFile.getContentType()+"\r\n\r\n");
       outStream.write(fileEntity.toString().getBytes());
       if(uploadFile.getInStream()!=null){
         byte[] buffer = new byte[1024];
         int len = 0;
         while((len = uploadFile.getInStream().read(buffer, 0, 1024))!=-1){
           outStream.write(buffer, 0, len);
         }
         uploadFile.getInStream().close();
       }else{
         outStream.write(uploadFile.getData(), 0, uploadFile.getData().length);
       }
       outStream.write("\r\n".getBytes());
    }
    //下面发送数据结束标志,表示数据已经结束
    outStream.write(endline.getBytes());

    BufferedReader reader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
    if(reader.readLine().indexOf("200")==-1){//读取web服务器返回的数据,判断请求码是否为200,如果不是200,代表请求失败
      return false;
    }
    outStream.flush();
    outStream.close();
    reader.close();
    socket.close();
    return true;
  }

  /**
   * 提交数据到服务器
   * @param path 上传路径(注:避免使用localhost或127.0.0.1这样的路径测试,因为它会指向手机模拟器,你可以使用http://www.itcast.cn或http://192.168.1.10:8080这样的路径测试)
   * @param params 请求参数 key为参数名,value为参数值
   * @param file 上传文件
   */
  public static boolean post(String path, Map<String, String> params, FormFile file) throws Exception{
    return post(path, params, new FormFile[]{file});
  }
}

4)、新建MainActivity类,实现每隔5秒上传一次

package com.ljq.activity;

import java.io.File;
import java.util.HashMap;
import java.util.Map;

import android.app.Activity;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.util.Log;

import com.ljq.utils.FormFile;
import com.ljq.utils.SocketHttpRequester;

public class MainActivity extends Activity {
  private File file;
  private Handler handler;
  private static final String TAG="MainActivity";

  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    Log.i(TAG, "onCreate");

    file = new File(Environment.getExternalStorageDirectory(), "123.rmvb");
    Log.i(TAG, "照片文件是否存在:"+file);
    handler=new Handler();
    handler.post(runnable);
  }

  Runnable runnable=new Runnable() {

    public void run() {
      Log.i(TAG, "runnable run");
      uploadFile(file);
      handler.postDelayed(runnable, 5000);
    }

  };

  /**
   * 上传图片到服务器
   *
   * @param imageFile 包含路径
   */
  public void uploadFile(File imageFile) {
    Log.i(TAG, "upload start");
    try {
      String requestUrl = "http://192.168.1.101:8083/upload/upload/execute.do";
      //请求普通信息
      Map<String, String> params = new HashMap<String, String>();
      params.put("username", "张三");
      params.put("pwd", "zhangsan");
      params.put("age", "21");
      params.put("fileName", imageFile.getName());
      //上传文件
      FormFile formfile = new FormFile(imageFile.getName(), imageFile, "image", "application/octet-stream");

      SocketHttpRequester.post(requestUrl, params, formfile);
      Log.i(TAG, "upload success");
    } catch (Exception e) {
      Log.i(TAG, "upload error");
      e.printStackTrace();
    }
    Log.i(TAG, "upload end");
  }
}

5)、修改清单文件

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
   package="com.ljq.activity"
   android:versionCode="1"
   android:versionName="1.0">
  <application android:icon="@drawable/icon" android:label="@string/app_name">
    <activity android:name=".MainActivity"
         android:label="@string/app_name">
      <intent-filter>
        <action android:name="android.intent.action.MAIN" />
        <category android:name="android.intent.category.LAUNCHER" />
      </intent-filter>
    </activity>

  </application>
  <uses-sdk android:minSdkVersion="4" />
  <uses-permission android:name="android.permission.INTERNET" /> 

</manifest> 

启动模拟器,运行如下:

二、服务器端用来获取Android端上传过来的文件信息

1)、新建一个web项目命名为upload,目录结构如下

注意:记得加入struts2 jar包,需加入的jar如下

2)、新建action类,命名为UploadAction,内容如下

package com.ljq.action;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;

import javax.servlet.http.HttpServletRequest;

import org.apache.struts2.ServletActionContext;

import com.opensymphony.xwork2.ActionSupport;

/**
 * 获取Android端上传过来的信息
 *
 * @author Administrator
 *
 */
@SuppressWarnings("serial")
public class UploadAction extends ActionSupport {
  // 上传文件域
  private File image;
  // 上传文件类型
  private String imageContentType;
  // 封装上传文件名
  private String imageFileName;
  // 接受依赖注入的属性
  private String savePath;

  @Override
  public String execute() {
    HttpServletRequest request=ServletActionContext.getRequest();
    FileOutputStream fos = null;
    FileInputStream fis = null;
    try {
      System.out.println("获取Android端传过来的普通信息:");
      System.out.println("用户名:"+request.getParameter("username"));
      System.out.println("密码:"+request.getParameter("pwd"));
      System.out.println("年龄:"+request.getParameter("age"));
      System.out.println("文件名:"+request.getParameter("fileName"));
      System.out.println("获取Android端传过来的文件信息:");
      System.out.println("文件存放目录: "+getSavePath());
      System.out.println("文件名称: "+imageFileName);
      System.out.println("文件大小: "+image.length());
      System.out.println("文件类型: "+imageContentType);

      fos = new FileOutputStream(getSavePath() + "/" + getImageFileName());
      fis = new FileInputStream(getImage());
      byte[] buffer = new byte[1024];
      int len = 0;
      while ((len = fis.read(buffer)) != -1) {
        fos.write(buffer, 0, len);
      }
      System.out.println("文件上传成功");
    } catch (Exception e) {
      System.out.println("文件上传失败");
      e.printStackTrace();
    } finally {
      close(fos, fis);
    }
    return SUCCESS;
  }

  /**
   * 文件存放目录
   *
   * @return
   */
  public String getSavePath() throws Exception{
    return ServletActionContext.getServletContext().getRealPath(savePath);
  }

  public void setSavePath(String savePath) {
    this.savePath = savePath;
  }

  public File getImage() {
    return image;
  }

  public void setImage(File image) {
    this.image = image;
  }

  public String getImageContentType() {
    return imageContentType;
  }

  public void setImageContentType(String imageContentType) {
    this.imageContentType = imageContentType;
  }

  public String getImageFileName() {
    return imageFileName;
  }

  public void setImageFileName(String imageFileName) {
    this.imageFileName = imageFileName;
  }

  private void close(FileOutputStream fos, FileInputStream fis) {
    if (fis != null) {
      try {
        fis.close();
        fis=null;
      } catch (IOException e) {
        System.out.println("FileInputStream关闭失败");
        e.printStackTrace();
      }
    }
    if (fos != null) {
      try {
        fos.close();
        fis=null;
      } catch (IOException e) {
        System.out.println("FileOutputStream关闭失败");
        e.printStackTrace();
      }
    }
  }

}

3)、配置struts.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
  "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
  "http://struts.apache.org/dtds/struts-2.0.dtd">

<struts>
  <!-- 该属性指定需要Struts2处理的请求后缀,该属性的默认值是action,即所有匹配*.action的请求都由Struts2处理。
    如果用户需要指定多个请求后缀,则多个后缀之间以英文逗号(,)隔开。 -->
  <constant name="struts.action.extension" value="do"/>
  <!-- 设置浏览器是否缓存静态内容,默认值为true(生产环境下使用),开发阶段最好关闭 -->
  <constant name="struts.serve.static.browserCache" value="false"/>
  <!-- 当struts的配置文件修改后,系统是否自动重新加载该文件,默认值为false(生产环境下使用),开发阶段最好打开 -->
  <constant name="struts.configuration.xml.reload" value="true"/>
  <!-- 开发模式下使用,这样可以打印出更详细的错误信息 -->
  <constant name="struts.devMode" value="true"/>
  <!-- 默认的视图主题 -->
  <constant name="struts.ui.theme" value="simple"/>
  <!--<constant name="struts.objectFactory" value="spring" />-->
  <!--解决乱码  -->
  <constant name="struts.i18n.encoding" value="UTF-8"/>
  <!-- 指定允许上传的文件最大字节数。默认值是2097152(2M) -->
  <constant name="struts.multipart.maxSize" value="22097152"/>
  <!-- 设置上传文件的临时文件夹,默认使用javax.servlet.context.tempdir -->
  <constant name="struts.multipart.saveDir " value="d:/tmp"/>

  <package name="upload" namespace="/upload" extends="struts-default">
    <action name="execute" class="com.ljq.action.UploadAction">
      <!-- 动态设置savePath的属性值 -->
      <param name="savePath">/image</param>
      <result name="success">/WEB-INF/page/message.jsp</result>
    </action>
  </package>
</struts>

4)、配置web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
  http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">

  <filter>
    <filter-name>struts2</filter-name>
    <filter-class>
      org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter
    </filter-class>
  </filter>
  <filter-mapping>
    <filter-name>struts2</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>

  <filter>
    <filter-name>strutsCleanup</filter-name>
    <filter-class>
      org.apache.struts2.dispatcher.ActionContextCleanUp
    </filter-class>
  </filter>
  <filter-mapping>
    <filter-name>strutsCleanup</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>

  <welcome-file-list>
    <welcome-file>index.jsp</welcome-file>
  </welcome-file-list>
</web-app>

运行结构如下:

获取Android端传过来的普通信息:
用户名:张三
密码:zhangsan
年龄:21
文件名:123.rmvb
获取Android端传过来的文件信息:
文件存放目录: D:\apache-tomcat-6.0.18\webapps\upload\image
文件名称: 123.rmvb
文件大小: 3962649
文件类型: application/octet-stream
文件上传成功

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索android上传服务器
php文件上传实例详解、服务器bios界面详解、ntp服务器配置详解、vpn服务器配置详解、wincc opc服务器实例,以便于您获取更多的相关知识。

时间: 2024-10-30 13:34:58

Android实现上传文件到服务器实例详解_Android的相关文章

Android编程之在SD卡上进行文件读写操作实例详解_Android

本文实例讲述了Android编程之在SD卡上进行文件读写操作的方法.分享给大家供大家参考,具体如下: 很多知识只有真正理解掌握之后才能运用自如,举一反三.对Java中的文件操作和android系统SD卡里面的文件操作,你觉得有区别吗,显然没有本质区别,如果勉强说有,那也是不足为道滴,但我们在实际运用中却要注意如下几点,不然问题会缠上你. 1.首先想要对android系统SD卡里文件操作需要添加使用权限: android系统是不会让外来程序随意动自己内存的,如果没有许可证,不好意思,不准你动我地盘

Android开发高仿课程表的布局实例详解_Android

先说下这个demo,这是一个模仿课程表的布局文件,虽然我是个菜鸟,但我还是想留给学习的人一些例子,先看下效果   然后再来看一下我们学校的app 布局分析 先上一张划分好了的布局图 首先整个页面放在一个LinearLayout布局下面,分为上面和下面两个部分,下面一个是显示课程表的详细信息 1:这个没什么好讲的,就是直接一个LinearLayout布局,然后将控件一个TextView用来显示年份,一个View用来当作竖线,一个Spinner用来显示选择周数 2:这个是显示星期几的部件,是我自定义

Android编程之文件的读写实例详解_Android

本文实例分析了Android编程之文件的读写方法.分享给大家供大家参考,具体如下: Android的文件读写与JavaSE的文件读写相同,都是使用IO流.而且Android使用的正是JavaSE的IO流,下面我们通过一个练习来学习Android的文件读写. 1.创建一个Android工程 Project name:File     BuildTarget:Android2.2     Application name:文件读写     Package name:test.file     Cre

JSP中图片的上传与显示方法实例详解_JSP编程

本文实例讲述了JSP中图片的上传与显示方法.分享给大家供大家参考.具体如下: 1.引言 数据库应用程序,特别是基于WEB的数据库应用程序,常会涉及到图片信息的存储和显示.通常我们使用的方法是将所要显示的图片存在特定的目录下,在数据库中保存相应的图片的名称,在JSP中建立相应的数据源,利用数据库访问技术处理图片信息.但是,如果我们想动态的显示图片,上述方法就不能满足需要了.我们必须把图片存入数据库,然后通过编程动态地显示我们需要的图片.实际操作中,可以利用JSP的编程模式来实现图片的数据库存储和显

Android自定义控件eBook实现翻书效果实例详解_Android

本文实例讲述了Android自定义控件eBook实现翻书效果的方法.分享给大家供大家参考,具体如下: 效果图: Book.java文件: package com.book; import android.app.Activity; import android.os.Bundle; import android.util.Log; import android.view.View; import android.widget.ImageView; public class Book extend

Android对sdcard扩展卡文件操作实例详解_Android

Android对sdcard扩展卡文件的操作其实就是普通的文件操作,但是仍然有些地方需要注意.比如: 1.加入sdcard操作权限: 2.确认sdcard的存在: 3.不能直接在非sdcard的根目录创建文件,而是需要先创建目录,再创建文件: 实例如下: (1)在AndroidManifest.xml添加sdcard操作权限 <!-- sdcard权限 --> <uses-permission android:name="android.permission.WRITE_EXT

Android编程之菜单的实现方法实例详解_Android

本文实例讲述了Android编程之菜单的实现方法.分享给大家供大家参考,具体如下: Options Menu 当用户按下menu button按钮时显示的菜单 Context Menu 当用户长久按住屏幕,被注册显示上下文菜单的视图时显示的菜单 Submenu    当用户按下一个菜单的某个选项时弹出的子菜单 以上三种菜单也就是我们经常所说的:选项菜单,上下文菜单和子菜单. 一.上下文菜单 1. 实现onCreateContextMenu即可创建该菜单 @Override public void

Android实现有道辞典查询功能实例详解_Android

本文实例讲述了Android实现有道辞典查询功能的方法.分享给大家供大家参考,具体如下: 这是我做的一个简单的有道Android的DEMO,只是简单的雏形.界面设计也有点丑陋呵呵~ 看看下面的效果图: 第一步:思路解析 从界面看一共用了三个控件EditText,Button,WebView.其实是四个,是当我们查询内容为空的时候用来提示的Toast控件. 我们在EditText输入查询内容,这里包括中文,英文.然后通过参数的形式,从http://dict.youdao.com/m取出数据把结果

Android 中SP与DP的区别实例详解_Android

从一开始写Android程序,就被告知这些常识 1.长度宽度的数值要使用dp作为单位放入dimens.xml文件中 2.字体大小的数值要使用sp作为单位,也放入dimens.xml文件中 然后,就没有然后了,仿佛潜台词就是说,你记住去用就行了. 偶然有一天,当我们阴差阳错地将字体写成了dp,也是可以工作,而且效果和sp一样. 这时候,就开始怀疑了,到底有啥区别呢,dp和sp有什么不同呢? 我们做个简单的Sample验证一下,如下,一个布局代码 <TextView android:layout_w