ios android多线程区别,【Android和iOS】快速切换到主线程更新UI

之前腾讯笔试有一道题目就是,请写出在子线程中更新UI的几种办法?

在实际项目中,我们也经常会遇到开辟子线程做事情的途中需要更新一下UI,这里总结一下Android和iOS快速切换到主线程更新UI的办法。

Android

方法一:view.post(Runnable action)

这是我认为最简单的方法了,比如你在子线程获得了多个数据,需要更新textview显示这些数据,可以这样做

textView.post(new Runnable() {

@Override

public void run() {

textView.setText("更新啦!");

//还可以更新其他的控件

imageView.setBackgroundResource(R.drawable.update);

}

});

这是view自带的方法,比较简单,如果你的子线程里可以得到要更新的view的话,可以用此方法进行更新。

view还有一个方法view.postDelayed(Runnable action, long delayMillis)用来延迟发送。

方法二:activity.runOnUiThread(Runnable action)

这是我认为第二简单的方法了,一般我的上下文(context)是大部分类都会传到的,而这个 context 其实就是我的 MainActivity,我会直接强制转换成 Activity 然后用activity.runOnUiThread(Runnable action)方法进行更新UI

/**

* 假设该更新方法在子线程中运行

* @param context 上下文

* /

public void update(final Context context) {

((MainActivity) context).runOnUiThread(new Runnable() {

@Override

public void run() {

//已在主线程中,可以更新UI

}

});

}

如果没有上下文(context)怎么办?

用view.getContext()可以得到上下文(不过你为什么不直接用方法一呢?)

跳过context直接用new Activity().runOnUiThread(Runnable action)来切换到主线程。

方法三:Handler

Handler 是最常用也是比上面稍微复杂一点的方法。

首先在主线程中定义Handler,Handler mainHandler = new Handler();(必须要在主线程中定义才能操作主线程,如果想在其他地方定义声明时要这样写Handler mainHandler = new Handler(Looper.getMainLooper()),来获取主线程的 Looper 和 Queue )

获取到 Handler 后就很简单了,用handler.post(Runnable r)方法把消息处理放在该 handler 依附的消息队列中(也就是主线程消息队列),这也是为什么我们第一步一定要获取主线程的 handler,如果在子线程中直接声明 handler,调用handler.post(Runnable r)其实还是在子线程中调用

//假设已在子线程

Handler mainHandler = new Handler(Looper.getMainLooper());

mainHandler.post(new Runnable() {

@Override

public void run() {

//已在主线程中,可以更新UI

}

});

Handler的方法稍微多一点

postAtTime(Runnable r, long uptimeMillis); //在某一时刻发送消息

postAtDelayed(Runnable r, long delayMillis); //延迟delayMillis毫秒再发送消息

postAtFrontOfQueue(Runnable r); //把消息放在队列的最前面(希望最先执行该消息)--这个方法慎用,可能会报错,因为有可能Looper循环到这个消息队列正在退出,准备进行下一轮消息循环

其实一般 Handler 是和 Message 一起使用的。

//假设在主线程中

Handler myHandler = new Handler() {

@Override

public void handleMessage(Message msg) {

switch(msg.what) {

case 0:

//xxx操作

break;

case 1:

//yyy操作

break;

default:

break;

}

}

}

之后可以把 mainHandler 当做参数传递在各个类之间,当需要更新UI时,可以调用sendMessage一系列方法来执行handleMessage里的操作。

//假设现在在子线程了

//获取消息

Message msg = myHandler.obtainMessage();

msg.what = 0; //消息标识

//msg.arg1 用来提供额外int型参数

//msg.obj 用来提供额外对象参数

//发送消息

myHandler.sendMessage(msg);

如上代码,只是发送了个消息标识,并没有传其他参数(可以用msg.arg1 和 msg.arg2用来提供额外int型参数,用msg.obj 用来提供额外对象参数),可以用简化方法sendEmptyMessage(int what)来减少不必要的代码

myHandler.sendEmptyMessage(0); //其实内部实现还是和上面一样

发送消息的其他方法有

sendEmptyMessageAtTime(int what, long uptimeMillis); //定时发送空消息

sendEmptyMessageDelayed(int what, long delayMillis); //延时发送空消息

sendMessageAtFrontOfQueue(Message msg); //最先处理消息(慎用)

sendMessageAtTime(Message msg, long uptimeMillis); //定时发送消息

sendMessageDelayed(Message msg, long delayMillis); //延时发送消息

iOS

方法一:[object performSelectorOnMainThread: withObject: waitUntilDone:]

属于NSObject的实例方法——也就是说所有对象都能够使用该方法达到快速切换到主线程更新UI的效果!(一般在 ViewController 中就可以直接用 self 调用)

- (void)xxx{

NSObject *object = [[NSObject alloc] init];

[object performSelectorOnMainThread:@selector(updateUI) withObject:nil waitUntilDone:false];

}

- (void)updateUI {

//更新UI

}

参数列表:

performSelectorOnMainThread:执行哪个方法

withObject:可以带一个参数(nil不带参)

waitUntilDone:是否要等待函数执行结束再继续

NSObject 以 “performSelector”开头的实例方法还有很多,再讲一个[object performSelector: onThread: withObject: waitUntilDone:]也可以实现切换到主线程,onThread:后跟要在哪个线程上执行 selector,而 NSThread 有个类方法[NSThread mainThread]就是用来得到主线程的,所以也可以这么写:

[object performSelector:@selector(updateUI) onThread:[NSThread mainThread] withObject:nil waitUntilDone:false];

方法二:GCD

GCD提供一个特殊的dispatch queue,可以在应用的主线程中执行任务。只要应用主线程设置了run loop(由CFRunLoopRef类型或NSRunLoop对象管理),就会自动创建这个queue,并且最后会自动销毁。

调用dispatch_get_main_queue函数获得应用主线程的dispatch queue,添加到这个queue的任务由主线程串行化执行

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

//子线程下载

...

dispatch_sync(dispatch_get_main_queue(), ^{

//主线程更新

...

});

});

如果你有任何问题,欢迎留言告诉我!~

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值