【问题标题】:What does FirebaseDatabase.getInstance().setPersistenceEnabled(true) really do?FirebaseDatabase.getInstance().setPersistenceEnabled(true) 到底做了什么?
【发布时间】:2016-10-26 09:29:55
【问题描述】:

我似乎不了解setPersistenceEnabled

我曾认为通过调用

FirebaseDatabase.getInstance().setPersistenceEnabled(true);

作为我的 Android 应用程序对象中的第一件事,即使在飞行模式下,我也可以在本地读取和写入 Firebase 数据,并且我的数据将存储在本地,因此即使在飞行模式下,下次应用启动时也可以使用.

我认为当应用程序离线时进行的写入会在网络再次可用时与 FB 数据库同步。

似乎根本不是这样。

[编辑] 在飞行模式下写入已完成,但该数据似乎无法读取。

addListenerForSingleValueEvent 和 ValueEventListener 不会检测到没有网络的数据库写入。

谁能帮我理解 setPersistenceEnabled(true) 的作用,以及它的典型用例可能是什么?

编辑添加:

我编写了我能想到的最小的 Android Firebase 数据库应用程序。

这是应用程序类:

 import android.app.Application;
import android.util.Log;

import com.google.firebase.database.FirebaseDatabase;

public class TheApp extends Application {
    @Override
    public void onCreate() {
        super.onCreate();
        // See https://firebase.google.com/docs/reference/android/com/google/firebase/database/FirebaseDatabase.html#public-methods
        FirebaseDatabase.getInstance().setPersistenceEnabled(true);
        Log.d("TheApp", "application created");
    }
}

所有主要活动都是在 EditText 旁边显示一个“保存”按钮。 在 onCreate MainActivity 登录一个知道的用户。当用户然后单击保存按钮时,EditText 中的字符串将写入 Firebase。

package com.grayraven.simpledb;

import android.os.Bundle;
import android.support.annotation.NonNull;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;

import com.google.android.gms.tasks.OnCompleteListener;
import com.google.android.gms.tasks.Task;
import com.google.firebase.auth.AuthResult;
import com.google.firebase.auth.FirebaseAuth;
import com.google.firebase.auth.FirebaseUser;
import com.google.firebase.database.DatabaseReference;
import com.google.firebase.database.FirebaseDatabase;
import com.google.firebase.database.ValueEventListener;

public class MainActivity extends AppCompatActivity {
    FirebaseAuth mAuth;
    FirebaseUser mUser;
    ValueEventListener mListener;
    private FirebaseAuth.AuthStateListener mAuthListener;
    private static final String TAG = "Main";
    FirebaseDatabase db = FirebaseDatabase.getInstance();
    DatabaseReference dbRef;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);
        dbRef = db.getReference("/data/strings/");

        Button btnSave = (Button)findViewById(R.id.btn_save);
        btnSave.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                EditText editText = (EditText)findViewById(R.id.editText);
                String text = String.valueOf(editText.getText());
                dbRef.setValue(text);
            }
        });

        signIn();

    }

    private void signIn() {
        mAuth= FirebaseAuth.getInstance();
        String email = "jim@example.com";  //valid credentials
        String password = "12345678";
        mAuth.signInWithEmailAndPassword(email, password)
                .addOnCompleteListener(this, new OnCompleteListener<AuthResult>() {
                    @Override
                    public void onComplete(@NonNull Task<AuthResult> task) {
                        Log.d(TAG, "signInWithEmail:onComplete:" + task.isSuccessful());

                        // If sign in fails, display a message to the user. If sign in succeeds
                        // the auth state listener will be notified and logic to handle the
                        // signed in user can be handled in the listener.
                        if (!task.isSuccessful()) {
                       //always fails when offline
                            Log.w(TAG, "signInWithEmail failed", task.getException());  
                        }
                    }
                });
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();

        //noinspection SimplifiableIfStatement
        if (id == R.id.action_settings) {
            return true;
        }

        return super.onOptionsItemSelected(item);
    }
}

只要设备连接到网络,这一切都可以工作。

如果应用程序在没有网络连接的情况下启动,则登录失败,因此用户无法执行任何操作。

如果在应用运行时网络连接断开,则不会保留任何已保存的字符串并在网络恢复时上传到 Firebase。

这是预期的行为吗?这种行为似乎与下面引用的文档不一致:

public synchronized void setPersistenceEnabled (boolean isEnabled)

Firebase 数据库客户端将缓存同步的数据并保留 跟踪您在应用程序运行时发起的所有写入 跑步。它无缝处理间歇性网络连接和 当网络连接恢复时重新发送写操作。 但是默认情况下,您的写入操作和缓存数据仅 存储在内存中,并在您的应用重新启动时丢失。通过设置 将此值设置为true,数据将被持久化到设备上(磁盘) 存储,因此将在应用程序重新启动时再次可用 (即使当时没有网络连接)。注意 必须在创建第一个数据库之前调用此方法 引用,每个应用程序只需要调用一次。

【问题讨论】:

标签: android firebase-realtime-database


【解决方案1】:

我的问题很简单。即使我有我的 TheApp 类,我也忘记更新清单,以便首先调用 TheApp。您可以使用“名称”应用程序标签来做到这一点:

<application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme"
        android:name="com.grayraven.simpledb.TheApp"> <!-- I forgot this! -->

现在应用程序似乎在离线或离线时都能正常工作,因为实际上调用了 setPersistenceEnabled(true)。唯一的限制是,如果您注销然后丢失网络,则无法进行身份验证。但似乎如果您已通过身份验证,您的应用程序将离线运行,即使您重新启动它也是如此。至少看起来是这样。

【讨论】:

  • 差不多 3 年后,我面临同样的问题,但没有任何线索。 :-(
猜你喜欢
  • 2018-03-19
  • 2022-11-10
  • 1970-01-01
  • 2011-12-10
  • 2017-04-20
  • 2012-05-19
  • 2019-09-05
  • 2012-10-06
  • 2022-01-10
相关资源
最近更新 更多