0

我在 Java 项目中使用 DataBinding 和 LiveData 时遇到问题。我在 Kotlin 中学习了以前的课程,当我尝试实现相同的行为时,我就是无法让它发挥作用。在理解方面,我显然缺少一些东西,所以我想请教您一些想法。

我将粘贴 Kotlin(工作)示例中的代码,然后粘贴 Java(不工作)示例。

科特林

score_fragment.xml

...
<data>

    <variable
        name="scoreViewModel"
        type="com.example.android.guesstheword.screens.score.ScoreViewModel" />
</data>

<androidx.constraintlayout.widget.ConstraintLayout
    android:id="@+id/score_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".screens.score.ScoreFragment">

    <TextView
        android:id="@+id/score_text"
        ...
        android:text="@{String.valueOf(scoreViewModel.score)}"
        .../>
...

分数片段.kt

class ScoreFragment : Fragment() {

private lateinit var viewModelFactory: ScoreViewModelFactory
private lateinit var viewModel: ScoreViewModel

override fun onCreateView(
        inflater: LayoutInflater,
        container: ViewGroup?,
        savedInstanceState: Bundle?
): View? {

        // Inflate view and obtain an instance of the binding class.
        val binding: ScoreFragmentBinding = DataBindingUtil.inflate(
                inflater,
                R.layout.score_fragment,
                container,
                false
        )

        // Get args using by navArgs property delegate
        val scoreFragmentArgs by navArgs<ScoreFragmentArgs>()

        viewModelFactory = ScoreViewModelFactory(scoreFragmentArgs.score)
        viewModel = ViewModelProviders.of(this, viewModelFactory)
                .get(ScoreViewModel::class.java)

        binding.scoreViewModel = viewModel
        binding.lifecycleOwner = this

        return binding.root
    }
}

ScoreViewModel.kt

class ScoreViewModel(finalScore: Int) : ViewModel() {

    private val _score = MutableLiveData<Int>()
    val score: LiveData<Int>
        get() = _score

    private val _eventPlayAgain = MutableLiveData<Boolean>()
    val eventPlayAgain: LiveData<Boolean>
        get() = _eventPlayAgain

    init {
        Timber.i("ScoreViewModel created")
        _score.value = finalScore
    }

    fun onPlayAgain() {
        _eventPlayAgain.value = true
    }

    fun onPlayAgainComplete() {
        _eventPlayAgain.value = false
    }

    override fun onCleared() {
        super.onCleared()
        Timber.i("ScoreViewModel cleared")
    }
}

解释:让我们只关注分值。在 ScoreViewModel 中,该值属于 LiveData 类型。当片段启动时,该值通过“@{String.valueOf(scoreViewModel.score)}”正确显示在屏幕上。这可以正常工作。

JAVA

activity_main.xml

<data>
    <variable
        name="noteViewModel"
        type="com.example.architectureapp.viewModel.NoteViewModel" />
</data>

<androidx.constraintlayout.widget.ConstraintLayout
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <TextView
        android:id="@+id/textview"
        android:text="@{noteViewModel.test}" />

</androidx.constraintlayout.widget.ConstraintLayout>

主要活动

public class MainActivity extends AppCompatActivity {
    private ActivityMainBinding binding;
    private NoteViewModel mNoteViewModel;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        binding = ActivityMainBinding.inflate(getLayoutInflater());
        mNoteViewModel = ViewModelProviders.of(this).get(NoteViewModel.class);
        binding.setNoteViewModel(mNoteViewModel);
        binding.setLifecycleOwner(this);
    }
}

NoteViewModel

public class NoteViewModel extends AndroidViewModel {
    public MutableLiveData<String> test = new MutableLiveData<>("TesT");

    public NoteViewModel(@NonNull Application application) {
        super(application);
    }
}

说明:这里我设置了一个值为“TestT”的 MutableLiveData 测试,然后我打算使用 android:text="@{noteViewModel.test}" 显示它。但是文本永远不会显示并保持空白。

显然有问题,但尽管两种实现之间存在语法差异,但我只是无法弄清楚为什么 Java 版本没有在 Textview 中显示值。

编辑

感谢 Rajnish suryavanshi,我没有以正确的方式进行绑定,我只需要使用:

binding = DataBindingUtil.setContentView(this, R.layout.activity_main);

这个使用提供的布局设置内容视图并返回绑定。或者,您可以这样做:

绑定 = ActivityMainBinding.inflate(getLayoutInflater()); (返回绑定但不设置内容视图)

setContentView(binding.getRoot()); (使用绑定根视图设置内容视图)

我发现这具有误导性-> https://developer.android.com/topic/libraries/data-binding/expressions

它指出我们可以替换

ActivityMainBinding 绑定 = DataBindingUtil.setContentView(this, R.layout.activity_main);

经过

ActivityMainBinding 绑定 = ActivityMainBinding.inflate(getLayoutInflater());

这是不一样的!

很高兴现在得到微妙之处!

4

1 回答 1

1

看看这两条线。

setContentView(R.layout.activity_main);
binding = ActivityMainBinding.inflate(getLayoutInflater());

在扩展布局时,您没有创建绑定。而不是setContentView使用DataBindingUtil.setContentView(this, R.layout.activity_main);

现在您可以使用布局充气器获取视图

binding = ActivityMainBinding.inflate(getLayoutInflater());
于 2019-09-22T14:01:57.427 回答