原文地址:JavaFx ObservableList的使用详解 | Stars-One的杂货小窝

最近在研究MVVM模式,发现可以将之前写的FxRecyclerView控件改造一下,便是开始尝试,尝试过程中发现了不少坑,但是网上资料不是太全面,这里便写一篇笔记记录一下,以供后来者的学习

注:由于本人使用的是TornadoFx来编写JavaFx项目,所以本文代码是使用Kotlin进行编写,需要有Kotlin基础,阅读前请须知,以免浪费你宝贵的时间

介绍

在MVVM模式火的今天,没想到之前JavaFx已经早有实现,估计现在市面这些都是玩别人剩下的了

这次也是想到了之前的FxRecyclerView控件可以拿MVVM改造一番

网上的资料并不多,仅有少有的一两篇,无奈之下,只得自己啃着官方文档,自己摸索,终于是把基本的使用摸清了

ObservableList,正如其名,可观察的List,它与List一样,也是个接口(Java的基础知识了..)

官方简短介绍:

A list that allows listeners to track changes when they occur.

大意为当数据改变时,ObservableList可以监听到这些改变,本质上是提供了一个监听器接口ListChangeListener.Change来进行相关的监听,

怎么监听呢?只需要设置监听器即可

val observableList = observableListOf(0,1,2,3)
observableList.onChange { change ->
while (change.next()) {
when {
change.wasPermutated() -> println("permutated (${change.from} ,${change.to})")
change.wasReplaced() -> println("replace (${change.from} ,${change.to})")
else -> {
when {
change.wasAdded() -> println("add (${change.from} ,${change.to})")
change.wasRemoved() -> println("remove (${change.from} ,${change.to})")
change.wasUpdated() -> println("update (${change.from} ,${change.to})")
}
}
}
}
}

官方的使用中,必须要求我们在监听之前调用next方法,其返回false表示当前已经是最后一次改变

wasPermutated()这种方法都是返回的boolean值,当数据发生对应符合的变化,各自对应的方法会返回true

change.fromchange.to两个属性在后面提及,这里稍微留意一下

三种监听类型

从上面的代码中,我们可以看到有几个分支条件,每个分支就是数据发生了某种改变,基本的有三种情况:

  1. List排列顺序改变
  2. List中的数据发生改变(数据更新)
  3. List的数据添加及删除(数据新增或删除)

PS:官方文档说明中,监听的顺序依次为wasPermutated、add/remove、update

下面对这几种情况进行说明

1.排列更新监听

顺序排列更新,对应的change.wasPermutated(),其返回值为boolean值,当ObservableList的顺序发生变化(即进行了排序操作),此方法就会返回为true

fun main() {
val observableList = observableListOf(0,1,2,3)
println(observableList)
observableList.onChange { change ->
while (change.next()) {
when {
change.wasPermutated() -> println("permutated (${change.from} ,${change.to})") change.wasReplaced() -> println("replace (${change.from} ,${change.to})")
else -> {
when {
change.wasAdded() -> println("add (${change.from} ,${change.to})")
change.wasRemoved() -> println("remove (${change.from} ,${change.to})")
change.wasUpdated() -> println("update (${change.from} ,${change.to})")
}
}
}
}
}
observableList.sortBy { it }
println(observableList)
observableList.sortByDescending { it }
println(observableList)
}

对应的输出结果为:

[0, 1, 2, 3]  //源数组
permutated (0 ,4) //回调监听器中的方法
[0, 1, 2, 3] //升序排列
permutated (0 ,4)
[3, 2, 1, 0] //降序排列

2.数据更新监听(replace)

这里大家可能会有点疑惑,最上面的代码不是有个wasUpdated()方法吗,这里怎么标的是replace?

我自己研究的时候,也是很奇怪...官方的那个wasUpdated()方法没有找到对应的回调方式,照理说我更新了数组中的一个数据,这个wasUpdated()应该返回的是true,但是实际测试的时候根本没有,反而是wasReplaced()返回了true

//上面省略相关代码...
//JavaFx对应是set(index,element)方法
observableList[1] = 12 //下标为1的对象更新为12
println(observableList)

输出结果:

[0, 1, 2, 3]
replace (1 ,2)
[0, 12, 2, 3]

3.数据添加与删除监听(add remove)

数据添加和删除就和上面同理了,当我们调用add()(包括addAll())和remove()方法,对应的wasAdd()wasRemove()方法就会返回true

另外,ObservableList提供了两个List供我们拿到添加的数据和移除的数据

  • change.addedSubList 新增的数据列表
  • change.removed 被移除的数据列表

补充:from与to

这里大家可能就注意到了change对象的fromto这两个属性的区别了

本质上,ObservableList里面定义的fromto这两个代表开始下标和结束下标,在数据发生改变的时候,记录了是哪几条数据发生了改变,之后提供给我们,我们在ObservableList.change监听器方法中就可以使用这两个变量来进行相关的逻辑操作

以下是官方对from的说明:

If wasAdded is true, the interval contains all the values that were added. If wasPermutated is true, the interval marks the values that were permutated. If wasRemoved is true and wasAdded is false, getFrom() and getTo() should return the same number - the place where the removed elements were positioned in the list.

注意重点:如果是数据移除操作,返回的from和to数值是相同的

ObservableList方法说明

  • addAll()
  • remove(from,to) 移除[from,to)之间的元素,from是包含,to是不包含的,注意与remove监听事件返回的from和to都是相同的
  • move(oldIndex,newIndex) 将下标为oldIndex移动至newIndex下标处

补充:移动监听(move)

之前没有找到更新的方法,发现了ObservableList有move的方法,没想到这个在ObservableList中其实是先做remove操作,之后再做add操作

