如何解决更改JButtons背景的最佳方法
|| 现在我通过使用更改按钮的背景色button.setBackground(Color.WHITE);
那是一个例子。
但是当我的jbuttons(1000+)中有一个巨大的网格时,仅运行for循环来更改每个按钮的背景就非常非常慢。您可以看到网格慢慢地变成白色,一格一格。我真的不想要这个
有没有更好的方法可以同时将网格上的每个JButton更改为相同的颜色?
这就是我制作网格的方式,使用的数字仅是示例...
grid = new JPanel(new GridLayout(64,64,0));
这是4096个按钮,大约需要30秒钟以上才能将每个按钮更改为相同的颜色。
编辑1:我需要按钮是可单击的,例如当我单击一个按钮时,它会变成蓝色。单击所有按钮后,将每个按钮的颜色更改为白色。现在我可以正常工作,但是更改每个按钮的颜色只是很慢。
编辑2:这就是我更改按钮的方式:
new javax.swing.Timer(300,new ActionListener() {
int counter = 0;
public void actionPerformed(ActionEvent e) {
if (counter >= counterMax) {
((Timer) e.getSource()).stop();
}
Color bckgrndColor = (counter % 2 == 0) ? flashColor : Color.white;
for (JButton button : gridButton) {
button.setBackground(bckgrndColor);
}
counter++;
}
}).start();
解决方法
您看到各个框分别被重新粉刷的事实表明关闭了双缓冲,或者按钮UI中的粉刷代码使用了
paintImmediately()
。
我使用64x64 JButton测试了您的设置,确保所有UI操作均在EDT(事件调度线程)中执行。我可以确认您看到的效果,更改所有按钮的背景大约需要1200毫秒,并且每个框都会立即重新绘制。
您可以通过将网格设置为不可见之前和更改背景后可见的方式来绕过立即重绘:
grid.setVisible(false);
for (Component comp : grid.getComponents()) {
comp.setBackground(color);
}
grid.setVisible(true);
这导致网格仅执行一次重绘,并将时间减少到〜300ms(因数4)。
对于频繁的更新而言,这仍然太慢,因此最好使用一个自定义组件来绘制网格,或者如果您希望允许网格单元是任意的,则可以使用一个轻量级的容器(问题注释中建议使用的垃圾桶)。组件。
,如果只需要重绘可见的按钮,那么您将获得可观的收益。在下面显示的MVC方法中,每个按钮都侦听一个定义其当前状态的模型。与重画相比,更新模型的速度非常快。尽管启动需要花费几秒钟,但我发现更新需要不到10毫秒的时间。在稳态。它的可扩展性不如ѭ5the所示的flyweight模式,但可以使用。
import java.awt.*;
import java.awt.event.*;
import java.util.Observable;
import java.util.Observer;
import java.util.Random;
import javax.swing.*;
/** @see https://stackoverflow.com/questions/6117908 */
public class UpdateTest {
private static final int ROW = 64;
private static final int COL = 64;
private static final int MAX = COL * ROW;
private final DataModel model = new DataModel(MAX);
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
new UpdateTest().create();
}
});
}
void create() {
JPanel panel = new JPanel();
panel.setLayout(new GridLayout(ROW,COL));
for (int i = 0; i < MAX; i++) {
panel.add(new ViewPanel(model,i));
}
Timer timer = new Timer(1000,new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
long start = System.nanoTime();
model.update();
System.out.println(
(System.nanoTime() - start) / (1000 * 1000));
}
});
JFrame f = new JFrame(\"JTextTest\");
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f.add(new JScrollPane(panel),BorderLayout.CENTER);
f.setPreferredSize(new Dimension(800,600));
f.pack();
f.setLocationRelativeTo(null);
f.setVisible(true);
timer.start();
}
private static class ViewPanel extends JPanel implements Observer {
private final JButton item = new JButton();
private DataModel model;
private int index;
public ViewPanel(DataModel model,int i) {
this.model = model;
this.index = i;
this.add(item);
item.setText(String.valueOf(i));
item.setOpaque(true);
item.setBackground(new Color(model.get(index)));
model.addObserver(this);
}
@Override
public void update(Observable o,Object arg) {
int value = model.get(index);
item.setBackground(new Color(value));
}
}
private static class DataModel extends Observable {
private final Random rnd = new Random();
private final int[] data;
public DataModel(int n) {
data = new int[n];
fillData();
}
public void update() {
fillData();
this.setChanged();
this.notifyObservers();
}
public int get(int i) {
return data[i];
}
private void fillData() {
for (int i = 0; i < data.length; i++) {
data[i] = rnd.nextInt();
}
}
}
}
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。