Android学习笔记(四六):互联网通信-文件下载

在Android 2.3引入了DownloadManager可以处理复杂的文件下载,包括检查用户是否有数据联系(WIFI或者移动数据),当用户从一个有数据连接的地方移动到无连接的地方(例如离开了wifi或者3G data的access point),确保设备在下载过程中保持awake状态。DownloadManager可以处理HTTP URLs,但是不能处理HTTPS(SSL) URLs。

设置下载文件条件许可

在这个例子,将学习通过DownloadManager从Internet下载文件,并存储在外部存储介质SD卡上。有以下需要注意:

  1. 由于不支持2.3之前的版本,需将最小版本设置为Android2.3或者以上。
  2. 在模拟器,我们需确保已设置SD卡,如右图所示。
  3. 程序具有Internet以及外部存储的访问权限,在Androidmanifest.xml中设置:

    <uses-permission android:name="android.permission.INTERNET"/> 
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    <application> 
        .... 
    </application>

小程序的设计

具体的xml文件略去。布局简单地分为3个button,如右图,第一个button设置android:onClick="startDownload",即点击后触发startDownload()方法,用于请求下载文件。第二个button触发queryStatus(),并disabled,点击触发下载的状态查询。第三个button触发viewLog(),调用系统提供的DownloadManager的Activity,用来查看历史下载情况。

请求文件下载

    privateDownloadManager mgr = null; 
    private long lastDownloadId = 0;

protected void onCreate(Bundle savedInstanceState) { 
        … …  
        // 步骤1 : 获取系统服务,并指明是下载服务,即DownloadManager。系统的这类服务大部分这些管理没有close() ,release()之类的由系统garbage收集来处理。我们只需获取这些服务的对象,并发出我们的请求
        mgr = (DownloadManager)getSystemService(DOWNLOAD_SERVICE); 
    }

public void startDownload(View v){  
        Uri uri = Uri.parse("http://commonsware.com/misc/test.mp4");
        //文件将存放在外部存储的确实download文件内,如果无此文件夹,创建之,如果有,下面将返回false。不同的手机不同Android版本的SD卡的挂载点可能会不一样,因此通过系统方式获取。
        Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS).mkdir();

//步骤2: 通过向下载服务发出enqueue()的请求,将放在下载队列中,通常会触发立即下载,并返回下载的ID号,根据这个号,可以查询相关的下载情况。分别设置请求的Uri,允许的数据访问方式,是否允许漫游,本地存储的位置,以及为这个下载设置title和描述信息。
        lastDownloadId = mgr.enqueue(new DownloadManager.Request(uri) 
           .setAllowedNetworkTypes(DownloadManager.Request.NETWORK_MOBILE | DownloadManager.Request.NETWORK_WIFI) 
           .setAllowedOverRoaming(false)           //缺省是true,所以天价漫游数据费的产生 
           .setTitle("MyTest")         //用于信息查看 
           .setDescription("Something Useful")   //用于信息查看 
           .setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, "test.mp4"));
        v.setEnabled(false); 
        findViewById(R.id.c25_query).setEnabled(true);  
    }

获取下载状态

通常会有一个后台运行来不断更新下载的情况,例子目的是如何获取,所以简单地通过点击第二个button触发查询下载状态。

public void queryStatus(View v){
    //关键:通过ID向下载管理查询下载情况,返回一个cursor 
    Cursor c = mgr.query(new DownloadManager.Query().setFilterById(lastDownloadId));
    if(c == null){ 
        Toast.makeText(this, "Download not found!", Toast.LENGTH_LONG).show(); 
    }else{  //以下是从游标中进行信息提取 
        c.moveToFirst(); 
        Log.d(getClass().getName(),"Column_id : " + 
                c.getLong(c.getColumnIndex(DownloadManager.COLUMN_ID))); 
        Log.d(getClass().getName(),"Column_bytes_downloaded so far : " + 
                c.getLong(c.getColumnIndex(DownloadManager.COLUMN_BYTES_DOWNLOADED_SO_FAR)));
        Log.d(getClass().getName(),"Column last modified timestamp : " + 
                c.getLong(c.getColumnIndex(DownloadManager.COLUMN_LAST_MODIFIED_TIMESTAMP)));
        Log.d(getClass().getName(),"Column local uri : " + 
                c.getString(c.getColumnIndex(DownloadManager.COLUMN_LOCAL_URI)));
        Log.d(getClass().getName(),"Column statue : " + 
                c.getInt(c.getColumnIndex(DownloadManager.COLUMN_STATUS))); 
        Log.d(getClass().getName(),"Column reason : " + 
                c.getInt(c.getColumnIndex(DownloadManager.COLUMN_REASON)));

Toast.makeText(this, statusMessage(c), Toast.LENGTH_LONG).show(); 
    } 
}

