当前位置: 首页 > 面试题库 >

JavaFX2-将自定义(fxml)面板动态添加到gridpane时,性能非常差

贾兴学
2023-03-14
问题内容

问题 我想在运行时将通过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都必须通过反射来查找类及其属性。这意味着:

  1. 对于每个import语句,尝试加载每个类,直到可以成功加载该类。
  2. 对于每个类,创建一个BeanAdapter来查找该类具有的所有属性,并尝试将给定的参数应用于该属性。
  3. 再次通过反射将参数应用于属性。
    当前load(),在代码中对后续对同一FXML文件的调用也没有任何改进。这意味着:不缓存找到的类,不缓存BeanAdapters等等。

但是,通过将自定义类加载器设置为FXMLLoader实例,可以解决第一步的性能问题:


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中已经有为此功能的请求。您很高兴支持此请求。

链接:

  • FXMLLoader应该能够在load()调用之间缓存导入和属性:https ://bugs.openjdk.java.net/browse/JDK-8090848

  • 将setAdapterFactory()添加到FXMLLoader:https ://bugs.openjdk.java.net/browse/JDK-8102624



 类似资料:
  • 问题我想在运行时向gridpane添加通过javafx scene builder构建的定制面板。我的定制面板有按钮,标签等。 我的尝试我试图从窗格扩展... ...然后在conroller的添加方法中使用此面板 它工作,但它的表现非常非常差。 我想要的是有一种方法可以通过javafx scene builder设计面板(结果是在fxml中使用这个面板),然后在运行时将其添加到gridpane中,

  • 我对java和javafx很陌生,有一个我无法解决的问题。我需要将新的自定义控件动态添加到 javafx 场景中。此外,我需要主控件和添加的控件之间的交互。我已经在网上找到了一些有用的信息,但无法将它们放在一起。 所以我举了一个小例子来解释: 主要类别: 主fxml: 及其控制器: 现在要动态添加的控件: 其影响: 和控制器: 用法是:应将fxml2添加到fxml1的hbox中。然后在fxml1中

  • 我正在构建一个Java程序。该程序的核心在JFrame中可视化,其中包含一个JMenuBar和各种JMenuItem和JMenu。关键是我在所有框架中添加了一个centralPanel,但是如果我在centralPanel中添加了一些内容,那么只有在调整主框架的大小、缩小或放大它时,它才会显示出来!代码如下: 这是构造函数: 在这里,我添加了中央面板,在这里,在ActionListener中,我试

  • 问题内容: 我已经构建了自己的自定义react-bootstrap Popover组件: 该组件的呈现方式如下: 现在,我想向组件中添加自定义道具,例如:我的文字,并使用新道具在弹出框中设置一些内容,例如- 但随后我在浏览器中收到此警告: 警告:标签上的未知道具。从元素中删除这些道具。 现在,我想我可以删除零件并逐个插入所有原始道具,而无需自定义道具,但是这样我就失去了“淡入淡出”效果,这也是处理

  • 这里的 Spring 文档 http://docs.spring.io/spring-data/data-jpa/docs/current/reference/html/repositories.html#repositories.custom-implementations 提供了将自定义功能添加到所有存储库或单个存储库的示例,而不是同时添加到两者。 假设我想向所有存储库添加一些自定义函数(使用

  • 我对JavaFX相对较新,我很困惑为什么我下面的代码没有产生预期的结果,即添加到网格中的标签。 我想做的是运行一个测试,将JavaFX标签添加到我的FXML GridPane,因为我想在不久的将来构建一个方法,允许用户选择一个文件,然后在用户选择文件时生成一个标签并将该标签添加到GridPane。 提前感谢, 代码: FXML代码是一个标准文件,其中定义了一个网格窗格,上面列出了fx:id。