Post #247,435
3/10/06 2:04:57 AM
8/21/07 6:24:22 AM
|
Erlang?
I saw a guy give a talk about erlang the other day and wondered if anyone has worked with it.
I was intrigued. Highly concurrent VM, GC, LWP's designed for low latency systems like phone switches. A different way to skin Stroustrup's cat.
"Whenever you find you are on the side of the majority, it is time to pause and reflect" --Mark Twain
"The significant problems we face cannot be solved at the same level of thinking we were at when we created them." --Albert Einstein
"This is still a dangerous world. It's a world of madmen and uncertainty and potential mental losses." --George W. Bush
Erlang?
I saw a guy give a talk about erlang the other day and wondered if anyone has worked with it.
I was intrigued. Highly concurrent VM, GC, LWP's designed for low latency systems like phone switches. A different way to skin Stroustrup's cat.
"Whenever you find you are on the side of the majority, it is time to pause and reflect" --Mark Twain
"The significant problems we face cannot be solved at the same level of thinking we were at when we created them." --Albert Einstein
"This is still a dangerous world. It's a world of madmen and uncertainty and potential mental losses." --George W. Bush
|
Post #247,453
3/10/06 8:50:15 AM
|
Threads, Messages, and Pattern Matching....
...that's the basic strengths of Erlang. No state to weigh down threads, so there can be hundreds of thousands of them running simultaneously. Each thread sending and receiving messages. And using patterns to filter the inbound messages. If you are building highly concurrent/distributed systems, Erlang is the easiest language to construct such solutions.
Erlang variables are single-assignment variables - once assigned, they are bound and cannot be changed. However, recursive functions (with tail-call-elimination) and threads can be used to capture state within a thread. And this capability of sending messages and emulating state means that you can mimic object oriented programming as well. And since threads are so cheap, you can run the same thread on multiple machines to ensure a sort of fault tolerance.
|
Post #247,545
3/10/06 11:58:03 PM
8/21/07 6:27:15 AM
|
Right so to me this sounds like the ultimate SOA language
and I can't help but wonder what my employer is doing wasting time with crap languages like C++ and Java trying to build highly concurrent distributed systems.
It sounds like the ultimate high volume website development language. I wonder why its not used? (Apart from the fact that most people who are paid to manage technology are total idiots).
"Whenever you find you are on the side of the majority, it is time to pause and reflect" --Mark Twain
"The significant problems we face cannot be solved at the same level of thinking we were at when we created them." --Albert Einstein
"This is still a dangerous world. It's a world of madmen and uncertainty and potential mental losses." --George W. Bush
|
Post #247,554
3/11/06 3:19:13 AM
|
I dunno
but after checking out the website, I think I'm going to look into it.
I think the reason it's not widely used is because it's a functional, not an imperative, language. I know that when I took the paradigms course, functional languages took a while for me to grok, and I did a lot better than some people in my class. I think there's a lot of people out there for whom Imperative Programming == Programming. This language doesn't fit into that paradigm, so if you're a dude looking for a tool, and you don't understand functional programming, then this tool is simply going to be discarded out of hand.
--\n-------------------------------------------------------------------\n* Jack Troughton jake at consultron.ca *\n* [link|http://consultron.ca|http://consultron.ca] [link|irc://irc.ecomstation.ca|irc://irc.ecomstation.ca] *\n* Kingston Ontario Canada [link|news://news.consultron.ca|news://news.consultron.ca] *\n-------------------------------------------------------------------
|
Post #247,556
3/11/06 8:33:22 AM
|
Quick CS101 pointer? Diff between fucntional and imperative?
===
Purveyor of Doc Hope's [link|http://DocHope.com|fresh-baked dog biscuits and pet treats]. [link|http://DocHope.com|http://DocHope.com]
|
Post #247,562
3/11/06 9:45:04 AM
|
Re: Quick CS101 pointer? Diff between fucntional and imperat
Imperative: variables, functions, loops, ... you're telling the computer exactly what to do (the imperative sense) in a series of operations.
Functional: just functions. No variables, no side effects, you don't store anything. The program is one humungous equation, and you always create new values from old ones instead of over-writing. True functional programs can be proven to be correct.
Those are the pure types; many languages are a combination of the two. For example, Python has the map function, continuations, closures, etc., which are typically considered features of functional languages.
Personally I hate the term "functional" because so many people confuse it with meanings that have nothing to do with the programming type itself. Using functions doesn't mean you are doing functional programming, and functional programming isn't stuff that is just plain old operational. :-)
Regards,
-scott anderson
"Welcome to Rivendell, Mr. Anderson..."
|
Post #247,563
3/11/06 10:01:55 AM
|
Okay, remember hearing that before, and I understand ...
... why it's hard to grok. How do you write a program without variables? Is there an example you can point to of how to do something useful? The definition I remember is that each function will always return the same value given the same inputs. What you've described seems to go several steps beyond that.
===
Purveyor of Doc Hope's [link|http://DocHope.com|fresh-baked dog biscuits and pet treats]. [link|http://DocHope.com|http://DocHope.com]
|
Post #247,574
3/11/06 1:19:24 PM
|
Re: Okay, remember hearing that before, and I understand ...
Strict functional programming has no side-effects. What this means is that you can't change anything. You can, however, return something calculated from your inputs. Simplistically, a true functional program has an entry point, and all of the functions within just return a modified version of what's passed in, and the output is the result of calling all of the functions. There are no global variables and no side-effects, and that's why a function will always return the same value given the same inputs. Nothing matters but the parameters that are passed in. Because of this you have to go through some hoops to perform some imperative constructs: looping is done through recursion, for example. // Crappy pseudocode example\nfunction forLoopPositiveIncrementsOnly(int from, int to, int inc, function dowhat)\n{\n if (from > to) return;\n dowhat(from);\n loop(from + inc, to, dowhat);\n} Note that there IS a stack, but you can't modify the values on the stack, just use them.
Regards,
-scott anderson
"Welcome to Rivendell, Mr. Anderson..."
|
Post #247,575
3/11/06 1:24:33 PM
|
So you can't have a (pseudo) random number generator?
|
Post #247,576
3/11/06 1:34:03 PM
|
Those don't have side-effects
And given the same seed, a pseudo-random number generator will always return the same sequence of "random" numbers.
Regards,
-scott anderson
"Welcome to Rivendell, Mr. Anderson..."
|
Post #247,577
3/11/06 1:52:48 PM
|
I was too tearse.
Some random number generators are more random than others. :-) I'm aware of the limitations of typical pseudo random number generators. I was too tearse in putting that in the Subject. (One of many true random number generators is [link|http://www.fourmilab.ch/hotbits/|HotBits].)
I wasn't questioning the "side-effects" aspect (I don't think), I was wondering whether it would be possible to hypotheically code a random number generator in a pure functional language given that, apparently, f(x0,t) = y0 for all x0 for a fuctional language, where a random number generator would give f(x0,t) = y(x0,t,gamma) (where gamma is some truly radom parameter). (If I didn't mess the notation up.) Presumably the code would talk to a radioactivity sensor or something similar that was truly random.
I would think that it must be possible, but wonder how it would be simply explained and whether it violates the premise of functional programming.
Thanks.
Cheers, Scott.
|
Post #247,578
3/11/06 2:00:46 PM
3/11/06 2:02:24 PM
|
For pure functional programming, no
Because the program won't be provable for a given set of inputs.
Unless you count the external random number as one of the inputs, I suppose.
Regards,
-scott anderson
"Welcome to Rivendell, Mr. Anderson..."
Edited by admin
March 11, 2006, 02:02:24 PM EST
|
Post #247,587
3/11/06 4:01:46 PM
|
You can if you thread the state through the functions...
...and hence the introduction of monads.
|
Post #247,564
3/11/06 10:16:20 AM
|
Is this worth going through?
[link|http://www.haskell.org/tutorial/|http://www.haskell.org/tutorial/]
|
Post #247,565
3/11/06 10:42:19 AM
|
Shorter read....
...would be this [link|http://www.haskell.org/haskellwiki/Introduction|Haskell Intro], which is just the highlights.
|
Post #247,570
3/11/06 11:22:16 AM
|
Already did that
So I was wondering it the other was worth the detail. I downloaded [link|ftp://ftp.geoinfo.tuwien.ac.at/navratil/HaskellTutorial.pdf|ftp://ftp.geoinfo.tu...skellTutorial.pdf] and started going through it. Seems interesting.
|
Post #247,591
3/11/06 5:16:48 PM
|
I didn't find the gentle intro useful when I read it...
...4 years ago. It assumes you have a lot of knowledge, especially with functional languages.
|
Post #247,566
3/11/06 10:56:30 AM
|
The three pillars
OOP is usually associated with: - encapsulation
- inheritance
- polymorphism
FP is usually associated with: - parametric genericity
- referential transparency
- higher order functions
|
Post #247,572
3/11/06 12:17:52 PM
|
You just lost me
Remember, I haven't taken the classes, so it's more often the terminology I don't know. And I have no idea what you meant by * parametric genericity * referential transparency * higher order functions
===
Purveyor of Doc Hope's [link|http://DocHope.com|fresh-baked dog biscuits and pet treats]. [link|http://DocHope.com|http://DocHope.com]
|
Post #247,588
3/11/06 4:17:40 PM
|
Google the terms
Basically he means that you have functions whose parameters can be of many different types, functions do the same thing every time you give them the same parameters, and there are functions that take or receive other functions as arguments. For example a quicksort written in Haskell will sort anything that < and <= are defined for. By contrast you have to write a different quicksort in C for every data type. That's parametric genericity. For example a function in Haskell will always do the same thing when you call it. (Not actually true, but you're told when it is not true. Incidentally this is how you can have a pseudo-random number work in Haskell.) A function in C might do something else depending on a global variable that got changed in between. And for higher order functions, the following piece of Perl demonstrates the idea: nested_for(\n sub {print "@_\\n";},\n [1..2], ['a'..'c'], ['A'..'D']\n);\n\nsub nested_for {\n ret_iter(@_)->();\n}\n\nsub ret_iter {\n my $fn = shift;\n my $range = shift;\n my $sub = sub {$fn->($_, @_) for @$range};\n return @_ ? ret_iter($sub, @_) : $sub;\n} Cheers, Ben
I have come to believe that idealism without discipline is a quick road to disaster, while discipline without idealism is pointless. -- Aaron Ward (my brother)
|
Post #247,589
3/11/06 4:46:51 PM
|
Quick explanation
Parametric Polymorphism: Strictly speaking, generics are really only a consideration in statically typed FP languages (e.g. ML and Haskell). This form of polymorphism is the type that C++ templates attempts (but like with everything else C++ it is way too complex). In ML, the polymorphism comes almost free. The bottom line is that you don't usually have to declare the type of a variable - the compiler will figure it out for you. A simple case would be calculating the length of a list: fun length [] = 0\n | length (x::xs) = 1 + length xs\n val a = length [1,2,3,4]\n val b = length ["a","b","c","d"] This should give the answer of 4. Notice no types were explicitly declared here. The compiler figures out that the first invocation is a length of integers and the second a list of strings. The length function can operate over lists of all types, so there is no need to declare the types. Referential Transparency: This roughly means that for a given set of inputs, the function should always generate the same results. State is the natural enemy of RI, in that it can change the result of a function. So the first rule of thumb is that there should not be a dependency upon external state. val a = ref 0\n fun nontransparent x = x + !a\n val b = nontransparent 1;\n a := !a + 1;\n val c = nontransparent 1 The results are b=1 and c=2, but the function calls were identical. The external state variable a got changed in between. This would be a violation of RI. The other enemy of RI is sequencing. FP languages come in two flavors - eager and lazy. Eager languages evaluate the function parameters prior to calling the function. Lazy languages don't evaluate parameters until the values are needed (forced). Higher Order Functions: Functions can be first class values. They can be parameters to functions and returned as values from functions. The three most common are: Map, Filter, and Fold (aka Reduce). val x = map (fn i => i+i) [1,2,3,4]\n (* result: x = [1,4,6,8] *)\n\n val y = filter (fn a => (a < 3)) [1,2,3,4]\n (* result: y = [1,2] *)\n\n val z = foldr (fn (x,y) => x+y) 0 [1,2,3,4]\n (* result: z = 10 *)\n\n val z = foldl (fn (x,y) => x+y) 0 [1,2,3,4]\n (* result: z = 10 *) Map applies a given function to each element within a list and returns the resulting list. Filter returns a list of items within the input list that have the function evaluate to true. Fold collapses the list by applying a function across the list and returning the value of the function. Fold can be either Left or Right, depending on whether you wont to fold the list starting from the left side and working your way right (or vice versa). As Scott said, many of these things are also in non-FP languages. But the point of FP is one of emphasis - having features which make working with functions easier. Haskell is the only FP language that's considered to be close to pure. ML and Scheme have state variables (like the above ML code). Haskell is a pursuit to use monads to handle the reality of needing to cope with external events but manage them in a pure FP fashion.
|
Post #247,590
3/11/06 5:06:03 PM
|
One other important concept worth mentioning is Currying
FPs generally allow functions to curry. A simple example would be an add function that takes two parameters - the values to be added: fun add x y = x + y\n\n val x = add 10\n\n val y = x 20 Notice that the calculation of the x value only calls the add function with one parameter (10). The result of this application is that x itself is a function that takes one value. Effectively it is the function: fun x(y) = 10 + y This concept of currying is part of the consideration for higher order functions.
|
Post #247,580
3/11/06 2:15:26 PM
|
Erlang does look very intriguing, will hopefully influence
other languages, for example, there's [link|http://candygram.sourceforge.net/|Candygram], an implementation of Erlang concurrency primitives for Python.
Note that Erlang was invented by Ericsson, and has been used by them in several very large telecom apps.
It's interesting, however, since I don't deal much with distributed systems (the most complex distributed system involved a PC, three PLCs, a robot, and two smart cameras -- and only the PC was capable of running Erlang) it'll be a while before I find time.
--Tony
|