private String statusMessage(Cursor c){  
    switch(c.getInt(c.getColumnIndex(DownloadManager.COLUMN_STATUS))){ 
    case DownloadManager.STATUS_FAILED:  
        return "Download failed";  
    case DownloadManager.STATUS_PAUSED:  
       return "Download paused";  
    case DownloadManager.STATUS_PENDING:  
        return "Download pending";  
    case DownloadManager.STATUS_RUNNING: 
        return "Download in progress!";  
    case DownloadManager.STATUS_SUCCESSFUL:  
        return "Download finished";  
    default: 
        return "Unknown Information";  
    } 
}

从信息中,我们可以看到下载的存放的位置,SD卡的挂点为/mnt/sdcard/我们可以通过$adb shell进入模拟器的控制台进行查看。另外获取文件的总体大小为COLUMN_TOTAL_SIZE_BYTES。

# pwd/mnt/sdcard/Download
# ls -l
----rwxr-x system sdcard_rw 6219229 2011-11-01 14:19 test.mp4

通过下载管理查看

public void viewLog(View v){ 
    startActivity(new Intent(DownloadManager.ACTION_VIEW_DOWNLOADS));
}

通过BoardReceiver获取实时事件触发

在上面的例子中,希望在一下载完进行触发,将第一个button恢复为enabled状态。在Android学习笔记(三四):再谈Intent(上)-一些知识中谈到,通过BoardReceiver从服务器中获取事件触发处理。

    protected void onCreate(Bundle savedInstanceState) {
        ... ... 
       //当下载结束时进行触发。 
        registerReceiver(onComplete, new IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE));
       //当点击一个正在下载的文件,如图所示 
        registerReceiver(onNotification,new IntentFilter(DownloadManager.ACTION_NOTIFICATION_CLICKED));
    } 
    BroadcastReceiver onComplete =  new BroadcastReceiver() {  
        public void onReceive(Context context, Intent intent) { 
           findViewById(R.id.c25_start).setEnabled(true);
        } 
    }; 
    BroadcastReceiver onNotification = new BroadcastReceiver() {  
        public void onReceive(Context context, Intent intent) { 
            Toast.makeText(context, "..............", Toast.LENGTH_LONG).show();
        } 
    };

对于DownloadManager,如果文件已经下载,第二次无需再下载。另外由于DownloadManager属于系统服务,不仅是你的app可以调用,也就是上面list的内容是全局的,可能部分并非你的app在下载,这样会使用户迷惑,在request请求中,我们可以同setVisibleInDownloadsUi(false),可以屏蔽之。

传统文件下载方式

ownloadManager服务需要Android版本2.3以上,如果不满足条件,可采用获取网络文件流的方式来处理,具体步骤如下:

  1. 建一个HttpURLConnection的对象,可以通过URL对象的openConnection()方法获取,例如:HttpURLConnection urlConn = (HttpURLConnection) url.openconnection();
  2. 获取一个InputStream对象:urlConn.getInputStream()。

有了InputStream,剩下的都是Java的标准I/O操作。

注意

对于Internet的访问,不要再应用的主线程进行,而应该在后台线程中处理HttpClient,HttpUrlConnection以及其他的Internet access API。Android学习笔记(四五):互联网通信-HttpClient、XML解析(W3C)的例子只是作为小例子简单清晰说明相关的使用方式。

相关链接:我的Andriod开发相关文章

时间: 2024-10-26 06:04:55

Android学习笔记(四六):互联网通信-文件下载的相关文章

Android 学习笔记 6 组件通信及广播消息(一)

Intent的概念 Intent的官方解释是“An Intent is a messaging object you can use to request an action from another app component. ”这里的app component就是指安卓activity,service,contentprovider,broadcastreceiver四大组件.不同的intent可以使这些组件产生相应的动作,为这些组件之间提供了交互能力.那么这个“messaging obj

