使用Firebase时,建议使用push()命令管理数据列表。 这真的很好,它为列表中推送的数据提供了唯一且有序的ID。

但是,当Firebase脱机(goOffline或NetworkOffline)时,如果应用尝试在列表中推送数据,则不会触发完成侦听器,直到应用程序重新联机:因此在该行重新打开之前没有唯一ID 。

1 /这是预期/正常行为吗?

2 /我没有在文档中看到(据我记得),push命令在脱机状态下的工作方式不同(或仅在onlinemode中)。我在某处错过了一条线路吗?

3 /我的用例处理拥有关系的数据。 这意味着我想在列表中创建一个对象(一种主类),然后重用这个主对象id(由完成侦听器提供)来构建主对象和所有其他相关对象之间的关系。 我该如何处理这个脱机状态?

代码示例:

findViewById(R.id.button3).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { new Firebase(getString(R.string.firebase_url)).child("stack").push().setValue(counter++, new Firebase.CompletionListener() { @Override public void onComplete(FirebaseError firebaseError, Firebase firebase) { ((TextView) findViewById(R.id.textView3)).setText(String.valueOf(counter) + " - " + firebase.toString()); } }); } });

编辑

以下是添加数据的4种方法:

用Listener推送

findViewById(R.id.button3).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { new Firebase(getString(R.string.firebase_url)).child("stack").push().setValue(counter++, new Firebase.CompletionListener() { @Override public void onComplete(FirebaseError firebaseError, Firebase firebase) { ((TextView) findViewById(R.id.textView3)).setText(String.valueOf(counter) + " - " + firebase.toString()); } }); } });

带监听器的SetValue

findViewById(R.id.button4).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { new Firebase(getString(R.string.firebase_url)).child("stackManual").child(UUID.randomUUID().toString()).setValue(counter++, new Firebase.CompletionListener() { @Override public void onComplete(FirebaseError firebaseError, Firebase firebase) { ((TextView) findViewById(R.id.textView4)).setText(String.valueOf(counter) + " - " + firebase.toString()); } }); } });

没有监听器的SetValue

findViewById(R.id.button5).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Firebase temp = new Firebase(getString(R.string.firebase_url)).child("stackManual").child(UUID.randomUUID().toString()); temp.setValue(counter++); ((TextView) findViewById(R.id.textView5)).setText(String.valueOf(counter) + " - " + temp.getKey().toString()); } });

推送没有听众

findViewById(R.id.button6).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Firebase temp = new Firebase(getString(R.string.firebase_url)).child("stack").push(); temp.setValue(counter++); ((TextView) findViewById(R.id.textView6)).setText(String.valueOf(counter) + " - " + temp.getKey().toString()); } });

它出现只有当数据在服务器上的数据库时,才会触发听众,而不是在本地数据库上! – 那是对的吗 ? – 那是文件吗?

因为知道这一点,直接的异步应用程序现在更难以构建:如果离线,可能无法执行异步作业的封装,可能吗?

所以在该线再次打开之前没有唯一的ID

最后一句话不是真的。 Firebase推送ID是在客户端生成的,并且在统计上保证是唯一的。

您可以通过拆分操作轻松检查:

Firebase ref = new Firebase(getString(R.string.firebase_url)); Firebase newStackRef = ref.child("stack").push(); System.out.println("New key/push id: "+newStackRef.key()); newStackRef.setValue(counter++, new Firebase.CompletionListener() {

即使您未连接到网络,日志记录输出也会显示新的推送ID。

它出现只有当数据在服务器上的数据库时,才会触发听众,而不是在本地数据库上! 那是对的吗 ?

只有在服务器上提交数据后,才会触发完成侦听器。

从保存数据的指南 :

如果您想知道数据何时提交,可以添加完成侦听器。 setValue()和updateChildren()都将获取一个可选的完成侦听器,该侦听器在写入已提交到数据库时调用。

但是,对于本地和远程编辑,将触发常规事件侦听器(如使用addValueEventListener()创建的侦听器)。 这些是您应该用于呈现UI等的侦听器。

关注@ Andrew-lee和@Frank-fan-puffelen(感谢双方同意),解决方案似乎是:

使用push()命令创建唯一的id

将侦听器(Single Value)添加到(1.)中先前创建的节点的firebase ref

在侦听器中安排您的工作(这里更新UI,但可能是另一个命令)

findViewById(R.id.button7).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Firebase temp = new Firebase(getString(R.string.firebase_url)).child("stack").push(); temp.addListenerForSingleValueEvent(new ValueEventListener() { @Override public void onDataChange(DataSnapshot dataSnapshot) { ((TextView) findViewById(R.id.textView7)).setText(String.valueOf(counter) + " - " + dataSnapshot.getKey().toString()); } @Override public void onCancelled(FirebaseError firebaseError) { } }); temp.setValue(counter++); } });

Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