FirebaseDatabase.getInstance()。setPersistenceEnabled(true)真的有用吗?

时间:2016-06-23 21:09:45

标签: android firebase-realtime-database

我似乎没有找到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,数据将持久保存到设备上(磁盘)   存储,因此在重新启动应用程序时将再次可用   (即使当时没有网络连接)。注意   必须在创建第一个数据库之前调用此方法   引用,只需要为每个应用程序调用一次。

1 个答案:

答案 0 :(得分:3)

我的问题非常简单。即使我有我的TheApp类,我忘了更新清单,以便首先调用TheApp。你这样做的名字是&#39;应用程序标签:

<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),因此应用程序似乎可以正常或脱机工作。唯一的限制是,如果您注销然后丢失了网络,则无法进行身份验证。但似乎如果您通过身份验证,即使重新启动,您的应用也会脱机工作。至少看起来是这样的。