移动开发iOS&Android对比学习--异步处理

在移动开发里很多时候需要用到异步处理。Android的主线程如果等待超过一定时间的时候直接出现ANR(对不熟悉Android的朋友这里需要解释一下什么叫ANR。ANR就是Application Not Responding,应用无响应的意思。系统在应用一段时间无响应的时候会弹出这个对话框。用户可以选择继续等待或者强制关闭)。这些还是次要的,最主要的还是心急的用户。让用户长时间等待是得罪他们的最好办法!

Android有一个很简单的办法实现异步处理:AnsyncTask。使用的时候你需要继承一个基类

public abstract class AsyncTask<Params, Progress, Result>

对java不熟的同学这里需要说明,尖括号里的是类型参数。这是java的一个语法,泛型。这三个参数分别指定的是输入参数的类型、任务执行进度值的类型和任务执行结果的类型。并不是所有的类型都被使用,不用的时候指定为void类型。

最简单的使用就是继承这个类以后Override方法doInBackground(Params... params)。使用的时候实例化你的AsyncTask实现,调用execute(Params... params)方法就开启了异步操作。例如:

布局代码:

<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="@+id/executeButton"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text=" Start "/>
    <ScrollView
        android:layout_width="fill_parent"
        android:layout_height="wrap_content">
        <TextView
            android:id="@+id/textView"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"/>
    </ScrollView> 

</LinearLayout>

 

java代码:

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

public class AsyncTaskDemo extends Activity {

    private Button executeButton;
    private TextView textView; 

    @Override
    public void onCreate(Bundle savedInstanceState){
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_demo);

        executeButton = (Button)findViewById(R.id.executeButton);
        textView = (TextView)findViewById(R.id.textView);

        executeButton.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                Log.i("-task-","Task started.");
                new MyAsyncTask().execute(" on click");
            }
        });
    }

    class MyAsyncTask extends AsyncTask<String, Integer, String>{

        @Override
        public String doInBackground(String... params){
            Log.i("-task-","doInBackground is called" + params[0]);
            try {
                // 线程暂停2秒模拟后台任务
                Thread.sleep(2000);
            } catch (Exception e) {
                return "";
            }
            return "Hello World from AsyncTask";
        }

        // 在doInBackground方法执行之后调用这个方法
        public void onPostExecute(String result){
            Log.i("-task-", "doPostExecute is called");
            textView.setText(result);
        }
    }
}

 

看到这里读者应该明白,doInBackground方法就是在后台执行的那个异步的方法。

但是AsyncTask的妙处不至于此。大家都知道,在UI线程意外不可以更新UI组件。如果后台线程执行完后一定要通知UI线程,由UI线程根据后台线程的执行结果更新UI组件。AsyncTask的妙处就在于,在它的实现中实现某些方法就可以直接更新UI。当然这些方法是在UI线程中的。这些方法有:

  1. onPreExecute() 在execute方法调用后立即执行,在doInBackground执行前执行。相当于在后台操作开始前做一些准备工作。
  2. doInBackground(Params... params) 在onPreExecute之后执行,处理需要在后台操作的任务。在其中可以通过publishProgress(Progress... values)通知UI线程更新进度。
  3. onProgressUpdate(Progress... values)在调用publishProgress后此方法执行。将进度更新到UI组件上。
  4. onPostExecute(Result result) 在doInBackground执行结束之后,后台执行结果作为参数传入这个方法并根据这个结果更新UI组件。
  5. onCancelled() 取消后台操作的时候调用方法cancel(boolean endTaskDuringExecuting),这个时候onCancelled方法会被调用。同事,isCancelled()方法返回true。在后台任务的执行中需要检查是否已经被取消,尤其是循环中。这样可以保证后台任务可以尽快退出。

下面就依据上面所说更新代码了:

layout:

<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="@+id/executeButton"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text=" Start "/>
    <Button
        android:id="@+id/cancelButton"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:enabled="false"
        android:text=" cancel "/>
    <ProgressBar
        android:id="@+id/progressBar"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:progress="0"
        android:max="100"
        style="?android:attr/progressBarStyleHorizontal"/>
    <ScrollView
        android:layout_width="fill_parent"
        android:layout_height="wrap_content">
        <TextView
            android:id="@+id/textView"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"/>
    </ScrollView> 

