android选择图片或拍照图片上传到服务器(包括上传参数) (转)

最近要搞一个项目,需要上传相册和拍照的图片,不负所望,终于完成了!  不过需要说明一下,其实网上很多教程拍照的图片,都是缩略图不是很清晰,所以需要在调用照相机的时候,事先生成一个地址,用于标识拍照的图片URI

具体上传代码:

1.选择图片和上传界面,包括上传完成和异常的回调监听

package com.spring.sky.image.upload;  
  
  
import java.util.HashMap;  
import java.util.Map;  
  
import android.app.Activity;  
import android.app.ProgressDialog;  
import android.content.Intent;  
import android.graphics.Bitmap;  
import android.graphics.BitmapFactory;  
import android.os.Bundle;  
import android.os.Handler;  
import android.os.Message;  
import android.util.Log;  
import android.view.View;  
import android.view.View.OnClickListener;  
import android.widget.Button;  
import android.widget.ImageView;  
import android.widget.ProgressBar;  
import android.widget.TextView;  
import android.widget.Toast;  
  
import com.spring.sky.image.upload.network.UploadUtil;  
import com.spring.sky.image.upload.network.UploadUtil.OnUploadProcessListener;  
/** 
 * @author spring sky<br> 
 * Email :vipa1888@163.com<br> 
 * QQ: 840950105<br> 
 * 说明:主要用于选择文件和上传文件操作 
 */  
public class MainActivity extends Activity implements OnClickListener,OnUploadProcessListener{  
    private static final String TAG = "uploadImage";  
      
    /** 
     * 去上传文件 
     */  
    protected static final int TO_UPLOAD_FILE = 1;    
    /** 
     * 上传文件响应 
     */  
    protected static final int UPLOAD_FILE_DONE = 2;  //  
    /** 
     * 选择文件 
     */  
    public static final int TO_SELECT_PHOTO = 3;  
    /** 
     * 上传初始化 
     */  
    private static final int UPLOAD_INIT_PROCESS = 4;  
    /** 
     * 上传中 
     */  
    private static final int UPLOAD_IN_PROCESS = 5;  
    /*** 
     * 这里的这个URL是我服务器的javaEE环境URL 
     */  
    private static String requestURL = "http://192.168.10.160:8080/fileUpload/p/file!upload";  
    private Button selectButton,uploadButton;  
    private ImageView imageView;  
    private TextView uploadImageResult;  
    private ProgressBar progressBar;  
      
    private String picPath = null;  
    private ProgressDialog progressDialog;  
      
    /** Called when the activity is first created. */  
    @Override  
    public void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.main);  
        initView();  
    }  
      
    /** 
     * 初始化数据 
     */  
    private void initView() {  
        selectButton = (Button) this.findViewById(R.id.selectImage);  
        uploadButton = (Button) this.findViewById(R.id.uploadImage);  
        selectButton.setOnClickListener(this);  
        uploadButton.setOnClickListener(this);  
        imageView = (ImageView) this.findViewById(R.id.imageView);  
        uploadImageResult = (TextView) findViewById(R.id.uploadImageResult);  
        progressDialog = new ProgressDialog(this);  
        progressBar = (ProgressBar) findViewById(R.id.progressBar1);  
    }  
  
    @Override  
    public void onClick(View v) {  
        switch (v.getId()) {  
        case R.id.selectImage:  
            Intent intent = new Intent(this,SelectPicActivity.class);  
            startActivityForResult(intent, TO_SELECT_PHOTO);  
            break;  
        case R.id.uploadImage:  
            if(picPath!=null)  
            {  
                handler.sendEmptyMessage(TO_UPLOAD_FILE);  
            }else{  
                Toast.makeText(this, "上传的文件路径出错", Toast.LENGTH_LONG).show();  
            }  
            break;  
        default:  
            break;  
        }  
    }  
  
    @Override  
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {  
        if(resultCode==Activity.RESULT_OK && requestCode == TO_SELECT_PHOTO)  
        {  
            picPath = data.getStringExtra(SelectPicActivity.KEY_PHOTO_PATH);  
            Log.i(TAG, "最终选择的图片="+picPath);  
            Bitmap bm = BitmapFactory.decodeFile(picPath);  
            imageView.setImageBitmap(bm);  
        }  
        super.onActivityResult(requestCode, resultCode, data);  
    }  
      
  
    /** 
     * 上传服务器响应回调 
     */  
    @Override  
    public void onUploadDone(int responseCode, String message) {  
        progressDialog.dismiss();  
        Message msg = Message.obtain();  
        msg.what = UPLOAD_FILE_DONE;  
        msg.arg1 = responseCode;  
        msg.obj = message;  
        handler.sendMessage(msg);  
    }  
      
    private void toUploadFile()  
    {  
        uploadImageResult.setText("正在上传中...");  
        progressDialog.setMessage("正在上传文件...");  
        progressDialog.show();  
        String fileKey = "pic";  
        UploadUtil uploadUtil = UploadUtil.getInstance();;  
        uploadUtil.setOnUploadProcessListener(this);  //设置监听器监听上传状态  
          
        Map<String, String> params = new HashMap<String, String>();  
        params.put("orderId", "11111");  
        uploadUtil.uploadFile( picPath,fileKey, requestURL,params);  
    }  
      
    private Handler handler = new Handler(){  
        @Override  
        public void handleMessage(Message msg) {  
            switch (msg.what) {  
            case TO_UPLOAD_FILE:  
                toUploadFile();  
                break;  
              
            case UPLOAD_INIT_PROCESS:  
                progressBar.setMax(msg.arg1);  
                break;  
            case UPLOAD_IN_PROCESS:  
                progressBar.setProgress(msg.arg1);  
                break;  
            case UPLOAD_FILE_DONE:  
                String result = "响应码:"+msg.arg1+"
响应信息:"+msg.obj+"
耗时:"+UploadUtil.getRequestTime()+"秒";  
                uploadImageResult.setText(result);  
                break;  
            default:  
                break;  
            }  
            super.handleMessage(msg);  
        }  
          
    };  
  
    @Override  
    public void onUploadProcess(int uploadSize) {  
        Message msg = Message.obtain();  
        msg.what = UPLOAD_IN_PROCESS;  
        msg.arg1 = uploadSize;  
        handler.sendMessage(msg );  
    }  
  
    @Override  
    public void initUpload(int fileSize) {  
        Message msg = Message.obtain();  
        msg.what = UPLOAD_INIT_PROCESS;  
        msg.arg1 = fileSize;  
        handler.sendMessage(msg );  
    }  
      
}  

2.选择图片界面,主要涉及两种方式:选择图片和及时拍照图片

package com.spring.sky.image.upload;  
  
import android.app.Activity;  
import android.content.ContentValues;  
import android.content.Intent;  
import android.database.Cursor;  
import android.net.Uri;  
import android.os.Bundle;  
import android.os.Environment;  
import android.provider.MediaStore;  
import android.util.Log;  
import android.view.MotionEvent;  
import android.view.View;  
import android.view.View.OnClickListener;  
import android.widget.Button;  
import android.widget.LinearLayout;  
import android.widget.Toast;  
  
/** 
 * @author spring sky<br> 
 * Email :vipa1888@163.com<br> 
 * QQ: 840950105<br> 
 * @version 创建时间:2012-11-22 上午9:20:03 
 * 说明:主要用于选择文件操作 
 */  
  
public class SelectPicActivity extends Activity implements OnClickListener{  
  
    /*** 
     * 使用照相机拍照获取图片 
     */  
    public static final int SELECT_PIC_BY_TACK_PHOTO = 1;  
    /*** 
     * 使用相册中的图片 
     */  
    public static final int SELECT_PIC_BY_PICK_PHOTO = 2;  
      
    /*** 
     * 从Intent获取图片路径的KEY 
     */  
    public static final String KEY_PHOTO_PATH = "photo_path";  
      
    private static final String TAG = "SelectPicActivity";  
      
    private LinearLayout dialogLayout;  
    private Button takePhotoBtn,pickPhotoBtn,cancelBtn;  
  
    /**获取到的图片路径*/  
    private String picPath;  
      
    private Intent lastIntent ;  
      
