java创建一个不可变对象_使用不可变对象创建值对象

java创建一个不可变对象

在回答我最近的文章中AutoValue:生成的不可变的值类 , 布兰登认为,这可能是有趣的,看看如何AutoValue比较项目Lombok和Immutables和凯文借调这一点。 我同意这是一个好主意,但是我首先将这篇文章发布为Immutables的简要概述,因为我已经为Lombok和AutoValue提供了类似的帖子。

可从Maven中央存储库中获得Immutables 2.2.5 , 其许可页面指出“ Imputables工具箱和所有必需的依赖项均在Apache软件许可2.0版中涵盖。” 开始吧! 页面指​​出“运行Immutables注释处理器需要Java 7或更高版本。”

像AutoValue这样的不可变对象使用编译时注释来生成定义不可变对象的类的源代码。 因为它们都使用这种方法,所以都只引入了编译时依赖性,并且在应用程序的运行时类路径上不需要它们各自的JAR。 换句话说,不可变的JAR必须位于编译器( javac )的类路径上,而不是位于Java启动器( java )的类路径上。

下一个代码清单( Person.java )中显示了“模板” Person类的代码清单。 它看起来与我在AutoValue演示中使用的Person.java非常相似。

人.java

package dustin.examples.immutables;import org.immutables.value.Value;/*** Represents an individual as part of demonstration of* the Immutables project (http://immutables.github.io/).*/
@Value.Immutable  // concrete extension will be generated by Immutables
abstract class Person
{/*** Provide Person's last name.** @return Last name of person.*/abstract String lastName();/*** Provide Person's first name.** @return First name of person.*/abstract String firstName();/*** Provide Person's birth year.** @return Person's birth year.*/abstract long birthYear();
}

我在AutoValue帖子中列出的“ template”类和“ template”类的唯一区别是包的名称,正在演示哪个产品的Javadoc注释以及(最重要的是)导入并应用于类。 在AutoValue示例中有一个特定的“ create”方法不在Immutables示例中,但这只是因为我没有演示AutoValue生成器的使用,这将使“ create”方法变得不必要。

当我在类路径上适当地指定使用Immutables并使用javac编译上述源代码时,将调用注释处理器并生成以下Java源代码:

ImmutablePerson.java

package dustin.examples.immutables;import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import javax.annotation.Generated;/*** Immutable implementation of {@link Person}.* <p>* Use the builder to create immutable instances:* {@code ImmutablePerson.builder()}.*/
@SuppressWarnings("all")
@Generated({"Immutables.generator", "Person"})
final class ImmutablePerson extends Person {private final String lastName;private final String firstName;private final long birthYear;private ImmutablePerson(String lastName, String firstName, long birthYear) {this.lastName = lastName;this.firstName = firstName;this.birthYear = birthYear;}/*** @return The value of the {@code lastName} attribute*/@OverrideString lastName() {return lastName;}/*** @return The value of the {@code firstName} attribute*/@OverrideString firstName() {return firstName;}/*** @return The value of the {@code birthYear} attribute*/@Overridelong birthYear() {return birthYear;}/*** Copy the current immutable object by setting a value for the {@link Person#lastName() lastName} attribute.* An equals check used to prevent copying of the same value by returning {@code this}.* @param lastName A new value for lastName* @return A modified copy of the {@code this} object*/public final ImmutablePerson withLastName(String lastName) {if (this.lastName.equals(lastName)) return this;String newValue = Objects.requireNonNull(lastName, "lastName");return new ImmutablePerson(newValue, this.firstName, this.birthYear);}/*** Copy the current immutable object by setting a value for the {@link Person#firstName() firstName} attribute.* An equals check used to prevent copying of the same value by returning {@code this}.* @param firstName A new value for firstName* @return A modified copy of the {@code this} object*/public final ImmutablePerson withFirstName(String firstName) {if (this.firstName.equals(firstName)) return this;String newValue = Objects.requireNonNull(firstName, "firstName");return new ImmutablePerson(this.lastName, newValue, this.birthYear);}/*** Copy the current immutable object by setting a value for the {@link Person#birthYear() birthYear} attribute.* A value equality check is used to prevent copying of the same value by returning {@code this}.* @param birthYear A new value for birthYear* @return A modified copy of the {@code this} object*/public final ImmutablePerson withBirthYear(long birthYear) {if (this.birthYear == birthYear) return this;return new ImmutablePerson(this.lastName, this.firstName, birthYear);}/*** This instance is equal to all instances of {@code ImmutablePerson} that have equal attribute values.* @return {@code true} if {@code this} is equal to {@code another} instance*/@Overridepublic boolean equals(Object another) {if (this == another) return true;return another instanceof ImmutablePerson&& equalTo((ImmutablePerson) another);}private boolean equalTo(ImmutablePerson another) {return lastName.equals(another.lastName)&& firstName.equals(another.firstName)&& birthYear == another.birthYear;}/*** Computes a hash code from attributes: {@code lastName}, {@code firstName}, {@code birthYear}.* @return hashCode value*/@Overridepublic int hashCode() {int h = 31;h = h * 17 + lastName.hashCode();h = h * 17 + firstName.hashCode();h = h * 17 + Long.hashCode(birthYear);return h;}/*** Prints the immutable value {@code Person} with attribute values.* @return A string representation of the value*/@Overridepublic String toString() {return "Person{"+ "lastName=" + lastName+ ", firstName=" + firstName+ ", birthYear=" + birthYear+ "}";}/*** Creates an immutable copy of a {@link Person} value.* Uses accessors to get values to initialize the new immutable instance.* If an instance is already immutable, it is returned as is.* @param instance The instance to copy* @return A copied immutable Person instance*/public static ImmutablePerson copyOf(Person instance) {if (instance instanceof ImmutablePerson) {return (ImmutablePerson) instance;}return ImmutablePerson.builder().from(instance).build();}/*** Creates a builder for {@link ImmutablePerson ImmutablePerson}.* @return A new ImmutablePerson builder*/public static ImmutablePerson.Builder builder() {return new ImmutablePerson.Builder();}/*** Builds instances of type {@link ImmutablePerson ImmutablePerson}.* Initialize attributes and then invoke the {@link #build()} method to create an* immutable instance.* <p><em>{@code Builder} is not thread-safe and generally should not be stored in a field or collection,* but instead used immediately to create instances.</em>*/static final class Builder {private static final long INIT_BIT_LAST_NAME = 0x1L;private static final long INIT_BIT_FIRST_NAME = 0x2L;private static final long INIT_BIT_BIRTH_YEAR = 0x4L;private long initBits = 0x7L;private String lastName;private String firstName;private long birthYear;private Builder() {}/*** Fill a builder with attribute values from the provided {@code Person} instance.* Regular attribute values will be replaced with those from the given instance.* Absent optional values will not replace present values.* @param instance The instance from which to copy values* @return {@code this} builder for use in a chained invocation*/public final Builder from(Person instance) {Objects.requireNonNull(instance, "instance");lastName(instance.lastName());firstName(instance.firstName());birthYear(instance.birthYear());return this;}/*** Initializes the value for the {@link Person#lastName() lastName} attribute.* @param lastName The value for lastName * @return {@code this} builder for use in a chained invocation*/public final Builder lastName(String lastName) {this.lastName = Objects.requireNonNull(lastName, "lastName");initBits &= ~INIT_BIT_LAST_NAME;return this;}/*** Initializes the value for the {@link Person#firstName() firstName} attribute.* @param firstName The value for firstName * @return {@code this} builder for use in a chained invocation*/public final Builder firstName(String firstName) {this.firstName = Objects.requireNonNull(firstName, "firstName");initBits &= ~INIT_BIT_FIRST_NAME;return this;}/*** Initializes the value for the {@link Person#birthYear() birthYear} attribute.* @param birthYear The value for birthYear * @return {@code this} builder for use in a chained invocation*/public final Builder birthYear(long birthYear) {this.birthYear = birthYear;initBits &= ~INIT_BIT_BIRTH_YEAR;return this;}/*** Builds a new {@link ImmutablePerson ImmutablePerson}.* @return An immutable instance of Person* @throws java.lang.IllegalStateException if any required attributes are missing*/public ImmutablePerson build() {if (initBits != 0) {throw new IllegalStateException(formatRequiredAttributesMessage());}return new ImmutablePerson(lastName, firstName, birthYear);}private String formatRequiredAttributesMessage() {List<String> attributes = new ArrayList<String>();if ((initBits & INIT_BIT_LAST_NAME) != 0) attributes.add("lastName");if ((initBits & INIT_BIT_FIRST_NAME) != 0) attributes.add("firstName");if ((initBits & INIT_BIT_BIRTH_YEAR) != 0) attributes.add("birthYear");return "Cannot build Person, some of required attributes are not set " + attributes;}}
}

