在我的应用程序的最新版本中,一些用户遇到了我无法重现的崩溃。目前只有运行 Lollipop 的 Samsung 设备有问题,但这可能只是巧合。
在分析了堆栈跟踪和相关代码之后,我认为我可能找到了罪魁祸首。为了检验我的假设,我将代码简化为以下代码 fragment :
public class TestActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Button b = new Button(this);
b.setText("Click me!");
b.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
new Handler().post(new Runnable() {
@Override
public void run() {
// This is the callback method
Log.d("TAG", "listenerNotified");
}
});
}
});
setContentView(b);
}
@Override
protected void onDestroy() {
super.onDestroy();
Log.d("TAG", "onDestroy");
}
}
每次我通过先点击Click me 按钮然后点击后退按钮来测试上述应用程序时,listenerNotified 会在 onDestroy() 之前打印到控制台。
但是我不确定我是否可以依赖这种行为。 Android 是否对上述场景做出任何保证?我是否可以安全地假设我的 Runnable 将始终在 onDestroy() 之前执行,或者是否存在不会发生这种情况的情况?在我的真实应用程序中,当然发生了更多事情(比如其他线程发布到主线程以及回调中发生的更多操作)。但这个简单的 fragment 似乎足以证明我的担忧。
是否有可能(可能是由于其他线程的影响或发布到主线程的回调)我得到下面的调试输出?
D/TAG: onDestroy
D/TAG: listenerNotified
我想知道这一点,因为可能的结果可以解释崩溃。
最佳答案
Is it possible for a callback method to be called after
onDestroy()?
是的。
让我们稍微更改一下关于将 Runnable 发送到 Handler 的示例代码。我还假设(根据您的描述)您可能有多个 Runnable 发布到主线程,因此在某些时候可能会有一个 Runnable 队列,这会带来我延迟了下面的实验:
public void onClick(View view) {
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
// This is the callback method
Log.d("TAG", "listenerNotified");
}
}, 3000);
}
现在按下按钮 b,然后按下后退按钮,您应该会看到有问题的输出。
这可能是您的应用程序崩溃的原因吗? 如果没有看到您得到的结果,很难说。我只想指出,当 new Handler() 在线程(您的情况下是主线程)上实例化时,Handler 与 相关联Looper线程的消息队列,发送和处理Runnable和来自队列的消息。这些 Runnable 和消息具有对目标 Handler 的引用。尽管 Activity 的 onDestroy() 方法不是“析构函数”,即当该方法返回 Activity 的实例时, t 被立即终止(see),由于隐式引用* 到 Activity,内存不能被 GC-ed。在 Runnable 从 Looper 的消息队列中出队并得到处理之前,您将一直处于泄漏状态。
可以在How to Leak a Context: Handlers & Inner Classes 上找到更详细的解释。
* 匿名内部类 Runnable 的实例引用了匿名内部类 View.OnClickListener 的实例,而后者又引用了Activity 实例。
关于java - onDestroy后是否可以调用回调方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39562914/
我正在学习如何使用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但我想要一些方法来使用
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
我正在尝试设置一个puppet节点,但rubygems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由rubygems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby
给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru
我想了解Ruby方法methods()是如何工作的。我尝试使用“ruby方法”在Google上搜索,但这不是我需要的。我也看过ruby-doc.org,但我没有找到这种方法。你能详细解释一下它是如何工作的或者给我一个链接吗?更新我用methods()方法做了实验,得到了这样的结果:'labrat'代码classFirstdeffirst_instance_mymethodenddefself.first_class_mymethodendendclassSecond使用类#returnsavailablemethodslistforclassandancestorsputsSeco
使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta
查看Ruby的CSV库的文档,我非常确定这是可能且简单的。我只需要使用Ruby删除CSV文件的前三列,但我没有成功运行它。 最佳答案 csv_table=CSV.read(file_path_in,:headers=>true)csv_table.delete("header_name")csv_table.to_csv#=>ThenewCSVinstringformat检查CSV::Table文档:http://ruby-doc.org/stdlib-1.9.2/libdoc/csv/rdoc/CSV/Table.html
我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer
设置:狂欢ruby1.9.2高线(1.6.13)描述:我已经相当习惯在其他一些项目中使用highline,但已经有几个月没有使用它了。现在,在Ruby1.9.2上全新安装时,它似乎不允许在同一行回答提示。所以以前我会看到类似的东西:require"highline/import"ask"Whatisyourfavoritecolor?"并得到:Whatisyourfavoritecolor?|现在我看到类似的东西:Whatisyourfavoritecolor?|竖线(|)符号是我的终端光标。知道为什么会发生这种变化吗? 最佳答案