@Documented
@Retention(value=SOURCE)
@Target(value=TYPE)
public @interface TupleConstructor
It allows you to write classes in this shortened form:
@groovy.transform.TupleConstructor
class Customer {
String first, last
int age
Date since
Collection favItems
}
def c1 = new Customer(first:'Tom', last:'Jones', age:21, since:new Date(), favItems:['Books', 'Games'])
def c2 = new Customer('Tom', 'Jones', 21, new Date(), ['Books', 'Games'])
def c3 = new Customer('Tom', 'Jones')
The @TupleConstructor
annotation instructs the compiler to execute an
AST transformation which adds the necessary constructor method to your class.
A tuple constructor is created with a parameter for each property (and optionally field and super properties). A default value is provided (using Java's default values) for all parameters in the constructor. Groovy's normal conventions then allows any number of parameters to be left off the end of the parameter list including all of the parameters - giving a no-arg constructor which can be used with the map-style naming conventions.
The order of parameters is given by the properties of any super classes with most super first
(if includeSuperProperties
is set) followed by the properties of the class followed
by the fields of the class (if includeFields
is set). Within each grouping the order
is as attributes appear within the respective class.
More examples:
//-------------------------------------------------------------------------- import groovy.transform.TupleConstructor @TupleConstructor() class Person { String name List likes private boolean active = false } def person = new Person('mrhaki', ['Groovy', 'Java']) assert person.name == 'mrhaki' assert person.likes == ['Groovy', 'Java'] person = new Person('mrhaki') assert person.name == 'mrhaki' assert !person.likes
//-------------------------------------------------------------------------- // includeFields in the constructor creation. import groovy.transform.TupleConstructor @TupleConstructor(includeFields=true) class Person { String name List likes private boolean active = false boolean isActivated() { active } } def person = new Person('mrhaki', ['Groovy', 'Java'], true) assert person.name == 'mrhaki' assert person.likes == ['Groovy', 'Java'] assert person.activated
//-------------------------------------------------------------------------- // use force attribute to force creation of constructor // even if we define our own constructors. import groovy.transform.TupleConstructor @TupleConstructor(force=true) class Person { String name List likes private boolean active = false Person(boolean active) { this.active = active } boolean isActivated() { active } } def person = new Person('mrhaki', ['Groovy', 'Java']) assert person.name == 'mrhaki' assert person.likes == ['Groovy', 'Java'] assert !person.activated person = new Person(true) assert person.activated
//-------------------------------------------------------------------------- // include properties and fields from super class. import groovy.transform.TupleConstructor @TupleConstructor(includeFields=true) class Person { String name List likes private boolean active = false boolean isActivated() { active } } @TupleConstructor(callSuper=true, includeSuperProperties=true, includeSuperFields=true) class Student extends Person { List courses } def student = new Student('mrhaki', ['Groovy', 'Java'], true, ['IT']) assert student.name == 'mrhaki' assert student.likes == ['Groovy', 'Java'] assert student.activated assert student.courses == ['IT']
Known Limitations:
@InheritConstructors
);
the order in which the particular transforms are processed becomes important in that case.
See the force
attribute for further details about customizing this behavior.@InheritConstructors
);
the order in which the particular transforms are processed becomes important in that case.
See the defaults
attribute for further details about customizing this behavior.LinkedHashMap
or if there is a single Map, AbstractMap or HashMap property (or field)Modifier and Type | Optional Element and Description |
---|---|
boolean |
allNames
Whether to include all fields and/or properties within the constructor, including those with names that are
considered internal.
|
boolean |
callSuper
Should super properties be called within a call to the parent constructor
rather than set as properties.
|
boolean |
defaults
Used to set whether default value processing is enabled (the default) or disabled.
|
java.lang.String[] |
excludes
List of field and/or property names to exclude from the constructor.
|
boolean |
force
By default, this annotation becomes a no-op if you provide your own constructor.
|
boolean |
includeFields
Include fields in the constructor.
|
boolean |
includeProperties
Include properties in the constructor.
|
java.lang.String[] |
includes
List of field and/or property names to include within the constructor.
|
boolean |
includeSuperFields
Include fields from super classes in the constructor.
|
boolean |
includeSuperProperties
Include properties from super classes in the constructor.
|
java.lang.Class |
post
A Closure containing statements which will be appended to the end of the generated constructor.
|
java.lang.Class |
pre
A Closure containing statements which will be prepended to the generated constructor.
|
boolean |
useSetters
By default, properties are set directly using their respective field.
|
public abstract java.lang.String[] excludes
public abstract java.lang.String[] includes
public abstract boolean includeFields
public abstract boolean includeProperties
public abstract boolean includeSuperFields
public abstract boolean includeSuperProperties
public abstract boolean callSuper
includeSuperProperties
.
Can't be true if using pre
with a super
first statement.public abstract boolean force
force=true
then the tuple constructor(s) will be added regardless of
whether existing constructors exist. It is up to you to avoid creating duplicate constructors.public abstract boolean defaults
public abstract boolean useSetters
useSetters=true
then a writable property will be set using its setter.
If turning on this flag we recommend that setters that might be called are
made null-safe wrt the parameter.public abstract boolean allNames
public abstract java.lang.Class pre
super(someArgs)
in which case the no-arg super constructor won't be called.