8

发生了一些奇怪的事情。我编写了一个 Java 程序,但我没有采取任何特殊方式处理未捕获的异常。但是当我在 Windows 7 中运行这个特定程序时,在从 main 调用的静态上下文中存在这个未捕获的异常,导致弹出一个窗口,显示异常。我试图编写一个小程序来复制这种效果,但无济于事。一个程序(我完全手工编写)会产生一个弹出窗口,而没有其他程序会这样做。

我想特别跟踪这一点,以便我可以添加使其他 CAUGHT 异常以类似方式显示堆栈跟踪的代码。

我在 IRC 中问过这个问题,但人们告诉我这不会发生。好吧,它确实发生了。下面有一个截图。

我认为我唯一的希望是如果其他人认识到这一点并能告诉我它来自哪里。

谢谢!

Java 异常

更新:很抱歉延迟获取一些代码。我不得不照顾一个绞痛的婴儿。请注意,这是一个桌面 Java 应用程序。它不是小程序,也不使用 webstart。

这是从获取对话框的程序中复制和粘贴的代码。我会再做一次编辑,让你知道我的同事(谁得到了例外)是否得到了这个案例的对话框。我一直小心地包括导致异常的所有内容。仅缺少 IPAddress 类实现,但这不参与异常,因为直到异常发生后才实际使用它。请注意发生异常的行之前的星号。该行代码与您在屏幕截图中看到的异常相对应。

package staticexception;

import java.net.InterfaceAddress;
import java.net.NetworkInterface;
import java.net.SocketException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Enumeration;
import java.util.List;
import javax.swing.UIManager;

public class StaticException {
    // Don't need this fully implemented.
    public static class IPAddress {

        public static IPAddress getBroadcast(IPAddress mask, IPAddress myip) {
            return new IPAddress();
        }

        public IPAddress() {}

        public IPAddress(int maskval) {}

        public IPAddress(byte[] addr) {}

        public IPAddress mask(IPAddress netmask) {
            return this;
        }

        public int prefixLength() {
            return 0;
        }

    }

    public static class Network {
        public IPAddress broadcast, netmask, ip;
        boolean remember;

        public Network(IPAddress br, IPAddress nm, IPAddress ip) {
            broadcast = br;
            netmask = nm;
            this.ip = ip;
        }

        boolean match(IPAddress ip) {
            IPAddress a = ip.mask(netmask);
            IPAddress b = this.ip.mask(netmask);
            return (a.equals(b));
        }

        @Override
        public String toString() {
            return ip.toString() + "/" + netmask.prefixLength();
        }
    }

    static List<Network> my_networks;

    static void enumerateNetworks() {
        my_networks = new ArrayList<Network>();

        Enumeration<NetworkInterface> nets = null;
        try {
            nets = NetworkInterface.getNetworkInterfaces();
        } catch (SocketException ex) {
            ex.printStackTrace();
        }
        for (NetworkInterface netint : Collections.list(nets)) {
            for (InterfaceAddress address : netint.getInterfaceAddresses()) {
                // *** Exception would occur on the next line when 
                // *** address.getAddress() would return null
                byte[] addr = address.getAddress().getAddress();
                if (addr.length == 4 && addr[0] != 127) {
                    int prefixlen = address.getNetworkPrefixLength();
                    int maskval = -1 << (32 - prefixlen);
                    IPAddress mask = new IPAddress(maskval);
                    //my_netmask = mask;
                    System.out.println("Netmask   = " + mask);

                    IPAddress myip = new IPAddress(addr);
                    //my_ip_address = myip;
                    System.out.println("Local IP  = " + myip);

                    IPAddress broadcast = IPAddress.getBroadcast(mask, myip);
                    System.out.println("Broadcast = " + broadcast);

                    my_networks.add(new Network(broadcast, mask, myip));

                    System.out.print(address.getAddress().getAddress().length + " ");
                    System.out.print(address.getAddress() + " ");
                    System.out.print(address.getAddress().getHostAddress() + " ");
                    System.out.println(address.getNetworkPrefixLength());
                }
            }
        }
    }

    static private void setupNetwork() {
        System.setProperty("java.net.preferIPv4Stack","true");

        enumerateNetworks();

        // ... stuff that would happen after the exception
    }

    public static void main(String[] args) {
        try {         
            UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());     
        } catch (Exception e) {}

        setupNetwork();

        // ... stuff that would happen after the exception
    }
}

第二次更新:我的同事报告说这个程序不会产生对话。这与获得弹出窗口的程序之间的唯一区别是获得弹出窗口的程序是从 AdvancedInstaller 生成的 exe 包装器启动的。除此之外,在 Java 程序中,执行顺序是相同的。我已经用谷歌搜索了这个,据我所知,AdvancedInstaller 根本没有做任何会导致生成此弹出窗口的操作。我不确定它是否可以在不修改 Java 程序的情况下(它没有),因为我不确定您是否可以从 Java 程序之外执行任何操作来实现这一点。除了可能捕获 stderr,但这并不能解释为什么由 AdvancedInstaller 包装的其他程序不'

4

1 回答 1

7

第二个答案 (在问题中添加了其他信息后)

高级安装程序具有如下所述的“启动失败检查”设置:

启动失败检查

任何来自主线程的未捕获异常都会在一个对话框中进行描述,该对话框将允许用户停止应用程序或忽略该异常。此选项仅适用于 GUI 应用程序。

更多信息: http: //www.advancedinstaller.com/user-guide/java-product-settings.html

第一个答案

这高度依赖于上下文,您没有提供足够的信息来获得非常具体的答案。loc2.LoC2.java 可能是自定义或项目特定代码。

不过,通常可以通过以下方式管理(或查询)未捕获的处理程序:

  • Thread.setDefaultUncaughtExceptionHandler
  • Thread.getDefaultUncaughtExceptionHandler

更多信息:

于 2012-07-31T18:59:28.003 回答