Si no confía en FindBug, puede ser una opción usar una aserción. De esta manera, evita los problemas mencionados por el usuario MSalters pero aún tiene un cheque. Por supuesto, este enfoque puede no ser aplicable si dicho enfoque de falla rápida no es factible, es decir, debe detectar alguna excepción.
Y para responder aún más a la pregunta de si es una mala práctica. Bueno, esto es discutible pero no lo creo. Considero esta anotación FindBug como una especificación ligera que sigue el diseño por principio de contrato.
En el diseño por contrato, establece un contrato entre un método y su interlocutor. El contrato consta de condiciones previas que la persona que llama acuerda cumplir al llamar al método y una condición posterior que a su vez se cumple con el método después de la ejecución si se cumple su condición previa.
Uno de los beneficios de este método de desarrollo es que puede evitar el llamado estilo de programación defensiva (en el que verifica explícitamente todos los valores de parámetros no válidos, etc.). En cambio, puede confiar en el contrato y evitar verificaciones redundantes para aumentar el rendimiento y la legibilidad.
Los contratos se pueden usar para la verificación de afirmación de tiempo de ejecución que sigue el principio de falla primero mencionado anteriormente en el que desea que su programa falle si se rompe un contrato, lo que le da una pista para identificar la fuente del error. (Una condición previa fallida significa que la persona que llamó hizo algo mal, una condición posterior fallida indica un error de implementación del método dado)
Además, los contratos se pueden utilizar para el análisis estático, que intenta sacar conclusiones sobre el código fuente sin ejecutarlo realmente.
La anotación @nonnull puede verse como una condición previa que la herramienta FindBugs utiliza para el análisis estático. Si prefiere un enfoque como la verificación de aserción en tiempo de ejecución, es decir, la estrategia de falla primero, debería considerar el uso de declaraciones de aserción como Java incorporado. O tal vez para adaptarse a una estrategia de especificación más sofisticada utilizando un lenguaje de especificación de interfaz de comportamiento como el Java Modeling Language (JML).
JML es una extensión de Java en la que la especificación está incrustada en comentarios especiales de Java. Una ventaja de este enfoque es que puede usar especificaciones sofisticadas, incluso usando un enfoque de desarrollo en el que solo confía en la especificación en lugar de en los detalles de implementación y usa diferentes herramientas que hacen uso de la especificación, como las herramientas mencionadas de verificación de afirmación de tiempo de ejecución, Generación automatizada de pruebas unitarias o documentación.
Si comparte mi punto de vista de que @nonnull es un contrato (ligero), sería una práctica común no agregar controles adicionales porque la idea original detrás de este principio es evitar la programación defensiva.
@Nonnull
: como una especificación de contrato. He estado eliminando los controles defensivos detrás del contrato y hasta ahora no he tenido problemas.