</LinearLayout>

java:

public class AsyncTaskDemo extends Activity {

    private Button executeButton;
    private Button cancelButton;
    private TextView textView;
    private ProgressBar progressBar;

    private MyAsyncTask myAsyncTask;

    @Override
    public void onCreate(Bundle savedInstanceState){
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_demo);

        executeButton = (Button)findViewById(R.id.executeButton);
        cancelButton = (Button)findViewById(R.id.cancelButton);
        textView = (TextView)findViewById(R.id.textView);
        progressBar = (ProgressBar)findViewById(R.id.progressBar);

        executeButton.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                Log.i("-task-","Task started.");

                // 每次开始的时候必须初始化一次。
                myAsyncTask = new MyAsyncTask();
                myAsyncTask.execute(" on click");

                cancelButton.setEnabled(true);
                executeButton.setEnabled(false);
            }
        });

        cancelButton.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View arg0) {
                Log.i("-task", "Task cancelled");

                // 取消任务
                myAsyncTask.cancel(true);
            }
        });
    }

    class MyAsyncTask extends AsyncTask<String, Integer, String>{

        @Override
        protected void onPreExecute() {
            Log.i("-task-", "onPreExecute is called");
            textView.setText("Mission is about to start dude...");
        }

        @Override
        public String doInBackground(String... params){
            Log.i("-task-","doInBackground is called" + params[0]);

            if (isCancelled()) {
                return "cancelled";
            }
            try {
                // 开始进度是0.
                publishProgress(0);
                Thread.sleep(2000);
                publishProgress(30);

                // 如果有循环的话需要在循环中每次都检查是否任务已经被取消。
                if (isCancelled()) {
                    return "cancelled";
                }

                // 线程暂停2秒模拟后台任务
                Thread.sleep(2000);
                publishProgress(100);
            } catch (Exception e) {
                return "";
            }
            return "Hello World from AsyncTask";
        }

        @Override
        protected void onProgressUpdate(Integer...progresses){
            Log.i("-task-", "onProgressUpdate is called");

            progressBar.setProgress(progresses[0]);
            textView.setText(progresses[0] + "% is handled...");
        }

        // 在doInBackground方法执行之后调用这个方法
        @Override
        protected void onPostExecute(String result){
            Log.i("-task-", "doPostExecute is called");
            textView.setText(result);

            executeButton.setEnabled(true);
            cancelButton.setEnabled(false);
        }

        @Override
        protected void onCancelled(){
            Log.i("-task-", "onCancelled is called");

            progressBar.setProgress(0);
            textView.setText("Cancelled");

            executeButton.setEnabled(true);
            cancelButton.setEnabled(false);
        }
    }
}

执行结果:

1. 没有取消的

2. 取消了的

这里有一点需要注意。AsyncTask对象,一个只跑一次任务!所以,这里你可以看到,每次在执行一个后台任务的时候都要初始化一次。从这里看到,AsyncTask非常好用。只要把后台任务放到doInBackground方法里,其他的在UI线程上执行的只要按照需要放在其他的方法中就可以了。简直是随用随调,方便已用。

在iOS里最方便的就是GCD了。GCD,Grand Central Dispatch。是OSX10.6 iOS4.0引入的,是一套新的编写并行程序的方法,是基于c开发的开源框架。GCD的便携很大程度上依赖于block。block就类似于lambda表达式。所以GCD的写法非常简单。GCD的异步操作是基于线程的,只不过是有GCD去管理这些线程,以发挥多核心CPU的优势。开发者就不用花时间在具体的线程管理的细节上。

block的例子:

void (^blockDemo)(void) = ^{
      // do something here...
}

关于block更详细的知识,请到这里自行补脑。

如果说block是异步任务的子弹的话,那么那把枪就是dispatch queue。dispatch queue可以是串行执行的也可以是并行执行的。串行dispatch queue按照先进先出的顺序执行任务,并行的dispatch queue只是按照这样的方式开始任务,至于具体的执行顺序会随具体的执行条件变化。

