为什么在构造函数上创建新线程是不好的做法? [复制]

2023-12-02

可能的重复:
Java:为什么不在构造函数中启动线程?如何终止?

我习惯了跑步FindBugs在我的代码上,以便发现错误或不良做法。 今天它抱怨我在类构造函数中启动一个线程。

真的是一件坏事吗?你能解释一下为什么吗?

如果我的课是期末考试至少是安全的吧?

EDIT:

该线程作为内部类实现,并且仅使用启动时已初始化的主类的字段:

public final class SingletonOuter {
    private static SingletonOuter ourInstance = new SingletonOuter();

    public static SingletonOuter getInstance() {
        return ourInstance;
    }

    private final SomeOtherClass aField;

    private SingletonOuter() {
        aField=new SomeOtherClass(); 
        thread=new InnerThread();
        thread.start();
    }

    private boolean pleaseStop;

    private synchronized boolean askedStop(){return pleaseStop;}
    public synchronized void stop(){
        pleaseStop=true;  
    }

    private final InnerThread thread ;
    private class InnerThread extends Thread{
        @Override public void run() {
            //do stuff with aField until askedStop()
        }
    }

}

EDIT:

我最终将线程的启动移至 getInstance 方法,以避免引入未来错误的可能性:

public final class SingletonOuter {
        private static SingletonOuter ourInstance

        public static SingletonOuter getInstance() {
            if (ourInstance==null){
                ourInstance= = new SingletonOuter();
                ourInstance.thread.start();
            }

            return ourInstance;
        }

        private final SomeOtherClass aField;

        private SingletonOuter() {
            aField=new SomeOtherClass(); 
            thread=new InnerThread();

        }
        ...

为什么在构造函数上创建新线程是不好的做法?

Findbugs 警告您有关对象构造的指令重新排序可能性的问题。尽管为新对象分配了内存空间,但不能保证在您创建对象时任何字段都已初始化。InnerThread已经开始了。虽然final fields will在构造函数完成之前初始化,不能保证如果InnerThread开始使用(例如)aField当它启动时,它将被初始化。 Java 编译器这样做是出于性能方面的考虑。它还可以选择将非最终字段的初始化移至after新实例已由构造函数返回。

如果您在构造函数中启动一个新线程,则该线程有可能处理部分初始化的对象。即使thread.start()是构造函数中的最后一个语句,由于重新排序,新线程可能正在访问部分构造的对象。这是 Java 语言规范的一部分。

这是关于该主题的一个很好的链接:在自己的构造函数中调用 thread.start()

它提到以下内容:

通过从构造函数内启动它,您肯定会违反 Java 内存模型指南。看Brian Goetz 的安全施工技术了解更多信息。

Edit:

由于您的代码正在启动一个正在访问的新线程afield,根据Java内存模型无法保证afield当线程开始运行时将被正确初始化。

我建议改为添加一个start()类上调用的方法thread.start()。这是一种更好的做法,并且使使用此类的其他类更清楚地看到在构造函数中创建了线程。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

为什么在构造函数上创建新线程是不好的做法? [复制] 的相关文章

随机推荐

  • .htaccess 重写规则不会使用 unicode 字符

    我正在使用以下 ModRewrite 来使我的网址看起来更干净 RewriteEngine On RewriteRule a zA Z0 9 index php key 1 它允许使用字母和数字就好 但当我尝试使用 时 它会产生 400 错
  • if 和 elseif 有什么区别?

    这应该是一个简单的问题 我有一个简单的 if else 语句 与此有区别吗
  • JavaFX 2 自定义弹出窗格

    JavaFX 2 颜色选择器有一个按钮 可以弹出颜色选择器窗格 如下所示 我想做类似的事情 因为我希望在单击按钮时弹出一个自定义窗格 并在单击其他内容时消失 在我的例子中是一些图像缩略图 实现这一目标的最佳方法是什么 我应该使用 Conte
  • iOS isKindOfClass 和 isMemberOfClass 之间的区别

    两者有什么区别isKind of aClass AnyClass 和isMember of aClass AnyClass 斯威夫特函数 Objective C 中的原始问题 两者有什么区别isKindOfClass Class aClas
  • datepicker 不会触发 onchange 事件

    我有两个输入日期字段 一个通过 JQuery datepicker 生成日期 我希望第二个通过 onblur 或 onchange 事件从第一个输入中获取值 当我手动输入日期时它工作正常 但一旦通过 datepicker 输入 它不会触发第
  • 调试代码时 Ninject 出现无源可用错误

