文件管理 · 2022年7月25日

asynctask下载文件|如何用队列管理AsyncTask

⑴ android asynctask过时了吗

注意事项;1. AsyncTask对象不可重复使用,也就是说一个AsyncTask对象只能execute()一次,否则会有异常抛出"java.lang.IllegalStateException: Cannot execute task: the task is already running"2. 在doInBackground()中要检查isCancelled()的返回值,如果你的异步任务是可以取消的话。cancel()仅仅是给AsyncTask对象设置了一个标识位,当调用了cancel()后,发生的事情只有:AsyncTask对象的标识位变了,和doInBackground()执行完成后,onPostExecute()不会被回调了,而doInBackground()和 onProgressUpdate()还是会继续执行直到doInBackground()结束。所以要在doInBackground()中不断的检查 isCancellled()的返回值,当其返回true时就停止执行,特别是有循环的时候。如上面的例子,如果把读取数据的isCancelled() 检查去掉,图片还是会下载,进度也一直会走,只是最后图片不会放到UI上(因为onPostExecute()没被回调)!这里的原因其实很好理解,想想Java SE的Thread吧,是没有方法将其直接Cacncel掉的,那些线程取消也无非就是给线程设置标识位,然后在run()方法中不断的检查标识而已。3. 如果要在应用程序中使用网络,一定不要忘记在AndroidManifest中声明INTERNET权限,否则会报出很诡异的异常信息,比如上面的例子,如果把INTERNET权限拿掉会抛出"UnknownHostException"。刚开始很疑惑,因为模拟器是可以正常上网的,后来Google了下才发现原来是没权限,但是疑问还是没有消除,既然没有声明网络权限,为什么不直接提示无网络权限呢?对比Java SE的ThreadThread是非常原始的类,它只有一个run()方法,一旦开始,无法停止,它仅适合于一个非常独立的异步任务,也即不需要与主线程交互,对于其他情况,比如需要取消或与主线程交互,都需添加额外的代码来实现,并且还要注意同步的问题。而AsyncTask是封装好了的,可以直接拿来用,如果你仅执行独立的异步任务,可以仅实现doInBackground()。所以,当有一个非常独立的任务时,可以考虑使用Thread,其他时候,尽可能的用 AsyncTask。-

⑵ 如何用队列管理AsyncTask

