无法通过 ObservableField<String> 使用双向数据绑定

Posted

技术标签:

【中文标题】无法通过 ObservableField<String> 使用双向数据绑定【英文标题】:Unable to use two-way databinding with ObservableField<String> 【发布时间】:2017-03-07 14:51:42 【问题描述】:

这是我的布局

<?xml version="1.0" encoding="utf-8"?>

<data>

    <import type="android.view.View" />

    <import type="android.databinding.ObservableField"/>

    <variable
        name="message"
        type="ObservableField&lt;String&gt;"/>


</data>


<LinearLayout
    android:layout_
    android:layout_
    android:orientation="vertical">


    <android.support.v7.widget.Toolbar
        android:id="@+id/toolbar"
        android:layout_
        android:layout_
        android:theme="@style/ToolbarTheme"
        app:elevation="4dp" />


    <RelativeLayout
        android:layout_
        android:layout_>


        <RelativeLayout
            android:layout_
            android:layout_
            android:layout_alignParentBottom="true">


            <RelativeLayout
                android:layout_
                android:layout_
                android:layout_centerVertical="true"
                android:layout_marginBottom="@dimen/activity_vertical_margin"
                android:layout_marginEnd="8dp"
                android:layout_marginLeft="8dp"
                android:layout_marginRight="8dp"
                android:layout_marginStart="8dp"
                android:layout_toLeftOf="@+id/send"
                android:layout_toStartOf="@+id/send"
                android:background="@drawable/balloon_outgoing_normal"
                android:padding="4dp">


                <android.support.v7.widget.AppCompatEditText
                    android:layout_
                    android:layout_
                    android:layout_centerVertical="true"
                    android:text="@=message.get()"
                    android:layout_marginLeft="8dp"
                    android:layout_marginStart="8dp"
                    android:layout_toLeftOf="@+id/audio"
                    android:layout_toStartOf="@+id/audio" />


                <android.support.v7.widget.AppCompatImageView
                    android:id="@+id/audio"
                    android:layout_
                    android:layout_
                    android:layout_centerVertical="true"
                    android:layout_toLeftOf="@+id/camera"
                    android:layout_toStartOf="@id/camera"
                    android:background="?attr/selectableItemBackground"
                    android:padding="8dp"
                    android:src="@drawable/ic_mic_white_24dp"
                    android:tint="@color/iconDefault"
                    android:visibility="@message.get().length()==0?View.VISIBLE:View.GONE" />


                <android.support.v7.widget.AppCompatImageView
                    android:id="@+id/camera"
                    android:layout_
                    android:layout_
                    android:layout_alignParentEnd="true"
                    android:layout_alignParentRight="true"
                    android:layout_centerVertical="true"
                    android:layout_marginEnd="8dp"
                    android:layout_marginRight="8dp"
                    android:background="?attr/selectableItemBackground"
                    android:padding="8dp"
                    android:src="@drawable/ic_photo_camera_black_24dp"
                    android:tint="@color/iconDefault"
                    android:visibility="@message.get().length()==0?View.VISIBLE:View.GONE" />


            </RelativeLayout>


            <android.support.design.widget.FloatingActionButton
                android:id="@+id/send"
                android:layout_
                android:layout_
                android:layout_alignParentEnd="true"
                android:layout_alignParentRight="true"
                android:layout_centerVertical="true"
                android:layout_marginBottom="@dimen/activity_vertical_margin"
                android:layout_marginEnd="8dp"
                android:layout_marginRight="8dp"
                android:src="@drawable/ic_send_white_24dp"
                app:elevation="6dp"
                app:fabSize="normal" />


        </RelativeLayout>


    </RelativeLayout>


</LinearLayout>
</layout>

双向部分在哪里:

<android.support.v7.widget.AppCompatEditText
                    android:layout_
                    android:layout_
                    android:layout_centerVertical="true"
                    android:text="@=message.get()"
                    android:layout_marginLeft="8dp"
                    android:layout_marginStart="8dp"
                    android:layout_toLeftOf="@+id/audio"
                    android:layout_toStartOf="@+id/audio" />

这是堆栈跟踪:

