Android HttpUrlConnection的使用(六)

  前面我简单的说了下, 客户端如何与服务器传输数据。之前的数据传输都是比较简单的,在这里我将说下,客户端如何从服务器上下载数据,同时也会解决在下载文件时,文件名出现中文的情况(其实这个方法,早在网上就有了,我只是在说下细节,楼主就是在细节出现问题,然后总是失败)。话不多说,直切主题。

1. 首先建一个xml布局文件(我的布局很简单)
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >
    <Button 
        android:id="@+idtton_load"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="@stringtton_string"
        />
    <TextView 
        android:id="@+id/textview"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        />
</LinearLayout>
2. Activity代码
package com.example.Download;

import java.net.URLEncoder;

import com.example.android_client.R;

import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;

public class MainActivity extends Activity implements OnClickListener{
    private Button button;
    private TextView textview = null;
    private int count = 0;
    private Handler handler = new Handler(){
        public void handleMessage(android.os.Message msg) {
            //这个Handler的作用主要是用来更新ui,它接收子线程传过来的消息,并且判断是否更新UI
            count +=msg.what;
            if(count == 3)
            {
                textview.setText("下载成功!");
            }
        };
    };
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.download);
        button = (Button) findViewById(R.id.button_load);
        textview = (TextView) findViewById(R.id.textview);
        button.setOnClickListener(this);
    }
    public void onClick(View v) {
        final Download download = new Download(handler);
        new Thread(){
            public void run() {
                
                try {
                    //将URl的格式转换为UTF-8
                    download.DownloadFile("http://222.196.200.63:8080/web/" + URLEncoder.encode("本兮 - 海誓山盟亦会分开 [mqms2].flac", "UTF-8").replace("+", "%20"));
                } catch (Exception e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
            }; 
        }.start();
    }

}
3. 下载线程的代码(新建了一个类,用来执行下载任务)
package com.example.Download;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.io.UnsupportedEncodingException;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.ProtocolException;
import java.net.URL;
import java.net.URLDecoder;
import java.net.URLEncoder;
import java.util.concurrent.Executor;
import java.util.concurrent.Executors;

import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.util.Log;

public class Download {

    //创建固定数量线程的线程池
    private Executor thread = Executors.newFixedThreadPool(3);
    private Handler handler = null;
    public Download(Handler handler)
    {
        this.handler = handler;
    }
    private class MyRunnable implements Runnable
    {
        private String url = null;
        private String fileName = null;
        private long start;
        private long end;
        private Handler handler = null;
        public MyRunnable(String url, String fileName, long start, long end, Handler handler)
        {
            this.url = url;
            this.fileName = fileName;
            this.end = end;
            this.start = start;
            this.handler = handler;
        }
        
        public void run() {
            try {
                URL httpurl = new URL(url);
                HttpURLConnection httpurlconnection = (HttpURLConnection) httpurl.openConnection();
                httpurlconnection.setReadTimeout(5000);
                httpurlconnection.setRequestMethod("GET");
                httpurlconnection.setRequestProperty("Range", "bytes=" + start + "-"+end);
                
                
                
                RandomAccessFile access = new RandomAccessFile(new File(fileName), "rwd");
                access.seek(start);
                InputStream is = httpurlconnection.getInputStream();
                byte []buff = new byte[1024 * 4];
                int len;
                while((len = is.read(buff)) != -1)
                {
                    access.write(buff,0,len);
                }
                if(access != null)
                {
                    access.close();
        
                }
                if(is != null)
                {
                    is.close();
                }
            } catch (MalformedURLException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } catch (ProtocolException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            //当当前的线程完成自己的任务是就像handler发送消息
            Message message = new Message();
            message.what = 1;
            handler.sendMessage(message);
        }
        
    }
    private String getfileName(String url)
    {
        //从url当中来获取文件名
        try {
            return URLDecoder.decode(url.substring(url.lastIndexOf("/") + 1), "UTF-8");
        } catch (UnsupportedEncodingException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return null;
    }
    public void DownloadFile(String url) throws Exception
    {
        Log.i("main", url);
        URL httpurl = new URL(url);
        HttpURLConnection httpurlconnection = (HttpURLConnection) httpurl.openConnection();
        httpurlconnection.setReadTimeout(5000);
        httpurlconnection.setRequestMethod("GET"); 

        //获取下载的文件的大小
        int leng = httpurlconnection.getContentLength();
        int block = leng / 3;
        
        String  fileName = getfileName(url);
        //获取手机内存卡的根目录
        File parent = Environment.getExternalStorageDirectory();
        //将文件写在根目录下
        File file = new File(parent, fileName);
        for(int i = 0; i < 3; i++)
        {
            //这个算法表示的意思是三个线程从不同索引开始下载
            //第一个是i * block 到 (i + 1) * block - 1
            //而最后一个是从i * block 到最后,因为不能保证文件长度恰好被3整除
            long start = i * block;
            long end = (i + 1) * block - 1;
            if(i == 2)
            {
                end = leng;
            }
            //连续创建三个线程,并将线程放在线程池中
            MyRunnable myrunnable = new MyRunnable(url, file.getAbsolutePath(), start, end, handler);
            //将线程放在线程池中并执行
            thread.execute(myrunnable);
        }
    }
}