通过检查生成的代码,可以得出一些结论(您会发现它们与我之前的文章中为AutoValue列出的观察结果非常相似):

  • 生成的类扩展了(实现继承)手写的抽象类,从而允许使用代码使用手写类的API,而不必知道正在使用生成的类。
  • 即使没有在源类中直接定义任何字段,也将生成字段; 不可变对象根据提供的abstract访问器方法解释字段。
  • 生成的类不为字段提供“设置” / mutator方法(get / accessor方法)。 这并不奇怪,因为值对象的关键概念是它们是不可变的,甚至这个项目的名称( Immutables )也暗示了这一特征。 请注意, Immutables确实为具有@ Value.Modifiable批注的可修改对象提供了一些功能。
  • 考虑到每个字段的类型,将自动为每个字段适当地生成equals(Object) , hashCode()和toString()的实现。
  • 在源类和方法上的Javadoc注释不会在生成的扩展类上重现。 相反,在生成的类的方法上提供了更简单(更通用)的Javadoc注释,在构建器类的方法上提供了更重要(但仍是通用的)的Javadoc注释。

正如我关于AutoValue所述,使用诸如Immutables生成之类的方法的主要优点之一是,开发人员可以专注于特定类应支持的更简单的高级概念,并且代码生成可以确保较低级别的细节。一致且正确地实施。 但是,使用这种方法时要记住一些事情。

  • 当开发人员受过足够的训练以检查和维护抽象的“源” Java类而不是生成的类时, 不可变对象最有帮助。
    • 下次注释处理再次生成该类时,对生成类的更改将被覆盖,否则必须停止该类的生成,以免发生这种情况。
  • 您需要设置build / IDE,以便将生成的类视为“源代码”,以便抽象类可以编译,并且对生成的类的任何依赖项也可以编译。
  • 如果要使不可变性与可变字段一起使用,则必须格外小心(通常选择使用不可变项或值对象通常是这种情况)。

结论

我的结论几乎与我在AutoValue上发表的文章一样。 Immutables允许开发人员编写更简洁的代码,重点放在高级细节上,并将繁琐的底层(通常容易出错)细节的实现委派给Immutables,以自动生成代码。 这类似于IDE的源代码生成可以执行的操作,但是Immutables相对于IDE方法的优势是Immutables可以在每次编译代码时重新生成源代码,从而使生成的代码保持最新。 Immutables的这一优势也是Java自定义注释处理功能强大的一个很好的例子。

翻译自: https://www.javacodegeeks.com/2016/06/creating-value-objects-immutables.html

java创建一个不可变对象

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/335923.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

混合高斯模型_高斯混合模型(GMM)

下图所示&#xff0c;显然用右边的图描述当前分布更加合理&#xff0c;即应用了两个高斯分布。图中每一个样本点同时属于任何一个高斯模型。高斯混合模型 从几何角度来理解&#xff0c;GMM是由多个高斯分布叠加而成&#xff0c;可以看做是多个高斯分布的加权平均。其中&#x…

