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