Implementing the Converter
An Attribute Converter allows us to implement methods to convert the value of an entity attribute to its database representation and back. I will not get into too much details on how to implement an Attribute Converter because I already did this in one of my former articles.
By implementing our own mapping, we can choose a compact database representation and make sure, that changing the enum in any way will not break the existing mapping. Therefore we add the shortName, which will be used as the database representation, as an additional property to the Enum. We also need a method to get the shortName property and to get the Enum for a given shortName:
Now we can implement the converter to use the shortName property to store the Enum in the database:
Now we can implement the converter to use the shortName property to store the Enum in the database:
The VehicleConverter maps the enum value to the shortName to store it in the database. By declaring it with @Converter(autoApply = true), we tell the JPA provider to use it to map all Vehicle enums. So we do not need to specify the converter at each entity attribute of type Vehicle.
But there is one thing we need to take care of and if you have read my former article about JPA Attribute Converter you might have wondered already. Attribute Converter cannot be applied to attributes annotated with @Enumerated. So we have to make sure that there is no @Enumerated annotation at our entity attributes of type Vehicle.
What do you think about using JPA Attribute Converter to persist enums? Please leave me a comment!
If you enjoyed reading this article and like to learn more about other Java EE7 features, make sure to subscribe to my RSS feed or follow me on twitter or google+.
And if you like to read more about the features introduced with JPA 2.1, have a look at my JPA 2.1 overview: JPA 2.1 - 12 features every developer should know.
But there is one thing we need to take care of and if you have read my former article about JPA Attribute Converter you might have wondered already. Attribute Converter cannot be applied to attributes annotated with @Enumerated. So we have to make sure that there is no @Enumerated annotation at our entity attributes of type Vehicle.
Using the Converter
OK, implementing the Converter was easy. But how do we use it in the application?
This is one of the best parts of the Attribute Converter. We do not have to do anything. The persistence provider will use it for all read and write operations, e.g. in named queries:
Conclusion
We implemented a simple Attribute Converter that uses our own rules to convert the Vehicle enum to its database representation. So we can make sure that changing the values of the Vehicle enum will not break the existing/remaining mappings.What do you think about using JPA Attribute Converter to persist enums? Please leave me a comment!
If you enjoyed reading this article and like to learn more about other Java EE7 features, make sure to subscribe to my RSS feed or follow me on twitter or google+.
And if you like to read more about the features introduced with JPA 2.1, have a look at my JPA 2.1 overview: JPA 2.1 - 12 features every developer should know.
Maybe the best solution would be use a generic type converter in order to manage all enums. In your example for each enum we have to create a converter.
ReplyDeleteHi Gamal,
Deletethanks for your comment.
I would love to create a generic type converter. But up to now, I have no idea how to implement the convertToEntityAttribute(String dbData) method in a generic way. The method has to create a specific enum and I have no idea how to get the required one. Any suggestions?
Regards,
Thorben
PS: If you want to try something, you can find the source on github: https://github.com/somethoughtsonjava/JPA2.1-EnumConverter
Even nicer would be to add a visitor to the enum and to use it inside the converter. That way the compiler would notify you in case the enum has changed.
ReplyDeleteHi Frisian
DeleteYou are right, using a visitor would be even better. I will use that for my real applications. Thanks!
Regards,
Thorben
I'd be happy if I could just change the JPA default from ORDINAL to STRING.
ReplyDelete