    我已经使用NuGet安装了最新版本的Ninject v2 2 1 4 然后我创建了自己的 NinjectDependencyResolver 归功于亚当 弗里曼和史蒂夫 桑德森 public class NinjectDependencyR
  • 第一次运行flutter时报错:没有找到pubspec.yaml文件

    我试着跟随Get Stated here https flutter io setup windows And got gt flutter run Error No pubspec yaml file found This command
  • Firebase 数据库规范化

    所以我制作了一个简单的应用程序 一个三轮车巡逻应用程序 您可以通过登录并填写表格来举报鲁莽的三轮车司机 在我们的城市这里没有任何问题 报告表格包含 created at description lat lng plateNumber 到目前
  • 用于按行“连接 If”的 VBA 用户定义函数

    我有一些关于是否为特定站点执行特定 服务 例如鸟类调查 的数据 每项服务都有 是 或 否 E G A 列包含站点名称 例如 A B C D 和 E A1 中标题为 站点名称 A2 中标题为 A 依此类推 B 列 B1 中包含 鸟类调查 然后
  • 捕获分段违规并继续生活

    我正在编写一个程序来检查它自己的地址空间 具体来说 我关心所有 malloc 的数据块 如果有一些系统调用来获取它们的列表 那就太棒了 对于我的应用程序 我不能使用 LD PRELOAD wrap 或任何额外的命令行选项 如果有办法做到这一
  • 使 arraylist 上的 select -unique 返回 arraylist 而不是字符串

    我在下面的类中有三个数组列表 我想让它们保持独特 但是 如果数组列表中只有一项 字符串 并且您使用 select unique 或任何其他方法来实现此目的 它将返回字符串而不是字符串列表 用 包围它也不起作用 因为这会将其转换为数组而不是数
  • Android内存泄漏,没有静态变量

    我是一名初级 Android 开发人员 作为一个实践项目 我正在尝试制作一个可以拍照 将其保存到外部存储并在 ImageView 中显示的活动 几乎一切似乎都正常 但是 我似乎有内存泄漏 当屏幕方向改变时 我相信活动会被破坏 然后重新创建
  • 存储过程中的账本报告逻辑

    我有一个名为 患者分类帐报告 的存储过程 我需要在其中显示患者的日常交易详细信息和余额金额 我正在为您提供以下代码中的一个采样数据 这些数据是如何插入到我的临时表中的sp create table Patient ledger PATIEN
  • Python 是否有相当于部分类的东西?

    使用 新 样式类 我使用的是 python 3 2 有没有办法将一个类拆分为多个文件 我有一个大类 从面向对象设计的角度来看 考虑到耦合等 它实际上应该是一个类 但是为了便于编辑类 最好将其拆分为几个文件 如果您的问题确实只是在编辑器中处理
  • “gcloud builds 提交...”有什么作用?

    我想知道什么gcloud 构建提交做 就我而言 我正在运行GCloud运行教程 The 官方文档指出它提交构建 这不是一个特别有用的信息 有人可以提供更多背景信息吗 什么是build 一个图像 jar 文件 这个 构建 被提交到哪里 提交
  • 更改 PHP 中包含内容的相对链接路径

    我的服务器根目录下有一个 PHP 文件 索引 php 哪个include的 DIR main php 现在 DIR main php 具有到许多附近文件的相对链接 所有相关链接均已损坏 我可以通过任何方式更改相对 URL 基本路径对于链接
  • 在Python 3中使用MagicMethods计算欧几里德距离

    我有一个为我计算欧几里德距离的代码 class Point A point in two dimensional space def init self x y self x x self y y def eq self other ret
  • 使用两个 android spinner,一个依赖于另一个

    我已经浏览了很多代码并试图找出我做错了什么 但我不及格并且浪费时间 这会很简单 但我无法弄清楚 我正在开发一个 Android 应用程序 需要在一项活动中放置 2 个旋转器 第二个微调器将根据第一个微调器中选择的条目进行填充 听起来很简单吧
  • Android 中的 Webview 能够运行 php

    我正在开发一个具有 Web 视图的 Android 应用程序 我还需要加载 html jquery js 和 php 现在我可以加载 html jquery css javascript 但我的问题是我无法在离线的 Web 视图中加载 ph
  • 为什么在构造函数上创建新线程是不好的做法? [复制]

    这个问题在这里已经有答案了 可能的重复 Java 为什么不在构造函数中启动线程 如何终止 我习惯了跑步FindBugs在我的代码上 以便发现错误或不良做法 今天它抱怨我在类构造函数中启动一个线程 真的是一件坏事吗 你能解释一下为什么吗 如果