我最近在写一个Android文件下载的模块,具体的实现是这样的。当用户选择多个文件或着文件夹后,获取到选择的文件路径,并放到一个数组中。使用循环遍历这个数组,为数组中的每一个文件创建一个AsyncTask实例(为每个文件开辟一个新线程),并执行它的execute方法。为了测试这个模块,我下载了20张图片,每张图片1兆左右,结果图片可以稳定地下载(只不过UI更新进度有问题,有的文件已经下载完毕却还显示的是下载到了百分之多少)。问题是,假如下载的不是20张图片,而是2000张图片的话,按照上面的做法,它会开启2000个线程。虽然Android SDK默认只执行前五个线程,其余线程处于等待。但是2000个AsyncTask实例已经调用了execute方法,系统已经分配了资源,所以我担心这样会拖慢系统性能,甚至内存溢出。既然这样设计是有缺陷的,为了避免这个问题,那么我想使用队列来管理这2000个AsyncTask实例,先拿出前5个执行,待执行完毕后,队列中的等待的实例再出列并调用它的execute方法。我对AsyncTask不熟悉,不知道这样想对不对呢?还望前辈们多多指点。method for add download task?12345678910111213 /** * Add a new download task */public int addDownloadTask(Account account, String repoName, String repoID, String path) { // omit lines… DownloadTask task = new DownloadTask(account, repoName, repoID, path); // execute download task serially task.execute(); return task.getTaskID();}call addDownloadTask in a loop ?12345678910111213141516 for (SeafDirent seafDirent : dirents) { if (!seafDirent.isDir()) { File localCachedFile = dataManager. getLocalCachedFile(repoName, repoID, seafDirent.name), seafDirent.id); if (localCachedFile == null) { txService.addDownloadTask(account, repoName, repoID, Utils.pathJoin(filePath, seafDirent.name)); } } }

⑶ 安装QQ时提示无法启动程序,计算机丢失asynctask.dll文件

解决方法如下:

1,.通过网络搜索下载一个asynvtask.dll文件。

⑷ AsyncTaskLoader和AsyncTask求教

没错,我现在发现这样一个问题我在处理下载的时候service中new 多个thread的话,如果个数比较多,就容易死掉某些线程,甚至崩溃,偶尔可能产生ANR。如果是asynctask的话,就没有这个情况。

⑸ 电脑出现了Asynctask. dll 打不开了

是QQ文件,可以卸载QQ再重新安装。

⑹ AsyncTask中的isCancelled是什么意思

你好很高兴为你解答答案是:就是任务被取消了 。 异步任务是可以取消的,比如你在下载文件,觉得下载的太慢,取消了下载,就可以将异步任务取消掉。满意请采纳,谢谢?

⑺ ASYnctask.dll是什么意思,电脑显示计算机中丢失这个。怎么办

某个软件的一个动态库文件。显示丢失,那么可以通过重新安装对应的软件即可修复。

⑻ android.os.asynctask需要什么jar包

在开发Android移动客户端的时候往往要使用多线程来进行操作,我们通常会将耗时的操作放在单独的线程执行,避免其占用主线程而给用户带来不好的用户体验。但是在子线程中无法去操作主线程(UI 线程),在子线程中操作UI线程会出现错误。因此android提供了一个类Handler来在子线程中来更新UI线程,用发消息的机制更新UI界面,呈现给用户。这样就解决了子线程更新UI的问题。但是费时的任务操作总会启动一些匿名的子线程,太多的子线程给系统带来巨大的负担,随之带来一些性能问题。因此android提供了一个工具类AsyncTask,顾名思义异步执行任务。这个AsyncTask生来就是处理一些后台的比较耗时的任务,给用户带来良好用户体验的,从编程的语法上显得优雅了许多,不再需要子线程和Handler就可以完成异步操作并且刷新用户界面。 先大概认识下Android.os.AsyncTask类: * android的类AsyncTask对线程间通讯进行了包装,提供了简易的编程方式来使后台线程和UI线程进行通讯:后台线程执行异步任务,并把操作结果通知UI线程。 * AsyncTask是抽象类.AsyncTask定义了三种泛型类型 Params,Progress和Result。 * Params 启动任务执行的输入参数,比如HTTP请求的URL。 * Progress 后台任务执行的百分比。 * Result 后台执行任务最终返回的结果,比如String,Integer等。 * AsyncTask的执行分为四个步骤,每一步都对应一个回调方法,开发者需要实现这些方法。 * 1) 继承AsyncTask * 2) 实现AsyncTask中定义的下面一个或几个方法 * onPreExecute(), 该方法将在执行实际的后台操作前被UI 线程调用。可以在该方法中做一些准备工作,如在界面上显示一个进度条,或者一些控件的实例化,这个方法可以不用实现。 * doInBackground(Params…), 将在onPreExecute 方法执行后马上执行,该方法运行在后台线程中。这里将主要负责执行那些很耗时的后台处理工作。可以调用 publishProgress方法来更新实时的任务进度。该方法是抽象方法,子类必须实现。 * onProgressUpdate(Progress…),在publishProgress方法被调用后,UI 线程将调用这个方法从而在界面上展示任务的进展情况,例如通过一个进度条进行展示。 * onPostExecute(Result), 在doInBackground 执行完成后,onPostExecute 方法将被UI 线程调用,后台的计算结果将通过该方法传递到UI 线程,并且在界面上展示给用户. * onCancelled(),在用户取消线程操作的时候调用。在主线程中调用onCancelled()的时候调用。为了正确的使用AsyncTask类,以下是几条必须遵守的准则: 1) Task的实例必须在UI 线程中创建 2) execute方法必须在UI 线程中调用 3) 不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params…), onProgressUpdate(Progress…)这几个方法,需要在UI线程中实例化这个task来调用。 4) 该task只能被执行一次,否则多次调用时将会出现异常 doInBackground方法和onPostExecute的参数必须对应,这两个参数在AsyncTask声明的泛型参数列表中指定,第一个为doInBackground接受的参数,第二个为显示进度的参数,第第三个为doInBackground返回和onPostExecute传入的参数。下面通过一个Demo来说明如何使用Android.os.AsyncTask类,通过进度条来显示进行的进度,然后用TextView来显示进度值。程序结构图如下:[1] \layout\main.xml 布局文件源码如下:[html] view plain<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <TextView android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="Hello , Welcome to Andy's Blog!"/> <Button android:id="@+id/download" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="Download"/> <TextView android:id="@+id/tv" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="当前进度显示"/> <ProgressBar android:id="@+id/pb" android:layout_width="fill_parent" android:layout_height="wrap_content" style="?android:attr/progressBarStyleHorizontal"/> </LinearLayout> [html] view plain <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <TextView android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="Hello , Welcome to Andy's Blog!"/> <Button android:id="@+id/download" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="Download"/> <TextView android:id="@+id/tv" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="当前进度显示"/> <ProgressBar android:id="@+id/pb" android:layout_width="fill_parent" android:layout_height="wrap_content" style="?android:attr/progressBarStyleHorizontal"/> </LinearLayout> [2] /src中的MainActivity.java源码如下:[html] view plainpackage com.andyidea.demo; import android.app.Activity; import android.os.AsyncTask; import android.os.Bundle; import android.view.View; import android.widget.Button; import android.widget.ProgressBar; import android.widget.TextView; public class MainActivity extends Activity { Button download; ProgressBar pb; TextView tv; /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); pb=(ProgressBar)findViewById(R.id.pb); tv=(TextView)findViewById(R.id.tv); download = (Button)findViewById(R.id.download); download.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { DownloadTask dTask = new DownloadTask(); dTask.execute(100); } }); } class DownloadTask extends AsyncTask<Integer, Integer, String>{ //后面尖括号内分别是参数(例子里是线程休息时间),进度(publishProgress用到),返回值 类型 @Override protected void onPreExecute() { //第一个执行方法 super.onPreExecute(); } @Override protected String doInBackground(Integer… params) { //第二个执行方法,onPreExecute()执行完后执行 for(int i=0;i<=100;i++){ pb.setProgress(i); publishProgress(i); try { Thread.sleep(params[0]); } catch (InterruptedException e) { e.printStackTrace(); } } return "执行完毕"; } @Override protected void onProgressUpdate(Integer… progress) { //这个函数在doInBackground调用publishProgress时触发,虽然调用时只有一个参数 //但是这里取到的是一个数组,所以要用progesss[0]来取值 //第n个参数就用progress[n]来取值 tv.setText(progress[0]+"%"); super.onProgressUpdate(progress); } @Override protected void onPostExecute(String result) { //doInBackground返回时触发,换句话说,就是doInBackground执行完后触发 //这里的result就是上面doInBackground执行后的返回值,所以这里是"执行完毕" setTitle(result); super.onPostExecute(result); } } } [html] view plain package com.andyidea.demo; import android.app.Activity; import android.os.AsyncTask; import android.os.Bundle; import android.view.View; import android.widget.Button; import android.widget.ProgressBar; import android.widget.TextView; public class MainActivity extends Activity { Button download; ProgressBar pb; TextView tv; /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); pb=(ProgressBar)findViewById(R.id.pb); tv=(TextView)findViewById(R.id.tv); download = (Button)findViewById(R.id.download); download.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { DownloadTask dTask = new DownloadTask(); dTask.execute(100); } }); } class DownloadTask extends AsyncTask<Integer, Integer, String>{ //后面尖括号内分别是参数(例子里是线程休息时间),进度(publishProgress用到),返回值 类型 @Override protected void onPreExecute() { //第一个执行方法 super.onPreExecute(); } @Override protected String doInBackground(Integer… params) { //第二个执行方法,onPreExecute()执行完后执行 for(int i=0;i<=100;i++){ pb.setProgress(i); publishProgress(i); try { Thread.sleep(params[0]); } catch (InterruptedException e) { e.printStackTrace(); } } return "执行完毕"; } @Override protected void onProgressUpdate(Integer… progress) { //这个函数在doInBackground调用publishProgress时触发,虽然调用时只有一个参数 //但是这里取到的是一个数组,所以要用progesss[0]来取值 //第n个参数就用progress[n]来取值 tv.setText(progress[0]+"%"); super.onProgressUpdate(progress); } @Override protected void onPostExecute(String result) { //doInBackground返回时触发,换句话说,就是doInBackground执行完后触发 //这里的result就是上面doInBackground执行后的返回值,所以这里是"执行完毕" setTitle(result); super.onPostExecute(result); } } }

⑼ 怎么用AsyncTask更新ui 然后切换Activity了再切换回来保持进度…

我有个想法,就是重载AsyncTask类,增加一个当前进度的属性,然后再写个当前所有异步的栈,每次onCreate的时候,重新去栈里读取listview中每一个的进度,这样,异步是不从属与activity的。。。你看行吗?

⑽ 关于AsyncTask嵌套

你可以自己定义一个函数:void startMyTask(AsyncTask asyncTask) { if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) asyncTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, params); else asyncTask.execute(params);}只要android的版本新于HONEYCOMB, 你可以同时在一个THREAD POOL 里运行两个或更多的 asyncTask。