Apache Groovy
Apache Groovy is a Java-syntax-compatible object-oriented programming language for the Java platform. It is both a static and dynamic language with features similar to those of Python, Ruby, and Smalltalk. It can be used as both a programming language and a scripting language for the Java Platform, is compiled to Java virtual machine bytecode, and interoperates seamlessly with other Java code and libraries. Groovy uses a curly-bracket syntax similar to Java's. Groovy supports closures, multiline strings, and expressions embedded in strings. Much of Groovy's power lies in its AST transformations, triggered through annotations.
Groovy 1.0 was released on January 2, 2007, and Groovy 2.0 in July, 2012. Since version 2, Groovy can be compiled statically, offering type inference and performance near that of Java. Groovy 2.4 was the last major release under Pivotal Software's sponsorship which ended in March 2015. Groovy has since changed its governance structure to a Project Management Committee in the Apache Software Foundation.
History
first talked about the development of Groovy on his blog in August 2003. In March 2004, Groovy was submitted to the JCP as JSR 241 and accepted by ballot. Several versions were released between 2004 and 2006. After the Java Community Process standardization effort began, the version numbering changed, and a version called "1.0" was released on January 2, 2007. After various betas and release candidates numbered 1.1, on December 7, 2007, Groovy 1.1 Final was released and immediately renumbered as Groovy 1.5 to reflect the many changes made.In 2007, Groovy won the first prize at JAX 2007 innovation award. In 2008, Grails, a Groovy web framework, won the second prize at JAX 2008 innovation award.
In November 2008, SpringSource acquired the Groovy and Grails company. In August 2009 VMware acquired SpringSource.
In April 2012, after eight years of inactivity, the Spec Lead changed the status of JSR 241 to dormant.
Strachan had left the project silently a year before the Groovy 1.0 release in 2007. In Oct 2016, Strachan stated "I still love groovy, java, go, typescript and kotlin".
On July 2, 2012, Groovy 2.0 was released, which, among other new features, added static compiling and static type checking.
When the Pivotal Software joint venture was spun-off by EMC Corporation and VMware in April 2013, Groovy and Grails formed part of its product portfolio. Pivotal ceased sponsoring Groovy and Grails from April 2015.
That same month, Groovy changed its governance structure from a Codehaus repository to a Project Management Committee in the Apache Software Foundation via its incubator.
Groovy graduated from Apache's incubator and became a top-level project in November 2015.
Features
Most valid Java files are also valid Groovy files. Although the two languages are similar, Groovy code can be more compact, because it does not need all the elements that Java needs. This makes it possible for Java programmers to learn Groovy gradually by starting with familiar Java syntax before acquiring more Groovy programming idioms.Groovy features not available in Java include both static and dynamic typing, operator overloading, native syntax for lists and associative arrays, native support for regular expressions, polymorphic iteration, string interpolation, added helper methods, and the safe navigation operator
?.
to check automatically for null pointers.Since version 2 Groovy also supports modularity, type checking, static compiling, Project Coin syntax enhancements, multicatch blocks and ongoing performance enhancements using the
invokedynamic
instruction introduced in Java 7.Groovy provides native support for various markup languages such as XML and HTML, accomplished via an inline Document Object Model syntax. This feature enables the definition and manipulation of many types of heterogeneous data assets with a uniform and concise syntax and programming methodology.
Unlike Java, a Groovy source code file can be executed as an script, if it contains code outside any class definition, if it is a class with a main method, or if it is a Runnable or GroovyTestCase. A Groovy script is fully parsed, compiled, and generated before executing. This occurs under the hood, and the compiled version is not saved as an artifact of the process.
GroovyBeans, properties
GroovyBeans are Groovy's version of JavaBeans. Groovy implicitly generates getters and setters. In the following code,setColor
and getColor
are implicitly generated. The last two lines, which appear to access color directly, are actually calling the implicitly generated methods.class AGroovyBean
def myGroovyBean = new AGroovyBean
myGroovyBean.setColor
assert myGroovyBean.getColor 'baby blue'
myGroovyBean.color = 'pewter'
assert myGroovyBean.color 'pewter'
Groovy offers simple, consistent syntax for handling lists and maps, reminiscent of Java's array syntax.
def movieList = // Looks like an array, but is a list
assert movieList 'Seven Samurai'
movieList = 'Casablanca' // Adds an element to the list
assert movieList.size 4
def monthMap = // Declares a map
assert monthMap 31 // Accesses an entry
monthMap = 30 // Adds an entry to the map
assert monthMap.size 4
Prototype extension
Groovy offers support for prototype extension throughExpandoMetaClass
, Extension Modules, Objective-C-like Categories and DelegatingMetaClass
.ExpandoMetaClass
offers a domain-specific language to express the changes in the class easily, similar to Ruby's open class concept:Number.metaClass
assert 9.sqrt 3
assert 4.sqrt 2
Groovy's changes in code through prototyping are not visible in Java, since each attribute/method invocation in Groovy goes through the metaclass registry. The changed code can only be accessed from Java by going to the metaclass registry.
Groovy also allows overriding methods as
getProperty
, propertyMissing
among others, enabling the developer to intercept calls to an object and specify an action for them, in a simplified aspect-oriented way. The following code enables the class java.lang.String
to respond to the hex
property:enum Color
String.metaClass.getProperty =
assert "WHITE".hex "#FFFFFF"
assert "BLUE".hex "#0000FF"
assert "BLACK".hex "#000000"
assert "GREEN".hex null
The Grails framework uses metaprogramming extensively to enable GORM dynamic finders, like
User.findByName
and others.Dot and parentheses
Groovy's syntax permits omitting parentheses and dots in some situations. The following groovy codetake.with.and
can be written as
take coffee with sugar, milk and liquor
enabling the development of domain-specific languages that look like plain English.
Functional programming
Although Groovy is mostly an object-oriented language, it also offers functional programming features.Closures
According to Groovy's documentation: "Closures in Groovy work similar to a 'method pointer', enabling code to be written and run in a later point in time". Groovy's closures support free variables, i.e. variables that have not been explicitly passed as a parameter to it, but exist in its declaration context, partial application, delegation, implicit, typed and untyped parameters.When working on Collections of a determined type, the closure passed to an operation on the collection can be inferred:
list =
/*
* Non-zero numbers are coerced to true, so when it % 2 0, it is false.
* The type of the implicit "it" parameter can be inferred as an Integer by the IDE.
* It could also be written as:
* list.findAll
* list.findAll
*/
def odds = list.findAll
assert odds
A group of expressions can be written in a closure block without reference to an implementation and the responding object can be assigned at a later point using delegation:
// This block of code contains expressions without reference to an implementation
def operations =
/*
* This class will handle the operations that can be used in the closure above. Another class
* could be declared having the same methods, but using, for example, webservice operations
* in the calculations.
*/
class Expression
// Here is defined who is going to respond the expressions in the block of code above.
operations.delegate = new Expression
operations
Curry
Usually called partial application, this Groovy feature allows closures' parameters to be set to a default parameter in any of their arguments, creating a new closure with the bound value. Supplying one argument to thecurry
method will fix argument one. Supplying N arguments will fix arguments 1.. N.def joinTwoWordsWithSymbol =
assert joinTwoWordsWithSymbol 'Hello#World'
def concatWords = joinTwoWordsWithSymbol.curry
assert concatWords 'Hello World'
def prependHello = concatWords.curry
//def prependHello = joinTwoWordsWithSymbol.curry
assert prependHello 'Hello World'
Curry can also be used in the reverse direction using
rcurry
.def power =
def square = power.rcurry
def cube = power.rcurry
assert power 4
assert square 16
assert cube 27
Groovy also supports lazy evaluation, reduce/fold, infinite structures and immutability, among others.
JSON and XML processing
On JavaScript Object Notation and XML processing, Groovy employs the Builder pattern, making the production of the data structure less verbose. For example, the following XML:can be generated via the following Groovy code:
def writer = new StringWriter
def builder = new groovy.xml.MarkupBuilder
builder.languages
and also can be processed in a streaming way through
StreamingMarkupBuilder
. To change the implementation to JSON, the MarkupBuilder
can be swapped to JsonBuilder
.To parse it and search for a functional language, Groovy's
findAll
method can serve:def languages = new XmlSlurper.parseText writer.toString
// Here is employed Groovy's regex syntax for a matcher that will be coerced to a
// boolean value: either true, if the value contains our string, or false otherwise.
def functional = languages.language.findAll
assert functional.collect
String interpolation
In Groovy, strings can be interpolated with variables and expressions by using GStrings:BigDecimal account = 10.0
def text = "The account shows currently a balance of $account"
assert text "The account shows currently a balance of 10.0"
GStrings containing variables and expressions must be declared using double quotes.
A complex expression must be enclosed in curly brackets. This prevents parts of it from being interpreted as belonging to the surrounding string instead of to the expression:
BigDecimal minus = 4.0
text = "The account shows currently a balance of $"
assert text "The account shows currently a balance of 6.0"
// Without the brackets to isolate the expression, this would result:
text = "The account shows currently a balance of $account - minus"
assert text "The account shows currently a balance of 10.0 - minus"
Expression evaluation can be deferred by employing arrow syntax:
BigDecimal tax = 0.15
text = "The account shows currently a balance of $"
tax = 0.10
// The tax value was changed AFTER declaration of the GString. The expression
// variables are bound only when the expression must actually be evaluated:
assert text "The account shows currently a balance of 9.000"
Abstract syntax tree transformation
According to Groovy's own documentation, "When the Groovy compiler compiles Groovy scripts and classes, at some point in the process, the source code will end up being represented in memory in the form of a Concrete Syntax Tree, then transformed into an Abstract Syntax Tree. The purpose of AST Transformations is to let developers hook into the compilation process to be able to modify the AST before it is turned into bytecode that will be run by the JVM. AST Transformations provides Groovy with improved compile-time metaprogramming capabilities allowing powerful flexibility at the language level, without a runtime performance penalty."Examples of ASTs in Groovy are:
- Category and Mixin transformation
- Immutable AST Macro
- Newify transformation
- Singleton transformation
Traits
According to Groovy's documentation, "Traits are a structural construct of the language that allows: composition of behaviors, runtime implementation of interfaces, behavior overriding, and compatibility with static type checking/compilation."Traits can be seen as interfaces carrying both default implementations and state. A trait is defined using the trait keyword:
trait FlyingAbility
Then, it can be used like a normal interface using the keyword
implements
:class Bird implements FlyingAbility /* Adds the trait FlyingAbility to the Bird class capabilities */
def bird = new Bird /* instantiate a new Bird */
assert bird.fly "I'm flying!" /* the Bird class automatically gets the behavior of the FlyingAbility trait */
Traits allow a wide range of abilities, from simple composition to testing.
Adoption
Notable examples of Groovy adoption include:- Adaptavist ScriptRunner, embeds a Groovy implementation to automate and extend Atlassian tools, in use by more than 20000 organisations around the world.
- Apache OFBiz, the open-source enterprise resource planning system, uses Groovy.
- Eucalyptus, a cloud management system, uses a significant amount of Groovy.
- Gradle is a popular build automation tool using Groovy.
- LinkedIn uses Groovy and Grails for some of their subsystems.
- LogicMonitor, a cloud-based monitoring platform, uses Groovy in script-based data sources.
- Jenkins, a platform for continuous integration. With version 2, Jenkins includes a Pipeline plugin that allows for build instructions to be written in Groovy.
- Liferay, uses groovy in their kaleo workflow
- Sky.com uses Groovy and Grails to serve massive online media content.
- SmartThings, an open platform for smart homes and the consumer Internet of Things, uses a security-oriented subset of Groovy
- SoapUI provides Groovy as a language for webservice tests development.
- Survata, a market research startup, uses Groovy and Grails.
- The European Patent Office developed a dataflow programming language in Groovy "to leverage similarities in the processes for communicating with each individual country’s patent office, and transform them into a single, universal process."
- Though Groovy can be integrated into any JVM environment, the JBoss Seam framework provides Groovy, besides Java, as a development language, out of the box.
- vCalc.com uses Groovy for all of the user defined mathematics in its math crowd-sourcing engine.
- Wired.com uses Groovy and Grails for the Product Reviews standalone section of the website.
- XWiki SAS uses Groovy as scripting language in their collaborative open-source product.
IDE support
- Android Studio, IDE used for making Android apps
- Atom editor
- Eclipse, through Groovy-Eclipse
- Emacs, using the groovy-emacs-mode project's groovy-mode.
- IntelliJ IDEA, Community Edition, Grails/Griffon in the Ultimate Edition only
- JDeveloper, for use with Oracle ADF
- jEdit, an advanced text editor for the Java platform
- Kate, an advanced text editor for KDE supports Groovy and over 200 other file formats
- NetBeans, since version 6.5
- Notepad++, an advanced text editor for Microsoft Windows
- Sublime Text, a cross platform text editor
- TextMate
- Visual Studio Code
- UltraEdit, general purpose program editor
Dialects
- Grooscript converts Groovy code to JavaScript code. Although Grooscript has some limitations compared to Apache Groovy, it can use domain classes in both the server and the client. Plugin support for Grails version 3.0 is provided, as well as online code conversions.
Citations