我主要为精通技术的人写一些小工具,例如:程序员,工程师等。由于这些工具通常会随着时间的推移而快速改进,我知道会有未处理的异常并且用户不会介意。我希望用户能够向我发送回溯,以便我可以检查发生了什么并可能改进应用程序。
我通常做wxPython编程,但最近我做了一些Java。我已将TaskDialog
类连接到Thread.UncaughtExceptionHandler()
,我对结果非常满意。特别是它可以捕获和处理来自任何线程的异常:
我在wxPython中做了类似的事情很长一段时间。但是:
这是Java和wxPython的代码,所以你可以看到我做了什么:
爪哇:
import java.awt.EventQueue;
import javax.swing.JFrame;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.JButton;
import java.awt.GridBagLayout;
import java.awt.GridBagConstraints;
import java.awt.event.ActionListener;
import java.awt.event.ActionEvent;
import com.ezware.dialog.task.TaskDialogs;
public class SwingExceptionTest {
private JFrame frame;
public static void main(String[] args) {
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
}
catch (ClassNotFoundException e) {
}
catch (InstantiationException e) {
}
catch (IllegalAccessException e) {
}
catch (UnsupportedLookAndFeelException e) {
}
Thread.setDefaultUncaughtExceptionHandler(new Thread.UncaughtExceptionHandler() {
public void uncaughtException(Thread t, Throwable e) {
TaskDialogs.showException(e);
}
});
EventQueue.invokeLater(new Runnable() {
public void run() {
try {
SwingExceptionTest window = new SwingExceptionTest();
window.frame.setVisible(true);
} catch (Exception e) {
e.printStackTrace();
}
}
});
}
public SwingExceptionTest() {
initialize();
}
private void initialize() {
frame = new JFrame();
frame.setBounds(100, 100, 600, 400);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
GridBagLayout gridBagLayout = new GridBagLayout();
gridBagLayout.columnWidths = new int[]{0, 0};
gridBagLayout.rowHeights = new int[]{0, 0};
gridBagLayout.columnWeights = new double[]{0.0, Double.MIN_VALUE};
gridBagLayout.rowWeights = new double[]{0.0, Double.MIN_VALUE};
frame.getContentPane().setLayout(gridBagLayout);
JButton btnNewButton = new JButton("Throw!");
btnNewButton.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent arg0) {
onButton();
}
});
GridBagConstraints gbc_btnNewButton = new GridBagConstraints();
gbc_btnNewButton.gridx = 0;
gbc_btnNewButton.gridy = 0;
frame.getContentPane().add(btnNewButton, gbc_btnNewButton);
}
protected void onButton(){
Thread worker = new Thread() {
public void run() {
throw new RuntimeException("Exception!");
}
};
worker.start();
}
}
的wxPython:
import StringIO
import sys
import traceback
import wx
from wx.lib.delayedresult import startWorker
def thread_guard(f):
def thread_guard_wrapper(*args, **kwargs) :
try:
r = f(*args, **kwargs)
return r
except Exception:
exc = sys.exc_info()
output = StringIO.StringIO()
traceback.print_exception(exc[0], exc[1], exc[2], file=output)
raise Exception("<THREAD GUARD>\n\n" + output.getvalue())
return thread_guard_wrapper
@thread_guard
def thread_func():
return 1 / 0
def thread_done(result):
r = result.get()
print r
class MainWindow(wx.Frame):
def __init__(self, *args, **kwargs):
wx.Frame.__init__(self, *args, **kwargs)
self.panel = wx.Panel(self)
self.button = wx.Button(self.panel, label="Throw!")
self.button.Bind(wx.EVT_BUTTON, self.OnButton)
self.sizer = wx.BoxSizer()
self.sizer.Add(self.button)
self.panel.SetSizerAndFit(self.sizer)
self.Show()
def OnButton(self, e):
startWorker(thread_done, thread_func)
app = wx.App(True)
win = MainWindow(None, size=(600, 400))
app.MainLoop()
现在的问题是:
我可以在wxPython中轻松完成与Java解决方案类似的操作吗?或者,在Java或wxPython中有更好的方法吗?
答案 0 :(得分:3)
在Python中,您可以将sys.execpthook
设置为要为未捕获的异常调用的函数。那么你不需要装饰器,你可以在钩子函数中集中处理异常。
而不是仅打印异常回溯文本并让它显示在stock stdout窗口中,您可以使用它做一些更智能的操作,比如使用对话框显示文本并具有允许用户发送错误的控件将信息反馈给开发人员,忽略未来错误,重新启动应用程序或任何您想要的信息。
答案 1 :(得分:3)
在Java中,如果TaskDialog
不可用,您可以使用JOptionPane
,如图here所示。在event dispatch thread以外的其他帖子中,按照建议here使用EventQueue.invokeLater()
打包电话。另请考虑添加可选配置以调用Desktop#mail()
。