  以上代码就是在客户端上的代码。当我们在服务器上放一下东西时,比如说在webcontent文件夹里面发一张图片,客户端在来下载。我们会发现,当文件名为英语时,可以正常下载。而当文件名是中文时或者含有中文就会下载失败,之后我们再去看编译器上的logcat,看到编译器抛出了一个FileNotFoundException异常,表示意思就是找不到这个文件。这是为什么呢?
  首先我们分析它的原因:
  服务器这边采用的是iso-8859-1编码格式,所以不支持中文,当服务器上有中文名的文件时,相当于的是该文件的Url地址就是含有中文,但是服务器的编码格式不支持中文,所以服务器这边的编码格式需要改变,改为支持中文的编码格式(我们统一使用UTF-8)。
  还有一个原因就是在客户端上。客户端这边提交的Url地址含有中文有问题,当我们使用java中的String采用的是unicode编码格式,与UTF-8编码格式不一致,所以客户端这边看似提交了一个正确的url地址,其实与服务器那边的Url不一致。在这里,不禁疑惑,为什么不一样?认真想一想,相同的中文,采用不同的编码格式,表示的形式还一样吗?所以这边的编码格式一定要转为统一的UTF-8格式。
  最后还有一个问题,当我们把服务器在JAVA EE上搭建好过后。记住,这里的顺序,是你先搭建好服务器,然后发现中文乱码了,再去使用网上的解决方案,发现还是不能正常的运行,这又是为什么呢?

  1.首先,我想说的是网上的解决方案没错。错误的原因在于我们的小细节出现了问题,问题在哪呢?

  想一想,我们是不是先在原来的apache Tomcat上搭建的系统,然后再去修改Apache Tomcat/conf/server.xml文件的内容。如果是这样的话,我会告诉你的是,这样的操作是大错特错的!!!正确的操作是在将原来的web服务器删除了,再去修改server的内容,将服务器的编码格式从默认的iso-8859-1格式改为UTF-8,再去以这个Apache Tomcat来创建一个web服务器。后面的操作与之前操作一样!这才是服务器的解决方案,客户端也要改变

  2.将客户端所提交的Url地址编码格式改为UTF-8

  首先,我再次说下,java的String类型采用的编码格式unicode格式编码,我们需要将它的编码格式改为UTF-8。那怎么修改呢?
代码如下(这是url地址码):

download.DownloadFile("http://222.196.200.63:8080/web/" + URLEncoder.encode("本兮 - 海誓山盟亦会分开 [mqms2].flac", "UTF-8").replace("+", "%20"));

  从这段代码中,我们可以看出我们将最后的带有中文的字符串的编码格式利用java中的URLEncoder转换为UTF-8格式,这样我们从客户端提交的url地址,服务器才能识别,因为此时服务器的编码格式也是UTF-8格式,这样服务器才能正确的返回我们需要的资源。
  可能还有人有疑问就是后面的replace("+", "%20")代表的是什么意思?其中URLEndcoder会使用+来代替空格,但是服务器并不能将+解码成空格,所以这里还有个小细节,就是将+换为%20。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,904评论 18 139
  • 一、概念(载录于:http://www.cnblogs.com/EricaMIN1987_IT/p/3837436...
    yuantao123434阅读 8,434评论 6 152
  • 《裕语言》速成开发手册3.0 官方用户交流:iApp开发交流(1) 239547050iApp开发交流(2) 10...
    叶染柒丶阅读 27,754评论 5 19
  • 1组原创清单如下 1号杨荣琳 1、2017.5/24第6周周检视 2、爱非坚持一一端午节也不放弃运动 3、5.31...
    Lily向丽阅读 457评论 0 1
  • 踏雪扬沙,飞云羽化。 人道是狸猫九命,毕竟空老去。 三千年回溯,一眼初秋。 问苍穹此生何苦?昨夜霜经处...
    北恨阅读 322评论 0 2