java - 将无状态 session bean 注入(inject)自定义 JsonDeserializer 失败

标签 java json jackson cdi

我正在构建一个使用 JPA (EclipseLink) 提供 JAX-RS REST 服务的应用程序。当通过 JSON 公开用户实体时,我在某些字段(例如密码字段)上使用 @XmlTransient 注释以将它们隐藏在 JSON 表示中。发送创建或更新 (POST/PUT) 操作时,我想再次填充缺失的字段,以便 JPA 正确执行操作。

我目前的方法是使用自定义 JsonDeserializer 来反序列化用户并添加缺失的字段。为此,我想注入(inject)(使用 @Inject)一个处理 JPA 内容的 UserFacadeREST bean。但是,此注入(inject)失败并且 bean 实例为 null(这当然会导致 NullPointerException)。

我的 UserFacadeREST bean 注释如下:

@Stateless
@LocalBean
@Path(UserFacadeREST.PATH)
public class UserFacadeREST extends AbstractFacade<User> {
    //...
}

我的UserDeserilizer(自定义JsonDeserializer):

public class UserDeserializer extends JsonDeserializer<User> {

  @Inject
  private UserFacadeREST userFacade;

  @Override
  public User deserialize(JsonParser parser, DeserializationContext context) throws IOException,
      JsonProcessingException {
    JsonNode node = parser.getCodec().readTree(parser);
    int userId = (Integer) ((IntNode) node.get("userID")).numberValue();
    System.out.println(userId);
    User user = userFacade.find(userId); // This line produces the NullPointerException
    return user;
  }

}

然后我使用 @JsonDeserialize 在我的用户实体上使用它:

@Entity
@Table(name = "User")
@XmlRootElement
@JsonDeserialize(using = UserDeserializer.class)
public class User implements Serializable {
    // ...
}

我在我的 WEB-INF 文件夹中包含了一个 bean.xml 文件,其中 bean-discovery-mode 设置为 all。我错过了什么?

最佳答案

Jon Peterson 为我指明了正确的方向。我最终选择以某种方式实现“hackish”解决方案。请注意,这里基本上有 2 个选项(如果您知道另一个,请告诉我!)。简短版本:

  1. Hackish 解决方案(我选择的解决方案):使用 javax.enterprise.inject.spi.CDI.current().select(UserFacadeRest.class).get( 以编程方式注入(inject)一个 bean )accepted answer 中所述Jon 或
  2. 提到的问题
  3. 更好(干净)的解决方案(但也更详细):按照 Jon 的建议,重新设计逻辑以填充反序列化后缺失的字段。

所以对于我的问题,解决方案如下所示:

1.

import javax.enterprise.inject.spi.CDI;

public class UserDeserializer extends JsonDeserializer<User> {

  private final UserFacadeREST userFacade =
      CDI.current().select(UserFacadeREST.class).get();

  // Rest as before
}

2。在这种情况下,在我的 JsonDeserializerdeserialize 方法中,我将构造一个只包含用户 ID 的用户。然后,在每个请求方法中,我必须检查所有用户并通过调用 EntityManager.find(User.class, user.getUserID()) 将它们替换为实际用户。这意味着在业务逻辑上需要付出更多努力,因为您必须记住,每次您需要在请求方法中处理 User 时,您首先必须进行查询以获取“完整”用户对象。在第一个解决方案中,此查询对业务逻辑是隐藏的,并且已在 JsonDeserializer 中发生。

public class UserDeserializer extends JsonDeserializer<User> {

  @Override
  public User deserialize(JsonParser parser, DeserializationContext context) throws IOException,
      JsonProcessingException {
    JsonNode node = parser.getCodec().readTree(parser);
    int userId = (Integer) ((IntNode) node.get("userID")).numberValue();
    return new User(userId); // Placeholder User object containing only the user ID, needs to be replaced in business logic
  }

}

关于java - 将无状态 session bean 注入(inject)自定义 JsonDeserializer 失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39111178/

相关文章:

java - 安卓数据验证

java - 在 Eclipse 中更新代号一

c# - 将数据行转换为 JSON 对象

java - 将毫秒时间戳反序列化为 java.time.Instant

java - 使用graphics2D 在图像上环绕/旋转文本

java - 如何将 NumberFormat 从特定货币更改为美元货币

javascript - 我们可以改变Json数据的Key值吗?

javascript - 通过 AJAX/jQuery 将特定值从数据库写入 HTML

java - 复合类的 Jackson CSV 建模器

java - 用于 xml 解码的 JAXB 或 Jackson?