【安卓开发 】Android初级开发(网络操作)

URI部分 URI详情 uri的具体案例使用参考&#xff0c;app与网页之间的页面跳转 H5唤醒app并跳转到指定页面 H5打开APP技术总结 H5页面唤醒app的方法 Android配置Scheme使用浏览器唤起APP的方式&#xff0c;以及不生效问题解决 网页唤起app,并传值到app中使用的全过程 and…

dynamodb java_使用Java第2部分查询DynamoDB项

dynamodb java在上一篇文章中&#xff0c;我们有机会发布了一些基本的DynamoDB查询操作。 但是&#xff0c;除了基本操作之外&#xff0c;DynamoDB api还为我们提供了一些额外的功能。 投影是具有类似选择功能的功能。 您选择应从DynamoDB项中提取哪些属性。 请记住&#xf…

【H.264/AVC视频编解码技术】第五章【哈夫曼编码】

本文章所需要的内容需要自行准备一个名为input.txt的文本文件作为案例演示。内容选择英语小短文即可 第一步,建立哈夫曼数 #include <iostream> #include <fstream> #include <queue> #include <vector> #include <string>using namespace st…

gitpython git diff_Python全栈开发-git常用命令

欢迎关注我的号Python全栈开发-git常用命令​mp.weixin.qq.com### Python全栈开发-git常用命令本节内容- github介绍- 安装- 仓库创建& 提交代码- 代码回滚- 工作区和暂存区- 撤销修改- 删除操作- 远程仓库- 分支管理- 多人协作- github使用- 忽略特殊文件.gitignore### 2.…

来的多可选_您的框架有多可扩展性?

来的多可选在参加会议时&#xff0c;我们总是会遇到高素质的决策者&#xff0c;他们经常问同样的问题&#xff1a; 您的框架有多可扩展性&#xff1f;如果我需要的比您开箱即用的功能还多呢&#xff1f; 。 这个问题非常合理&#xff0c;因为他们只是不想被卡在开发曲线的中间&…

【H.264/AVC视频编解码技术】第六章【指数哥伦布编码】

H264中语法元素描述符 指数哥伦布 (Exponential-Golomb) 熵编码 指数哥伦布编码同哈夫曼编码一样,都是变长编码。 二者的显著区别: 信源相关性:哈夫曼编码依赖于信源的概率分布;指数哥伦布与信源无关。 额外信息:哈夫曼编码的数据必须额外携带与信源匹配的码表;指…

python素描效果_python实现图片素描效果

代码如下&#xff1a;from PIL import Image #图像处理模块import numpy as npa np.asarray(Image.open("这里是原图片的路径").convert(L)).astype(float)#将图像以灰度图的方式打开并将数据转为float存入np中depth 10. # (0-100)grad np.gradient(a) #取图像灰度…

java 认证_Java认证:认证或不认证

java 认证专业认证始终是一个有争议的主题&#xff0c;有资格的人在争论收益与成本/时间的关系。 通过Oracle的Java认证&#xff0c;我认为有两个主要的受众可以从中受益&#xff1a; 那些开始从事软件事业的人。 扎实的工作经验和可证明的代码将永远是潜在雇主的首要考虑因素…

C++ 11 深度学习(七)位运算常见操作

1. 取出数中任意k位置的二进制位是0还是1 n >> k & 1 原理&#xff1a;先把想要取出的位置移动到个位&#xff0c;1的二进制是 0001 , 进行与操作就可以提取出最后一位是0还是1&#xff1b;二进制是从右向左&#xff0c;由低到高&#xff0c;从0到7。 2. 右移操作 …

聚合项目访问后台接口失败_聚合支付系统和免签支付系统对未来支付市场有哪些影响...