* 出了什么问题:
任务 ':app:compileDebugJavaWithJavac' 执行失败。
> java.lang.RuntimeException: 失败,详情请查看日志。
  无法生成视图活页夹 java.lang.NullPointerException
    在 android.databinding.tool.expr.MethodCallExpr.generateCode(MethodCallExpr.java:72)
    在 android.databinding.tool.expr.Expr.toFullCode(Expr.java:745)
    在 android.databinding.tool.expr.Expr.assertIsInvertible(Expr.java:767)
    在 android.databinding.tool.BindingTarget.addInverseBinding(BindingTarget.java:68)
    在 android.databinding.tool.LayoutBinder.(LayoutBinder.java:229)
    在 android.databinding.tool.DataBinder.(DataBinder.java:52)
    在 android.databinding.tool.CompilerChef.ensureDataBinder(CompilerChef.java:86)
    在 android.databinding.tool.CompilerChef.sealModels(CompilerChef.java:200)
    在 android.databinding.annotationprocessor.ProcessExpressions.writeResourceBundle(ProcessExpressions.java:149)
    在 android.databinding.annotationprocessor.ProcessExpressions.onHandleStep(ProcessExpressions.java:82)
    在 android.databinding.annotationprocessor.ProcessDataBinding$ProcessingStep.runStep(ProcessDataBinding.java:154)
    在 android.databinding.annotationprocessor.ProcessDataBinding$ProcessingStep.access$000(ProcessDataBinding.java:139)
    在 android.databinding.annotationprocessor.ProcessDataBinding.process(ProcessDataBinding.java:66)
    在 com.sun.tools.javac.processing.JavacProcessingEnvironment.callProcessor(JavacProcessingEnvironment.java:794)
    在 com.sun.tools.javac.processing.JavacProcessingEnvironment.discoverAndRunProcs(JavacProcessingEnvironment.java:705)
    在 com.sun.tools.javac.processing.JavacProcessingEnvironment.access 1800 美元(JavacProcessingEnvironment.java:91)
    在 com.sun.tools.javac.processing.JavacProcessingEnvironment$Round.run(JavacProcessingEnvironment.java:1035)
    在 com.sun.tools.javac.processing.JavacProcessingEnvironment.doProcessing(JavacProcessingEnvironment.java:1176)
    在 com.sun.tools.javac.main.JavaCompiler.processAnnotations(JavaCompiler.java:1170)
    在 com.sun.tools.javac.main.JavaCompiler.compile(JavaCompiler.java:856)
    在 com.sun.tools.javac.main.Main.compile(Main.java:523)
    在 com.sun.tools.javac.api.JavacTaskImpl.doCall(JavacTaskImpl.java:129)
    在 com.sun.tools.javac.api.JavacTaskImpl.call(JavacTaskImpl.java:138)
    在 org.gradle.api.internal.tasks.compile.JdkJavaCompiler.execute(JdkJavaCompiler.java:46)
    在 org.gradle.api.internal.tasks.compile.JdkJavaCompiler.execute(JdkJavaCompiler.java:33)
    在 org.gradle.api.internal.tasks.compile.NormalizingJavaCompiler.delegateAndHandleErrors(NormalizingJavaCompiler.java:104)
    在 org.gradle.api.internal.tasks.compile.NormalizingJavaCompiler.execute(NormalizingJavaCompiler.java:53)
    在 org.gradle.api.internal.tasks.compile.NormalizingJavaCompiler.execute(NormalizingJavaCompiler.java:38)
    在 org.gradle.api.internal.tasks.compile.CleaningJavaCompilerSupport.execute(CleaningJavaCompilerSupport.java:35)
    在 org.gradle.api.internal.tasks.compile.CleaningJavaCompilerSupport.execute(CleaningJavaCompilerSupport.java:25)
    在 org.gradle.api.tasks.compile.JavaCompile.performCompilation(JavaCompile.java:163)
    在 org.gradle.api.tasks.compile.JavaCompile.compile(JavaCompile.java:145)
    在 org.gradle.api.tasks.compile.JavaCompile.compile(JavaCompile.java:93)
    在 com.android.build.gradle.tasks.factory.AndroidJavaCompile.compile(AndroidJavaCompile.java:49)
    在 sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    在 sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    在 sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    在 java.lang.reflect.Method.invoke(Method.java:498)
    在 org.gradle.internal.reflect.JavaMethod.invoke(JavaMethod.java:75)
    在 org.gradle.api.internal.project.taskfactory.AnnotationProcessingTaskFactory$IncrementalTask​​Action.doExecute(AnnotationProcessingTaskFactory.java:245)
    在 org.gradle.api.internal.project.taskfactory.AnnotationProcessingTaskFactory$StandardTaskAction.execute(AnnotationProcessingTaskFactory.java:221)
    在 org.gradle.api.internal.project.taskfactory.AnnotationProcessingTaskFactory$IncrementalTask​​Action.execute(AnnotationProcessingTaskFactory.java:232)
    在 org.gradle.api.internal.project.taskfactory.AnnotationProcessingTaskFactory$StandardTaskAction.execute(AnnotationProcessingTaskFactory.java:210)
    在 org.gradle.api.internal.tasks.execution.ExecuteActionsTaskExecuter.executeAction(ExecuteActionsTaskExecuter.java:80)
    在 org.gradle.api.internal.tasks.execution.ExecuteActionsTaskExecuter.executeActions(ExecuteActionsTaskExecuter.java:61)
    在 org.gradle.api.internal.tasks.execution.ExecuteActionsTaskExecuter.execute(ExecuteActionsTaskExecuter.java:46)
    在 org.gradle.api.internal.tasks.execution.PostExecutionAnalysisTaskExecuter.execute(PostExecutionAnalysisTaskExecuter.java:35)
    在 org.gradle.api.internal.tasks.execution.SkipUpToDateTaskExecuter.execute(SkipUpToDateTaskExecuter.java:66)
    在 org.gradle.api.internal.tasks.execution.ValidatingTaskExecuter.execute(ValidatingTaskExecuter.java:58)
    在 org.gradle.api.internal.tasks.execution.SkipEmptySourceFilesTaskExecuter.execute(SkipEmptySourceFilesTaskExecuter.java:52)
    在 org.gradle.api.internal.tasks.execution.SkipTaskWithNoActionsExecuter.execute(SkipTaskWithNoActionsExecuter.java:52)
    在 org.gradle.api.internal.tasks.execution.SkipOnlyIfTaskExecuter.execute(SkipOnlyIfTaskExecuter.java:53)
    在 org.gradle.api.internal.tasks.execution.ExecuteAtMostOnceTaskExecuter.execute(ExecuteAtMostOnceTaskExecuter.java:43)
    在 org.gradle.execution.taskgraph.DefaultTaskGraphExecuter$EventFiringTaskWorker.execute(DefaultTaskGraphExecuter.java:203)
    在 org.gradle.execution.taskgraph.DefaultTaskGraphExecuter$EventFiringTaskWorker.execute(DefaultTaskGraphExecuter.java:185)
    在 org.gradle.execution.taskgraph.Abs​​tractTaskPlanExecutor$TaskExecutorWorker.processTask(AbstractTaskPlanExecutor.java:66)
    在 org.gradle.execution.taskgraph.Abs​​tractTaskPlanExecutor$TaskExecutorWorker.run(AbstractTaskPlanExecutor.java:50)
    在 org.gradle.execution.taskgraph.DefaultTaskPlanExecutor.process(DefaultTaskPlanExecutor.java:25)
    在 org.gradle.execution.taskgraph.DefaultTaskGraphExecuter.execute(DefaultTaskGraphExecuter.java:110)
    在 org.gradle.execution.SelectedTaskExecutionAction.execute(SelectedTaskExecutionAction.java:37)
    在 org.gradle.execution.DefaultBuildExecuter.execute(DefaultBuildExecuter.java:37)
    在 org.gradle.execution.DefaultBuildExecuter.access$000(DefaultBuildExecuter.java:23)
    在 org.gradle.execution.DefaultBuildExecuter$1.proceed(DefaultBuildExecuter.java:43)
    在 org.gradle.execution.DryRunBuildExecutionAction.execute(DryRunBuildExecutionAction.java:32)
    在 org.gradle.execution.DefaultBuildExecuter.execute(DefaultBuildExecuter.java:37)
    在 org.gradle.execution.DefaultBuildExecuter.execute(DefaultBuildExecuter.java:30)
    在 org.gradle.initialization.DefaultGradleLauncher$4.run(DefaultGradleLauncher.java:153)
    在 org.gradle.internal.Factories$1.create(Factories.java:22)
    在 org.gradle.internal.progress.DefaultBuildOperationExecutor.run(DefaultBuildOperationExecutor.java:91)
    在 org.gradle.internal.progress.DefaultBuildOperationExecutor.run(DefaultBuildOperationExecutor.java:53)
    在 org.gradle.initialization.DefaultGradleLauncher.doBuildStages(DefaultGradleLauncher.java:150)
    在 org.gradle.initialization.DefaultGradleLauncher.access $200(DefaultGradleLauncher.java:32)
    在 org.gradle.initialization.DefaultGradleLauncher$1.create(DefaultGradleLauncher.java:98)
    在 org.gradle.initialization.DefaultGradleLauncher$1.create(DefaultGradleLauncher.java:92)
    在 org.gradle.internal.progress.DefaultBuildOperationExecutor.run(DefaultBuildOperationExecutor.java:91)
    在 org.gradle.internal.progress.DefaultBuildOperationExecutor.run(DefaultBuildOperationExecutor.java:63)
    在 org.gradle.initialization.DefaultGradleLauncher.doBuild(DefaultGradleLauncher.java:92)
    在 org.gradle.initialization.DefaultGradleLauncher.run(DefaultGradleLauncher.java:83)
    在 org.gradle.launcher.exec.InProcessBuildActionExecuter$DefaultBuildController.run(InProcessBuildActionExecuter.java:99)
    在 org.gradle.tooling.internal.provider.runner.BuildModelActionRunner.run(BuildModelActionRunner.java:46)
    在 org.gradle.launcher.exec.ChainingBuildActionRunner.run(ChainingBuildActionRunner.java:35)
    在 org.gradle.tooling.internal.provider.runner.SubscribableBuildActionRunner.run(SubscribableBuildActionRunner.java:58)
    在 org.gradle.launcher.exec.ChainingBuildActionRunner.run(ChainingBuildActionRunner.java:35)
    在 org.gradle.launcher.exec.InProcessBuildActionExecuter.execute(InProcessBuildActionExecuter.java:48)
    在 org.gradle.launcher.exec.InProcessBuildActionExecuter.execute(InProcessBuildActionExecuter.java:30)
    在 org.gradle.launcher.exec.ContinuousBuildActionExecuter.execute(ContinuousBuildActionExecuter.java:81)
    在 org.gradle.launcher.exec.ContinuousBuildActionExecuter.execute(ContinuousBuildActionExecuter.java:46)
    在 org.gradle.launcher.daemon.server.exec.ExecuteBuild.doBuild(ExecuteBuild.java:52)
    在 org.gradle.launcher.daemon.server.exec.BuildCommandOnly.execute(BuildCommandOnly.java:36)
    在 org.gradle.launcher.daemon.server.api.DaemonCommandExecution.proceed(DaemonCommandExecution.java:120)
    在 org.gradle.launcher.daemon.server.exec.WatchForDisconnection.execute(WatchForDisconnection.java:37)
    在 org.gradle.launcher.daemon.server.api.DaemonCommandExecution.proceed(DaemonCommandExecution.java:120)
    在 org.gradle.launcher.daemon.server.exec.ResetDeprecationLogger.execute(ResetDeprecationLogger.java:26)
    在 org.gradle.launcher.daemon.server.api.DaemonCommandExecution.proceed(DaemonCommandExecution.java:120)
    在 org.gradle.launcher.daemon.server.exec.RequestStopIfSingleUsedDaemon.execute(RequestStopIfSingleUsedDaemon.java:34)
    在 org.gradle.launcher.daemon.server.api.DaemonCommandExecution.proceed(DaemonCommandExecution.java:120)
    在 org.gradle.launcher.daemon.server.exec.ForwardClientInput$2.call(ForwardClientInput.java:74)
    在 org.gradle.launcher.daemon.server.exec.ForwardClientInput$2.call(ForwardClientInput.java:72)
    在 org.gradle.util.Swapper.swap(Swapper.java:38)
    在 org.gradle.launcher.daemon.server.exec.ForwardClientInput.execute(ForwardClientInput.java:72)
    在 org.gradle.launcher.daemon.server.api.DaemonCommandExecution.proceed(DaemonCommandExecution.java:120)
    在 org.gradle.launcher.daemon.server.health.DaemonHealthTracker.execute(DaemonHealthTracker.java:47)
    在 org.gradle.launcher.daemon.server.api.DaemonCommandExecution.proceed(DaemonCommandExecution.java:120)
    在 org.gradle.launcher.daemon.server.exec.LogToClient.doBuild(LogToClient.java:60)
    在 org.gradle.launcher.daemon.server.exec.BuildCommandOnly.execute(BuildCommandOnly.java:36)
    在 org.gradle.launcher.daemon.server.api.DaemonCommandExecution.proceed(DaemonCommandExecution.java:120)
    在 org.gradle.launcher.daemon.server.exec.EstablishBuildEnvironment.doBuild(EstablishBuildEnvironment.java:72)
    在 org.gradle.launcher.daemon.server.exec.BuildCommandOnly.execute(BuildCommandOnly.java:36)
    在 org.gradle.launcher.daemon.server.api.DaemonCommandExecution.proceed(DaemonCommandExecution.java:120)
    在 org.gradle.launcher.daemon.server.health.HintGCAfterBuild.execute(HintGCAfterBuild.java:41)
    在 org.gradle.launcher.daemon.server.api.DaemonCommandExecution.proceed(DaemonCommandExecution.java:120)
    在 org.gradle.launcher.daemon.server.exec.StartBuildOrRespondWithBusy$1.run(StartBuildOrRespondWithBusy.java:50)
    在 org.gradle.launcher.daemon.server.DaemonStateCoordinator$1.run(DaemonStateCoordinator.java:237)
    在 org.gradle.internal.concurrent.ExecutorPolicy$CatchAndRecordFailures.onExecute(ExecutorPolicy.java:54)
    在 org.gradle.internal.concurrent.StoppableExecutorImpl$1.run(StoppableExecutorImpl.java:40)
    在 java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
    在 java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
    在 java.lang.Thread.run(Thread.java:745)

