JavaFX技巧来节省内存! 属性和可观察物的阴影场
在 JavaFX的世界中, Properties API允許UI開發(fā)人員將值綁定到UI控件。 這種功能非常容易,但是當對象模型經(jīng)常使用屬性時,應(yīng)用程序可能會很快耗盡內(nèi)存。 我通常會編寫兩個單獨的對象,例如pojo類和表示模型對象。 此技術(shù)通常在基于Swing的應(yīng)用程序中使用。 從JavaFX角度來看,您可以只創(chuàng)建一個具有屬性的對象,以允許觀察者(偵聽器)更新值。 聽起來不錯吧? 不完全是因為主要問題是當所有對象的(pojo)屬性(字段)都是還包裝實際值的屬性時,程序員(api的用戶)可能根本不想綁定或使用屬性,而只想訪問實際值。 那么,JavaFX開發(fā)人員該做什么?
一個TreeTable控件持有雇員。
我經(jīng)常訪問Dirk Lemmermann的博客Pixel Perfect ,該博客經(jīng)常發(fā)布非常有用的JavaFX技巧。 最近,Dirk在博客上發(fā)表了一篇關(guān)于如何使用一種有趣的模式“ Shadow Fields ”來節(jié)省內(nèi)存的博客。 要查看他的帖子,請訪問他的博客條目JavaFX Tip 23:節(jié)省內(nèi)存! 屬性的陰影字段。 。 Dirk的JavaFX技巧確實有助于解決上述問題(減少堆),但是我注意到必須存在樣板代碼才能( 聰明地 確定 )為調(diào)用者提供返回值是實際對象還是屬性包裝器對象。 例如,該代碼將返回int或Integer值,而不是在調(diào)用get或set方法時返回IntegerProperty對象,從而節(jié)省了內(nèi)存。 此外,代碼聲明了兩個變量來保存兩種值類型之一。 例如:
private String _title = "Untitled"; // shadow field private StringProperty title;我覺得我可以使事情更簡潔,并可能節(jié)省更多的內(nèi)存。 并減少樣板代碼。 我決定使用Java 8的默認方法創(chuàng)建一個接口,該接口將處理管理實際值和屬性。 API的用戶將只創(chuàng)建一個實現(xiàn)以下接口的域類:
界面屬性訪問器
一個提供訪問器方法的接口,以提供對實際對象值或JavaFX屬性包裝器對象的智能確定。 API的用戶必須實現(xiàn)一種稱為getModelProperties()的方法,該方法返回屬性名(字符串)和值(對象)的映射。 該值可以是實際對象或?qū)傩灶愋蛯ο蟆?下面的代碼也將支持可觀察列表。
package com.jfxbe;import javafx.beans.property.Property; import javafx.collections.FXCollections; import javafx.collections.ObservableList;import java.lang.reflect.Constructor; import java.util.ArrayList; import java.util.List; import java.util.Map;/*** Provide default methods to support the similar* capability of the shadow fields pattern.* To save memory object values don't have to be* wrapped into a Property object when using getters* and setters, however when calling property type methods* values will be wrapped into a property object.** Default methods for Observable lists are provided too.** Created by cpdea on 4/3/16.*/ public interface PropertyAccessors {Map<String, Object> getModelProperties();default <T> T getValue(String name, Object defaultVal) {Object p = getModelProperties().get(name);p = p==null ? defaultVal : p;return (T) ((p instanceof Property) ? ((Property) p).getValue(): p);}default void setValue(String name, Object value) {Object p = getModelProperties().get(name);if (p instanceof Property) {((Property)p).setValue(value);} else {getModelProperties().put(name, value);}}default <T> T refProperty(String name, Class propClass, Class rawValType) {Object p = getModelProperties().get(name);Property prop = null;try {if (p == null) {Class[] constructorTypes =new Class[]{Object.class, String.class};Constructor<Property> propConstr =propClass.getDeclaredConstructor(constructorTypes);prop = propConstr.newInstance(this, name);} else if (rawValType.isInstance(p)) {Class[] constructorTypes = new Class[]{Object.class,String.class, rawValType};Constructor<Property> propConstr =propClass.getDeclaredConstructor(constructorTypes);prop = propConstr.newInstance(this, name, p);} else {prop = (Property) p;}getModelProperties().put(name, prop);} catch (Exception e) {e.printStackTrace();}return (T) prop;}default <T> List<T> getValues(String name, List<T> defaultValue) {Object p, o = getModelProperties().get(name);p = o;o = o==null ? defaultValue : o;if (!o.equals(p)) {getModelProperties().put(name, o);}return (List<T>) o;}default <T> void setValues(String name, List<T> newList) {Object list = getModelProperties().get(name);if (list == null || !(list instanceof ObservableList)) {getModelProperties().put(name, newList);} else {// Should the list be totally replaced? below clears and adds all itemsObservableList<T> observableList = (ObservableList<T>) list;observableList.clear();observableList.addAll(newList);}}default <T> ObservableList<T> refObservables(String name) {List list = (List) getModelProperties().get(name);if (list == null) {list = FXCollections.observableArrayList(getValues(name, new ArrayList<>()));getModelProperties().put(name, list);}if (! (list instanceof ObservableList)) {list = FXCollections.observableArrayList(list);getModelProperties().put(name, list);}return (ObservableList<T>) list;} }員工階層
一個名為Employee的類,它實現(xiàn)PropertyAccessor接口。 在下面,您會注意到每個字段的屬性名稱都是使用public static final String聲明的 。 例如,員工的名字是:
public static final String NAME_PROPERTY = “name”;對于諸如getter,setter和xyzProperty()之類的訪問器方法,您會在PropertyAccessor接口中注意到對默認方法的調(diào)用。
package com.jfxbe;import javafx.beans.property.ObjectProperty; import javafx.beans.property.SimpleStringProperty; import javafx.beans.property.StringProperty; import javafx.collections.ObservableList;import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map;/*** A hybrid domain and model object using the shadow field pattern to save memory.* Created by cpdea*/ public class Employee implements PropertyAccessors{/** This is a map to hold properties and observables */private Map<String, Object> modelProperties;public static final String NAME_PROPERTY = "name";public static final String POWERS_PROPERTY = "powers";public static final String SUPERVISOR_PROPERTY = "supervisor";public static final String MINIONS_PROPERTY = "minions";public Employee(String name, String powers) {setName(name);setPowers(powers);}@Overridepublic Map<String, Object> getModelProperties() {if (modelProperties == null) {modelProperties = new HashMap<>();}return modelProperties;}public final String getName() {return getValue(NAME_PROPERTY, "");}public final void setName(String name) {setValue(NAME_PROPERTY, name);}public final StringProperty nameProperty() {return refProperty(NAME_PROPERTY, SimpleStringProperty.class, String.class);}public String getPowers() {return getValue(POWERS_PROPERTY, "");}public final StringProperty powersProperty() {return refProperty(POWERS_PROPERTY, StringProperty.class, String.class);}public final void setPowers(String powers) {setValue(POWERS_PROPERTY, powers);}public final Employee getSupervisor() {return getValue(SUPERVISOR_PROPERTY, null);}public final ObjectProperty<Employee> supervisorProperty() {return refProperty(SUPERVISOR_PROPERTY, ObjectProperty.class, Employee.class);}public final void setSupervisor(Employee supervisor) {setValue(SUPERVISOR_PROPERTY, supervisor);}public final List<Employee> getMinions() {return getValues(MINIONS_PROPERTY, new ArrayList<>());}public final ObservableList<Employee> minionsObservables() {return refObservables(MINIONS_PROPERTY);}public final void setMinions(List<Employee> minions) {setValues(MINIONS_PROPERTY, minions);}}結(jié)論
因此,您已找到解決方案,嘗試消除兩個變量和其他樣板代碼。 我實際上并沒有使用大量數(shù)據(jù)來測試代碼,因此也許在另一篇文章中或某個幸運的讀者會創(chuàng)建一個測試,將所有三個(對象具有所有屬性,Dirk和mine的)實現(xiàn)進行比較。
與RMI服務(wù)器一起使用時,此方法的可能缺點可能是序列化對象。 我敢肯定還有其他可能的缺點,但是對于大多數(shù)用例而言,這可能更容易處理和更簡潔。
請隨意發(fā)表評論!
參考文獻
- https://dlemmermann.wordpress.com
- https://docs.oracle.com/javase/8/javafx/api/javafx/beans/property/Property.html
- http://blog.netopyr.com/2011/05/13/javafx-properties/
- https://projectlombok.org
- http://www.jgoodies.com/freeware/libraries/binding
翻譯自: https://www.javacodegeeks.com/2016/04/javafx-tips-save-memory-shadow-fields-properties-observables.html
總結(jié)
以上是生活随笔為你收集整理的JavaFX技巧来节省内存! 属性和可观察物的阴影场的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 女士内衣品牌前十大排名(女性内衣文胸十大
- 下一篇: esim是什么意思(esim是什么意思?