    private Uri photoUri;  
    @Override  
    protected void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.select_pic_layout);  
        initView();  
    }  
    /** 
     * 初始化加载View 
     */  
    private void initView() {  
        dialogLayout = (LinearLayout) findViewById(R.id.dialog_layout);  
        dialogLayout.setOnClickListener(this);  
        takePhotoBtn = (Button) findViewById(R.id.btn_take_photo);  
        takePhotoBtn.setOnClickListener(this);  
        pickPhotoBtn = (Button) findViewById(R.id.btn_pick_photo);  
        pickPhotoBtn.setOnClickListener(this);  
        cancelBtn = (Button) findViewById(R.id.btn_cancel);  
        cancelBtn.setOnClickListener(this);  
          
        lastIntent = getIntent();  
    }  
  
    @Override  
    public void onClick(View v) {  
        switch (v.getId()) {  
        case R.id.dialog_layout:  
            finish();  
            break;  
        case R.id.btn_take_photo:  
            takePhoto();  
            break;  
        case R.id.btn_pick_photo:  
            pickPhoto();  
            break;  
        default:  
            finish();  
            break;  
        }  
    }  
  
    /** 
     * 拍照获取图片 
     */  
    private void takePhoto() {  
        //执行拍照前,应该先判断SD卡是否存在  
        String SDState = Environment.getExternalStorageState();  
        if(SDState.equals(Environment.MEDIA_MOUNTED))  
        {  
              
            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);//"android.media.action.IMAGE_CAPTURE"  
            /*** 
             * 需要说明一下,以下操作使用照相机拍照,拍照后的图片会存放在相册中的 
             * 这里使用的这种方式有一个好处就是获取的图片是拍照后的原图 
             * 如果不实用ContentValues存放照片路径的话,拍照后获取的图片为缩略图不清晰 
             */  
            ContentValues values = new ContentValues();    
            photoUri = this.getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);    
            intent.putExtra(android.provider.MediaStore.EXTRA_OUTPUT, photoUri);  
            /**-----------------*/  
            startActivityForResult(intent, SELECT_PIC_BY_TACK_PHOTO);  
        }else{  
            Toast.makeText(this,"内存卡不存在", Toast.LENGTH_LONG).show();  
        }  
    }  
  
    /*** 
     * 从相册中取图片 
     */  
    private void pickPhoto() {  
        Intent intent = new Intent();  
        intent.setType("image/*");  
        intent.setAction(Intent.ACTION_GET_CONTENT);  
        startActivityForResult(intent, SELECT_PIC_BY_PICK_PHOTO);  
    }  
      
    @Override  
    public boolean onTouchEvent(MotionEvent event) {  
        finish();  
        return super.onTouchEvent(event);  
    }  
      
      
    @Override  
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {  
        if(resultCode == Activity.RESULT_OK)  
        {  
            doPhoto(requestCode,data);  
        }  
        super.onActivityResult(requestCode, resultCode, data);  
    }  
      
    /**  
     * 选择图片后,获取图片的路径  
     * @param requestCode  
     * @param data  
     */  
    private void doPhoto(int requestCode,Intent data)  
    {  
        if(requestCode == SELECT_PIC_BY_PICK_PHOTO )  //从相册取图片,有些手机有异常情况,请注意  
        {  
            if(data == null)  
            {  
                Toast.makeText(this, "选择图片文件出错", Toast.LENGTH_LONG).show();  
                return;  
            }  
            photoUri = data.getData();  
            if(photoUri == null )  
            {  
                Toast.makeText(this, "选择图片文件出错", Toast.LENGTH_LONG).show();  
                return;  
            }  
        }  
        String[] pojo = {MediaStore.Images.Media.DATA};  
        Cursor cursor = managedQuery(photoUri, pojo, null, null,null);     
        if(cursor != null )  
        {  
            int columnIndex = cursor.getColumnIndexOrThrow(pojo[0]);  
            cursor.moveToFirst();  
            picPath = cursor.getString(columnIndex);  
    try {  
                //4.0以上的版本会自动关闭 (4.0--14;; 4.0.3--15)  
                if(Integer.parseInt(Build.VERSION.SDK) < 14)  
                {  
                    cursor.close();  
                }  
            }catch(Exception e){  
                Log.e(TAG, "error:" + e);  
            }  
        }  
        Log.i(TAG, "imagePath = "+picPath);  
        if(picPath != null && ( picPath.endsWith(".png") || picPath.endsWith(".PNG") ||picPath.endsWith(".jpg") ||picPath.endsWith(".JPG")  ))  
        {  
            lastIntent.putExtra(KEY_PHOTO_PATH, picPath);  
            setResult(Activity.RESULT_OK, lastIntent);  
            finish();  
        }else{  
            Toast.makeText(this, "选择图片文件不正确", Toast.LENGTH_LONG).show();  
        }  
    }  
}  

3. 上传工具类,主要实现了图片的上传,上传过程的初始化监听和上传完成的监听,还有上传耗时的计算

