我创建了一个包含多个文本节点的动画。用户应该在从服务器接收到文本时阅读它。问题是,在几分钟内(大约5分钟),性能开始下降。从60 fps降至30 fps及以下。因此,文本很难阅读。
编辑2:
我创建了一个最小、完整和可验证的示例:
项目中有3个文件:
MainFxApp:
即使是这个简单的例子,性能下降也非常明显。场景中有20个节点,FPS下降到20以下。我的CPU是i5-4440 CPU(3.10GHz × 4)。无论我在哪个平台上测试它 - JavaFX 8、JavaFX 9和Ubuntu 16.04,问题都会出现。
编辑3:
该问题似乎只存在于Linux平台上。
然而,即使在Windows上,当我通过Platform.runLater方法频繁访问JavaFX线程时,动画似乎也不太流畅,尽管它保持在60fps。有人知道如何改进吗?
谢谢,Jan
编辑2:
我创建了一个最小、完整和可验证的示例:
项目中有3个文件:
MainFxApp:
import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.layout.Pane;
import javafx.scene.paint.Color;
import javafx.scene.text.Font;
import javafx.stage.Stage;
import java.util.List;
public class MainFxApp extends Application {
@Override
public void start(Stage primaryStage) {
Pane root = new Pane();
root.setStyle("-fx-background-color: black;");
MyAnimationTimer myAnimationTimer = new MyAnimationTimer((List<MyText>) (List<?>) root.getChildren());
MyText newText;
for (int i = 65; i < 85; i++) {
newText = new MyText("" + ((char) i));
newText.setFill(Color.GREEN);
newText.setFont(Font.font(40));
myAnimationTimer.addNode(newText);
}
Scene scene = new Scene(root, 1200, 600);
primaryStage.setTitle("Performance test");
primaryStage.setScene(scene);
primaryStage.show();
myAnimationTimer.start();
}
public static void main(String[] args) {
launch(args);
}
}
我的动画计时器:
import javafx.animation.AnimationTimer;
import javafx.application.Platform;
import javafx.scene.CacheHint;
import javafx.scene.Node;
import java.util.List;
public class MyAnimationTimer extends AnimationTimer {
private List<MyText> nodes;
private double panelWidth;
private double panelHeight;
private double basicVelocity; // Distance per nanosecond
private long lastFrameTime;
private long timeCount = 0;
private int frameCount = 0;
MyAnimationTimer(List<MyText> nodes) {
super();
this.nodes = nodes;
this.panelWidth = 1200;
this.panelHeight = 600;
this.setBasicVelocity();
}
@Override
public void start() {
this.lastFrameTime = System.nanoTime();
super.start();
}
@Override
public void handle(long now) {
long deltaT = now - lastFrameTime;
double deltaX = this.basicVelocity * deltaT;
for (MyText node : this.nodes) {
node.setTranslateX(node.getTranslateX() + node.direction * deltaX);
if (node.getTranslateX() < 0) {
node.direction = 1;
} else if (node.getTranslateX() > 1200) {
node.direction = -1;
}
}
this.lastFrameTime = now;
this.timeCount += deltaT;
this.frameCount++;
if (timeCount > 1000000000) {
System.out.println(this.frameCount / (double) timeCount * 1000000000);
this.frameCount = 0;
this.timeCount = 0;
}
}
void addNode(final MyText node) { // Not sure about the final thing
Platform.runLater(() -> {
node.setCache(true);
node.setCacheHint(CacheHint.SPEED);
node.setTranslateY(panelHeight / 2);
double nodePositionX = panelWidth - 20;
if (nodes.size() >= 1) {
Node lastNode = nodes.get(nodes.size() - 1);
double lastNodeEnd = lastNode.getTranslateX() + 50;
if (lastNodeEnd > nodePositionX) {
nodePositionX = lastNodeEnd;
}
}
node.setTranslateX(nodePositionX);
nodes.add(node);
});
}
private void setBasicVelocity() {
Platform.runLater(() -> {
this.basicVelocity = ((panelWidth / 4) * 3 / (double) 5000 / 1000000.0);
});
}
}
我的文本:
import javafx.scene.text.Font;
import javafx.scene.text.Text;
class MyText extends Text {
int direction = -1;
MyText(String text) {
super(text);
this.setFont(new Font("Arial Regular", 40));
}
}
即使是这个简单的例子,性能下降也非常明显。场景中有20个节点,FPS下降到20以下。我的CPU是i5-4440 CPU(3.10GHz × 4)。无论我在哪个平台上测试它 - JavaFX 8、JavaFX 9和Ubuntu 16.04,问题都会出现。
编辑3:
该问题似乎只存在于Linux平台上。
然而,即使在Windows上,当我通过Platform.runLater方法频繁访问JavaFX线程时,动画似乎也不太流畅,尽管它保持在60fps。有人知道如何改进吗?
谢谢,Jan
-Dprism.verbose=true
运行吗?另外,请查看此问题:https://dev59.com/2Zvga4cB1Zd3GeqP9OkZ#40239829 - Itai