怎么实现关闭窗口

思路:在java源文件中查找源代码
首先,我们知道JFrame.setDefaultCloseOperation(int operation)中提供了四种方式去实现关闭窗口
JFrame在包javax.Swing中,所以他继承了java.awt.Frame

setDefaultCloseOperation(int operation)

下面贴上 JFrame.setDefaultCloseOperation(int operation) 的代码

    public void setDefaultCloseOperation(int operation) {
    //判断参数是否正确,错误则抛出非法参数异常
        if (operation != DO_NOTHING_ON_CLOSE &&
            operation != HIDE_ON_CLOSE &&
            operation != DISPOSE_ON_CLOSE &&
            operation != EXIT_ON_CLOSE) {
            throw new IllegalArgumentException("defaultCloseOperation must be one of: DO_NOTHING_ON_CLOSE, HIDE_ON_CLOSE, DISPOSE_ON_CLOSE, or EXIT_ON_CLOSE");
        }

        if (operation == EXIT_ON_CLOSE) {
            SecurityManager security = System.getSecurityManager();、
            //lang.SecurityManager是一个安全管理器,可以预判操作是否正确,以决定是否要执行这个操作
            //判断当前应用程序是否建立了安全管理器,如果没建立,则返回Null,此时不可以强退
            if (security != null) {
                security.checkExit(0);
            }
        }
        //判断改变前后属性是否改变,不改变则不用执行操作
        if (this.defaultCloseOperation != operation) {
            int oldValue = this.defaultCloseOperation;
            this.defaultCloseOperation = operation;
            //JFrame完全继承了父类Frame的firePropertyChang的方法
            firePropertyChange("defaultCloseOperation", oldValue, operation);
        }
    }

类SecurityManager是在java.lang包中的
可以通过静态方法System.getSecurityManager()获得当前程序的安全管理器(如果没有建立安全管理器,则返回null)
checkExit(int status):int
如果不允许调用线程使用特定的状态码暂停 Java 虚拟机,则抛出 SecurityException。
所以可以来判断此时退出是否安全

四种参数表示的意思:
DO_NOTHING_ON_CLOSE(在 WindowConstants中定义):不执行任何操作;要求程序在已注册的 WindowListener 对象的 windowClosing 方法中处理该操作。
HIDE_ON_CLOSE(在 WindowConstants 中定义):调用任意已注册的 WindowListener 对象后自动隐藏该窗体。
DISPOSE_ON_CLOSE(在 WindowConstants 中定义):调用任意已注册 WindowListener 的对象后自动隐藏并释放该窗体。
EXIT_ON_CLOSE(在 JFrame 中定义):使用 System exit 方法退出应用程序。仅在应用程序中使用。

WindowConstans(窗口常量)是在包Javax.Swing中的接口
然而窗口即使不注册指定的监视器也可以最大化最小化,所以推测,当初始化窗口的时候,窗口会默认注册监视器

从上面看出,下面要去找firePropertyChange()方法

firePropertyChange()

从方法名可以看出,这个方法是用于改变配置的
又这个方法在Component中,所以一定还有其他的修改配置方法会通过firePropertyChange()方法去修改
如果改变颜色啊 字体啊 大小啊 什么什么的

firePropertyChange()在java.awt.Component中

下面贴出在java.awt.Component中的firePropertyChange()

    protected void firePropertyChange(String propertyName,
                                      Object oldValue, Object newValue) {
        PropertyChangeSupport changeSupport;
        synchronized (getObjectLock()) {
            changeSupport = this.changeSupport;
        }
        //在类中找到了changeSupport 的定义 但是没有找到他的初始化 这是怎么回事?
        if (changeSupport == null ||
            (oldValue != null && newValue != null && oldValue.equals(newValue))) {
            return;
        }
        changeSupport.firePropertyChange(propertyName, oldValue, newValue);
    }
    //重载形式,参数为int
    protected void firePropertyChange(String propertyName,
                                      int oldValue, int newValue) {
        PropertyChangeSupport changeSupport = this.changeSupport;
        if (changeSupport == null || oldValue == newValue) {
            return;
        }
        changeSupport.firePropertyChange(propertyName, oldValue, newValue);
    }

从上面看 我们还需要去PropertyChangeSupport中去找

PropertyChangeSupport中的firePropertyChange()

贴代码

    public void firePropertyChange(String propertyName, Object oldValue, Object newValue) {
        if (oldValue == null || newValue == null || !oldValue.equals(newValue)) {
            firePropertyChange(new PropertyChangeEvent(this.source, propertyName, oldValue, newValue));
        }
    }
  //重载形式,参数为整型
   public void firePropertyChange(String propertyName, int oldValue, int newValue) {
        if (oldValue != newValue) {
        //将int参数包装成Integer对象,再调用参数为Object的方法
            firePropertyChange(propertyName, Integer.valueOf(oldValue), Integer.valueOf(newValue));
        }
    }
   public void firePropertyChange(PropertyChangeEvent event) {
        Object oldValue = event.getOldValue();
        Object newValue = event.getNewValue();
        if (oldValue == null || newValue == null || !oldValue.equals(newValue)) {
            String name = event.getPropertyName();

            PropertyChangeListener[] common = this.map.get(null);
            PropertyChangeListener[] named = (name != null)
                        ? this.map.get(name)
                        : null;

            fire(common, event);
            fire(named, event);
        }
    }
    //实际在监听器中改变事件
    private static void fire(PropertyChangeListener[] listeners, PropertyChangeEvent event) {
        if (listeners != null) {
            for (PropertyChangeListener listener : listeners) {
                listener.propertyChange(event);
            }
        }
    }

然后一查 PropertyChangeListener 是一个接口,有好多好多的实现类 , 然后我就不知道怎么查了

其实从那四个参数的意思上面看也能看出个大概

答案

DO_NOTHING_ON_CLOSE:什么都不做,那我就根本不用实现关闭按钮了
HIDE_ON_CLOSE:隐藏窗口,那我把窗口设为invisible就可以啦
DISPOSE_ON_CLOSE:退出时释放窗口,那就把窗口对象清理掉就行拉
EXIT_ON_CLOSE:这个肯定是直接退出程序啦

原文地址:https://www.cnblogs.com/wewill/p/5588758.html