2012-09-01 13 views
5

que tienen la clase siguiente:¿Validar campos tanto en el constructor como en el setter es considerado un código redundante malo?

public class Project { 

    private int id; 
    private String name; 

    public Project(int id, String name) { 
     if(name == null){ 
      throw new NullPointerException("Name can't be null"); 
     } 

     if(id == 0){ 
      throw new IllegalArgumentException("id can't be zero"); 
     } 

      this.name = name; 
      this.id = id; 

    } 

    private Project(){} 

    public int getId() { 
     return id; 
    } 

    public void setId(int id) { 
     if(id == 0){ 
      throw new IllegalArgumentException("id can't be zero"); 
     } 
     this.id = id; 
    } 

    public String getName() 
     return name; 
    } 

    public void setName(String name) { 
     if(name == null){ 
      throw new NullPointerException("Name can't be null"); 
     } 
     this.name = name; 
    } 

} 

Si notas que setName y setId comparten la misma validación de sus campos con el constructor. ¿Este código redundante podría causar problemas en el futuro (por ejemplo, si alguien edita el setter para permitir 0 para el id y evitar -1 en su lugar, pero no cambió el constructor)? . ¿Debo usar un método privado para hacer el control y compartirlo entre el constructor y el colocador, que parece demasiado si hay muchos campos?

Nota: Esta es la razón por la que no estoy usando los setters en el constructor. https://stackoverflow.com/a/4893604/302707

+4

Use setters en ctors Si usted está preocupado. –

+2

O (en caso de que no se pueda llamar a un colocador desde Constructor), pase la lógica de validación a un método privado compartido. – SJuan76

+0

¿Qué pasa con el constructor 'privado' y la validación en el getter? – mre

Respuesta

1

Recomiendo que debe definir un método por campo como isValid() y luego llamar al mismo método en setter y en Constructor.

+0

Pero, ¿cómo crees que es válido para ID y nombre? – Jimmy

0

Yo diría que sí. En lugar de eso, simplemente llame a los emisores de su constructor:

public Project(int id, String name, Date creationDate, int fps, List<String> frames) { 
    setName(name); 
    setId(id); 
    // other stuff with creationDate, fps and frames? 
} 

Además, no se debe comprobar si hay un nulo name en getName - lo hacen en setName. De lo contrario, los bugs serán difíciles de rastrear; querrás capturar el nombre inválido tan pronto como ingrese, no cuando se use (lo cual puede ser mucho más tarde).

+0

En mi humilde opinión, este es un mal consejo ... a menos que haya hecho los métodos de los mutadores 'final'. – mre

+0

malo porque si alguien anula 'setName' y' setId' podría ser una catástrofe.no se recomienda llamar al método no privado o no final en el constructor dentro de la misma clase – gigadot

+0

Buena captura para ambos; deberías hacer que los métodos de los mutadores (o toda la clase) sean "finales" si sigues este enfoque. Por supuesto, si utiliza un enfoque como 'isValid (String)' o 'checkName (String)', se aplica la misma condición. – yshavit

0

si hace que Project sea inmutable, eliminará el código redundante. pero por ahora, creo que arrojar excepciones explícitamente en los métodos constructor y mutador está bien.

y no llamaría a los métodos de mutador dentro del constructor por muchas razones, incluyendo this. Además, eliminaría el código de validación en el método de acceso ... no es necesario.

2

Aquí está el código revisado:

public class Project { 

    private int id; 
    private String name; 

    public Project(int id, String name, Date creationDate, int fps, List<String> frames) { 

     checkId(id); 
      checkName(name); 
      //Insted of lines above you can call setters too. 

      this.name = name; 
      this.id = id; 

    } 

    private Project(){} 

    public int getId() { 
     return id; 
    } 

    public void setId(int id) { 
     checkId(id); 
     this.id = id; 
    } 

    public String getName() 
     return name; 
    } 

    public void setName(String name) { 
      checkName(name); 
     this.name = name; 
    } 

    private void checkId(int id){ 
     if(id == 0){ 
      throw new IllegalArgumentException("id can't be zero"); 
     } 
    } 

    private void checkName(String name){ 
      if(name == null){ 
      throw new NullPointerException("Name can't be null"); 
     } 
    } 

} 
+0

Si bien es una solución, todavía no aborda las preocupaciones de los OP. –

+0

esto es una compensación y depende del requisito del desarrollador. depende de la forma en que mantienen su código. – Heidarzadeh

Cuestiones relacionadas