All Downloads are FREE. Search and download functionalities are using the official Maven repository.

groovy.transform.Canonical Maven / Gradle / Ivy

The newest version!
/*
 *  Licensed to the Apache Software Foundation (ASF) under one
 *  or more contributor license agreements.  See the NOTICE file
 *  distributed with this work for additional information
 *  regarding copyright ownership.  The ASF licenses this file
 *  to you under the Apache License, Version 2.0 (the
 *  "License"); you may not use this file except in compliance
 *  with the License.  You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 *  Unless required by applicable law or agreed to in writing,
 *  software distributed under the License is distributed on an
 *  "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 *  KIND, either express or implied.  See the License for the
 *  specific language governing permissions and limitations
 *  under the License.
 */
package groovy.transform;

import org.codehaus.groovy.transform.GroovyASTTransformationClass;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * Class annotation used to assist in the creation of mutable classes.
 * 

* It allows you to write classes in this shortened form: *

 * {@code @Canonical} class Customer {
 *     String first, last
 *     int age
 *     Date since
 *     Collection favItems = ['Food']
 *     def object 
 * }
 * def d = new Date()
 * def anyObject = new Object()
 * def c1 = new Customer(first:'Tom', last:'Jones', age:21, since:d, favItems:['Books', 'Games'], object: anyObject)
 * def c2 = new Customer('Tom', 'Jones', 21, d, ['Books', 'Games'], anyObject)
 * assert c1 == c2
 * 
* * You don't need to provide all arguments in constructor calls. If using named parameters, any property names not * referenced will be given their default value (as per Java's default unless an explicit initialization constant is * provided when defining the property). If using a tuple constructor, parameters are supplied in the order in which * the properties are defined. Supplied parameters fill the tuple from the left. Any parameters missing on the right * are given their default value. *
 * def c3 = new Customer(last: 'Jones', age: 21)
 * def c4 = new Customer('Tom', 'Jones')
 *
 * assert null == c3.since
 * assert 0 == c4.age
 * assert c3.favItems == ['Food'] && c4.favItems == ['Food']
 * 
* * The {@code @Canonical} annotation instructs the compiler to execute an * AST transformation which adds positional constructors, * equals, hashCode and a pretty print toString to your class. There are additional * annotations if you only need some of the functionality: {@code @EqualsAndHashCode}, * {@code @ToString} and {@code @TupleConstructor}. In addition, you can add one of * the other annotations if you need to further customize the behavior of the * AST transformation. *

* A class created in this way has the following characteristics: *

    *
  • A no-arg constructor is provided which allows you to set properties by name using Groovy's normal bean conventions. *
  • Tuple-style constructors are provided which allow you to set properties in the same order as they are defined. *
  • Default {@code equals}, {@code hashCode} and {@code toString} methods are provided based on the property values. * Though not normally required, you may write your own implementations of these methods. For {@code equals} and {@code hashCode}, * if you do write your own method, it is up to you to obey the general contract for {@code equals} methods and supply * a corresponding matching {@code hashCode} method. * If you do provide one of these methods explicitly, the default implementation will be made available in a private * "underscore" variant which you can call. E.g., you could provide a (not very elegant) multi-line formatted * {@code toString} method for {@code Customer} above as follows: *
     *     String toString() {
     *        _toString().replaceAll(/\(/, '(\n\t').replaceAll(/\)/, '\n)').replaceAll(/, /, '\n\t')
     *    }
     * 
    * If an "underscore" version of the respective method already exists, then no default implementation is provided. *
*

* If you want similar functionality to what this annotation provides but also require immutability, see the * {@code @}{@link Immutable} annotation. *

* Limitations: *

    *
  • If you explicitly add your own constructors, then the transformation will not add any other constructor to the class
  • *
  • Groovy's normal map-style naming conventions will not be available if the first property * has type {@code LinkedHashMap} or if there is a single Map, AbstractMap or HashMap property
  • *
* * @author Paulo Poiati * @author Paul King * @see groovy.transform.EqualsAndHashCode * @see groovy.transform.ToString * @see groovy.transform.TupleConstructor * @see groovy.transform.Immutable * @since 1.8.0 */ @java.lang.annotation.Documented @Retention(RetentionPolicy.SOURCE) @Target({ElementType.TYPE}) @GroovyASTTransformationClass("org.codehaus.groovy.transform.CanonicalASTTransformation") public @interface Canonical { /** * List of field and/or property names to exclude. * Must not be used if 'includes' is used. For convenience, a String with comma separated names * can be used in addition to an array (using Groovy's literal list notation) of String values. * * If the {@code @Canonical} behavior is customised by using it in conjunction with one of the more specific * related annotations (i.e. {@code @ToString}, {@code @EqualsAndHashCode} or {@code @TupleConstructor}), then * the value of this attribute can be overridden within the more specific annotation. */ String[] excludes() default {}; /** * List of field and/or property names to include. * Must not be used if 'excludes' is used. For convenience, a String with comma separated names * can be used in addition to an array (using Groovy's literal list notation) of String values. * * If the {@code @Canonical} behavior is customised by using it in conjunction with one of the more specific * related annotations (i.e. {@code @ToString}, {@code @EqualsAndHashCode} or {@code @TupleConstructor}), then * the value of this attribute can be overridden within the more specific annotation. */ String[] includes() default {}; }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy