Ответ 1
Добавление к предложению @camickr:
Создайте BufferedImage (BI), оберните его в IconImage, установите его как значок для JLabel. Paint изменится на BI и вызовет JLabel repaint(), чтобы очистить их. изменения на экране. Кэширование частичного изображения в BI уменьшает объем работы для процедуры рисования. Он просто должен разжечь изображение, а не отображать изображение. Используйте SwingWorkers для развертывания фоновых потоков для выполнения вычислений и передачи результатов обратно в EDT для рисования.
Пока вы говорите о статическом изображении, это будет работать нормально. Если вы рассматриваете нечто вроде видеоигры (некоторые фиксированные изображения и другие движущиеся изображения), посмотрите на VolatileImage. Здесь есть хорошее описание: http://gpwiki.org/index.php/Java:Tutorials:VolatileImage
Update:
Ниже приведено немного более 80 кадров в секунду:
public class Demo extends javax.swing.JPanel {
private Image src = null;
public Demo() {
new Worker().execute();
}
protected void paintComponent(Graphics g) {
super.paintComponent(g);
if (src != null) g.drawImage(src, 0, 0, this);
}
private class Worker extends SwingWorker<Void, Image>{
private final Color[] colors = { Color.red, Color.green, Color.blue };
protected void process(List<Image> chunks){
for (Image bufferedImage : chunks){
src = bufferedImage;
repaint();
}
}
protected Void doInBackground() throws Exception{
int frames = 0;
int[] mem = new int[1024 * 768];
long start = System.currentTimeMillis();
long end = start + 15000;
long last = start;
while (last < end){
int col = colors[frames % colors.length].getRGB();
for (int y = 0; y < 768; y++)
for (int x = 0; x < 1024; x++)
mem[x + y * 1024] = col;
Image img = createImage(new MemoryImageSource(1024, 768, mem, 0, 1024));
BufferedImage bi = new BufferedImage(1024, 768, BufferedImage.TYPE_INT_ARGB);
Graphics2D g2 = bi.createGraphics();
g2.drawImage(img, 0, 0, null);
g2.dispose();
publish(bi);
last = System.currentTimeMillis();
frames++;
}
System.err.println("Frames = " + frames + ", fps = " + ((double) frames / (last - start) * 1000));
return null;
}
}
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run(){
JFrame jf = new JFrame();
jf.getContentPane().add(new Demo(), BorderLayout.CENTER);
jf.setSize(1024, 768);
jf.setVisible(true);
}
});
}
}