GCD包括三中queue:

  1. Main queue:系统定义,用dispatch_get_main_queue获取。提交到这个queue的任务会在UI线程中执行。
  2. Global queue:系统定义,用dispatch_get_global_queue获取。global queue就是并行queue,一共有四种优先级。
  3. Private queue:用 dispatch_queue_create创建。默认为串行执行。可以用DISPATCH_QUEUE_CONCURRENT指定为并行队列(只在iOS5或以后的系统中有效)。

下面通过代码展示使用GCD实现异步操作有多简单:

// 创建一个private queue
dispatch_queue_t myBackgroundQueue;
myBackgroundQueue = dispatch_queue_create("com.company.subsystem.task", NULL);

// 异步处理
// Dispatch a private queue
dispatch_async(myBackgroundQueue, ^(void) {
    // do some time consuming things here
});

// Release queue。iOS6以后系统自动管理内存。
dispatch_release(myBackgroundQueue);

// 获取global queue
dispatch_queue_t myGlobalQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

// 异步操作
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0)
, ^(void) {
    // do some time consuming things here
});

只要给你的queue dispatch_async一个任务(block)就可以。

由此可见,用dispatch queue实现异步比Android的AsyncTask简单了很多。只需要简单的几句话就可以实现。不用实现抽象类什么的繁琐操作。当你调用dipatch_async后,代码立即返回分发出去的任务由系统分配资源在后台异步执行。

参考上面Android的例子,在dispatch queue执行的过程中要更新异步任务执行进度怎么办呢,要在界面上现实执行了百分之多少,progress bar要现实对应的进度怎么办呢?UI组件的更新需要在UI线程中执行。这个时候就需要用到上面说的main queue了。我们上面已经说过使用dispatch_get_main_queue方法来获取main queue也就是UI线程的queue。下面通过一个简单的例子说明在dispatch queue异步操作中如何更新UI组件。

dispatch_async(myBackgroundQueue, ^(void) {

// 异步操作在这里。。。

    dispatch_async(dispatch_get_main_queue(), ^{

        // 操作UI组件等。。。
    });
});

另外需要说明一点:如果一个变量需要在queue中操作,准确的说是在block中操作的话,需要有一个特别的声明:在常用的声明前加__block。

那么上面在Android中示例用ObjC的GCD该如何实现呢?下面给出答案。

界面布局:

头文件代码:

#import <UIKit/UIKit.h>

@interface CMRootViewController : UIViewController{
@private
    __block BOOL _cancelTask;
}

@property (assign, nonatomic, readonly) __block BOOL cancelled;
@property (strong, nonatomic) dispatch_queue_t taskQueue;

@property (weak, nonatomic) IBOutlet UIProgressView *progressBar;
@property (weak, nonatomic) IBOutlet UILabel *progressLabel;
@property (weak, nonatomic) IBOutlet UIButton *startButton;
@property (weak, nonatomic) IBOutlet UIButton *cancelButton;

- (IBAction)startAction:(id)sender;
- (IBAction)cancelAction:(id)sender;

@end

源文件:

#import "CMRootViewController.h"

@implementation CMRootViewController

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
    self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
    if (self) {
        // 初始化线程
        self.taskQueue = dispatch_queue_create("com.queue.demo", NULL);
        _cancelTask = NO;
        _cancelled = NO;

    }
    return self;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
    self.title = @"Queue Demo";
    self.progressBar.progress = .0f;
    self.progressLabel.text = @"0%";
}

