我可以打电话Snackbar.make()从后台线程没有任何问题。这让我感到惊讶,因为我认为 UI 操作只能从 UI 线程进行。但这里绝对不是这种情况。
究竟是什么Snackbar.make()不同的?当您从后台线程修改它时,为什么这不会像任何其他 UI 组件一样导致异常?
最佳答案
首先:make()不执行任何 UI 相关的操作,它只是创建一个新的 Snackbar实例。这是给show()的电话这实际上添加了 Snackbar到 View 层次结构并执行其他危险的 UI 相关任务。但是,您可以从任何线程安全地执行此操作,因为它被实现为在 UI 线程上调度任何显示或隐藏操作,而不管哪个线程调用 show()。 .
要获得更详细的答案,让我们仔细看看 Snackbar 的源代码中的行为。 :
让我们从一切开始,调用 show() :
public void show() {
SnackbarManager.getInstance().show(mDuration, mManagerCallback);
}
show() 的调用获取 SnackbarManager 的实例然后将持续时间和回调传递给它。 SnackbarManager是单例。它是负责显示、调度和管理 Snackbar 的类。 .现在让我们继续执行 show()在 SnackbarManager :public void show(int duration, Callback callback) {
synchronized (mLock) {
if (isCurrentSnackbarLocked(callback)) {
// Means that the callback is already in the queue. We'll just update the duration
mCurrentSnackbar.duration = duration;
// If this is the Snackbar currently being shown, call re-schedule it's
// timeout
mHandler.removeCallbacksAndMessages(mCurrentSnackbar);
scheduleTimeoutLocked(mCurrentSnackbar);
return;
} else if (isNextSnackbarLocked(callback)) {
// We'll just update the duration
mNextSnackbar.duration = duration;
} else {
// Else, we need to create a new record and queue it
mNextSnackbar = new SnackbarRecord(duration, callback);
}
if (mCurrentSnackbar != null && cancelSnackbarLocked(mCurrentSnackbar,
Snackbar.Callback.DISMISS_EVENT_CONSECUTIVE)) {
// If we currently have a Snackbar, try and cancel it and wait in line
return;
} else {
// Clear out the current snackbar
mCurrentSnackbar = null;
// Otherwise, just show it now
showNextSnackbarLocked();
}
}
}
synchronized围绕此阻止确保调用 show() 的线程安全.synchronized阻止经理负责关闭当前显示的 Snackbars如果您 show() 更新持续时间或重新安排两次相同的,当然创建新的 Snackbars .每个Snackbar SnackbarRecord创建的包含最初传递给 SnackbarManager 的两个参数,持续时间和回调:mNextSnackbar = new SnackbarRecord(duration, callback);
showNextSnackbarLocked() .这是魔法发生的地方,下一个 Snackbar 排队 - 至少是这样。showNextSnackbarLocked()的源代码:private void showNextSnackbarLocked() {
if (mNextSnackbar != null) {
mCurrentSnackbar = mNextSnackbar;
mNextSnackbar = null;
final Callback callback = mCurrentSnackbar.callback.get();
if (callback != null) {
callback.show();
} else {
// The callback doesn't exist any more, clear out the Snackbar
mCurrentSnackbar = null;
}
}
}
mNextSnackbar 来检查 Snackbar 是否已排队。不为空。如果不是,我们设置 SnackbarRecord作为当前 Snackbar并从记录中检索回调。现在发生了一些事情,在进行简单的空检查以查看回调是否有效之后,我们调用 show()在回调上,这是在 Snackbar 中实现的类 - 不在 SnackbarManager - 实际显示 Snackbar屏幕上。SnackbarManager只负责跟踪Snackbars的状态并协调它们,它不在乎如何Snackbar看起来,它是如何显示的,甚至是什么,它只是调用 show()方法在正确的时间回调告诉Snackbar来展示自己。synchronized阻止在 show() SnackbarManager的方法确保没有其他 Thread可以干扰我们所做的一切,但主要是什么安排节目和解散事件Thread仍然失踪。然而,当我们查看 Snackbar 中回调的实现时,这将立即改变。类(class):private final SnackbarManager.Callback mManagerCallback = new SnackbarManager.Callback() {
@Override
public void show() {
sHandler.sendMessage(sHandler.obtainMessage(MSG_SHOW, Snackbar.this));
}
@Override
public void dismiss(int event) {
sHandler.sendMessage(sHandler.obtainMessage(MSG_DISMISS, event, 0, Snackbar.this));
}
};
MSG_SHOW显示 Snackbar或 MSG_DISMISS再次隐藏它。 Snackbar本身作为有效负载附加到消息中。现在我们一看静态处理程序的声明就差不多完成了:private static final Handler sHandler;
private static final int MSG_SHOW = 0;
private static final int MSG_DISMISS = 1;
static {
sHandler = new Handler(Looper.getMainLooper(), new Handler.Callback() {
@Override
public boolean handleMessage(Message message) {
switch (message.what) {
case MSG_SHOW:
((Snackbar) message.obj).showView();
return true;
case MSG_DISMISS:
((Snackbar) message.obj).hideView(message.arg1);
return true;
}
return false;
}
});
}
Looper.getMainLooper() 所示)。消息的有效载荷 - Snackbar - 被强制转换,然后取决于消息的类型 showView()或 hideView()在 Snackbar 上被调用. 这两个方法现在都在 UI 线程上执行! View到 View 层次结构,在它出现和消失时对其进行动画处理,处理 CoordinatorLayout.Behaviours和其他关于 UI 的东西。关于java - 从非 UI 线程调用 Snackbar.make() 是如何工作的?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37510565/
我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
我正在使用i18n从头开始构建一个多语言网络应用程序,虽然我自己可以处理一大堆yml文件,但我说的语言(非常)有限,最终我想寻求外部帮助帮助。我想知道这里是否有人在使用UI插件/gem(与django上的django-rosetta不同)来处理多个翻译器,其中一些翻译器不愿意或无法处理存储库中的100多个文件,处理语言数据。谢谢&问候,安德拉斯(如果您已经在rubyonrails-talk上遇到了这个问题,我们深表歉意) 最佳答案 有一个rails3branchofthetolkgem在github上。您可以通过在Gemfi
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-
给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru
我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t
我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚
Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack
在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/