Saturday, January 23, 2010

Groovy: Injecting New Methods on Existing Java Classes

One of Groovy's most significant "goodies" is the ability to intercept calls, even to standard Java classes, and either override existing methods or even add new methods dynamically. In this blog post, I will demonstrate how to add a method to the Calendar that provides the appropriate Zodiak Tropical Sign.

Groovy offers rich dynamic capabilities via its MetaClass support. In this blog post, I will take advantage of this feature to have a method called getZodiakTropicalSign() injected into the SDK-provided Calendar. The purpose of this method is to provide the applicable Zodiak sign for a given instance of Calendar.

The portion of this script most relevant to this post appears at the top of the script. The syntax Calendar.metaClass.getZodiakTropicalSign = { -> defines a new method getZodiakTrpopicalSign on Calendar and then uses delegate to work with the underlying instance of java.util.Calendar.


  1. #!/usr/bin/env groovy  
  2.   
  3. Calendar.metaClass.getZodiakTropicalSign = { ->  
  4.    def humanMonth = delegate.get(Calendar.MONTH) + 1  
  5.    def day = delegate.get(Calendar.DATE)  
  6.    def zodiakTropicalSign = "Unknown"  
  7.    switch (humanMonth)  
  8.    {  
  9.       case 1 :  
  10.          zodiakTropicalSign = day < 20 ? "Capricorn" : "Aquarius"   
  11.          break  
  12.       case 2 :  
  13.          zodiakTropicalSign = day < 19 ? "Aquarius" : "Pisces"  
  14.          break  
  15.       case 3 :  
  16.          zodiakTropicalSign = day < 21 ? "Pisces" : "Aries"  
  17.          break  
  18.       case 4 :  
  19.          zodiakTropicalSign = day < 20 ? "Aries" : "Taurus"  
  20.          break  
  21.       case 5 :  
  22.          zodiakTropicalSign = day < 21 ? "Taurus" : "Gemini"  
  23.          break  
  24.       case 6 :  
  25.          zodiakTropicalSign = day < 21 ? "Gemini" : "Cancer"  
  26.          break  
  27.       case 7 :  
  28.          zodiakTropicalSign = day < 23 ? "Cancer" : "Leo"  
  29.          break  
  30.       case 8 :   
  31.          zodiakTropicalSign = day < 23 ? "Leo" : "Virgo"  
  32.          break  
  33.       case 9 :  
  34.          zodiakTropicalSign = day < 23 ? "Virgo" : "Libra"  
  35.          break  
  36.       case 10 :  
  37.          zodiakTropicalSign = day < 23 ? "Libra" : "Scorpio"  
  38.          break  
  39.       case 11 :  
  40.          zodiakTropicalSign = day < 22 ? "Scorpio" : "Sagittarius"  
  41.          break  
  42.       case 12 :  
  43.          zodiakTropicalSign = day < 22 ? "Sagittarius" : "Capricorn"  
  44.          break  
  45.    }  
  46.    return zodiakTropicalSign  
  47. }  
  48.   
  49. def calendar = Calendar.instance  
  50. // First argument to script is expected to be month integer between 1 (January)  
  51. // and 12 (December).  Second argument is expected to be the date of that month.  
  52. // If fewer than two arguments are provided, will use today's date (default) instead.  
  53. if (args.length > 1)  
  54. {  
  55.    try  
  56.    {  
  57.       calendar.set(Calendar.MONTH, (args[0] as Integer) - 1)  
  58.       calendar.set(Calendar.DATE, args[1] as Integer)  
  59.    }  
  60.    catch (NumberFormatException nfe)  
  61.    {  
  62.       println "Arguments ${args[0]} and ${args[1]} are not valid respective month and date integers."  
  63.       println "Using today's date (${calendar.get(Calendar.MONTH)+1}/${calendar.get(Calendar.DATE)}) instead."  
  64.    }  
  65. }  
  66. print "A person born on ${calendar.getDisplayName(Calendar.MONTH, Calendar.LONG, Locale.US)}"  
  67. println " ${calendar.get(Calendar.DATE)} has the Zodiak sign ${calendar.getZodiakTropicalSign()}"  


The final line of the above script calls the injected getZodiakTropicalSign() method on Calendar. This is demonstrated in the next screen snapshot.



The first execution of the script in the above screen snapshot is executed without parameters and so it prints out the Zodiak Sign for today (23 January). The remainder of the script runs are for different dates. This script could have been made more user-friendly using Groovy's built-in CLI support, but the focus here is on the dynamically injected method.

Conclusion

Groovy allows the developer to inject methods on instances of existing classes even when the developer does not own or have control over the existing class. This can be a powerful capability to bending an existing class to one's will.

No comments: