今天一直在调试安卓向服务器上传的图片的程序,在网上下了个可以单个上传照片的程序,做了一些修改,写点文字记录一下改动的内容和心得加深一下记忆。中午在百度里也有搜到安卓图片批量上传的代码,但是基于servlet不保证在Struts2下可以运行,考虑了一下果断放弃了,觉得还是修改单个图片上传的学习成本较低。一开始做了很多尝试,发现凡是空指针的异常的都可以通过单点调试找到位置。

           首先文件上传要定义一个接口,在文件上传类中需要保存一个接口实例。一开始没有加abstract关键字,但是实现类也需要重写override这三个方法,后来看到别的接口定义的时候加了abstract关键字就加上了。查了之后发现其实不加没有关系,接口中的方法本来就是默认是abstract的。之前对接口一直处于概念阶段,现在终于见识到它的实用性了。图片上传过程不能放在主线程中,多个图片上传需要线程池的支持,这里的接口实现了在子线程中调用在主线程里创建的接口实例。

public interface OnUploadProcessListener { /**  * 上传响应  * @param responseCode  * @param message  */ public abstract void onUploadDone(int responseCode, String message); /**  * 上传中  * @param uploadSize  */ public abstract void onUploadProcess(int uploadSize); /**  * 准备上传  * @param fileSize  */ public abstract void initUpload(int fileSize); }

 

        这个接口用来控制文件图片上传的三个过程,开始上传、上传中和结束上传。使用时在主线程创建一个接口的实例,重写这三个方法,并把这个实例作为参数传递给图片上传类,在图片上传的线程中调用接口函数。

