2

すでに実装されているパフォーマンス ブースター: - 描画する GraphicsConfiguration の互換性のあるイメージを取得します - 1.5 で OpenGL パイプラインを有効にします: 深刻なアーティファクトのため不可能です

プログラムの主なボトルネックは、数千のタイルで画像を描画することです。残念ながら、それは規則的ではありません。それ以外の場合は、ピクセルを設定してスケーリングするだけです。VolatileImages と独自のレンダリング ルーチンを使用して画像を高速化しました (再描画を無視し、タイマーでそれ自体を描画します)。結果は満足のいくものであり、十分でした。ただし、JMenu がオーバードローされているため、画像の一部に通常どおりホバーする JMenu を選択すると、非常に妨げられます。容認できず、レイアウトを変更できませんでした。

JOGLのGLJPanelを試してみましたが、目に見えるパフォーマンスの向上は見られません。VolatileImages (または GLCanvas のような他の高速化された軽量コンポーネント) を使用して、通常の JMenu 表示を取得する可能性はありますか?

4

3 に答える 3

2

ポップアップを非軽量に設定してみてください。それが機能するかどうかはよくわかりませんが、ポップアップはネイティブコンポーネントであり、上書きされないため、機能する可能性があります。ポップアップをヘビーウェイトに設定: JPopupMenu.setDefaultLightWeightPopupEnabled(false)

詳細情報:重いコンポーネントと軽いコンポーネントの混合

于 2008-10-30T11:40:38.130 に答える
1

コード例を次に示します。

import javax.swing.*;
import java.awt.*;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.awt.image.BufferStrategy;
import java.awt.image.BufferedImage;
import java.awt.image.VolatileImage;
import java.io.File;
import java.io.IOException;
import java.util.Timer;
import java.util.TimerTask;

public final class FastDraw extends JFrame {

    private static final transient double NANO = 1.0e-9;
    private BufferStrategy bs;
    private BufferedImage frontImg;
    private BufferedImage backImg;
    private int PIC_WIDTH,
    PIC_HEIGHT;
    private Timer timer;

    public FastDraw() {
        timer = new Timer(true);
        JMenu menu = new JMenu("Dummy");
        menu.add(new JMenuItem("Display me !"));
        menu.add(new JMenuItem("Display me, too !"));
        JMenuBar menuBar = new JMenuBar();
        menuBar.add(menu);
        setJMenuBar(menuBar);
        setIgnoreRepaint(true);
        setVisible(true);

        addWindowListener(new WindowAdapter() {
            public void windowClosing(WindowEvent evt) {
                super.windowClosing(evt);
                timer.cancel();
                dispose();
                System.exit(0);
            }
        });

        try {
            backImg = javax.imageio.ImageIO.read(new File(<insert a jpg picture here>));
            frontImg = javax.imageio.ImageIO.read(<here, too>));
        }
        catch (IOException e) {
            System.out.println(e.getMessage());
        }

        PIC_WIDTH = backImg.getWidth();
        PIC_HEIGHT = backImg.getHeight();
        setSize(PIC_WIDTH, PIC_HEIGHT);

        createBufferStrategy(1); // Double buffering
        bs = getBufferStrategy();
        timer.schedule(new Drawer(),0,20);

    }

    public static void main(String[] args) {
        new FastDraw();
    }

    private class Drawer extends TimerTask {
        private VolatileImage img;

        public void run() {
            long begin = System.nanoTime();
            Graphics2D  g  = (Graphics2D) bs.getDrawGraphics();
            GraphicsConfiguration gc = g.getDeviceConfiguration();

            if (img == null)
                img = gc.createCompatibleVolatileImage(PIC_WIDTH, PIC_HEIGHT);

            Graphics2D g2 = img.createGraphics();

            do {
                int valStatus = img.validate(gc);

                if (valStatus == VolatileImage.IMAGE_OK)
                    g2.drawImage(backImg,0,0,null);
                else {
                    g.drawImage(frontImg, 0, 0, null);
                }
                // volatile image is ready
                g.drawImage(img,0,50,null);
                bs.show();
            } while (img.contentsLost());
        }
    }
}

ウィンドウのサイズを変更して、JMenuBar表示できるようにします。メニューポイントを選択してみてください。見る?

于 2008-10-30T11:51:59.747 に答える
0

問題を完全に理解しているかどうかはわかりませんが、主な問題は画像が表示されるパネルの再描画にあるようです。パネルに画像を描画する描画ルーチンを提供してください。

于 2008-10-30T10:51:36.967 に答える