• 通过android 客户端上传图片到服务器


    昨天,(在我的上一篇博客中)写了通过浏览器上传图片到服务器(php),今天将这个功能付诸实践.(还完善了服务端的代码)

    不试不知道,原来通过android 向服务端发送图片还真是挺麻烦的一件事.

    上传代码:

    /* 上传文件至Server的方法 */
        private void uploadFile()
        {
            String end = "
    ";
            String twoHyphens = "--";
            String boundary = "*****";
            try
            {
                URL url = new URL(postUrl);
                HttpURLConnection con = (HttpURLConnection) url.openConnection();
              /* Output to the connection. Default is false,
                 set to true because post method must write something to the connection */
                con.setDoOutput(true);
              /* Read from the connection. Default is true.*/
                con.setDoInput(true);
              /* Post cannot use caches */
                con.setUseCaches(false);
              /* Set the post method. Default is GET*/
                con.setRequestMethod("POST");
              /* 设置请求属性 */
                con.setRequestProperty("Connection", "Keep-Alive");
                con.setRequestProperty("Charset", "UTF-8");
                con.setRequestProperty("Content-Type", "multipart/form-data;boundary=" + boundary);
              /*设置StrictMode 否则HTTPURLConnection连接失败,因为这是在主进程中进行网络连接*/
                StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder().detectDiskReads().detectDiskWrites().detectNetwork().penaltyLog().build());
              /* 设置DataOutputStream,getOutputStream中默认调用connect()*/
                DataOutputStream ds = new DataOutputStream(con.getOutputStream());  //output to the connection
                ds.writeBytes(twoHyphens + boundary + end);
                ds.writeBytes("Content-Disposition: form-data; " +
                        "name="file";filename="" +
                        fileName + """ + end);
                ds.writeBytes(end);
              /* 取得文件的FileInputStream */
                FileInputStream fStream = new FileInputStream(uploadFile);
              /* 设置每次写入8192bytes */
                int bufferSize = 8192;
                byte[] buffer = new byte[bufferSize];   //8k
                int length = -1;
              /* 从文件读取数据至缓冲区 */
                while ((length = fStream.read(buffer)) != -1)
                {
                /* 将资料写入DataOutputStream中 */
                    ds.write(buffer, 0, length);
                }
                ds.writeBytes(end);
                ds.writeBytes(twoHyphens + boundary + twoHyphens + end);
              /* 关闭流,写入的东西自动生成Http正文*/
                fStream.close();
              /* 关闭DataOutputStream */
                ds.close();
              /* 从返回的输入流读取响应信息 */
                InputStream is = con.getInputStream();  //input from the connection 正式建立HTTP连接
                int ch;
                StringBuffer b = new StringBuffer();
                while ((ch = is.read()) != -1)
                {
                    b.append((char) ch);
                }
              /* 显示网页响应内容 */
    //            Toast.makeText(MainActivity.this, b.toString().trim(), Toast.LENGTH_SHORT).show();//Post成功
                System.out.println(b.toString());
            } catch (Exception e)
            {
                /* 显示异常信息 */
    //            Toast.makeText(MainActivity.this, "Fail:" + e, Toast.LENGTH_SHORT).show();//Post失败
                System.out.println(e);
            }
        }

    上篇博文有说上传图片必须使用POST方法,这里当然使用的就是post方法.这里上传跟普通上传写法没啥变化.

    不过我还是不习惯使用这个方法(我一直使用volley框架, 有兴趣可以百度一下.一个官方推出的很好用的框架,要不是需要用到传图我还不会研究这个写法)

    使用原生POST方法的基本步骤:

      1.使用 HttpURLConnection 建立连接

      2.设置请求属性

      3.根据请求格式组装参数(很麻烦 , 可以通过浏览器发送POST请求,取出调试信息对比)

      4.建立输入流,并写入数据(FileInputStream 和下面循环写入哪里)

      5.获取返回流,并处理返回参数

      6.关闭所有流

    该部分代码几乎完全借鉴大神的写法.里面注释也很多我也不多说了.

    值得一提的是在4.0(android 版本)以后就已经不支持在主线程中执行上传,下载等耗时操作了.

    这也很合理,假如你在上传文件的时候,突然你的网络不好,你就会一直在上传状态.体验很不好.

    所以调用上面的方法需要新建一个线程:

     new Thread(new Runnable() {
                        @Override
                        public void run() {
                            uploadFile();
                        }
                    }).start();

    这个调用方法是在上传方法与该方法同文件时可以这样写,如果上传方法在独立的class文件中需要先实例化然后通过

    实例化对象名.uploadFile();

    的方法调用.

    服务器端

      上篇博文里面只是简单的实现了,接收并转存图片.这里对该服务端进行完善:

    <?php
    /**
     * 接收上传的图片
     * 
     * ------------
     * 200 ok
     * 401 Error method
     * 500 Internal error
     * ------------
     */
    // ini_set("display_errors", "On");
    // error_reporting(E_ALL | E_STRICT);
    header('Access-Control-Allow-Origin: *');   // 解决前段javascript跨域请求
    
    $fileInfo = $_FILES['file'];
    $maxSize=2097152;//允许的最大值
    
    $allowExt=array('jpeg','jpg','png','gif','wbmp');
    $flag=true;//检测是否为真实图片类型
    
    if($fileInfo['error']==0){
        //判断上传文件的大小
        
        if($fileInfo['size']>$maxSize){
            $data = '上传文件过大';
            
            return Response::show(201,'error1', $data);
            
            exit();
        }
        //$ext=strtolower(end(explode('.',$fileInfo['name'])));获取后缀
        
        $ext=pathinfo($fileInfo['name'],PATHINFO_EXTENSION);
        if(!in_array($ext,$allowExt)){
            
            $data = '非法文件类型';
            
            return Response::show(202 , 'error2' , $data);
            
            exit();
            
        }
        //判断文件是否是通过HTTP POST方式上传来的
        
        if(!is_uploaded_file($fileInfo['tmp_name'])){
            
            $data = '文件不是通过HTTP POST方式上传来的';
            
            return Response::show(203 , 'error3' , $data);
            
            exit();
        }
        //检测是否为真实的图片类型
        
        if($flag){
            if(!getimagesize($fileInfo['tmp_name'])){
                
            $data = '不是真正图片类型';
                
            return Response::show(204 , 'error4' , $data);
                
            exit();
            
                
            }
        }
        
        //创建与id对应的文件夹 
        
        $id = $_GET['id'];
        if(!file_exists($id)){
            mkdir('../files/'.$id);
            //chmod($id , 0777);
        }
        
        if(@move_uploaded_file($fileInfo['tmp_name'],'../files/'.$id.'/'.$fileInfo['name'])){
            $data = '文件上传成功';
                
             return Response::show(200 , 'ok' , $data);
             
        }else{
            $data = '文件上传失败';
                
             return Response::show(404 , 'error5' , $data);
        }
    }else{
        switch($fileinfo['error']){
            case 1:
                $data = '上传文件超过了PHP配置文件中upload_max_filesize选项的值';
                
                return Response::show(401 , 'error5' , $data);
                
                break;
                
            case 2:
                
                $data = '超过了表单MAX_FILE_SIZE限制的大小';
                
                return Response::show(402 , 'error5' , $data);
                
                break;
            case 3:
                
                $data = '文件部分被上传';
                
                return Response::show(403 , 'error5' , $data);
                
                break;
            case 4:
            
                $data = '没有选择上传文件';
                
                return Response::show(405 , 'error5' , $data);
                
                break;
            case 6:
                
                $data = '没有找到临时目录';
                
                return Response::show(405 , 'error5' , $data);
                
                break;
            case 7:
            case 8:
                
                $data = '系统错误';
                
                return Response::show(405 , 'error5' , $data);
                
                break;
        }
    }

    这里对图片进行了各种检测.比昨天的更可靠了是吧.还统一了返回的json格式方便客户端进行解析.(还有就是,上面的服务端的代码也是借鉴了慕课大神的写法,略作修改)

  • 相关阅读:
    MapReduce案例
    Hive学习笔记九
    大数据技术之Hive
    Hive学习笔记八
    Hive学习笔记七
    Hive学习笔记六
    大数据应用技术课程实践--选题与实践方案
    15.手写数字识别-小数据集
    14.深度学习-卷积
    13.垃圾邮件分类2
  • 原文地址:https://www.cnblogs.com/wobeinianqing/p/5484035.html
Copyright © 2020-2023  润新知