78

这可能是一个非常菜鸟的问题。我刚开始学习Java

我不明白paintComponent方法的操作。我知道如果我想画一些东西,我必须重写paintComponent 方法。

public void paintComponent(Graphics g)
{
   ...
}

但是什么时候调用呢?我从来没有见过像“object.paintComponent(g)”这样的东西,但它仍然是在程序运行时绘制的。

什么是 Graphics 参数?这个从哪里来?调用方法时必须提供参数。但正如我之前所说,似乎从未显式调用此方法。那么这个参数是谁提供的呢?为什么我们必须将其转换为 Graphics2D?

public void paintComponent(Graphics g)
{
    ...
    Graphics2D g2= (Graphics2D) g;
    ...
}
4

5 回答 5

51

对您的问题的(非常)简短的回答paintComponent是“当它需要时”。有时更容易将 Java Swing GUI 系统视为一个“黑盒”,其中大部分内部结构都在没有太多可见性的情况下处理。

决定何时需要重新绘制组件的因素有很多,包括移动、调整大小、更改焦点、被其他帧隐藏等等。其中许多事件是自动检测到的,并paintComponent在确定需要该操作时在内部调用。

我使用 Swing 已经很多年了,我认为我从来没有直接调用过paintComponent,甚至没有看到它直接从其他东西调用过。我最接近的是使用这些repaint()方法以编程方式触发某些组件的重绘(我假设它在paintComponent下游调用了正确的方法。

以我的经验,paintComponent很少被直接覆盖。我承认有些自定义渲染任务需要这种粒度,但 Java Swing 确实提供了一组(相当)健壮的 JComponents 和 Layouts,可以用来完成大部分繁重的工作,而无需直接覆盖paintComponent。我想我的意思是在你开始尝试推出你自己的自定义渲染组件之前,确保你不能用原生的 JComponents 和 Layouts 做一些事情。

于 2013-03-21T10:45:35.680 回答
16

你可以在这里做两件事:

  1. 在 AWT 和 Swing 中阅读绘画
  2. 使用调试器并在paintComponent 方法中放置一个断点。然后向上移动堆栈跟踪并查看如何提供 Graphics 参数。

仅供参考,这是我从最后发布的代码示例中获得的堆栈跟踪:

Thread [AWT-EventQueue-0] (Suspended (breakpoint at line 15 in TestPaint))  
    TestPaint.paintComponent(Graphics) line: 15 
    TestPaint(JComponent).paint(Graphics) line: 1054    
    JPanel(JComponent).paintChildren(Graphics) line: 887    
    JPanel(JComponent).paint(Graphics) line: 1063   
    JLayeredPane(JComponent).paintChildren(Graphics) line: 887  
    JLayeredPane(JComponent).paint(Graphics) line: 1063 
    JLayeredPane.paint(Graphics) line: 585  
    JRootPane(JComponent).paintChildren(Graphics) line: 887 
    JRootPane(JComponent).paintToOffscreen(Graphics, int, int, int, int, int, int) line: 5228   
    RepaintManager$PaintManager.paintDoubleBuffered(JComponent, Image, Graphics, int, int, int, int) line: 1482 
    RepaintManager$PaintManager.paint(JComponent, JComponent, Graphics, int, int, int, int) line: 1413  
    RepaintManager.paint(JComponent, JComponent, Graphics, int, int, int, int) line: 1206   
    JRootPane(JComponent).paint(Graphics) line: 1040    
    GraphicsCallback$PaintCallback.run(Component, Graphics) line: 39    
    GraphicsCallback$PaintCallback(SunGraphicsCallback).runOneComponent(Component, Rectangle, Graphics, Shape, int) line: 78    
    GraphicsCallback$PaintCallback(SunGraphicsCallback).runComponents(Component[], Graphics, int) line: 115 
    JFrame(Container).paint(Graphics) line: 1967    
    JFrame(Window).paint(Graphics) line: 3867   
    RepaintManager.paintDirtyRegions(Map<Component,Rectangle>) line: 781    
    RepaintManager.paintDirtyRegions() line: 728    
    RepaintManager.prePaintDirtyRegions() line: 677 
    RepaintManager.access$700(RepaintManager) line: 59  
    RepaintManager$ProcessingRunnable.run() line: 1621  
    InvocationEvent.dispatch() line: 251    
    EventQueue.dispatchEventImpl(AWTEvent, Object) line: 705    
    EventQueue.access$000(EventQueue, AWTEvent, Object) line: 101   
    EventQueue$3.run() line: 666    
    EventQueue$3.run() line: 664    
    AccessController.doPrivileged(PrivilegedAction<T>, AccessControlContext) line: not available [native method]    
    ProtectionDomain$1.doIntersectionPrivilege(PrivilegedAction<T>, AccessControlContext, AccessControlContext) line: 76    
    EventQueue.dispatchEvent(AWTEvent) line: 675    
    EventDispatchThread.pumpOneEventForFilters(int) line: 211   
    EventDispatchThread.pumpEventsForFilter(int, Conditional, EventFilter) line: 128    
    EventDispatchThread.pumpEventsForHierarchy(int, Conditional, Component) line: 117   
    EventDispatchThread.pumpEvents(int, Conditional) line: 113  
    EventDispatchThread.pumpEvents(Conditional) line: 105   
    EventDispatchThread.run() line: 90  

Graphics 参数来自这里:

RepaintManager.paintDirtyRegions(Map) line: 781 

涉及的片段如下:

Graphics g = JComponent.safelyGetGraphics(
                        dirtyComponent, dirtyComponent);
                // If the Graphics goes away, it means someone disposed of
                // the window, don't do anything.
                if (g != null) {
                    g.setClip(rect.x, rect.y, rect.width, rect.height);
                    try {
                        dirtyComponent.paint(g); // This will eventually call paintComponent()
                    } finally {
                        g.dispose();
                    }
                }

如果你看一下它,你会看到它从 JComponent 本身(间接地javax.swing.JComponent.safelyGetGraphics(Component, Component))检索图形,它本身最终从它自己从它的第一个“重量级父”(剪辑到组件边界)中获取它对应的原生资源。

关于您必须将a 强制Graphics转换为 a的事实,Graphics2D碰巧在使用 Window Toolkit 时,Graphics实际上 extends Graphics2D,但您可以使用其他Graphics“不必” extends Graphics2D(它不经常发生,但 AWT/ Swing 允许您这样做)。

import java.awt.Color;
import java.awt.Graphics;

import javax.swing.JFrame;
import javax.swing.JPanel;

class TestPaint extends JPanel {

    public TestPaint() {
        setBackground(Color.WHITE);
    }

    @Override
    public void paintComponent(Graphics g) {
        super.paintComponent(g);
        g.drawOval(0, 0, getWidth(), getHeight());
    }

    public static void main(String[] args) {
        JFrame jFrame = new JFrame();
        jFrame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        jFrame.setSize(300, 300);
        jFrame.add(new TestPaint());
        jFrame.setVisible(true);
    }
}
于 2013-03-21T10:41:51.690 回答
3

GUI 系统的内部调用该方法,并将Graphics参数作为您可以在其上绘制的图形上下文传递。

于 2013-03-21T10:11:54.473 回答
2

调用object.paintComponent(g)是错误的。

而是在创建面板时自动调用此方法。该paintComponent()方法也可以由类中repaint()定义的方法显式调用Component

调用的效果repaint()是Swing自动清空面板上的图形,并执行paintComponent方法重绘该面板上的图形。

于 2014-10-23T00:17:32.900 回答
0

void paintComponent(Graphics g){}如果您希望任何先前的绘图在组件上永久存在,您可能必须重新定义该方法。您需要通过显式调用升序类的方法来做到这一点,例如super.painComponent();. 这样,任何时候java都需要使用该paintComponent方法,您将保持所做的更改。

这是因为如果你不这样做,超类将通过简单地调用它自己的方法来撤销你所做的一切,完全忽略任何更改。

于 2021-03-26T17:36:28.533 回答