草庐IT

Java 最终修饰符

coder 2023-05-13 原文

有人告诉我,我误解了 final 的影响. final有什么作用关键词?

这是我的想法的简短概述,我知道:

Java final modifier (aka aggregation relation)

primitive variables: can be set only once. (memory and performance gain)
objects variables: may be modified, final applies to object reference.
fields: can be set only once.
methods: can't be overridden, hidden.
classes: can't be extended.
garbage collection: will force Java generational garbage collection mark-sweep to double sweep.



jar 头和 jar 头
  • 可以使克隆失败(这有好有坏)
  • 可以制作不可变的原语,又名 const
  • 可以使空白不可变 - 在创建时初始化又名只读
  • 可以使对象浅不可变
  • 可以使范围/可见性不可变
  • 可以让方法调用开销更小(因为不需要虚表)
  • 可以将方法参数用作 final(即使你不是)
  • 可以使对象线程安全(如果对象定义为 final,则不会使方法参数成为 final)
  • 可以进行模拟测试(不是说你可以做任何事情 - 你可以说是有缺陷的)
  • 不能交 friend (与其他 friend 可变,休息不可变)
  • 不能使以后更改为不可变的可变(但可以使用工厂模式如修复)
  • 不能使数组元素不可变,也就是深度不可变
  • 无法创建对象的新实例(这既好又坏)
  • 无法使序列化工作

  • 没有替代品 final ,但有包装器 + 私有(private)和枚举。

    最佳答案

    依次回答你的每一点:

    primitive variables: can be set only once. (memory and performance gain)



    是的,但没有内存增益,也没有性能增益。 (您假设的性能提升来自仅设置一次......而不是来自 final 。)

    objects variables: may be modified, final applies to object reference.



    是的。 (但是,这个描述忽略了这一点,即这与 Java 语言的其余部分处理对象/引用二元性的方式完全一致。例如,当对象作为参数传递并作为结果返回时。)

    fields: can be set only once.



    真正的答案是:与变量相同。

    methods: can't be overridden, hidden.



    是的。但还要注意,这里发生的事情是 final关键字在不同的句法上下文中用于表示与 final 不同的含义对于字段/变量。

    classes: can't be extended.



    是的。但也请参见上面的注释。

    garbage collection: will force Java generational garbage collection mark-sweep to double sweep.



    这是无稽之谈。 final关键字与垃圾收集没有任何关系。您可能会感到困惑 final最终确定......它们是无关的。

    但即使是终结器也不会强制进行额外的扫描。发生的情况是需要终结的对象被设置在一侧,直到主 GC 完成。 GC 然后在对象上运行 finalize 方法并设置它的标志......并继续。下次 GC 运行时,该对象将被视为普通对象:
  • 如果可访问,则标记并复制
  • 如果无法访问,则不会对其进行标记。

  • (您的表征-“Java 分代垃圾收集标记-清除”是乱码。垃圾收集器可以是“标记-扫描”或“分代”(“复制”的子类)。不能两者兼而有之。Java 通常使用分代收集,并且仅在紧急情况下回退到标记清除;即当空间不足或低暂停收集器无法跟上时。)

    Can make clone fail (this is both good and bad)



    我不这么认为。

    Can make immutable primitives aka const



    是的。

    Can make blank immutable - initialized at creation aka readonly



    是的......虽然我以前从未听说过“空白不可变”这个词。

    Can make objects shallowly immutable



    对象可变性是关于可观察状态是否可能改变。因此,声明属性 final可能会也可能不会使对象表现为不可变的。此外,“浅不可变”的概念没有得到很好的定义,尤其是因为如果没有对类语义的深入了解,就无法映射“浅”的概念。

    (需要明确的是,变量/字段的可变性在 JLS 的上下文中是一个定义明确的概念。从 JLS 的角度来看,它只是 undefined object 可变性的概念。)

    Can make scope / visibility immutable



    术语错误。可变性与对象状态有关。可见性和范围不是。

    Can make method invocation overhead smaller (because it does not need virtual table)



    在实践中,这无关紧要。如果非 final方法没有被应用程序实际使用的任何类覆盖,现代 JIT 编译器也会对非 final方法进行这种优化。 (聪明的事情发生了......)

    Can make method arguments used as final (even if thy are not)



    嗯?我无法解析这句话。

    Can make objects threadsafe



    在某些情况下是的。

    (if object is defined as final, it wont make method arguments final)



    是的,如果您的意思是类(class)是否是最终的。对象不是最终的。

    Can make mock tests (not that you could do anything about it - you can say bugs are intended)



    不会解析。

    Can't make friends (mutable with other friends and immutable for rest)



    Java 没有“ friend ”。

    Can't make mutable that is changed to be immutable later (but can with factory pattern like fix)



    是的,第一个 final字段不能从可变切换到不可变。

    不清楚你说的第二部分是什么意思。确实可以使用工厂(或构建器)模式来构造不可变对象(immutable对象)。但是,如果您使用 final对于对象字段,对象在任何时候都不会是可变的。

    或者,您可以实现使用非最终字段来表示不可变状态的不可变对象(immutable对象),并且您可以设计 API,以便您可以“翻转开关”使以前的可变对象从现在开始不可变。但是如果你采用这种方法,你需要更加小心同步......如果你的对象需要是线程安全的。

    Can't make array elements immutable aka deeply immutable



    是的,但是您的术语已损坏;请参阅上面关于“浅可变性”的评论。

    Can't make new instances of object (this is both good and bad)



    不会。没有什么能阻止您使用 final 字段或 final 类或 final 方法创建对象的新实例。

    Can't make serialization work



    不。序列化有效。 (当然,使用自定义 final 方法对 readObject 字段进行反序列化存在问题……尽管您可以使用反射技巧来解决这些问题。)

    There are no alternatives to final,



    正确的。

    but there is wrapper + private



    是的,模(严格来说)非最终字段的非同步 getter 可能是非线程安全的......即使它在对象构造期间被初始化,然后永远不会改变!

    and enums.



    解决一个不同的问题。和 enums可以是可变的。

    关于Java 最终修饰符,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4012167/

    有关Java 最终修饰符的更多相关文章

    1. java - 等价于 Java 中的 Ruby Hash - 2

      我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

    2. java - 从 JRuby 调用 Java 类的问题 - 2

      我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

    3. java - 我的模型类或其他类中应该有逻辑吗 - 2

      我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我

    4. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

      什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

    5. Observability:从零开始创建 Java 微服务并监控它 (二) - 2

      这篇文章是继上一篇文章“Observability:从零开始创建Java微服务并监控它(一)”的续篇。在上一篇文章中,我们讲述了如何创建一个Javaweb应用,并使用Filebeat来收集应用所生成的日志。在今天的文章中,我来详述如何收集应用的指标,使用APM来监控应用并监督web服务的在线情况。源码可以在地址 https://github.com/liu-xiao-guo/java_observability 进行下载。摄入指标指标被视为可以随时更改的时间点值。当前请求的数量可以改变任何毫秒。你可能有1000个请求的峰值,然后一切都回到一个请求。这也意味着这些指标可能不准确,你还想提取最小/

    6. 【Java 面试合集】HashMap中为什么引入红黑树,而不是AVL树呢 - 2

      HashMap中为什么引入红黑树,而不是AVL树呢1.概述开始学习这个知识点之前我们需要知道,在JDK1.8以及之前,针对HashMap有什么不同。JDK1.7的时候,HashMap的底层实现是数组+链表JDK1.8的时候,HashMap的底层实现是数组+链表+红黑树我们要思考一个问题,为什么要从链表转为红黑树呢。首先先让我们了解下链表有什么不好???2.链表上述的截图其实就是链表的结构,我们来看下链表的增删改查的时间复杂度增:因为链表不是线性结构,所以每次添加的时候,只需要移动一个节点,所以可以理解为复杂度是N(1)删:算法时间复杂度跟增保持一致查:既然是非线性结构,所以查询某一个节点的时候

    7. 【Java入门】使用Java实现文件夹的遍历 - 2

      遍历文件夹我们通常是使用递归进行操作,这种方式比较简单,也比较容易理解。本文为大家介绍另一种不使用递归的方式,由于没有使用递归,只用到了循环和集合,所以效率更高一些!一、使用递归遍历文件夹整体思路1、使用File封装初始目录,2、打印这个目录3、获取这个目录下所有的子文件和子目录的数组。4、遍历这个数组,取出每个File对象4-1、如果File是否是一个文件,打印4-2、否则就是一个目录,递归调用代码实现publicclassSearchFile{publicstaticvoidmain(String[]args){//初始目录Filedir=newFile("d:/Dev");Datebeg

    8. java - 为什么 ruby​​ modulo 与 java/other lang 不同? - 2

      我基本上来自Java背景并且努力理解Ruby中的模运算。(5%3)(-5%3)(5%-3)(-5%-3)Java中的上述操作产生,2个-22个-2但在Ruby中,相同的表达式会产生21个-1-2.Ruby在逻辑上有多擅长这个?模块操作在Ruby中是如何实现的?如果将同一个操作定义为一个web服务,两个服务如何匹配逻辑。 最佳答案 在Java中,模运算的结果与被除数的符号相同。在Ruby中,它与除数的符号相同。remainder()在Ruby中与被除数的符号相同。您可能还想引用modulooperation.

    9. java - Ruby 相当于 Java 的 Collections.unmodifiableList 和 Collections.unmodifiableMap - 2

      Java的Collections.unmodifiableList和Collections.unmodifiableMap在Ruby标准API中是否有等价物? 最佳答案 使用freeze应用程序接口(interface):Preventsfurthermodificationstoobj.ARuntimeErrorwillberaisedifmodificationisattempted.Thereisnowaytounfreezeafrozenobject.SeealsoObject#frozen?.Thismethodretur

    10. java - Java 的 StringReader 的 Ruby 等价物是什么? - 2

      在Java中,可以像这样从一个字符串创建一个IO流:Readerr=newStringReader("mytext");我希望能够在Ruby中做同样的事情,这样我就可以获取一个字符串并将其视为一个IO流。 最佳答案 r=StringIO.new("mytext")和here'sthedocumentation. 关于java-Java的StringReader的Ruby等价物是什么?,我们在StackOverflow上找到一个类似的问题: https://st

    随机推荐