问题
我想通过 javafx场景构建器将定制的面板添加到运行时的网格窗格.我的定制面板出现了按钮,标签等.
我想通过 javafx场景构建器将定制的面板添加到运行时的网格窗格.我的定制面板出现了按钮,标签等.
我的尝试
我试图从窗格中扩展…
public class Celli extends Pane{ public Celli() throws IOException{ Parent root = FXMLLoader.load(getClass().getResource("Cell.fxml")); this.getChildren().add(root); } }
@FXML private void textChange(KeyEvent event) { GridPane g = new GridPane(); for (int i=0 : i<100; i++){ g.getChildren().add(new Celli()); } } }
它的工作,但它表现非常非常穷.
我在找什么
有没有办法通过javafx场景生成器设计面板(因此在fxml中有这些面板),然后在运行时将其添加到网格窗格中,而无需为每个实例使用此fxmlloader.因为fxml加载器,我觉得它的性能很差.当我添加标准按钮,例如whitout fxml是非常快的.
解决方法
简短的答案:不,它不是(从JavaFX 2.x和8.0开始).它可能在将来的版本(JFX> 8)
长答案:
FXMLLoader目前不是作为一个模板提供者来实现一个又一个的实例化相同的项目.相反,它意味着成为大型GUI的一次性加载器(或将其序列化).
性能较差,因为FXML文件在每次调用load()时,FXMLLoader都必须通过反射来查找类及其属性.这意味着:
>对于每个import语句,尝试加载每个类,直到类可以成功加载.
>对于每个类,创建一个BeanAdapter来查找此类所有的属性,并尝试将给定的参数应用于该属性.
>通过反射再次对属性的参数进行应用.
对于在代码中完成的同一个FXML文件的后续调用load(),目前还没有任何改进.这意味着:没有缓存找到的类,没有缓存BeanAdapters等等.
但是,通过将自定义类加载器设置为FXMLLoader实例,可以解决步骤1的性能问题:
import java.io.IOException; import java.net.URL; import java.util.Enumeration; import java.util.HashMap; import java.util.Map; public class MyClassLoader extends ClassLoader{ private final Map<String,Class> classes = new HashMap<String,Class>(); private final ClassLoader parent; public MyClassLoader(ClassLoader parent) { this.parent = parent; } @Override public Class<?> loadClass(String name) throws ClassNotFoundException { Class<?> c = findClass(name); if ( c == null ) { throw new ClassNotFoundException( name ); } return c; } @Override protected Class<?> findClass( String className ) throws ClassNotFoundException { // System.out.print("try to load " + className); if (classes.containsKey(className)) { Class<?> result = classes.get(className); return result; } else { try { Class<?> result = parent.loadClass(className); // System.out.println(" -> success!"); classes.put(className,result); return result; } catch (ClassNotFoundException ignore) { // System.out.println(); classes.put(className,null); return null; } } } // ========= delegating methods ============= @Override public URL getResource( String name ) { return parent.getResource(name); } @Override public Enumeration<URL> getResources( String name ) throws IOException { return parent.getResources(name); } @Override public String toString() { return parent.toString(); } @Override public void setDefaultAssertionStatus(boolean enabled) { parent.setDefaultAssertionStatus(enabled); } @Override public void setPackageAssertionStatus(String packageName,boolean enabled) { parent.setPackageAssertionStatus(packageName,enabled); } @Override public void setClassAssertionStatus(String className,boolean enabled) { parent.setClassAssertionStatus(className,enabled); } @Override public void clearAssertionStatus() { parent.clearAssertionStatus(); } }
用法:
public static ClassLoader cachingClassLoader = new MyClassLoader(FXMLLoader.getDefaultClassLoader()); FXMLLoader loader = new FXMLLoader(resource); loader.setClassLoader(cachingClassLoader);
这显着加快了性能.但是,步骤2没有解决方法,因此这可能仍然是一个问题.
但是,官方JavaFX jira中已经有了这个功能请求.支持这个请求是很好的.
链接: