- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
这很难解释,所以我举个例子:
@Override
public void start(Stage primaryStage) throws Exception
{
final VBox vbox = new VBox();
final Scene sc = new Scene(vbox);
primaryStage.setScene(sc);
final TableView<Person> table = new TableView<>();
final TableColumn<Person, String> columnName = new TableColumn<Person, String>("Name");
table.getColumns().add(columnName);
final ObservableList<Person> list = FXCollections.observableArrayList();
list.add(new Person("Hello"));
list.add(new Person("World"));
Bindings.bindContent(table.getItems(), list);
columnName.setCellValueFactory(new PropertyValueFactory<>("name"));
vbox.getChildren().add(table);
final Button button = new Button("test");
button.setOnAction(event ->
{
final Person removed = list.remove(0);
removed.setName("Bye");
list.add(0, removed);
});
vbox.getChildren().add(button);
primaryStage.show();
}
public static class Person
{
private String name = "";
public Person(String n)
{
name = n;
}
public String getName()
{
return name;
}
public void setName(String n)
{
name = n;
}
}
在这个例子中,我展示了一个 TableView
,其中只有一个名为“Name”的列。运行此示例代码,您将得到两行:第一行“Name”列中包含“Hello”;第二行的“名称”列中包含“世界”。
此外,还有一个按钮,此按钮从列表中删除第一个 Person
对象,然后对该对象进行一些更改,然后将其添加回相同的索引处。这样做会导致添加到 ObservableList
的任何 ListChangeListener
被触发,我已经测试过这是真的。
我希望带有“Hello”的行被替换为“Bye”,但 TableView
似乎继续显示“Hello”。如果我在将删除的 Person
对象添加回列表之前使用 TimeLine
添加延迟,它将变为“Bye”。
final Timeline tl = new Timeline(new KeyFrame(Duration.millis(30), ae -> list.add(0, removed)));
tl.play();
API 有什么奇怪的地方吗?有没有办法解决这个问题?
最佳答案
这本质上是预期的行为。
请注意(我猜您正在尝试解决此问题),如果您只是调用
list.get(0).setName("Bye");
在底层数据方面具有相同的效果,表不会更新,因为无法通知元素中的 String
字段 name
列表的内容已更改。
代码
Person removed = list.remove(0);
removed.setName("Bye");
list.add(0, removed);
实际上等同于 list.get(0).setName("Bye");
:您只是在更改之前暂时从列表中删除该项目,然后再将其添加回去。就列表而言,最终结果是相同的。我猜你这样做是希望从列表中删除和替换项目会说服表注意到项目的状态已经改变。不能保证会是这种情况。这是正在发生的事情:
两个列表之间的绑定(bind):
Bindings.bindContent(table.getItems(), list);
像任何其他绑定(bind)一样工作:它定义如何获取绑定(bind)的值(list
的元素),并在 list
无效时将数据标记为无效随时。后者发生在您从 list
中添加和删除元素时。
TableView
不会在每次对列表的绑定(bind)发生变化时执行布局;相反,当 then 绑定(bind)无效(添加或删除元素)时, TableView 将自身标记为可能需要重绘。然后,在下一个渲染脉冲中,表格将检查数据并查看是否真的需要重绘,并在需要时重新渲染。此实现具有明显的性能节省功能。
那么您的代码发生的情况是从列表中删除了一个项目,导致绑定(bind)被标记为无效。然后更改项目(通过调用 setName(...)
),然后将相同的项目添加回列表的相同位置。这也会导致绑定(bind)被标记为无效,这没有任何效果(它已经无效)。
在删除和重新添加此元素之间不会出现渲染脉冲。因此,表第一次实际查看对列表所做的更改必须在整个删除-更改-添加过程之后。届时,该表将看到该列表仍然包含与它之前包含的完全相同的顺序的完全相同的元素。 (其中一个元素的内部状态已经改变,但由于这不是一个可观察的值——不是 JavaFX 属性——表不知道这一点。)因此,表看不到任何变化(或看到所有更改都相互抵消了),并且不会重新呈现。
在您添加暂停的情况下,在移除项目和重新添加项目之间会出现一个(或两个)渲染帧。因此,表格实际上在没有项目的情况下呈现一两个帧,当它被重新添加时,它会重新添加并呈现当前值。 (您可能可以通过暂停 16 或 17 毫秒来使行为不可预测,这恰好是一个渲染帧的时间点。)
不清楚您真正打算做什么。如果您试图在不使用 JavaFX 属性的情况下说服表进行更新,您可以这样做
list.get(0).setName("Bye");
table.refresh();
虽然这不是一个非常令人满意的解决方案。
还要注意
list.remove(0);
list.add(0, new Person("Bye"));
也将起作用(因为现在添加的元素与删除的元素不同)。
更好的方法是使用 JavaFX 属性实现您的模型类:
public static class Person
{
private final StringProperty name = new SimpleStringProperty("");
public Person(String n)
{
setName(n);
}
public StringProperty nameProperty() {
return name ;
}
public final String getName()
{
return nameProperty().get();
}
public final void setName(String n)
{
nameProperty().set(n);
}
}
然后简单地调用
list.get(0).setName("Bye");
将更新表格(因为单元格将观察该属性)。
关于JavaFX TableView : Rapid change in items list not reflected,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43060044/
我有: func NewMethodDescriptor(typ interface{}) *MethodDescriptor { reflectedMethod := reflect.Val
我需要确定地检查 reflect.Type 是否是一个错误。 错误没有反射(reflect)类型。在 go reflect 中检查类型错误的正式/惯用方式是什么? Go Playground Full
根据 reflect 文档 reflect.Value.MapIndex() 应返回一个 reflect.Value,它表示存储在 map 特定键处的数据的值.所以我的理解是以下两个表达式应该是相同的
与 reflect pkg 有点混淆 所有示例都使用 reflect.NewValue() 来获取 var 的 reflect.Value,但是 func NewValue 未记录在 http://g
在计算机语言的上下文中,我从未找到关于反射的词源的明确解释,所以我想在这里澄清一下。 “Reflection”源于拉丁语,有以下definitions : bend back turn back tu
我写了一个漂亮的函数,它可以接受 system.object ,反射(reflect)其属性并将对象序列化为 JSON 字符串。它看起来像这样: public class JSONSerializer
我正在尝试创建一个函数 import Language.Reflection foo : Type -> TT 我尝试使用reflect 策略: foo = proof { intro t
最近我和一位同事谈论 C++,感叹没有办法获取带有类字段名称的字符串并提取具有该名称的字段;换句话说,它缺乏反射(reflection)。他困惑地看着我,并问什么时候有人需要做这样的事情。 除了“嘿,
我正在考虑允许模块与属性文件中的类一起使用的想法;像 availableModules.properties Contact=org.addressbook.ContactMain Business=
这个问题特别与为具有大量字段的对象覆盖 equals() 方法有关。首先,让我说这个大对象不能在不违反 OO 原则的情况下分解成多个组件,所以告诉我“没有类应该有超过 x 个字段”无济于事。 继续前进
例子 router.Get(path, handler) // works fine methodStr = "Get" router.methodStr(path, handler) // e
我一直坚持使用反射库的问题。由于很多推荐,我决定使用它,但我只是在学习,有些部分并不是很容易.. 我有这部分代码: func countDataByName(sourceName string, s
我有一个包含一些 url 参数的特定结构,我想使用 reflect 构建一个 url 参数字符串以遍历结构字段,这样我就不会关心结构真正包含什么。 假设我有一个这样的结构: type Student
我正在尝试从 reflect.Value 中检索字符串值, 我希望 value.String()成为okok但我得到了相反。 我错过了什么吗? package main import ( "f
为了避免创建 org.reflections.Reflections 类的多个实例,我只想创建一个并根据需要重用。有谁知道这个类是否是线程安全的? 如果它不是线程安全的,我知道我可以使用 Java 的
我最近对引用、具体化和反射(reflection)感到困惑。有人可以很好地解释他们的关系和差异(如果有的话)吗? 最佳答案 引用 这可能是最简单的一个。考虑一下当您在 REPL 中键入以下内容时会发生
less main.go输出: ``` package main import ( "reflect" "net/url" "fmt" ) type User struct {
我在 golang 中使用 gorm 包 ( https://github.com/jinzhu/gorm ) 作为我的数据库库。我有很多类(数据库表),如“酒店”或“套餐”。复制代码不是好的编程习惯
我有代码 var t reflect.Type = LaunchController(route.controller) // create controller ptr . var
是否有可能以及如何在不从类型创建对象并调用它的情况下获取类型的 reflect.Type reflect.TypeOf(obj) Java 中的内容是:MyType.class 最佳答案 您可以使用以
我是一名优秀的程序员,十分优秀!