IDEA @Contract annotation
-----------------------------------
http://www.jetbrains.com/help/idea/contract-annotations.html
@Contract
The @Contract
annotation is used for defining a contract that a method must meet. This lets the IDE find problems in methods which call methods that you have annotated. You can use this annotation not only for annotating your own code but also for other existing libraries.
The @Contract
annotation has two attributes — value
and pure
. The value
attribute contains clauses describing causal relationship between arguments and the returned value.
The pure
attribute is intended for methods that do not change the state of their objects, but just return a new value. If its return value is not used, removing its invocation will not affect program state or change the semantics, unless the method call throws an exception (exception is not considered to be a side effect).
A method should not be marked as pure if it does not produce a side effect by itself, but it could be used to establish the happens-before relation between an event in another thread, so that changes performed in another thread might become visible in current thread after invocation of this method. On the other hand, some synchronized methods could be marked as pure, because the purpose of synchronization here is to keep the collection internal integrity rather than to wait for an event in another thread. "Invisible" side effects (such as logging) that do not affect important program semantics are allowed.
A contract is a set of clauses that describe an input and an output. They are separated with the ->
symbol: "A -> B"
. This forms a contract meaning that when you provide A to a method, you will always get B. Clauses in a contract must be separated with the ;
(semicolon) symbol. For example:
@Contract("_, null -> null") |
The method returns |
@Contract("_, null -> null; _, !null -> !null") |
The method returns |
@Contract("true -> fail") |
A typical |
@Contract("_ -> this") |
The method always returns its qualifier (e.g. |
@Contract("null -> fail; _ -> param1") |
The method throws an exception if the first argument is null, otherwise it returns the first argument (e.g. |
@Contract("!null, _ -> param1; null, !null -> param2; null, null -> fail") |
The method returns the first non-null argument, or throws an exception if both arguments are null (e.g. |
Define a contract
-
Press Alt+Enter on a method, and select Add method contract or Edit method contract.
-
Configure the contract and apply the changes.
Syntax
The @Contract
annotation value has the following syntax:
contract ::= (clause ‘;’)* clause
clause ::= args ‘->’ effect
args ::= ((arg ‘,’)* arg )?
arg ::= value-constraint
value-constraint ::= ‘_’ | ‘null’ | ‘!null’ | ‘false’ | ‘true’
effect ::= ‘_’ | ‘null’ | ‘!null’ | ‘false’ | ‘true’ | ‘fail’ | ‘new’ | ‘this’ | ‘param’ number
number ::= [1-9] [0-9]*
The constraints are:
_ |
Any value |
|
Null value |
|
Value statically proved not to be null |
|
True boolean value |
|
False boolean value |
|
The method throws an exception if arguments meet argument constraints |
|
Every time the method is executed, it returns a non-null new object that is distinct from other objects existing in the heap prior to method execution. If the method is pure, the new object is not stored in any field or array and will be lost if method return value is not used. |
|
The method returns non-null |
|
The method returns its first (second, third, etc.) argument |
----------------------------------
来源 https://blog.jetbrains.com/idea/2013/10/better-control-flow-analysis-with-contract-annotations-and-intellij-idea-13/
Better Control Flow Analysis with Contract Annotations and IntelliJ IDEA 13
As we’re closing in on the release date, we’re going to publish more details on the new features of IntelliJ IDEA 13. Today we’d like to tell you about the new @Contract annotation.
If you are aware of @Nullable/@NotNull annotations, you probably know how helpful they are against NullPointerException. The new @Contract annotation brings one more layer of safety for your code by defining dependencies between method arguments and return values. This information lets IntelliJ IDEA provide smarter control flow analysis for your code and avoid probable errors.
Here’s an example: say we’ve got a method sort() that returns a sorted list or a null value if the argument is null.
Now if we try to to call this method with a null value and check the result against null, IntelliJ IDEA will not complain because it doesn’t know that a null input yields a null output.
How can we help IntelliJ IDEA? That’s right… Decorate the sort() method with a @Contractannotation, specifying that null inputs yield null outputs:
Now IntelliJ IDEA can see that the sort() method always returns null and the if statement is redundant. The corresponding message appears in the editor.
The @Contract annotation value has the following syntax:
- contract ::= (clause ‘;’)* clause
- clause ::= args ‘->’ effect
- args ::= ((arg ‘,’)* arg )?
- arg ::= value-constraint
- value-constraint ::= ‘any’ | ‘null’ | ‘!null’ | ‘false’ | ‘true’
- effect ::= value-constraint | ‘fail’
The constraints denote the following:
- _ – any value
- null – null value
- !null – a value statically proved to be not-null
- true – true boolean value
- false – false boolean value
- fail – the method throws exception, if the arguments satisfy argument constraints
The @Contract annotation is a powerful and flexible tool that helps make your APIs safer. However, you can use it not only for annotating your own code but also for other existing libraries.
Once you’ve configured the annotations for the project libraries, the IDE stores this information in simple XML files so it can be shared with the team via version control.
To enable the annotations in your project, just add <IntelliJ IDEA Home>/lib/annotations.jar to the classpath via Project Structure.
UPDATE: Please find the latest annotations.jar in Maven repository.
Develop with Pleasure!
--------------------------
https://scratchpad101.com/2015/08/06/annotations-are-evil/
Are annotations bad?
We sacrifice the art of writing good and performant code for the short term gains of improving developer productivity.
Annotations can be powerful but only when used to add context and information to the code. But trying to configure your application with them is nothing less that a crime.
============= End