They are both pretty much equivalent in terms of what you can do with them, i.e.:
- They execute at compile time
- They can perform arbitrary transformation and generation of code (exploiting the homoiconicity of Lisp code)
- They are suitable for "extending the language" with new language constructs or DSLs
- You feel very powerful, and can be very productive (in the beating the averages way)
And now for some differences:
Common Lisp also allows reader macros which allow you to change the behaviour of the reader. This allows you to introduce completely new syntax (e.g. for data structure literals). This might be the reason that your friend describes Clojure's macro system as "weaker" since Clojure does not allow reader macros. In Clojure you are basically stuck with the syntax (macro-name ....)
but apart from that you can do anything you want. Opinion is divided as to whether reader macros a good thing or not: my personal view is not, as it doesn't give you any extra "power" and has the potential to cause extreme confusion.
Clojure has, in my view, a nicer implementation of namespaces that I think make Clojure's macro system easier to use. Every symbol in Clojure is namespace-qualified, so different libraries can define the same symbol different in their own namespace. So +
can be defined separately as clojure.core/+
and my.vector.library/+
without any risk of conflicts. In your own namespace, you can use definitions from another namespace, which will mean that you can choose to take +
from either clojure.core
or my.vector.library
as needed.
Clojure on the other hand has extra literals for maps {}
and vectors []
. These give you a bit more expressivity (in the sense of concise readable syntax) than traditional Lisp s-expressions. In particular, the use of [] for binding forms is a convention in Clojure that I think works well both for macros and normal code - it makes them stand out clearly from the other parentheses.
Clojure is also a Lisp-1 (like Scheme) so it doesn't have a separate namespace for functions and data. Common Lisp is a Lisp-2 which has separate function and data namesapces (so you can have both a function called foo and a data item called foo). I slightly prefer the Lisp-1 approach, since it is simpler and the division between code and data seems a bit arbitrary when you are writing in a functional langauge. This is probably a personal taste thing though.
Overall, the differences are relatively minor. I think Clojure is a bit simpler and more elegant, whereas Common Lisp has some extra features (use at your own risk!). Both are extremely capable, so you can't go wrong choosing either.