草庐IT

java - 设计模式 - 如何仅在某些情况下强制执行对象属性(构建器模式、依赖注入(inject))

coder 2024-03-19 原文

我正在编写的其中一个类处于非常特殊的情况。我有一个名为 User 的类,它看起来像这样:

public class User {
    private long id; // + getters and setters
    private boolean isDeletable; // + getters and setters
    private String name; // + getters and setters
    private String password; // + getters and setters
    private String email; // + getters and setters
    private String authenticationRealm; // + getters and setters
    private String displayName; // + getters and setters
    private Date deletedDate; // + getters and setters
}

在我的代码中有几种情况,我只需要一个 User 类型的空对象,因此只需使用默认构造函数构建它:new User()

但是,我有另一个名为 CreateUserRequest 的类,它模拟 REST 请求以在服务器中创建用户。 最小 负载必须包含namepasswordemailauthenticationRealm 属性以 JSON 格式发送。

现在我通过在请求的构造函数中检查这些参数来处理这个问题:

public CreateUserRequest(User user) {
    if(user.getName() == null || user.getPassword() == null || user.getEmail() == null || user.getAuthenticationRealm() == null)
        throw new RuntimeException("Not enough attributes in User object. Minimum: name, password, e-mail and authentication realm.");
}

这工作正常,但有点痒...我想以更安全的方式强制执行此操作,以便代码强制填充属性,而不会抛出异常。

我觉得一定有更好的方法可以用设计模式来做到这一点。我想创建一个 UserRequestBuilder 类,但这也可能意味着在 build() 方法中抛出异常(否则,有没有办法保证属性在 build()? 之前填充。依赖注入(inject)听起来也有可能,但我不确定我将如何在这个特定示例中将其放置到位......

有什么想法吗?

最佳答案

如何使您的 REST 服务在用户上运行 DTO ? (当然,UserDTO 可以替换为 User 的子类)。

您可以使用 @NonNull 注释 UserDTO 上的字段、 setter 或构造函数参数,并使用 Checker Framework将空值而不是名称密码、电子邮件等传递给 UserDTO 时发出编译器警告。

使用类似 Mapstruct 的框架,REST 服务 DTO 和后端对象之间的映射非常简单:

@Mapper
public interface UserMapper {

    public static final UserMapper INSTANCE = Mappers.getMapper(UserMapper.class);

    UserDTO map(User user);

    User map(UserDTO userDTO);
}

以上代码将在编译时生成一个 UserMapper 实现,其中包含指定方法的自动生成代码( - 并且自动生成的代码只是简单地配对类似命名的 getter 和 setter。你可以自己做,但是有很多 DTO/实体会变得很耗时和无聊)。

在 DTO 中,您可以排除所有不想公开的字段。

附言。我自己对上面提到的用法是这样的:我正在创建一个基于 Jersey 的 REST 服务器,即 JAX-RS 的引用实现。这个项目,称之为 A,只知道 DTO。 REST 方法调用另一个项目 B,它从数据库中检索对象,并将它们映射到相应的 DTO,然后返回给项目 A。这种模式的部分原因是由于历史原因,项目 B 的实体是杂乱无章的方法/功能,不应该暴露给项目A。至于健全性检查(JSON到DTO),jersey支持Bean Validation,也就是说,框架将验证每个rest资源的输入bean,如果它们被注释与@Valid。 也可以创建您自己的自定义注释,其中定义了 ConstraintValidator。 bean 验证框架将检查这些对带注释的 jersey REST 方法参数的约束。 参见 https://jersey.java.net/documentation/latest/bean-validation.html#d0e13690

关于java - 设计模式 - 如何仅在某些情况下强制执行对象属性(构建器模式、依赖注入(inject)),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36157937/

有关java - 设计模式 - 如何仅在某些情况下强制执行对象属性(构建器模式、依赖注入(inject))的更多相关文章

  1. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

  2. ruby-openid:执行发现时未设置@socket - 2

    我在使用omniauth/openid时遇到了一些麻烦。在尝试进行身份验证时,我在日志中发现了这一点:OpenID::FetchingError:Errorfetchinghttps://www.google.com/accounts/o8/.well-known/host-meta?hd=profiles.google.com%2Fmy_username:undefinedmethod`io'fornil:NilClass重要的是undefinedmethodio'fornil:NilClass来自openid/fetchers.rb,在下面的代码片段中:moduleNetclass

  3. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  4. ruby - 默认情况下使选项为 false - 2

    这是在Ruby中设置默认值的常用方法:classQuietByDefaultdefinitialize(opts={})@verbose=opts[:verbose]endend这是一个容易落入的陷阱:classVerboseNoMatterWhatdefinitialize(opts={})@verbose=opts[:verbose]||trueendend正确的做法是:classVerboseByDefaultdefinitialize(opts={})@verbose=opts.include?(:verbose)?opts[:verbose]:trueendend编写Verb

  5. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

  6. ruby - 在没有 sass 引擎的情况下使用 sass 颜色函数 - 2

    我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re

  7. ruby-on-rails - 使用 rails 4 设计而不更新用户 - 2

    我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它​​不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数

  8. ruby - Chef 执行非顺序配方 - 2

    我遵循了教程http://gettingstartedwithchef.com/,第1章。我的运行list是"run_list":["recipe[apt]","recipe[phpap]"]我的phpapRecipe默认Recipeinclude_recipe"apache2"include_recipe"build-essential"include_recipe"openssl"include_recipe"mysql::client"include_recipe"mysql::server"include_recipe"php"include_recipe"php::modul

  9. 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/

  10. ruby - 为什么 Ruby 的 each 迭代器先执行? - 2

    我在用Ruby执行简单任务时遇到了一件奇怪的事情。我只想用每个方法迭代字母表,但迭代在执行中先进行:alfawit=("a".."z")puts"That'sanalphabet:\n\n#{alfawit.each{|litera|putslitera}}"这段代码的结果是:(缩写)abc⋮xyzThat'sanalphabet:a..z知道为什么它会这样工作或者我做错了什么吗?提前致谢。 最佳答案 因为您的each调用被插入到在固定字符串之前执行的字符串文字中。此外,each返回一个Enumerable,实际上您甚至打印它。试试

随机推荐