Android学习笔记之AsyncTask实现文件下载任务

(1)该文件下载主要演示异步任务下载图片 (2)布局代码如下:一个ImageView用于存放下载的图片,Button用于开始异步任务下载 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent&q

Android 学习笔记 7 组件通信及广播消息(二)

Intent隐式启动Activity 隐式启动的好处在于不需要在第一个组件中指明需要启动另外的哪一个组件,而由Android系统来决定,这样有利于降低组件之间的耦合度. 选择隐式启动Activity,Android系统会在程序运行时解析Intent,并根据一定的规则对Intent和组件进行匹配,使Intent上的action.data和category与目标Activity吻合.匹配的组件可以是程序本身的Activity,也可以是Android系统内置的Activity,还可以是第三方应用程序提

Android学习笔记(四五):互联网通信-HttpClient、XML解析(W3C)

前几日Android发布了4.0 Icecream,昨天上网发现Begining Book中有Edition 3的版本,比对一下,还是有相当的改动,不仅仅增加了tablet的部分,对原有的章节有有一些修订,前后的调整等等.先按Edtion 2的顺序看,相同章节的看Edtion 3,然后回头看Edition 3的Chapter 24.25(E2的36).26.27.28.29.44.45.46.47几个新增章节.同时将模拟器改为Android 2.3的版本,已适应可能新增的改动. 访问Intern

【转】 Pro Android学习笔记(七五):HTTP服务(9):DownloadManager

目录(?)[-] 小例子 保存在哪里下载文件信息设置和读取 查看下载状态和取消下载 文章转载只能用于非商业性质,且不能带有虚拟货币.积分.注册等附加条件,转载须注明出处:http://blog.csdn.net/flowingflying/ 应用有时需要从web中下载一个大文件,并保存在本地,这个操作过程是标准的,因此在Android2.3引入了DownloadManager类.相关的学习也可以参考Android学习笔记(四六):互联网通信-文件下载. 小例子 先看一个小例子,如下图所示.lay

【转】 Pro Android学习笔记(七四):HTTP服务(8):使用后台线程AsyncTask

目录(?)[-] 5秒超时异常 AsyncTask 实现AsyncTask抽象类 对AsyncTask的调用 在哪里运行 其他重要method 文章转载只能用于非商业性质,且不能带有虚拟货币.积分.注册等附加条件,转载须注明出处:http://blog.csdn.net/flowingflying/ 之前,我们直接在activity中执行http通信,在通信过程中可能会出现连接超时.socket超时等情况,超时阈值一般是秒级,例如AndroidHttpClient中设置的20秒,如果出现超时,就

【转】 Pro Android学习笔记(六七):HTTP服务(1):HTTP GET

目录(?)[-] HTTP GET小例子 简单小例子 出现异常NetworkOnMainThreadException 通过StrictMode进行处理 URL带键值对 Andriod应用可利用service提供很多功能,例如利用Google Maps API,现在我们将聚焦在HTTP serice中. Android SDK提供HttpClient,和J2EE中的接口非常相似.最常用的就是HTTP GET和HTTP POST.相关内容也可以阅读Android学习笔记(四五):互联网通信-Htt

Android学习笔记(二二): 多页显示-Tag的使用

在手机屏幕中,Tab也是比较常用的,通常和List结合,例如我们手机的通信录.下面是Tag的结构. TabHost是整个Tab的容器,包括两部分,TabWidget和FrameLayout.TabWidget就是每个tab的标签,FrameLayout则是tab内容. 如果我们使用extends TabAcitivty,如同ListActivity,TabHost必须设置为@android:id/tabhost TabWidget必须设置android:id为@android:id/tabs F

Android学习笔记(十六)——碎片之间进行交互(附源码)

碎片之间进行交互 点击下载源码 很多时候,一个活动中包含一个或者多个碎片,它们彼此协作,向用户展示一个一致的UI.在这种情况下,碎片之间能进行通信并交换数据十分重要. 1.使用上一篇中创建的同一个项目,在fragment.xml中添加TextView的标识id: android:id="@+id/lblFragment1" 2.在fragment2.xml中添加一个Button,用于与fragment1进行交互: <Button android:id="@+id/btn