//代码与上面的一致,已省略
//将下标1的数据移动到下标3的位置
observableList.moveAt(1,3)
println(observableList)

输出结果:

[0, 1, 2, 3]
remove (1 ,1)
add (3 ,4)
[0, 2, 3, 1]

参考

JavaFx ObservableList的使用详解的更多相关文章

  1. JavaFx TableView疑难详解

    TableView是个十分有用的控件,适应性和灵活性非常强,可以对它进行任意的修改,比如界面样式.功能.本文将从一步步提问的方式讲解TableView 创建已知列的TableView 已知列的表格的创 ...

  2. WordPress代码高亮插件SyntaxHighlighter终极使用详解

    子曰: 工欲善其事,必先利其器.作为码农一枚,再加上站长这个已经不再光鲜的称呼,岂能没有一款经济实用.操作简单.而且功能必须强大.样式也必须好看的Wordpress代码高亮插件?!作为一个视代码如生命 ...

  3. Linq之旅:Linq入门详解(Linq to Objects)

    示例代码下载:Linq之旅:Linq入门详解(Linq to Objects) 本博文详细介绍 .NET 3.5 中引入的重要功能:Language Integrated Query(LINQ,语言集 ...

  4. 架构设计:远程调用服务架构设计及zookeeper技术详解(下篇)

    一.下篇开头的废话 终于开写下篇了,这也是我写远程调用框架的第三篇文章,前两篇都被博客园作为[编辑推荐]的文章,很兴奋哦,嘿嘿~~~~,本人是个很臭美的人,一定得要截图为证: 今天是2014年的第一天 ...

  5. EntityFramework Core 1.1 Add、Attach、Update、Remove方法如何高效使用详解

    前言 我比较喜欢安静,大概和我喜欢研究和琢磨技术原因相关吧,刚好到了元旦节,这几天可以好好学习下EF Core,同时在项目当中用到EF Core,借此机会给予比较深入的理解,这里我们只讲解和EF 6. ...

  6. Java 字符串格式化详解

    Java 字符串格式化详解 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 文中如有纰漏,欢迎大家留言指出. 在 Java 的 String 类中,可以使用 format() 方法 ...

  7. Android Notification 详解(一)——基本操作

    Android Notification 详解(一)--基本操作 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Notification 文中如有纰 ...

  8. Android Notification 详解——基本操作

    Android Notification 详解 版权声明:本文为博主原创文章,未经博主允许不得转载. 前几天项目中有用到 Android 通知相关的内容,索性把 Android Notificatio ...

  9. Git初探--笔记整理和Git命令详解

    几个重要的概念 首先先明确几个概念: WorkPlace : 工作区 Index: 暂存区 Repository: 本地仓库/版本库 Remote: 远程仓库 当在Remote(如Github)上面c ...

随机推荐

  1. 使用pdf.js aspose各种文档转PDF 版本对应license.xml 去水印破解

    在使用pdf.js途中,使用aspose转换的文件一直有水印,在网上找了许多破解办法都是已经失效的,于是乎,就查看了一下jar的源码,找到了版本对应的破解字符(如下):对应版本为 aspose-wor ...

  2. 修改/查看ssh端口

    修改ssh端口 vi /etc/ssh/sshd_config 将Port修改为需要的端口 Port 212 重启ssh服务 service sshd restart 查看ssh端口 netstat ...

  3. sql操作数据库(3)-->外键约束、数据库表之间的关系、三大范式、多表查询、事务

    外键约束 在新表中添加外键约束语法: constraint 外键约束名称 foreign key(外键的字段名称) references 主表表名(主键字段名) 在已有表中添加外键约束:alter t ...

  4. 使用docker制作Mysql镜像

     一.过程说明 通过Dockerfile创建mysql镜像,使用的mysql软件包为mariadb二进制分发版,最终在宿主机启动mysql容器从而能在mysql容器外部访问mysql服务. 宿主机IP ...

  5. MP(MyBatis-Plus)的自动填充功能

    什么是自动填充 有些表中会有更新时间.创建时间.更新人或者创建人这些字段. 每次对数据进行新增.删除.修改时都需要对这些字段进行设置.传统的做法是在进行这些操作前,对Entity的字段进行set设置, ...

  6. Faceto_object_programdesign

    一面向对象程序设计思想 1 和面向过程关注点不同 过程: 程序执行流程和逻辑 (局部功能) 对象: 程序中的对象 (程序结构) 2 对象 现实世界中的实体和事物 可以看成是一种具有自身属性和功能的构件 ...

  7. 【Git】命令行操作

    Git 命令行操作 1 本地库初始化 git init:初始化本地仓库 效果 注意:.git目录中存放的是本地库相关的子目录和文件,不要删除,也不要胡乱修改. 2 设置签名 形式: 用户名:tom E ...

  8. 【SpringBoot1.x】 Docker

    SpringBoot1.x Docker 核心概念 Docker 是一个开源的应用容器引擎,是一个轻量级容器技术.Docker 支持将软件编译成一个镜像,然后在镜像中各种软件做好配置,将镜像发布出去, ...

  9. LeetCode485 最大连续1的个数

    给定一个二进制数组, 计算其中最大连续1的个数. 示例 1: 输入: [1,1,0,1,1,1] 输出: 3 解释: 开头的两位和最后的三位都是连续1,所以最大连续1的个数是 3. 注意: 输入的数组 ...

  10. LeetCode739 每日温度

    根据每日 气温 列表,请重新生成一个列表,对应位置的输入是你需要再等待多久温度才会升高的天数.如果之后都不会升高,请输入 0 来代替. 例如,给定一个列表 temperatures = [73, 74 ...