top | item 450886

Clojure: Genetic Mona Lisa problem in 250 beautiful lines

66 points| ynd | 17 years ago |npcontemplation.blogspot.com | reply

31 comments

order
[+] tlrobinson|17 years ago|reply
"Lambdas are not garbage collected.

Yes. That means lambdas can be the cause of memory leaks.

As described by Charles Nutter, each lambda in Clojure is an anonymous class in Java. The problem is that classes are never garbage collected. They reside in a special place called the PermGen."

I know very little about Clojure, but that seems like a huge problem for Clojure, and other functional-ish JVM languages. Is it?

I've used Rhino (JavaScript on JVM) extensively, and I don't think it suffers from this.

[+] jstraszheim|17 years ago|reply
He means the compiled code of a lambda can't be GC'ed. He was creating new functions inside of an eval.

The individual closures created as code executes are GC'ed like everything else.

Since eval is rather specialized, and everyone recommends it be avoided in "normal" code, this is not a major problem.

Plus, it is fixed.

[+] icey|17 years ago|reply
"As of r1232, lambdas created in eval can be GCed."

- Comment from cgrand at the bottom of the article.

[+] GeoJawDguJin|17 years ago|reply
Honestly, the JVM is pretty crappy right now for functional language support. A lot of stuff has to be implemented by hand in either the compiler or the program itself. Several features in Clojure take the latter track at the moment.

I advocate tracking down people from Sun and asking them in a personable, friendly tone of voice to please fix this kind of problem. On their front lawns.

[+] dreish|17 years ago|reply
In this thread:

http://markmail.org/message/axarlob7wwmnm2xe#query:clojure%2...

Rich Hickey claims that the PermGen issue alleged in this article doesn't actually exist. A new class is created every time a (fn) is compiled, but only a new instance of that class is created each time it is invoked.

The only thing that generates a new class each time it is called is (eval), which as we all remember from reading our PG, is generally regarded as suspect when used in production code. Obviously you can see this effect at the Clojure REPL, however.

[+] michaelneale|17 years ago|reply
Yes the PermGen issue in general seems a major weakness of the "Sun" VM (now OpenJDK) - although its not an intractable problem - just an irritation.

It seems an out dated idea today to have a separate space/special rules for generated code versus the rest of the heap - I think other VMs like JRockit didn't do that and didn't have that issue.

Although - I think the class/classloader relationship is in the spec, so to allow easy GC of generated classes you need to have 1 to 1 with classloaders (I hear that all this is going to change anyway to reflect how people really use things).

[+] Zak|17 years ago|reply
Genetic programming generally won't appear in production code, though it might be used to generate production code. Using eval seems reasonable in the context of executing code generated by your program. I'd be interested in hearing about alternatives.

In response to this problem, Clojure now uses an ephemeral classloader. I don't know the JVM that well, but Rich Hickey says this means the anonymous classes generated by evaling fn forms can be collected.

[+] smanek|17 years ago|reply
Incidentally, I wrote the same thing for a class recently in Common Lisp. See http://github.com/smanek/ga

The biggest portion of the code was the bitmap/image manipulation libraries. Excluding those, it was about the same size.

[+] apgwoz|17 years ago|reply
... how long did you let it run? The results in the presentation did not turn out very well.
[+] paulgb|17 years ago|reply
In the informal algorithm description, it says "Create Children from the best programs by mating and mutating them". By mating, do you mean actually taking two (or more) "programs" and merging the contents? I looked over the code, but I don't know enough clojure to find any part that looked like it merged two programs.

Incidentally, now I want to learn clojure :).

[+] Zak|17 years ago|reply
I do know Clojure and I didn't see any mating either - just mutating. I haven't had any caffeine yet though.

I'm also working on genetic programming in Clojure, and with the PermGen fix in rev. 1232, it seems to be a great tool for the job.

[+] ynd|17 years ago|reply
I removed mating in the end because it made the program converge slower.

I'm glad that got you interested in Clojure!

[+] biohacker42|17 years ago|reply
As python is beautiful, lisp is parenthesis-y.

Has anyone thought about a Lisp dialect that uses white space, like Python, to get rid of the ()?

[+] dimitar|17 years ago|reply
Why don't you make your editor to paint them white and indent them?
[+] nihilocrat|17 years ago|reply
Make an editor which will randomly color-code each set of parentheses such that they or their background is in a set of colors which don't appear as other keywords (like function defs or strings or whatever) so as not to add confusion. You now have a way of more easily seeing every matching pair of parens without having to hover over each one. However, now you have to deal with a strange assortment of colors on your screen.

Oh wait, you can just use a gradient of colors instead of random ones, with the more-nested paren pairs having colors farther in on the gradient. Silly me.

[+] icey|17 years ago|reply
Some of us prefer the parens.
[+] ramchip|17 years ago|reply
Parens let you have multiple expressions in one line, which indentation alone could not really express (except perhaps in a Haskell-like with currying, . and $ operators, etc).

For example, I don't see how (map (fn [x] (+ x 2)) (concat (take 5 my-series) etc...)) could be broken down with indentation alone without taking quite a few extra lines.

[+] peregrine|17 years ago|reply
I like this, it gives a clear set of code with a simple non important purpose.

Neat.

[+] toodlestech|17 years ago|reply
This has already been done and much better at that. Don't give props to someone copying anothers work a half year after the fact.

Credit and originality goes to : http://rogeralsing.com/2008/12/07/genetic-programming-evolut...

[+] smanek|17 years ago|reply
The author of this article links to that same post and gives credit to Roger Alsing.

He was just using it as a demonstration of Clojure, nothing more.

Read the article before criticizing.