- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我想知道 Looper 类实际上是如何处理 Looper 附加到的 Thread 中的 Runnables(通过 Handler 类)的?如果一个 looper 在它的 messageQueue 中循环,那么它肯定是那个线程的阻塞操作?我想它本身一定在执行一些线程技巧,但它如何将发布的 Runnables run() 方法添加到主机线程堆栈中?
很多问题!任何帮助将非常感激。谢谢!
编辑:
通过 Looper 查看 class file我看到下面的类,这让我更加困惑,因为所有评论都提到了在主线程中运行的循环程序,而且它在等待 MessageQueue 上的新消息时也是一个阻塞操作。这怎么不阻塞 UI/主线程???
/**
* Run the message queue in this thread. Be sure to call
* {@link #quit()} to end the loop.
*/
public static final void loop() {
Looper me = myLooper();
MessageQueue queue = me.mQueue;
// Make sure the identity of this thread is that of the local process,
// and keep track of what that identity token actually is.
Binder.clearCallingIdentity();
final long ident = Binder.clearCallingIdentity();
while (true) {
Message msg = queue.next(); // might block
//if (!me.mRun) {
// break;
//}
if (msg != null) {
if (msg.target == null) {
// No target is a magic identifier for the quit message.
return;
}
if (me.mLogging!= null) me.mLogging.println(
">>>>> Dispatching to " + msg.target + " "
+ msg.callback + ": " + msg.what
);
msg.target.dispatchMessage(msg);
if (me.mLogging!= null) me.mLogging.println(
"<<<<< Finished to " + msg.target + " "
+ msg.callback);
// Make sure that during the course of dispatching the
// identity of the thread wasn't corrupted.
final long newIdent = Binder.clearCallingIdentity();
if (ident != newIdent) {
Log.wtf("Looper", "Thread identity changed from 0x"
+ Long.toHexString(ident) + " to 0x"
+ Long.toHexString(newIdent) + " while dispatching to "
+ msg.target.getClass().getName() + " "
+ msg.callback + " what=" + msg.what);
}
msg.recycle();
}
}
}
/**
* Returns the application's main looper, which lives in the main thread of the application.
*/
public synchronized static final Looper getMainLooper() {
return mMainLooper;
}
/**
* Return the Looper object associated with the current thread. Returns
* null if the calling thread is not associated with a Looper.
*/
public static final Looper myLooper() {
return (Looper)sThreadLocal.get();
}
// sThreadLocal.get() will return null unless you've called prepare().
private static final ThreadLocal sThreadLocal = new ThreadLocal();
/** Initialize the current thread as a looper.
* This gives you a chance to create handlers that then reference
* this looper, before actually starting the loop. Be sure to call
* {@link #loop()} after calling this method, and end it by calling
* {@link #quit()}.
*/
public static final void prepare() {
if (sThreadLocal.get() != null) {
throw new RuntimeException("Only one Looper may be created per thread");
}
sThreadLocal.set(new Looper());
}
最佳答案
你完全正确。调用构造 Looper.start() 的原始线程成为处理所有已发布的 Runnable 和消息的线程。这确实意味着这个线程被阻塞了。
关于Android Looper 和调用栈,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6984263/
我的应用程序有问题。我需要在特定时间间隔获取位置更新,因此基本上需要能够很好地控制 GPS 模块,这对于 Android 操作系统来说并不容易。基本上我需要以 5 分钟的间隔打开 GPS 2 分钟。我
考虑以下 fragment : Looper.prepare(); handler = new Handler() { @Override public void ha
在 Android 中,如果是 UI 线程,我们只需创建 Handler,因为主 UI 线程已经有了它的 Looper。 我想知道在框架代码中为 UI 线程调用 Looper.prepare 和 Lo
有人可以简单地向我解释一下以下术语以及它们之间的关系吗? handler - 每个线程都有一个处理程序?每个 View 都有一个处理程序?执行post()。操作系统主线程有默认处理程序吗? 循环器 -
我正在创建一个线程来处理请求。在线程中,我必须调用 Looper.prepare() ,这是它正在使用的其他一些功能所需要的,然后我调用 Looper.loop() 。但是当请求的连接关闭时,我会在不
所以我有这个循环器,我用它来执行长时间运行的任务。我将 Worker 对象传递给它,它本质上是一个 Runnable 的包装器 我注意到它似乎正在泄漏大小完全相同的 Message 对象 知道为什么会
我试图理解 android 中的循环器和处理程序,但被编写的示例卡住了。我想做的是,向线程添加一个循环程序,使线程在 run() 方法中连续运行。然后将消息或 runnables 发布到 hanlde
我有一个在互联网上请求数据信息的应用程序(客户端-服务器应用程序),但是这种通信非常慢,因此我决定创建一个 AsyncTask 来管理延迟。在 doInBackground 中,我调用了 Looper
我想知道 Looper 类实际上是如何处理 Looper 附加到的 Thread 中的 Runnables(通过 Handler 类)的?如果一个 looper 在它的 messageQueue 中循
我在使用 Android looper 时遇到问题。我有一个扩展了 AsynTask 的类。在 doInBackground() 方法中,我有 Looper.prepare() 和下面的一些代码。 它
我有一个线程,用于定期更新 Activity 中的数据。我创建了线程并启动了一个循环程序,以便将处理程序与 postDelay() 一起使用。在我的 Activity 的 onDestroy() 中,
我是 Android 新手。我想知道 Looper 类的作用以及如何使用它。我已阅读 Android Looper class documentation但我无法完全理解它。我在很多地方都见过它,但无
我有一个帮助类,用于在数据库 Realm 做一些工作。如您所知,我们在使用 Realm 时有一些限制,例如: Realm 实例不会在非循环线程上自动刷新。 Realm 对象只能在创建它们的线程上访问
我有一个子线程正在运行以无限地执行任务。我想(1)不断将数据发送回UI线程,并且(2)偶尔将数据(对应于按钮)发送到子线程以暂停/继续无限任务。我的问题是子线程卡在了循环程序中,这意味着任务无法执行。
这个问题已经有答案了: Can't create handler inside thread that has not called Looper.prepare() (30 个回答) 已关闭 1 年
我在这段代码中循环遍历对象的attaylist并为每个对象分配一个回调,但是当我开始使用CountDownTimer时,它崩溃了Can't create handler inside thread t
我的 Android 应用程序允许用户从 Fragment 更新他们的 Facebook 状态。我已经实现了以下 AsyncTask,以便他们可以登录: public class update
这个问题已经有答案了: Can't create handler inside thread that has not called Looper.prepare() (30 个回答) Threadi
我正在使用附加到 Android HandlerThread 的 Looper执行长时间运行的异步任务: HandlerThread handlerThread = new HandlerThread
我已经尝试了 mContext.getMainLooper() 和 Looper.getMainLooper()。两者都返回相同的结果,但我想知道哪种方法正确? 我还从 Android 开发人员链接中
我是一名优秀的程序员,十分优秀!