spring Java -需要在构造函数调用super()之前初始化变量

1tu0hz3e  于 2024-01-05  发布在  Spring
关注(0)|答案(2)|浏览(359)

注意:部分代码将使用西班牙语。

嗨,我为我的Sping Boot 应用程序做了一个很好的异常模型,这样我抛出的每个个性化异常都会创建一个个性化的响应(在本例中,它是一个名为ErrorModel得类),这样我就可以用@ExceptionHandler捕获@RestControllerAdvice中得异常,并且只返回该特定异常创建的errorModel。在个性化异常中创建的每个errorModel的特殊之处是codedescription

  1. @Slf4j
  2. @RestControllerAdvice
  3. public class GlobalExceptionHandler {
  4. @ExceptionHandler(ClienteSinMorasException.class)
  5. @ResponseStatus(HttpStatus.NOT_FOUND)
  6. @ResponseBody
  7. public ErrorModel manejoDeClienteSinMoras(ClienteSinMorasException ex) {
  8. return ex.getErrorModel();
  9. }
  10. }

个字符
因此,因为每个异常都是相同的,并且工作方式相同,唯一的区别是codecodigo)和descriptiondescipcion),所以我创建了这个,它工作得很好:

  1. public abstract class IErrorException extends RuntimeException {
  2. @Getter
  3. private final ErrorModel errorModel;
  4. public abstract String getCodigo();
  5. public abstract String getDescripcion();
  6. public IErrorException() {
  7. super();
  8. this.errorModel = new ErrorModel(this.getCodigo(), this.getDescripcion(), null);
  9. }
  10. public IErrorException(ArrayList<HashMap<String, String>> errores) {
  11. super();
  12. this.errorModel = new ErrorModel(this.getCodigo(), this.getDescripcion(), errores);
  13. }
  14. public IErrorException(String errorKey, String errorValue) {
  15. super();
  16. ArrayList<HashMap<String, String>> errores = new ArrayList<>();
  17. errores.add(new HashMap<>(Map.of(errorKey, errorValue)));
  18. this.errorModel = new ErrorModel(this.getCodigo(), this.getDescripcion(), errores);
  19. }
  20. }
  1. // One of the exceptions
  2. @Getter
  3. public class ClienteSinMorasException extends IErrorException {
  4. private final String codigo = "code";
  5. private final String descripcion = "desc";
  6. public ClienteSinMorasException() {
  7. super();
  8. }
  9. public ClienteSinMorasException(ArrayList<HashMap<String, String>> errores) {
  10. super(errores);
  11. }
  12. public ClienteSinMorasException(String errorKey, String errorValue) {
  13. super(errorKey, errorValue);
  14. }
  15. }

的字符串
这是完美的(除了那些超级,但无论如何),但最近,他们要求我改变它,使codedescription是从.yml属性读取,这样的值可以在运行时改变,而不需要重新运行应用程序。
我开始明白,这起作用的唯一原因是,当我在抽象类中执行this.getCodigo()时,explicit primitive类型的final字段“有点已经存在(在本例中,返回“code”)。如果不是final,也不是explicit primitive类型,则不起作用,它在构造函数执行后加载所有内容,因此我最终创建了一个new ErrorModel(null, null, null)。另外,super()必须是构造函数中的第一个元素,因此我无法正确地初始化变量。

  1. // One of the exceptions
  2. @Getter
  3. public class ClienteSinMorasException extends IErrorException {
  4. private final String codigo = Yml.get("cod");
  5. private final String descripcion = Yml.get("desc");
  6. public ClienteSinMorasException() {
  7. super();
  8. }
  9. public ClienteSinMorasException(ArrayList<HashMap<String, String>> errores) {
  10. super(errores);
  11. }
  12. public ClienteSinMorasException(String errorKey, String errorValue) {
  13. super(errorKey, errorValue);
  14. }
  15. }


这不管用,老实说,我不知道该怎么办。

f5emj3cl

f5emj3cl1#

字段初始化器直到稍后才运行。然而,这两个字段似乎没有任何功能。你不需要它们。它们最多是作为缓存Yml读取器值的一种有点蹩脚的方式存在。蹩脚的是,每个异常的示例都会重新创建它们,如果缓存是重点,在Yml类本身中这样做会更有效。
因此,完全删除字段。要么它们根本没有任何意义,要么它们的意义是作为缓存(即避免重复锤击磁盘以多次重新读取YML文件)。如果它是缓存-这是它的错误位置,请编辑Yml的代码以在那里添加缓存行为。无论哪种方式,对这种异常类型的正确处理都是:

  1. public class ClienteSinMorasException extends IErrorException {
  2. @Override public String getCodigo() {
  3. return Yml.get("cod");
  4. }
  5. @Override public String getDescripcion() {
  6. return Yml.get("desc");
  7. }
  8. public ClienteSinMorasException() {
  9. super();
  10. }
  11. public ClienteSinMorasException(ArrayList<HashMap<String, String>> errores) {
  12. super(errores);
  13. }
  14. public ClienteSinMorasException(String errorKey, String errorValue) {
  15. super(errorKey, errorValue);
  16. }
  17. }

字符串
这避免了问题:这里没有字段,所以也没有字段初始化器,所以你不会遇到任何关于初始化序列的问题。

展开查看全部
oug3syen

oug3syen2#

所以我是最初发布这个问题的人。上面的答案是我用的.....直到我最近才明白,从我的Angular 来看,这一切都是愚蠢的。这里是我的新答案。
所以,基本上,我是搬起石头砸自己的脚。原因很简单:**我不需要在初始化类时做所有这些!!! 我可以在调用.getErrorModel()的时候做,这是我需要的时候,而不是之前。这是一个如此明显的疏忽,似乎我被@Getter蒙蔽了双眼,尝试使用它,而不是自己编写任何类型的getter。这是新的解决方案。

  1. public abstract class IErrorException extends RuntimeException {
  2. protected ArrayList<HashMap<String, String>> errores = null;
  3. public abstract String getCodigo();
  4. public abstract String getDescripcion();
  5. public IErrorException() {
  6. super();
  7. }
  8. public IErrorException(ArrayList<HashMap<String, String>> errores) {
  9. super();
  10. this.errores = errores;
  11. }
  12. public IErrorException(String errorKey, String errorValue) {
  13. super();
  14. this.errores = new ArrayList<>();
  15. this.errores.add(new HashMap<>(Map.of(errorKey, errorValue)));
  16. }
  17. public ErrorModel getErrorModel() {
  18. return new ErrorModel(this.getCodigo(), this.getDescripcion(), this.errores);
  19. }
  20. }

字符串
然后:

  1. public class ClienteSinMorasException extends IErrorException {
  2. public ClienteSinMorasException() {
  3. super();
  4. }
  5. public ClienteSinMorasException(ArrayList<HashMap<String, String>> errores) {
  6. super(errores);
  7. }
  8. public ClienteSinMorasException(String errorKey, String errorValue) {
  9. super(errorKey, errorValue);
  10. }
  11. @Override public String getCodigo() {
  12. return "Codigo";
  13. }
  14. @Override public String getDescripcion() {
  15. return "descripcion";
  16. }
  17. }


我认为这是目前为止最好的解决方案,因为它不会不必要地接近边缘。

展开查看全部

相关问题