如果我不使用双向数据绑定将 android:text="@=message.get()" 替换为 android:text="@message.get()" 一切正常。那么正确的做法是什么?

【问题讨论】:

你的message有二传手吗? @Amylinn "message" 是一个 ObservableField 它应该已经像 ObservableBoolean、ObservableInt... 一样被处理(适用于双向数据绑定) 【参考方案1】:

您使用的是ObservableField - 它的工作方式与内置的ObservableBooleanObservableInt 略有不同。

使用message.get(),您只会得到一个存储在ObservableField 中的StringDataBinding 不知道如何在其中存储输入的数据。按照 George Mount(DataBinding 库的开发者)的example,您应该执行以下操作:

    在您的班级中创建一个public final ObservableField&lt;String&gt;。该类可以是一个简单的 Java 类,您所在的活动或片段。

    将对类的引用传递给您的xml

    使用android:text="@myClass.myField"

完整示例:

public class MainActivity extends AppCompatActivity 
    public final ObservableField<String> message = new ObservableField<>();

    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        DataBindingUtil.setContentView(this, R.layout.activity_main);
        message.set("Hello"); //initial data
    

还有activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<layout>
    <data>
        <variable
            name="myClass"
            type="com.example.yourpackage.test.MainActivity"/>
    </data>

    <RelativeLayout      
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"        
        android:layout_
        android:layout_>

        <android.support.v7.widget.AppCompatEditText
            android:layout_
            android:layout_
            android:text="@=myClass.message" />
    </RelativeLayout>
</layout>

你也可以用一个额外的类包裹你的Fields,以保持你的MainActivity干净。

【讨论】:

我的错误是认为 ObservableFields 和 ObservableBoolean 以相同的方式工作,不幸的是,使它工作的唯一方法是你的(通过将活动添加为变量) 在写这篇文章之前我自己试过了;)我通常使用一种 MVVM 模式,在我的ViewModel 中存储必要的变量,效果很好。 如果您对这个主题感兴趣,我建议您深入研究android-architecture 示例。

以上是关于无法通过 ObservableField<String> 使用双向数据绑定的主要内容,如果未能解决你的问题,请参考以下文章

MutableLiveData和ObservableField对比

LiveData与 ObservableField的区别

s-s-rS 报告无法通过计划订阅工作,但可以正常工作

Flutter开发 - 使用GetX框架实现类似MVVM架构

通过正则表达式解析 CSS

Vuejs-nuxt(s-s-r 模式)无法通过插件内部的 getter 获取 UserUUID。它显示未定义的 GetUserUUID