FileUploadRequest request = new FileUploadRequest(url,picPath, parameters,new OnUploadProcessListener() {     @Override     public void onUploadDone(int responseCode,String message) {     if (null != handler) {          Message msg = handler.obtainMessage();          Bundle data = new Bundle();          data.putBoolean(Uri.KEY_FLAG, true);          data.putString(Uri.KEY_MSG, message);          msg.setData(data);          msg.what = Uri.Upload.MSG_BRIDGEUSUALEXAM_PHOTO;                  handler.sendMessage(msg);          }     }        @Override     public void onUploadProcess(int uploadSize) {     // TODO Auto-generated method stub              }     @Override     public void initUpload(int fileSize) {     // TODO Auto-generated method stub              } }); ThreadPool.getSingle().execute(request);

 

        图片上传类实现了Runable接口,并且保存了OnUploadProcessListener接口实例:

public class FileUploadRequest implements Runnable{ private Map<String, String> parameters; private OnUploadProcessListener onUploadProcessListener; private String picPath; /**  * @param url  请求地址  * @param params 请求参数  * @param requestResultCallback 请求回调函数  * */ public FileUploadRequest(String url,String picPath,Map<String, String> parameters,OnUploadProcessListener MyUploadProcessListener){ this.url = url; this.parameters = parameters; this.onUploadProcessListener = MyUploadProcessListener; this.picPath = picPath; } //线程函数 @Override public void run() { try { String fileKey = "img";// 与服务器端的File名字相同 UploadUtil uploadUtil = UploadUtil.getInstance(); uploadUtil.uploadFile(picPath, fileKey, url, parameters,onUploadProcessListener); } catch(IllegalArgumentException e){ e.printStackTrace(); }  super.run(); } }

 

             UploadUtil是图片上传工具函数,和线程池一样采用d单例模式,线程池采用单例模式很好理解,因为这个进程中只有一个线程池来统筹所有子线程的有序运转,那为什么这个也是单例模式呢?其实我也没理解

public class UploadUtil { private static UploadUtil uploadUtil; private static final String BOUNDARY =  UUID.randomUUID().toString(); // 边界标识 随机生成 private static final String PREFIX = "--"; private static final String LINE_END = "\r\n"; private static final String CONTENT_TYPE = "multipart/form-data"; // 内容类型 private UploadUtil() { } /**  * 单例模式获取上传工具类  * @return  */ public static UploadUtil getInstance() { if (null == uploadUtil) { uploadUtil = new UploadUtil(); } return uploadUtil; } private static final String TAG = "UploadUtil"; private int readTimeOut = 10 * 1000; // 读取超时 private int connectTimeout = 10 * 1000; // 超时时间 /***  * 请求使用多长时间  */ private static int requestTime = 0; private static final String CHARSET = "utf-8"; // 设置编码 /***  * 上传成功  */ public static final int UPLOAD_SUCCESS_CODE = 1; /**  * 文件不存在  */ public static final int UPLOAD_FILE_NOT_EXISTS_CODE = 2; /**  * 服务器出错  */ public static final int UPLOAD_SERVER_ERROR_CODE = 3; protected static final int WHAT_TO_UPLOAD = 1; protected static final int WHAT_UPLOAD_DONE = 2; /**  * android上传文件到服务器  *   * @param filePath  *            需要上传的文件的路径  * @param fileKey  *            在网页上<input type=file name=xxx/> xxx就是这里的fileKey  * @param RequestURL  *            请求的URL  */ public void uploadFile(String filePath, String fileKey, String RequestURL, Map<String, String> param,OnUploadProcessListener MyUploadProcessListener) { this.onUploadProcessListener = MyUploadProcessListener; if (filePath == null) { onUploadProcessListener.onUploadDone(UPLOAD_FILE_NOT_EXISTS_CODE,"文件不存在"); return; } try { File file = new File(filePath); uploadFile(file, fileKey, RequestURL, param); } catch (Exception e) { onUploadProcessListener.onUploadDone(UPLOAD_FILE_NOT_EXISTS_CODE,"文件不存在"); e.printStackTrace(); return; } } /**  * android上传文件到服务器  *   * @param file  *            需要上传的文件  * @param fileKey  *            在网页上<input type=file name=xxx/> xxx就是这里的fileKey  * @param RequestURL  *            请求的URL  */ public void uploadFile(final File file, final String fileKey, final String RequestURL, final Map<String, String> param) { if (file == null || (!file.exists())) { onUploadProcessListener.onUploadDone(UPLOAD_FILE_NOT_EXISTS_CODE,"文件不存在"); return; } Log.i(TAG, "请求的URL=" + RequestURL); Log.i(TAG, "请求的fileName=" + file.getName()); Log.i(TAG, "请求的fileKey=" + fileKey); toUploadFile(file, fileKey, RequestURL, param); } private void toUploadFile(File file, String fileKey, String RequestURL, Map<String, String> param) { String result = null; requestTime= 0; long requestTime = System.currentTimeMillis(); long responseTime = 0; try { URL url = new URL(RequestURL); HttpURLConnection conn = (HttpURLConnection) url.openConnection(); conn.setReadTimeout(readTimeOut); conn.setConnectTimeout(connectTimeout); conn.setDoInput(true); // 允许输入流 conn.setDoOutput(true); // 允许输出流 conn.setUseCaches(false); // 不允许使用缓存 conn.setRequestMethod("POST"); // 请求方式 conn.setRequestProperty("Charset", CHARSET); // 设置编码 conn.setRequestProperty("connection", "keep-alive"); conn.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1)"); conn.setRequestProperty("Content-Type", CONTENT_TYPE + ";boundary=" + BOUNDARY); /**  * 当文件不为空,把文件包装并且上传  */ DataOutputStream dos = new DataOutputStream(conn.getOutputStream()); StringBuffer sb = null; String params = ""; /***  * 以下是用于上传参数  */ if (param != null && param.size() > 0) { Iterator<String> it = param.keySet().iterator(); while (it.hasNext()) { sb = null; sb = new StringBuffer(); String key = it.next(); String value = param.get(key); sb.append(PREFIX).append(BOUNDARY).append(LINE_END); sb.append("Content-Disposition: form-data; name=\"").append(key).append("\"").append(LINE_END).append(LINE_END); sb.append(value).append(LINE_END); params = sb.toString(); Log.i(TAG, key+"="+params+"##"); dos.write(params.getBytes()); } } sb = null; params = null; sb = new StringBuffer(); /**  * 这里重点注意: name里面的值为服务器端需要key 只有这个key 才可以得到对应的文件  * filename是文件的名字,包含后缀名的 比如:abc.png  */ sb.append(PREFIX).append(BOUNDARY).append(LINE_END); sb.append("Content-Disposition:form-data; name=\"" + fileKey + "\"; filename=\"" + file.getName() + "\"" + LINE_END); sb.append("Content-Type:image/pjpeg" + LINE_END); // 这里配置的Content-type很重要的 ,用于服务器端辨别文件的类型的 sb.append(LINE_END); params = sb.toString(); sb = null; Log.i(TAG, file.getName()+"=" + params+"##"); dos.write(params.getBytes()); /**上传文件*/ InputStream is = new FileInputStream(file); onUploadProcessListener.initUpload((int)file.length()); byte[] bytes = new byte[1024]; int len = 0; int curLen = 0; while ((len = is.read(bytes)) != -1) { curLen += len; dos.write(bytes, 0, len); onUploadProcessListener.onUploadProcess(curLen); } is.close(); dos.write(LINE_END.getBytes()); byte[] end_data = (PREFIX + BOUNDARY + PREFIX + LINE_END).getBytes(); dos.write(end_data); dos.flush(); /**  * 获取响应码 200=成功 当响应成功,获取响应的流  */ int res = conn.getResponseCode(); responseTime = System.currentTimeMillis(); this.requestTime = (int) ((responseTime-requestTime)/1000); Log.e(TAG, "response code:" + res); if (res == 200) { Log.e(TAG, "request success"); InputStream input = conn.getInputStream(); StringBuffer sb1 = new StringBuffer(); int ss; while ((ss = input.read()) != -1) { sb1.append((char) ss); } result = sb1.toString(); Log.e(TAG, "result : " + result); onUploadProcessListener.onUploadDone(UPLOAD_SUCCESS_CODE, "上传结果:" + result); return; } else { Log.e(TAG, "request error"); onUploadProcessListener.onUploadDone(UPLOAD_SERVER_ERROR_CODE,"上传失败:code=" + res); return; } } catch (MalformedURLException e) { onUploadProcessListener.onUploadDone(UPLOAD_SERVER_ERROR_CODE,"上传失败:error=" + e.getMessage()); e.printStackTrace(); return; } catch (IOException e) { onUploadProcessListener.onUploadDone(UPLOAD_SERVER_ERROR_CODE,"上传失败:error=" + e.getMessage()); e.printStackTrace(); return; } } public int getReadTimeOut() { return readTimeOut; } public void setReadTimeOut(int readTimeOut) { this.readTimeOut = readTimeOut; } public int getConnectTimeout() { return connectTimeout; } public void setConnectTimeout(int connectTimeout) { this.connectTimeout = connectTimeout; } /**  * 获取上传使用的时间  * @return  */ public static int getRequestTime() { return requestTime; } }

 

        在修改代码的过程中,遇见最多的是空指针异常,这个需要有耐心去加断点跟踪代码,很快会发现是哪个对象的问题。还有就是conn.getResponseCode()发生IO异常,这个并不是代码的问题,而是服务端缺少文件上传包,分别是commons-fileupload-1.2.1.jar和commons-io-1.3.2.jar.感觉在Java程序中遇到莫名其妙的问题都要考虑是不是缺包。

        服务端需要配置Struts文件

<struts> <constant name="struts.custom.i18n.resources" value="globalMessages"/> <constant name="struts.i18n.encoding" value="utf-8"/> <constant name="struts.multipart.saveDir" value="D:/upfile"/> <constant name="struts.multipart.maxSize" value="30000000"></constant> <package name="extjs" extends="json-default" namespace="/json"> <action name="photoUpload" class="com.example.action.FileUpload" method="upload"> <interceptor-ref name="fileUpload"> <param name="maximumSize">524288000</param> </interceptor-ref> <interceptor-ref name="defaultStack"/> <!-- 保存路径为当前项目WebRoot/upload --> <param name="savePath">/upload</param> <result name="success">index.jsp</result> <result name="input">index.jsp</result> </action> </package> </struts>

 

struts.multipart.maxSize的值,否则也会报错,具体看报错信息再做修改吧。

        到这里是一个线程的图片上传,那多张图片上传就创建多个线程去上传,感觉还需要考虑同步函数的问题。