- (IBAction)startAction:(id)sender {
    NSLog(@"Start button is clicked.");

    _startButton.enabled = NO;
    _cancelButton.enabled = YES;
    _cancelled = NO;
    _cancelTask = NO;

    // 这里用sleep方法暂停线程,代表那些费时的操作。
    // 在任务执行的过程中check是否已经取消了操作,尤其在循环里。这样可以尽快停止后台操作
    dispatch_async(self.taskQueue, ^{
        if (_cancelTask) {
            NSLog(@"Task cancelled");
            _cancelButton.enabled = NO;
            _startButton.enabled = YES;
            _cancelled = YES;
            return;
        }
        sleep(2);
        NSLog(@"Task: 10%% is completed.");
        dispatch_async(dispatch_get_main_queue(), ^{
            self.progressBar.progress = .1f;
            self.progressLabel.text = @"10%";
        });

        if (_cancelTask) {
            NSLog(@"Task cancelled");
            _cancelButton.enabled = NO;
            _startButton.enabled = YES;
            _cancelled = YES;
            return;
        }
        sleep(3);
        NSLog(@"Task: 60%% is completed.");
        dispatch_async(dispatch_get_main_queue(), ^{
            self.progressBar.progress = .6f;
            self.progressLabel.text = @"60%";
        });

        if (_cancelTask) {
            NSLog(@"Task cancelled");
            _cancelButton.enabled = NO;
            _startButton.enabled = YES;
            _cancelled = YES;
            return;
        }
        sleep(2);
        NSLog(@"Task: 80%% is completed.");
        dispatch_async(dispatch_get_main_queue(), ^{
            self.progressBar.progress = .8f;
            self.progressLabel.text = @"80%";
        });

        if (_cancelTask) {
            NSLog(@"Task cancelled");
            _cancelButton.enabled = NO;
            _startButton.enabled = YES;
            _cancelled = YES;
            return;
        }
        sleep(1);
        NSLog(@"Task: 100%% is completed.");
        dispatch_async(dispatch_get_main_queue(), ^{
            self.progressBar.progress = 1.f;
            self.progressLabel.text = @"100%";
        });

    });
}

- (IBAction)cancelAction:(id)sender {
    NSLog(@"Cancel button is clicked.");
    _cancelTask = YES;

    _startButton.enabled = YES;
    _cancelButton.enabled = NO;
}

@end

 

Android和iOS对比着学习可以对这两个平台的各种技术留下深刻的印象。毕竟都是移动平台,大多数的语言之外的东西有很多东西都是想通的。Android有着广大的用户群体,而且在不断的进步。所以不必把两个平台完全独立或者对立起来。对于开发者来说得到实惠才是最重要的!

 

欢迎加群互相学习,共同进步。QQ群:iOS: 58099570 | Android: 330987132 | Go:217696290 | Python:336880185 | 做人要厚道,转载请注明出处!http://www.cnblogs.com/sunshine-anycall/p/3411112.html

时间: 2024-12-30 12:01:13

移动开发iOS&Android对比学习--异步处理的相关文章

Xamarin体验:使用C#开发iOS/Android应用

原文:Xamarin体验:使用C#开发iOS/Android应用 Xamarin是Mono创始人Miguel de Icaza创建的公司,旨在让开发者可以用C#编写iOS, Android, Mac应用程序,也就是跨平台移动开发.   简介 Xamarin是基于Mono的平台,目前主要有以下产品(更具体请见:http://xamarin.com/products): Xamarin Studio:IDE,是从原来的MonoDevelop改名而来.现在从MonoDevelop官方网站下载的其实也是

Android编程学习之异步加载图片的方法_Android

本文实例讲述了Android编程学习之异步加载图片的方法.分享给大家供大家参考,具体如下: 最近在android开发中碰到比较棘手的问题,就是加载图片内存溢出.我开发的是一个新闻应用,应用中用到大量的图片,一个界面中可能会有上百张图片.开发android应用的朋友可能或多或少碰到加载图片内存溢出问题,一般情况下,加载一张大图就会导致内存溢出,同样,加载多张图片内存溢出的概率也很高. 列一下网络上查到的一般做法: 1.使用BitmapFactory.Options对图片进行压缩 2.优化加载图片的

《Android安全技术揭秘与防范》——第1章,第1.3节Android和iOS系统对比

1.3 Android和iOS系统对比 很多人喜欢拿iOS系统来与Android系统做比较.这是由于它俩是目前市面上最流行的手机操作系统.但是,我们只要从专业角度来看,会发现它们有许多不同点. iOS是由苹果公司开发的手持设备操作系统,最初是设计给iPhone使用的,后来陆续套用到iPod touch.iPad以及Apple TV等苹果产品上.它也是以Darwin为基础的,因此同样属于类UNIX的商业操作系统. Android是一种以Linux为基础的开放源码的操作系统,Android操作系统最

