Clojure interoperability with Java
This guide covers:
- How to instantiate Java classes
- How to invoke Java methods
- How to extend Java classes with proxy
- How to implement Java interfaces with reify
- How to generate Java classes with gen-class
- Other topics related to interop
This work is licensed under a Creative Commons Attribution 3.0 Unported License (including images & stylesheets). The source is available on Github.
What Version of Clojure Does This Guide Cover?
This guide covers Clojure 1.5.
Overview
Clojure was designed to be a hosted language that directly interoperates with its host platform (JVM, CLR and so on).
Clojure code is compiled to JVM bytecode. For method calls on Java objects, Clojure compiler
will try to emit the same bytecode javac
would produce.
It is possible to implement interfaces, extend and generate Java classes in Clojure.
Clojure also provides convenient functions and macros that make consuming of Java libraries easier and often more concise than it would be in Java code.
Imports
Java classes can be referenced either using their fully-qualified names (FQNs) such as
java.util.Date
or be imported in the current Clojure namespace using clojure.core/import
and
referenced by short names:
java.util.Date ; ⇒ java.util.Date
(import java.util.Date)
Date ; ⇒ java.util.Date
ns
macro supports imports, too:
(ns myservice.main
(:import java.util.Date))
More about the ns
macro can be found in the article on Clojure namespaces.
Dynamic (at runtime) imports are usually only used in the REPL and cases when there are multiple implementations of a particular protocol/service/feature and it is not possible to tell which one should be used until run time.
Automatic Imports For java.lang.*
Classes from the java.lang
package are automatically imported. For example, you can use String
or Math
without explicitly importing them:
(defn http-uri?
[^String uri]
(.startsWith (.toLowerCase uri) "http"))
(Math/round 0.7886)
Inner (Nested) Classes
In Java, classes can be nested inside other classes. They are called inner classes and by convention,
separated from their outer class by a dollar sign ($
):
(import java.util.Map$Entry)
Map$Entry ; ⇒ java.util.Map$Entry
;; this example assumes RabbitMQ Java client is on classpath
(import com.rabbitmq.client.AMQP$BasicProperties)
AMQP$BasicProperties ; ⇒ com.rabbitmq.client.AMQP$BasicProperties
Note that if you need to use both a class and one or more of its inner classes, they all need to be imported separately. As far as JVM is concerned, they are all separate classes, there is no "imports hierarchy".
How to Instantiate Java Classes
Java classes are instantiated using the new
special form:
(new java.util.Date) ; ⇒ #inst "2012-10-09T21:23:57.278-00:00"
However, the Clojure reader provides a bit of syntactic sugar and you are much more likely to see this:
(java.util.Date.) ; ⇒ #inst "2012-10-09T21:24:43.878-00:00"
It is possible to use fully qualified names (e.g. java.util.Date
) or short names with imports:
(import java.util.Date)
(Date.) ; ⇒ #inst "2012-10-09T21:24:27.229-00:00"
An example with constructor arguments:
(java.net.URI. "http://clojure.org") ; ⇒ #<URI http://clojure.org>
How to Invoke Java Methods
Instance Methods
Instance methods are invoked using the .
special form:
(let [d (java.util.Date.)]
(. d getTime)) ; ⇒ 1349819873183
Just like with object instantiation, it is much more common to see an alternative version:
(let [d (java.util.Date.)]
(.getTime d)) ; ⇒ 1349819873183
Static Methods
Static methods can be invoked with the same .
special form:
(. Math floor 5.677) ; ⇒ 5.0
or (typically) to sugared version, ClassName/methodName
:
(Math/floor 5.677) ; ⇒ 5.0
(Boolean/valueOf "false") ; ⇒ false
(Boolean/valueOf "true") ; ⇒ true
Chained Calls With The Double Dot Form
It is possible to chain method calls using the ..
special form:
(.. (java.util.Date.) getTime toString) ; ⇒ "1349821993809"
Multiple Calls On the Same Object
If you need to call a bunch of methods on a mutable object, you
can use the doto
macro:
(doto (java.util.Stack.)
(.push 42)
(.push 13)
(.push 7)) ; ⇒ #<Stack [42, 13, 7]>
(let [pt (Point. 0 0)]
(doto pt
(.move 10 0))) ; ⇒ #<Point java.awt.Point[x=10, y=0]
(let [pt (Point. 0 0)]
(doto pt
(.move 10 0)
(.translate 0 10))) ; ⇒ #<Point java.awt.point[x=10,y=10]
The doto
macro returns its first argument as a result.
How to Access Java Fields
Public mutable fields are not common in Java libraries but sometimes you need to access them. It's done with the same dot special form:
(import java.awt.Point)
(let [pt (Point. 0 10)]
(. pt x)) ; ⇒ 0
(let [pt (Point. 0 10)]
(. pt y)) ; ⇒ 10
and just like with instance methods, it is much more common to see the following version:
(import java.awt.Point)
(let [pt (Point. 0 10)]
(.x pt)) ; ⇒ 0
(let [pt (Point. 0 10)]
(.y pt)) ; ⇒ 10
How to Set Java Fields
To set a public mutable field, use clojure.core/set!
that takes a field in the dot notation
demonstrated earlier and a new value:
(import java.awt.Point)
(let [pt (Point. 0 10)]
(set! (.y pt) 100)
(.y pt)) ; ⇒ 100
Fortunately, mutable public fields are rare to meet in the JVM ecosystem so you won't need to do this often.
How To Work With Enums
Enums (enumeration) type values are accessed the same way as fields, except on enum classes and not objects:
java.util.concurrent.TimeUnit/MILLISECONDS ; ⇒ #< MILLISECONDS>
Determining Classes of Java Objects
To get class of a particular value, pass it to clojure.core/class
:
(class 1) ; ⇒ java.lang.Long
(class 1.0) ; ⇒ java.lang.Double
(class "docs") ; ⇒ java.lang.String
(class (java.net.URI. "https://github.com")) ; ⇒ java.net.URI
As this example demonstrates, Clojure strings are JVM strings, integer literals are compiled as longs and floating point literals are compiled as doubles.
You can also use clojure.core/type
to return either the class of the
Java object, or the :type
metadata if it exists:
(def foo (with-meta [1 2 3] {:type :bar}))
(type foo)
;; ⇒ :bar
(type [1 2 3])
;; ⇒ clojure.lang.PersistentVector
How To Get a Java Class Reference By Name
To obtain a class reference by its string name (fully qualified), use Class/forName
via Java interop:
(Class/forName "java.util.Date") ; ⇒ java.util.Date
Array Types, Primitives
JVM has what is called primitive types (numerics, chars, booleans) that are not "real" objects.
In addition, array types have pretty obscure internal names. If you need to obtain a reference to
an array of longs, for example, pass "[[J"
to Class/forName
. Below is the full table:
Internal JVM class name | Array of ? (type) |
---|---|
"[[S" | short |
"[[I" | integer |
"[[J" | long |
"[[F" | float |
"[[D" | double |
"[[B" | byte |
"[[C" | char |
"[[Z" | boolean |
If this does not make much sense, don't worry. Just remember to come back to this guide when you need to extend a protocol for an array of primitives.
Implementing Java Interfaces With reify
It is possible to implement Java interfaces in Clojure. It is typically needed to interact with Java libraries that take arguments implementing a particular interface.
Interfaces are implemented using the reify
special form.
Given the following Java interface:
public
interface FilenameFilter {
/**
* Tests if a specified file should be included in a file list.
*
* @param dir the directory in which the file was found.
* @param name the name of the file.
* @return <code>true</code> if and only if the name should be
* included in the file list; <code>false</code> otherwise.
*/
boolean accept(File dir, String name);
}
here is how to implement it in Clojure:
;; a FileFilter implementation that accepts everything
(reify java.io.FilenameFilter
(accept [this dir name]
true))
reify
takes an interface (fully-qualified name or short name) and one or more
method implementations that mimic function definitions without the defn
and with
this (as in Java, JavaScript or self in Ruby, Python) reference being the first argument:
(accept [this dir name]
true)
With reify
, generally there is no need to add type hints on arguments: Clojure
compiler typically will detect the best matching method (by name and number of arguments).
reify
returns a Java class instance. Clojure compiler will generate a class that implements
the interface and instantiate it. To demonstrate that reified objects indeed implement
the interface:
(let [ff (reify java.io.FilenameFilter
(accept [this dir name]
true))]
(instance? java.io.FileFilter ff)) ; ⇒ true
reify
can be used to implement multiple interfaces at once:
(let [ff (reify java.io.FilenameFilter
(accept [this dir name]
true)
java.io.FileFilter
(accept [this dir]
true))]
(instance? java.io.FileFilter ff)) ; ⇒ true
reify, Parameter Destructuring and Varargs
reify
does not support destructuring or variadic number of arguments in method signatures.
For example, the following will not work and won't even compile in Clojure 1.5:
(reify com.megacorp.api.AnInterface
(aMethod [a [b c]]
(comment ...))
(anotherMethod [a & rest]
(comment ...)))
Example 1
The following example demonstrates how instances created with reify
are passed around
as regular Java objects:
(import java.io.File)
;; a file filter implementation that keeps only .clj files
(let [ff (reify java.io.FilenameFilter
(accept [this dir name]
(.endsWith name ".clj")))
dir (File. "/Users/antares/Development/ClojureWerkz/neocons.git/")]
(into [] (.listFiles dir ff)))
;; ⇒ [#<File /Users/antares/Development/ClojureWerkz/neocons.git/project.clj>]
reify
forms a closure: it will capture locals in its scope. This can be used to make implemented
methods delegate to Clojure functions. The same example, rewritten with delegation:
user> (import java.io.File)
;; a file filter implementation that keeps only .clj files
(let [f (fn [^File dir ^String name]
(.endsWith name ".clj"))
ff (reify java.io.FilenameFilter
(accept [this dir name]
(f dir name)))
dir (File. "/Users/antares/Development/ClojureWerkz/neocons.git/")]
(into [] (.listFiles dir ff)))
;; ⇒ [#<File /Users/antares/Development/ClojureWerkz/neocons.git/project.clj>]
Note that unlike in the "inline" implementation, Clojure compiler cannot infer types of
dir
and name
parameters in the function that does the filtering, so we added type hints
to avoid reflective calls. When methods are implemented "inline", types can be inferred from
method signatures in the interface.
Extending Java Classes With proxy
proxy
is one of two ways to generate instances of anonymous classes in Clojure.
proxy
takes two vectors: one listing its superclass and (optional) interfaces, another constructor signatures, as well as
method implementations. Method implementations are basically identical to reify
except that the this
argument is
not necessary.
A very minimalistic example, we instantiate an anonymous class that extends java.lang.Object
, implements no
interfaces, has no explictly defined constructors and overrides #toString
:
(proxy [Object] []
(toString []
"I am an instance of an anonymous class generated via proxy"))
;; ⇒ #<Object$0 I am an instance of an anonymous class generated via proxy>
Clojure compiler will generate an anonymous class for this proxy
and at runtime, the cost of
a proxy
call is the cost of instantiating this class (the class is not generated anew on every single call).
A slightly more complex example where the generated class also implements java.lang.Runnable
(runnable objects
are commonly used with threads and java.util.concurrent
classes) which defines one method, #run
:
;; extends java.lang.Object, implements java.lang.Runnable
(let [runnable (proxy [Object Runnable] []
(toString []
"I am an instance of an anonymous class generated via proxy")
(run []
(println "Run, proxy, run")))]
(.run runnable)) ; ⇒ nil
;; outputs "Run, proxy, run"
proxy
forms a closure: it will capture locals in its scope. This is very often used to create an instance
that delegates to a Clojure function:
(let [f (fn [] (println "Executed from a function"))
obj (proxy [Object Runnable] []
(run []
(f)))]
(.run obj)) ; ⇒ nil
;; outputs "Executed from a function"
TBD: more realistic examples | How to Contribute
Clojure Functions Implement Runnable and Callable
Note that Clojure functions implement java.lang.Runnable
and
java.util.concurrent.Callable
directly so you can pass functions to
methods found in various classes from the java.util.concurrent
package.
For example, to run a function in a new thread:
(let [t (Thread. (fn []
(println "I am running in a separate thread")))]
(.start t))
Or submit a function for execution to a thread pool (in JDK terms: an execution service):
(import '[java.util.concurrent Executors ExecutorService Callable])
(let [^ExecutorService pool (Executors/newFixedThreadPool 16)
^Callable clbl (cast Callable (fn []
(reduce + (range 0 10000))))
task (.submit pool clbl)]
(.get task))
;; ⇒ 49995000
Note that without the cast, Clojure compiler would not be able to determine
which exact version of the method we intend to invoke, because java.util.concurrent.ExecutionService/submit
has two versions, one for Runnable
and one for Callable
. They work very much the same but return
slightly different results (Callable
produces a value while Runnable
always returns nil when
executed).
The exception we would get without the cast is
CompilerException java.lang.IllegalArgumentException: More than one matching method found: submit, compiling:(NO_SOURCE_PATH:2)
gen-class and How to Implement Java Classes in Clojure
Overview
gen-class
is a Clojure feature for implementing Java classes in Clojure. It is relatively
rarely used compared to proxy
and reify
but is needed to implement executable classes
(that java
runner and IDEs can as program entry points).
Unlike proxy
and reify
, gen-class
defines named classes. They can be passed to Java
APIs that expect class references. Classes defined with gen-class
can extend
base classes, implement any number of Java interfaces, define any number of constructors
and define both instance and static methods.
AOT
gen-class
requires ahead-of-time (AOT) compilation. It means that
before using the classes defined with gen-class
, the Clojure
compiler needs to produce .class
files from gen-class
definitions.
Class Definition With clojure.core/gen-class
clojure.core/gen-class
is a macro that uses a DSL for defining class
methods, base class, implemented interfaces and so on.
It takes a number of options:
:name
(a symbol): defines generated class name:extends
(a symbol): name of the base class:implements
(a collection): interfaces the class implements:constructors
(a map): constructor signatures:methods
(a collection): lists methods that will be implemented:init
(symbol): defines a function that will be invoked with constructor arguments:post-init
(symbol): defines a function that will be called with a constructed instance as its first argument:state
(symbol): if supplied, a public final instance field with the given name will be created. Only makes sense when used with:init
. State field value should be an atom or other ref type to allow state mutation.:prefix
(string, default:"-"
): methods will call functions named as(str prefix method-name)
, e.g.-getName
forgetName
.:main
(boolean): iftrue
, a public static main method will be generated for the class. It will delegate to a function named main with the prefix ((str prefix "main")
),-main
by default:exposes
: TBD:exposes-methods
: TBD:factory
: TBD:load-impl-ns
: TBD:impl-ns
: TBD
The :name Option
TBD
The :extends Option
TBD
The :implements Option
TBD
The :constructors Option
TBD
The :methods Option
TBD
The :init Option
TBD
The :post-init Option
TBD
The :state Option
TBD
The :prefix Option
TBD
The :main Option
TBD
The :exposes Option
TBD
The :exposes-methods Option
TBD
The :factory Option
TBD
The :load-impl-ns Option
TBD
The :impl-ns Option
TBD
gen-class In The ns Macro
gen-class
can be used with existing namespaces by adding (:gen-class)
to the
ns
macro. Here is a "hello, world" example command line app that uses gen-class
to generate a class that JVM launcher (java
) can run:
(ns genclassy.core
(:gen-class))
(defn -main
[& args]
(println "Hello, World!"))
This will use the name of the namespace for class name and use the namespace for method
implementation (see the :impl-ns
option above).
Examples
A medium size example taken from an open source library:
(ns clojurewerkz.quartzite.listeners.amqp.PublishingSchedulerListener
(:gen-class :implements [org.quartz.SchedulerListener]
:init init
:state state
:constructors {[com.rabbitmq.client.Channel String String] []})
(:require [langohr.basic :as lhb]
[clojure.data.json :as json])
(:use [clojurewerkz.quartzite.conversion])
(:import [org.quartz SchedulerListener SchedulerException Trigger TriggerKey JobDetail JobKey]
[com.rabbitmq.client Channel]
[java.util Date]
[clojurewerkz.quartzite.listeners.amqp PublishingSchedulerListener]))
(defn publish
[^PublishingSchedulerListener this payload ^String type]
(let [{ :keys [channel exchange routing-key] } @(.state this)
payload (json/json-str payload)]
(lhb/publish channel exchange routing-key payload :type type)))
(defn -init
[^Channel ch ^String exchange ^String routing-key]
[[] (atom { :channel ch :exchange exchange :routing-key routing-key })])
(defmacro payloadless-publisher
[method-name message-type]
`(defn ~method-name
[this#]
(publish this# (json/json-str {}) ~message-type)))
(payloadless-publisher -schedulerStarted "quartz.scheduler.started")
(payloadless-publisher -schedulerInStandbyMode "quartz.scheduler.standby")
(payloadless-publisher -schedulingDataCleared "quartz.scheduler.cleared")
(payloadless-publisher -schedulerShuttingDown "quartz.scheduler.shutdown")
(defn -schedulerError
[this ^String msg ^SchedulerException cause]
(publish this (json/json-str { :message msg :cause (str cause) }) "quartz.scheduler.error"))
(defn -jobScheduled
[this ^Trigger trigger]
(publish this (json/json-str { :group (-> trigger .getKey .getGroup) :key (-> trigger .getKey .getName) :description (.getDescription trigger) }) "quartz.scheduler.job-scheduled"))
(defn -jobUnscheduled
[this ^TriggerKey key]
(publish this (json/json-str { :group (.getGroup key) :key (.getName key) }) "quartz.scheduler.job-unscheduled"))
(defn -triggerFinalized
[this ^Trigger trigger]
(publish this (json/json-str { :group (-> trigger .getKey .getGroup) :key (-> trigger .getKey .getName) :description (.getDescription trigger) }) "quartz.scheduler.trigger-finalized"))
(defn -triggerPaused
[this ^TriggerKey key]
(publish this (json/json-str { :group (.getGroup key) :key (.getName key) }) "quartz.scheduler.trigger-paused"))
(defn -triggersPaused
[this ^String trigger-group]
(publish this (json/json-str { :group trigger-group }) "quartz.scheduler.triggers-paused"))
(defn -triggerResumed
[this ^TriggerKey key]
(publish this (json/json-str { :group (.getGroup key) :key (.getName key) }) "quartz.scheduler.trigger-resumed"))
(defn -triggersResumed
[this ^String trigger-group]
(publish this (json/json-str { :group trigger-group }) "quartz.scheduler.triggers-resumed"))
(defn -jobAdded
[this ^JobDetail detail]
(publish this (json/json-str { :job-detail (from-job-data (.getJobDataMap detail)) :description (.getDescription detail) }) "quartz.scheduler.job-added"))
(defn -jobDeleted
[this ^JobKey key]
(publish this (json/json-str { :group (.getGroup key) :key (.getName key) }) "quartz.scheduler.job-deleted"))
(defn -jobPaused
[this ^JobKey key]
(publish this (json/json-str { :group (.getGroup key) :key (.getName key) }) "quartz.scheduler.job-paused"))
(defn -jobsPaused
[this ^String job-group]
(publish this (json/json-str { :group job-group }) "quartz.scheduler.jobs-paused"))
(defn -jobResumed
[this ^JobKey key]
(publish this (json/json-str { :group (.getGroup key) :key (.getName key) }) "quartz.scheduler.job-resumed"))
(defn -jobsResumed
[this ^String job-group]
(publish this (json/json-str { :group job-group }) "quartz.scheduler.jobs-resumed"))
Inspecting Class Signatures
When using gen-class
for interoperability purposes, sometimes it is necessary to inspect the API
of the class generated by gen-class
.
It can be inspected using javap. Given the following Clojure namespace:
(ns genclassy.core
(:gen-class))
(defn -main
[& args]
(println "Hello, World!"))
We can inspect the produced class like so:
# from target/classes, default .class files location used by Leiningen
javap genclassy.core
will output
public class genclassy.core {
public static {};
public genclassy.core();
public java.lang.Object clone();
public int hashCode();
public java.lang.String toString();
public boolean equals(java.lang.Object);
public static void main(java.lang.String[]);
}
How To Extend Protocols to Java Classes
Clojure protocols can be extended to any java class (including
Clojure's internal types) very easily using extend
:
Using the example of a json library, we can define our goal as getting to the point where the following works:
(json-encode (java.util.UUID/randomUUID))
First, let's start with the protocol for json encoding an object:
(defprotocol JSONable
(json-encode [obj]))
So, everything that is "JSONable" implements a json-encode
method.
Next, let's define a dummy method to do the "encoding" (in this example, it just prints to standard out instead, it doesn't actually do any json encoding):
(defn encode-fn
[x]
(prn x))
Now, define a method that will encode java objects by calling bean
on them, then making each value of the bean map a string:
(defn encode-java-thing
[obj]
(encode-fn
(into {}
(map (fn [m]
[(key m) (str (val m))])
(bean obj)))))
Let's try it on an example object, a UUID:
(encode-java-thing (java.util.UUID/randomUUID))
;; ⇒ {:mostSignificantBits "-6060053801408705927",
;; :leastSignificantBits "-7978739947533933755",
;; :class "class java.util.UUID"}
The next step is to extend the protocol to the java type, telling
clojure which java type to extend, the protocol to implement and the
method to use for the json-encode
method:
(extend java.util.UUID
JSONable
{:json-encode encode-java-thing})
Alternatively, you could use the extend-type
macro, which actually
expands into calls to extend
:
(extend-type java.util.UUID
JSONable
(json-encode [obj] (encode-java-thing obj)))
Now we can use json-encode
for the object we've extended:
(json-encode (java.util.UUID/randomUUID))
;; ⇒ {:mostSignificantBits "3097485598740136901",
;; :leastSignificantBits "-9000234678473924364",
;; :class "class java.util.UUID"}
You could also write the function inline in the extend block, for
example, extending nil
to return a warning string:
(extend nil
JSONable
{:json-encode (fn [x] "x is nil!")})
(json-encode nil)
;; ⇒ "x is nil!"
The encode-java-thing
method can also be reused for other Java types
we may want to encode:
(extend java.net.URL
JSONable
{:json-encode encode-java-thing})
(json-encode (java.net.URL. "http://aoeu.com"))
;; ⇒ {:path "",
;; :protocol "http",
;; :authority "aoeu.com",
;; :host "aoeu.com",
;; :ref "",
;; :content "sun.net.www.protocol.http.HttpURLConnection$HttpInputStream@4ecac02f",
;; :class "class java.net.URL",
;; :defaultPort "80",
;; :port "-1",
;; :query "",
;; :file "",
;; :userInfo ""}
Using Intrinsic Locks ("synchronized") in Clojure
Every object on the JVM has an intrinsic lock (also referred to as monitor lock or simply monitor). While very rarely necessary, Clojure provides support for operations that acquire intrinsic lock of a mutable Java object.
This is covered in the Concurrency and Parallelism guide.
Wrapping Up
TBD: How to Contribute
Contributors
Michael Klishin michael@defprotocol.org (original author) Lee Hinman lee@writequit.org gsnewmark gsnewmark@meta.ua