无法通过 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<String>"/>
</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$IncrementalTaskAction.doExecute(AnnotationProcessingTaskFactory.java:245) 在 org.gradle.api.internal.project.taskfactory.AnnotationProcessingTaskFactory$StandardTaskAction.execute(AnnotationProcessingTaskFactory.java:221) 在 org.gradle.api.internal.project.taskfactory.AnnotationProcessingTaskFactory$IncrementalTaskAction.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.AbstractTaskPlanExecutor$TaskExecutorWorker.processTask(AbstractTaskPlanExecutor.java:66) 在 org.gradle.execution.taskgraph.AbstractTaskPlanExecutor$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您使用的是ObservableField
- 它的工作方式与内置的ObservableBoolean
或ObservableInt
略有不同。
使用message.get()
,您只会得到一个存储在ObservableField
中的String
,DataBinding
不知道如何在其中存储输入的数据。按照 George Mount(DataBinding 库的开发者)的example,您应该执行以下操作:
在您的班级中创建一个public final ObservableField<String>
。该类可以是一个简单的 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对比
Flutter开发 - 使用GetX框架实现类似MVVM架构
Vuejs-nuxt(s-s-r 模式)无法通过插件内部的 getter 获取 UserUUID。它显示未定义的 GetUserUUID