【跨平台开发神器】用c#开发iOS、Android 与Windows 应用,你也可以!

问题描述 TUP第三十期:移动跨平台开发,一步到位!时间:11月7日(周四)13:30-16:30地点:北京市朝阳区望京街8号利星行广场C座(微软望京306办公室)议程:1.如何用VisualStudio开发iOS.Android与Windows应用2.抢鲜预览MicrosoftVisualStudio2013IDE功能的创新与突破你熟悉VisualStudio开发平台么?你想用c#开发iOS及Android应用么?你想免费获得跨平台解决方案么?那你绝对不能错过这期"移动跨平台开发一步到位&qu

突破瓶颈,对比学习:Eclipse开发环境与VS开发环境的调试对比

曾经看了不少Java和Android的相关知识,不过光看不练易失忆,所以,还是写点文字,除了加强下记忆,也证明我曾经学过~~~ 突破瓶颈,对比学习: 学习一门语言,开发环境很重,对于VS的方形线条开发环境有先入为主的先天因素. 因此多年以来(从学校开java课),对Eclipse那椭圆形的线条,似乎有点抗拒或格格不入的情怀.   抗拒情怀其实是一种瓶颈,如果不突破,很容易被局限,不管是跨语言的水平扩展突破,还是语言内的突破,都是同理. 只要是开发者,总会遇到到瓶颈,而这瓶颈久久不能突破,就是抗拒

微软官方回应:暂不开发 iOS 或 Android 版 IE

过去一年多,微软一直在贯彻跨平台战略,比如说,面向一系列非 Windows 设备推出软件及服务.在 "移动为王" 的当下,它是否会考虑开发 iOS 或 Android 版 IE 浏览器呢?IE 团队最近给出了答复:否否否否否. 8月14日,在 Reddit 随性问(Ask Me Anything)环节,当被问到推出 iOS 或 Android 版 IE 时,一名 IE 团队成员答: "Right now, we're focused on building a great m

Swift开发iOS应用过程中的问题和解决记录

Swift开发iOS应用过程中的问题和解决记录 虚拟机里安装OSX+XCode开发环境 用真机的请直接跳过这个部分. 主要是在VitrualBox里安装mac系统和xcode,参考这篇教程,VirtualBox的版本是4.3.18 r96156,OSX版本是10.11 El Capitan,XCode版本是7.1 (7B91b). 经过几天的初步使用,感觉用虚拟机开发iOS基本能够满足要求,i5/8g/ssd的配置目测能达到真机70%的开发效率.主要存在的问题是:1)在宽屏上无法满屏,两边会留有

开发iOS应用程序前需要解决的十大问题

在本文中我们将为您展示在开发iOS平台(iPhone.iPad和iPod)应用前,您可能想问的10个问题. 1. 我们目标应该是哪个平台? 只有iOS (iPhone and iPad) 以及Android有足够的下载量.购买量以及使用量.其他平台也都有应用程序,但用户大部份都太少,或者应用程序的下载量都不足以支撑.也许未来可能会改变,但是以今日来说,这是不争的事实. 也许有人会争辩WP.Bada都可撑起一片江山,但在没有真切的放大量出来,不会建议一下子就冒进. 2. 我们需要针对不同平台来开发

服务器-Android如何在一个异步任务类结束时再次开启一个异步任务类

问题描述 Android如何在一个异步任务类结束时再次开启一个异步任务类 本人Android新手,在学习中遇到了一个难题,希望大神给我一点启发,问题是这样的,我从服务器获取JSON数据进行解析,在onPostExecute中获取了一个图片的URL,当我在onPostExecute进行图片的读取时程序报错,查找了一些资料,说需要在onPostExecute里发送一个消息在开启一个任务类,请问大神,该怎么在onPostExecute里发送一个消息通知线程开启任务类呢? 解决方案 直接在onPostE