In the section Swaps, you were introduced to annotations that can be applied to bean
classes, methods, fields, and constructors such as {@link oaj.annotation.Bean @Bean}:
| // Address class with only street/city/state properties (in that order).
| // All other properties are ignored.
| @Bean(properties="street,city,state")
| public class Address { ... }
An alternate way of applying these annotations is to attach them to unrelated classes and methods and then
tell your serializer or parser where to find them.
| // Unannotated class.
| public class Address { ... }
|
| @Bean(onClass=Address.class, properties="street,city,state")
| public static class DummyClass {}
|
| WriterSerializer serializer = JsonSerializer
| .create()
| .applyAnnotations(DummyClass.class)
| .build();
|
| String json = serializer.toString(addressBean);
The advantage to this approach is it allows you to use Juneau annotations on classes/methods/fields/constructors
where you might not have access to the source code, or when you only want to selectively apply the annotation
under certain scenarios instead of globally.
For example, the following shows the @Bean annotation being selectively applied on a single REST method
(described later in juneau-rest-server):
| @RestGet
| @Bean(onClass=Address.class, properties="street,city,state")
| public List<Address> getAddresses() {}
Any Juneau annotation that has an on()/onClass() method can be applied dynamically this way.
These include:
- {@link oaj.annotation.Bean}
- {@link oaj.annotation.Beanc}
- {@link oaj.annotation.BeanIgnore}
- {@link oaj.annotation.Beanp}
- {@link oaj.annotation.Example}
- {@link oaj.annotation.NameProperty}
- {@link oaj.annotation.ParentProperty}
- {@link oaj.annotation.Swap}
- {@link oaj.annotation.Uri}
- {@link oaj.csv.annotation.Csv}
- {@link oaj.html.annotation.Html}
- {@link oaj.json.annotation.Json}
- {@link oaj.annotation.Schema}
- {@link oaj.msgpack.annotation.MsgPack}
- {@link oaj.oapi.annotation.OpenApi}
- {@link oaj.plaintext.annotation.PlainText}
- {@link oaj.soap.annotation.SoapXml}
- {@link oaj.uon.annotation.Uon}
- {@link oaj.urlencoding.annotation.UrlEncoding}
- {@link oaj.xml.annotation.Xml}
The valid pattern matches are:
- Classes:
- Fully qualified:
- Fully qualified inner class:
- "com.foo.MyClass$Inner1$Inner2"
- Simple:
- Simple inner:
- "MyClass$Inner1$Inner2"
- "Inner1$Inner2"
- "Inner2"
- Methods:
- Fully qualified with args:
- "com.foo.MyClass.myMethod(String,int)"
- "com.foo.MyClass.myMethod(java.lang.String,int)"
- "com.foo.MyClass.myMethod()"
- Fully qualified:
- "com.foo.MyClass.myMethod"
- Simple with args:
- "MyClass.myMethod(String,int)"
- "MyClass.myMethod(java.lang.String,int)"
- "MyClass.myMethod()"
- Simple:
- Simple inner class:
- "MyClass$Inner1$Inner2.myMethod"
- "Inner1$Inner2.myMethod"
- "Inner2.myMethod"
- Fields:
- Fully qualified:
- "com.foo.MyClass.myField"
- Simple:
- Simple inner class:
- "MyClass$Inner1$Inner2.myField"
- "Inner1$Inner2.myField"
- "Inner2.myField"
- Constructors:
- Fully qualified with args:
- "com.foo.MyClass(String,int)"
- "com.foo.MyClass(java.lang.String,int)"
- "com.foo.MyClass()"
- Simple with args:
- "MyClass(String,int)"
- "MyClass(java.lang.String,int)"
- "MyClass()"
- Simple inner class:
- "MyClass$Inner1$Inner2()"
- "Inner1$Inner2()"
- "Inner2()"
- A comma-delimited list of anything on this list.