package com.spring.sky.image.upload.network;  
  
import java.io.DataOutputStream;  
import java.io.File;  
import java.io.FileInputStream;  
import java.io.IOException;  
import java.io.InputStream;  
import java.net.HttpURLConnection;  
import java.net.MalformedURLException;  
import java.net.URL;  
import java.util.Iterator;  
import java.util.Map;  
import java.util.UUID;  
  
import android.util.Log;  
  
/** 
 *  
 * 上传工具类 
 * @author spring sky<br> 
 * Email :vipa1888@163.com<br> 
 * QQ: 840950105<br> 
 * 支持上传文件和参数 
 */  
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 = "
";  
    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) {  
        if (filePath == null) {  
            sendMessage(UPLOAD_FILE_NOT_EXISTS_CODE,"文件不存在");  
            return;  
        }  
        try {  
            File file = new File(filePath);  
            uploadFile(file, fileKey, RequestURL, param);  
        } catch (Exception e) {  
            sendMessage(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())) {  
            sendMessage(UPLOAD_FILE_NOT_EXISTS_CODE,"文件不存在");  
            return;  
        }  
  
        Log.i(TAG, "请求的URL=" + RequestURL);  
        Log.i(TAG, "请求的fileName=" + file.getName());  
        Log.i(TAG, "请求的fileKey=" + fileKey);  
        new Thread(new Runnable() {  //开启线程上传文件  
            @Override  
            public void run() {  
                toUploadFile(file, fileKey, RequestURL, param);  
            }  
        }).start();  
          
    }  
  
    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);  
//          conn.setRequestProperty("Content-Type", "application/x-www-form-urlencoded");  
              
            /** 
             * 当文件不为空,把文件包装并且上传 
             */  
            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());  
//                  dos.flush();  
                }  
            }  
              
            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();  
//            
//          dos.write(tempOutputStream.toByteArray());  
            /** 
             * 获取响应码 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);  
                sendMessage(UPLOAD_SUCCESS_CODE, "上传结果:"  
                        + result);  
                return;  
            } else {  
                Log.e(TAG, "request error");  
                sendMessage(UPLOAD_SERVER_ERROR_CODE,"上传失败:code=" + res);  
                return;  
            }  
        } catch (MalformedURLException e) {  
            sendMessage(UPLOAD_SERVER_ERROR_CODE,"上传失败:error=" + e.getMessage());  
            e.printStackTrace();  
            return;  
        } catch (IOException e) {  
            sendMessage(UPLOAD_SERVER_ERROR_CODE,"上传失败:error=" + e.getMessage());  
            e.printStackTrace();  
            return;  
        }  
    }  
  
    /** 
     * 发送上传结果 
     * @param responseCode 
     * @param responseMessage 
     */  
    private void sendMessage(int responseCode,String responseMessage)  
    {  
        onUploadProcessListener.onUploadDone(responseCode, responseMessage);  
    }  
      
    /** 
     * 下面是一个自定义的回调函数,用到回调上传文件是否完成 
     *  
     * @author shimingzheng 
     *  
     */  
    public static interface OnUploadProcessListener {  
        /** 
         * 上传响应 
         * @param responseCode 
         * @param message 
         */  
        void onUploadDone(int responseCode, String message);  
        /** 
         * 上传中 
         * @param uploadSize 
         */  
        void onUploadProcess(int uploadSize);  
        /** 
         * 准备上传 
         * @param fileSize 
         */  
        void initUpload(int fileSize);  
    }  
    private OnUploadProcessListener onUploadProcessListener;  
      
      
  
    public void setOnUploadProcessListener(  
            OnUploadProcessListener onUploadProcessListener) {  
        this.onUploadProcessListener = onUploadProcessListener;  
    }  
  
    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;  
    }  
      
    public static interface uploadProcessListener{  
          
    }  
      
      
      
      
}
注意:一般是下面两种情况:
1、如果摄像的时候手动传入一个URI,则onActivityResult不会返回URI。
2、如果摄像的时候没有传入一个URI,则onActivityResult会返回URI,然后查询得到文件名。
 
还有极少数比较特别的情况,但都是一些定制系统的问题。 
 
解析Intent的Data要用ContentProvider去查系统数据库,得到文件路径,进而得到文件名。




转自: http://blog.csdn.net/blueangle17/article/details/9613807
原文地址:https://www.cnblogs.com/lufengdie/p/3644817.html