时势所趋&#xff0c;在如今支付通道不稳定的情况下&#xff0c;四方聚合支付的出现弥补了通道不稳的情况&#xff0c;四方聚合支付可以接入多个三方&#xff0c;实现在三方不稳的情况直接后台切换三方&#xff0c;实现一秒切换&#xff0c;还可以接入个人免签支付系统&#xf…

activemq消息持久化_ActiveMQ 5.x中的消息持久性

activemq消息持久化我被问了很多关于ActiveMQ如何存储消息&#xff08;或在某些情况下不存储&#xff09;的基本知识。 这是它的高级解释。 注意&#xff0c;上下文在JMS中。 如果您使用ActiveMQ的非JMS客户端&#xff08;即STOMP&#xff0c;AMQP&#xff0c;MQTT等&#xff0…

【安卓开发 】Android初级开发(十一)Android中多线程

线程的创建 1.创建一个Thread类&#xff0c;或者创建一个Thread子的对象&#xff1b; 2.创建一个Runnable接口的类对象; 传入Runnable对象创建线程 package com.sina.baode;import android.util.Log;/** 自定义一个继承于Runnable*/public class DemoThread implements Run…

td 内单选框不可用_在TD,我和曾经的老师变成了同事,也收获了最满意的“课外活动”...

早就想找个时间好好写一篇文章记录一下我在TD的学习工作经历了&#xff0c;同时表达一下我对TD深深的爱和感谢。一、结缘TD一年前&#xff0c;我第一次报了TD的AP环境科学寒假班。由于当时我在冲刺三月的SAT&#xff0c;整个寒假我一心扑在了SAT上&#xff0c;没来得及听TD的直…

activiti脚本任务_Activiti中的安全脚本如何工作

activiti脚本任务最近的Activiti 5.21.0版本的突出特点之一是“安全脚本”。 Activiti用户指南中详细介绍了启用和使用此功能的方法 。 在这篇文章中&#xff0c;我将向您展示我们如何实现其最终实现以及它在幕后所做的事情。 当然&#xff0c;因为这是我通常的签名风格&#x…

C++ 11 深度学习(八)重定义override

1.动态联编&#xff0c;本质是在运行时多态的表现。 2.其本质是因为维护了一张虚函数表&#xff0c;虚函数表以链表的形式存在。每个结点存储了对象指针的地址&#xff0c;通过一个指针进行遍历索引。 #include <iostream> #include <armadillo> using namespace …

逻辑回归模型_联邦学习体系下——逻辑回归模型

联邦学习的体系我们在前期介绍过&#xff0c;这里我们简单回顾一下纵向联邦学习的定义&#xff1a;在两个数据集的用户重叠较多而用户特征重叠较少的情况下&#xff0c;将数据集按照纵向 (即特征维度)切分&#xff0c;并取出双方用户相同而用户特征不完全相同的那部分数据进行训…

C++ 11 深度学习(九)C++文件IO

1.将数据写入文件 #include <iostream> #include <fstream> using namespace std;int main() {ofstream p1;p1.open("outfile.txt");p1 << "向文件写入信息" << endl;p1.close();return 0; } 2.将数据从文件中读出 #inclu…

plsql例外_大例外背后的真相

plsql例外异常可能是最被滥用的Java语言功能。 这就是为什么 让我们打破一些神话。 没有牙仙子。 圣诞老人不是真实的。 TODO评论。 finalfinalversion-final.pdf。 无皂肥皂。 而且…例外实际上是例外。 后者可能需要更多说服力&#xff0c;但是我们可以帮助您。 在这篇文章…

滴滴java开发面试题_Java开发经典面试题(十二)

好久没有来更新我的面试题了&#xff0c;不知道关注我的小伙伴有没有失联啊&#xff1f;&#xff01;呼叫&#xff01;好了开始我们今天的正题分享&#xff01;1、如何从FutureTask不阻塞获取结果get(long timeout,TimeUnit unit)&#xff0c;超时则返回轮询&#xff0c;先通过…