Android使用Handler實(shí)現(xiàn)下載文件功能
本文實(shí)例為大家分享了Android實(shí)現(xiàn)下載文件的具體代碼,供大家參考,具體內(nèi)容如下
1.實(shí)現(xiàn)效果
直接上圖:

2.代碼實(shí)現(xiàn)
在AndroidManifest.xml文件中聲明申請(qǐng)的權(quán)限,如下所示:
<uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
新建一個(gè)名為DownloadDemo的項(xiàng)目,activity_main.xml代碼如下:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:gravity="center_horizontal" android:orientation="vertical" tools:context=".MainActivity"> <Button android:id="@+id/button" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginTop="10dp" android:text="下載" /> <ProgressBar android:id="@+id/progressBar" style="?android:attr/progressBarStyleHorizontal" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_margin="10dp" android:max="100" android:progress="0" /> </LinearLayout>
MainActivity.class代碼如下:
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
public static final String APP_URL = "http://download.sj.qq.com/upload/connAssitantDownload/upload/MobileAssistant_1.apk";
public static final int DOWNLOAD_MESSAGE_CODE = 100001;
private static final int DOWNLOAD_MESSAGE_FAIL_CODE = 100002;
public static final int REQUEST_CODE = 1;
private Button button;
private ProgressBar progressBar;
private MyHandler mHandler;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
askPermission();
mHandler = new MyHandler(this);
}
public static class MyHandler extends Handler {
final WeakReference<MainActivity> mWeakReference;
public MyHandler(MainActivity activity) {
this.mWeakReference = new WeakReference<>(activity);
}
@Override
public void handleMessage(Message msg) {
MainActivity activity = mWeakReference.get();
super.handleMessage(msg);
switch (msg.what) {
case DOWNLOAD_MESSAGE_CODE:
activity.progressBar.setProgress((Integer) msg.obj);
if (((Integer) msg.obj) == 100) {
Toast.makeText(activity, "文件下載已完成!", Toast.LENGTH_SHORT).show();
}
break;
case DOWNLOAD_MESSAGE_FAIL_CODE:
Toast.makeText(activity, "文件下載失敗!", Toast.LENGTH_SHORT).show();
break;
}
}
}
/**
* 申請(qǐng)權(quán)限
*/
private void askPermission() {
//將所需申請(qǐng)的權(quán)限添加到List集合中
List<String> permissionList = new ArrayList<>();
if (ContextCompat.checkSelfPermission(MainActivity.this,
Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
permissionList.add(Manifest.permission.WRITE_EXTERNAL_STORAGE);
}
//判斷權(quán)限列表是否為空,若不為空,則向用戶(hù)申請(qǐng)權(quán)限,否則則直接執(zhí)行操作
if (!permissionList.isEmpty()) {
String[] permissions = permissionList.toArray(new String[permissionList.size()]);
ActivityCompat.requestPermissions(MainActivity.this, permissions, REQUEST_CODE);
} else {
//TODO
button.setOnClickListener(this);
}
}
/**
* 用戶(hù)對(duì)請(qǐng)求權(quán)限的dialog做出響應(yīng)之后,會(huì)處理請(qǐng)求權(quán)限的響應(yīng)
*
* @param requestCode 請(qǐng)求碼
* @param permissions 權(quán)限的集合
* @param grantResults 權(quán)限授予的結(jié)果
*/
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
switch (requestCode) {
case REQUEST_CODE:
if (grantResults.length > 0) {
for (int result : grantResults) {
if (result != PackageManager.PERMISSION_GRANTED) {
Toast.makeText(this, "必須同意所有權(quán)限才能使用本程序", Toast.LENGTH_SHORT).show();
finish();
return;
}
}
//TODO
button.setOnClickListener(this);
} else {
Toast.makeText(this, "權(quán)限申請(qǐng)失?。?, Toast.LENGTH_SHORT).show();
finish();
}
break;
default:
break;
}
}
/**
* 點(diǎn)擊事件
*
* @param v
*/
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.button:
new Thread(new Runnable() {
@Override
public void run() {
download(APP_URL);
}
}).start();
break;
}
}
/**
* 下載
*
* @param appUrl
*/
private void download(String appUrl) {
try {
//實(shí)例化一個(gè)url對(duì)象
URL url = new URL(appUrl);
//獲取URLConnection對(duì)象
URLConnection urlConnection = url.openConnection();
//獲取輸入流
InputStream inputStream = urlConnection.getInputStream();
//獲取文件的總長(zhǎng)度
int contentLength = urlConnection.getContentLength();
//設(shè)置軟件下載到手機(jī)存儲(chǔ)的位置文件夾名稱(chēng)
String downloadFolderName = Environment.getExternalStorageDirectory()
+ File.separator + "MyApp" + File.separator;
File file = new File(downloadFolderName);
//若文件夾不存在則進(jìn)行創(chuàng)建
if (!file.exists()) {
file.mkdir();
}
String fileName = downloadFolderName + "chaixingsi.apk";
File apkFile = new File(fileName);
if (apkFile.exists()) {
apkFile.delete();
}
int downloadContentLength = 0;
int length = 0;
byte[] bytes = new byte[1024];
OutputStream outputStream = new FileOutputStream(fileName);
while ((length = inputStream.read(bytes)) != -1) {
outputStream.write(bytes, 0, length);
downloadContentLength += length;
/**
* 發(fā)送消息通知主線(xiàn)程去更新UI
*/
Message message = Message.obtain();
message.obj = downloadContentLength * 100 / contentLength;
message.what = DOWNLOAD_MESSAGE_CODE;
mHandler.sendMessage(message);
}
inputStream.close();
outputStream.close();
} catch (MalformedURLException e) {
//下載失敗的話(huà)也發(fā)送消息
notifyDownloadFailed();
e.printStackTrace();
} catch (IOException e) {
notifyDownloadFailed();
e.printStackTrace();
}
}
/**
* 通知下載失敗
*/
private void notifyDownloadFailed() {
Message message = Message.obtain();
message.what = DOWNLOAD_MESSAGE_FAIL_CODE;
mHandler.sendMessage(message);
}
/**
* 初始化視圖
*/
private void initView() {
button = findViewById(R.id.button);
progressBar = findViewById(R.id.progressBar);
}
}
3.快捷鍵總結(jié)
提取方法:Ctrl+Alt+M;
提取變量:Ctrl+Alt+V;
提取常量:Ctrl+Alt+C;
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android 下載文件通知欄顯示進(jìn)度條功能的實(shí)例代碼
- Android中使用AsyncTask實(shí)現(xiàn)下載文件動(dòng)態(tài)更新進(jìn)度條功能
- android中實(shí)現(xiàn)OkHttp下載文件并帶進(jìn)度條
- android實(shí)現(xiàn)多線(xiàn)程下載文件(支持暫停、取消、斷點(diǎn)續(xù)傳)
- Android實(shí)現(xiàn)Service下載文件,Notification顯示下載進(jìn)度的示例
- 使用Android系統(tǒng)提供的DownloadManager來(lái)下載文件
- Android通過(guò)SOCKET下載文件的方法
- Android實(shí)現(xiàn)多線(xiàn)程下載文件的方法
- Android實(shí)現(xiàn)下載文件功能的方法
相關(guān)文章
Android開(kāi)發(fā)VR實(shí)戰(zhàn)之播放360度全景視頻
這篇文章主要為大家詳細(xì)介紹了Android開(kāi)發(fā)VR實(shí)戰(zhàn)之播放360度全景視頻,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-12-12
Android通知欄增加快捷開(kāi)關(guān)的功能實(shí)現(xiàn)教程
對(duì)于Android來(lái)說(shuō)其中一項(xiàng)很方便的操作便是下拉菜單,下拉菜單欄可以快捷打開(kāi)某項(xiàng)設(shè)置,這篇文章主要給大家介紹了關(guān)于Android通知欄增加快捷開(kāi)關(guān)的功能實(shí)現(xiàn),需要的朋友可以參考下2023-01-01
Kotlin開(kāi)發(fā)筆記之委托屬性與區(qū)間(譯)
最近在學(xué)習(xí)kotlin,發(fā)現(xiàn)了一些比較重要的知識(shí)點(diǎn),所以下面這篇文章主要給大家介紹了關(guān)于Kotlin開(kāi)發(fā)筆記之委托屬性與區(qū)間的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面來(lái)一起看看吧。2017-12-12
Android自定義View播放Gif動(dòng)畫(huà)的示例
本篇文章主要介紹了Android自定義View播放Gif動(dòng)畫(huà)的示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-10-10
Intent傳遞對(duì)象之Serializable和Parcelable的區(qū)別
Intent在不同的組件中傳遞對(duì)象數(shù)據(jù)的應(yīng)用非常普遍,大家都知道在intent傳遞對(duì)象的方法有兩種:1、實(shí)現(xiàn)Serializable接口、2、實(shí)現(xiàn)Parcelable接口,接下來(lái)通過(guò)本文給大家介紹Intent傳遞對(duì)象之Serializable和Parcelable的區(qū)別,感興趣的朋友一起學(xué)習(xí)吧2016-01-01
Android AndBase框架使用封裝好的函數(shù)完成Http請(qǐng)求(三)
這篇文章主要介紹了Android AndBase框架使用封裝好的函數(shù)完成Http請(qǐng)求的相關(guān)資料,感興趣的小伙伴們可以參考一下2016-03-03
Android編程基于自定義View實(shí)現(xiàn)絢麗的圓形進(jìn)度條功能示例
這篇文章主要介紹了Android編程基于自定義View實(shí)現(xiàn)絢麗的圓形進(jìn)度條功能,結(jié)合實(shí)例形式詳細(xì)分析了Android自定義view實(shí)現(xiàn)圓形進(jìn)度條的具體步驟與相關(guān)操作技巧,需要的朋友可以參考下2017-01-01

