如何命名 SwingWorker 线程?开放代码或最佳实践

发布于 2024-08-16 02:32:49 字数 168 浏览 6 评论 0原文

我正在尝试调试 SwingWorker 线程中发生的可怕异常。如果我可以命名我的 SwingWorker 线程,那么我可以更好地解释我收到的堆栈跟踪。

似乎不可能命名 SwingWorker 线程(是吗?)。我正在寻找任何最佳实践或“技巧”来帮助我正确识别我的工作线程,而不仅仅是在我的代码中添加日志语句。

I'm trying to debug a horrible exception that is occurring in a SwingWorker thread. If I could name my SwingWorker threads, then I could better interpret the stack traces I am receiving.

It doesn't seem possible to name a SwingWorker thread (is it?). I'm looking for any best practices or 'tricks' to help me properly identify my worker threads, short of just peppering my code w/ log statements.

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(5

錯遇了你 2024-08-23 02:32:49

虽然您无权访问 SwingWorker 正在使用的 ThreadFactory,但您可以在 SwingWorker 正在运行的任何类的 run() 或 call() 方法中调用 Thread.currentThread().setName() 。

只需确保在线程实际执行代码时设置该值即可

While you don't have access to the ThreadFactory that SwingWorker is using, you can call Thread.currentThread().setName() in your run() or call() method of whatever class the SwingWorker is running.

Just be sure that you're setting that value when the thread is actually executing the code

欲拥i 2024-08-23 02:32:49

SwingWorker 使用的线程来自 Executors.defaultThreadFactory,这意味着它来自池。因此,如果“MyXYZThread”被系统中另一个无法重命名线程的 SwingWorker 重用,则命名线程可能会产生误导。

我将扩展 SwingWorker 以包含名称并在引发异常时记录该名称。

注意:我不想让调用者负责处理异常,但我没有找到一种方法可以在不改变 SwingWorker 合同的情况下做到这一点。

private static abstract class MySwingWorker<X, Y> extends SwingWorker<X, Y>
{
    private final String taskName;
    public MySwingWorker(String name)
    {
        this.taskName = name;
    }
    public String getTaskName() {
        return taskName;
    }
}

用法:

new MySwingWorker<Object,Object>("my task") {
    protected Object doInBackground() {
        try {
            return someCall();
        }
        catch(Exception e) {
            Logger.getLogger(getClass().getName()).log(Level.SEVERE, 
                "Exception in " + getTaskName(), e);
            throw e;
        }
    }
}.execute();

The Thread used by SwingWorker is from an Executors.defaultThreadFactory which means it's from a pool. So, naming the thread could be misleading if "MyXYZThread" gets reused for another SwingWorker in the system that fails to rename the thread.

I'd extend the SwingWorker to include a name and log that name when throwing an exception.

Note: I'd rather not make the caller responsible for handling the exception, but I don't see a way to do it without altering the contract of SwingWorker.

private static abstract class MySwingWorker<X, Y> extends SwingWorker<X, Y>
{
    private final String taskName;
    public MySwingWorker(String name)
    {
        this.taskName = name;
    }
    public String getTaskName() {
        return taskName;
    }
}

Usage:

new MySwingWorker<Object,Object>("my task") {
    protected Object doInBackground() {
        try {
            return someCall();
        }
        catch(Exception e) {
            Logger.getLogger(getClass().getName()).log(Level.SEVERE, 
                "Exception in " + getTaskName(), e);
            throw e;
        }
    }
}.execute();
谈下烟灰 2024-08-23 02:32:49

您应该能够使用反射来获取线程并调用 setName。您需要使用 setAccessible 方法来执行此操作,并进行一些挖掘以查找需要访问的变量的名称。

我不喜欢这种方法,但它可能和其他方法一样“hacky”......

You should be able to use reflection to get the thread and call setName as well. You would need to make use of the setAccessible method to do that and some digging around to find the names of the variables you need to access.

I would not prefer this method, but it may be just as "hacky" as any other way...

倦话 2024-08-23 02:32:49

最佳实践是使用已经提到的 Thread.currentThread().setName(String)。

一个“技巧”是为 SwingWorkers 设置 ThreadFactory:

sun.awt.AppContext context = sun.awt.AppContext.getAppContext();

int maxWorkers = 10;
ThreadFactory factory = ...
ExecutorService executor = Executors.newFixedThreadPool(maxWorkers, factory);

context.put(SwingWorker.class, executor);

它使用 sun.awt 包中的一个类,因此您已收到警告。

The best practice would be to use the already mentioned Thread.currentThread().setName(String).

A "trick" would be setting the ThreadFactory for SwingWorkers:

sun.awt.AppContext context = sun.awt.AppContext.getAppContext();

int maxWorkers = 10;
ThreadFactory factory = ...
ExecutorService executor = Executors.newFixedThreadPool(maxWorkers, factory);

context.put(SwingWorker.class, executor);

This uses a class from the sun.awt package, so you've been warned.

还给你自由 2024-08-23 02:32:49

导入 javax.swing.SwingWorker;

public class SwingWorkerName {

    public static void main(String[] args) throws Exception {
        new SwingWorker() {
            @Override
            protected Object doInBackground() throws Exception {
                final String previousThreadName = Thread.currentThread().getName();
                Thread.currentThread().setName("My Worker Thread");
                try {

                    // DO YOUR WORK HERE
                    Thread.sleep(100000L);

                    return null; // or return whatever you want
                } finally {
                    Thread.currentThread().setName(previousThreadName);
                }
            }
        }.execute();        
        Thread.sleep(100000L);
    }

}

注意: 在 NetBeans 中,直到我暂停应用程序或命中断点后,swingworker 线程的新名称才会出现在调试器中。

import javax.swing.SwingWorker;

public class SwingWorkerName {

    public static void main(String[] args) throws Exception {
        new SwingWorker() {
            @Override
            protected Object doInBackground() throws Exception {
                final String previousThreadName = Thread.currentThread().getName();
                Thread.currentThread().setName("My Worker Thread");
                try {

                    // DO YOUR WORK HERE
                    Thread.sleep(100000L);

                    return null; // or return whatever you want
                } finally {
                    Thread.currentThread().setName(previousThreadName);
                }
            }
        }.execute();        
        Thread.sleep(100000L);
    }

}

NOTE: In NetBeans, the new name of the swingworker thread did not appear in the debugger until I paused the application or hit a breakpoint.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文