So from what I've read about combinators I can't quite tell how they're different from simply chaining function calls. I know I must be missing something but I'm not figuring out what I'm missing. I mean would g(f(x)) (where f and g are functions) effectively be a combinator if I gave it a different name?
Functional Programming – Difference Between a Combinator and Function Chaining
functional programmingmath
Related Solutions
there are 2 ways to pass data into a function: parameters or globals, in small scripts globals are acceptable but really try to avoid them
it's easier to simply extract y this is also better when you need to change y later
you can use lazy initialization here:
var subresult=undefined function sub(){ if(subresult===undefined){ subresult=//calculate... } return subresult; }
there are 2 competing principles here SRP and YAGNI:
A. Single Responsibility Principle means essentially that each function should do a single thing and do it correctly
B. You Aren't Going to Need It: don't waste time on stuff that may or may not be needed in the future, focus on what you need now
You can't create a pure function called random
that will give a different result every time it is called. In fact, you can't even "call" pure functions. You apply them. So you aren't missing anything, but this doesn't mean that random numbers are off-limits in functional programming. Allow me to demonstrate, I'll use Haskell syntax throughout.
Coming from an imperative background, you may initially expect random to have a type like this:
random :: () -> Integer
But this has already been ruled out because random cannot be a pure function.
Consider the idea of a value. A value is an immutable thing. It never changes and every observation that you can make about it is consistent for all time.
Clearly, random can't produce a Integer value. Instead, it produces a Integer random variable. It's type might look like this:
random :: () -> Random Integer
Except that passing an argument is completely unnecessary, functions are pure, so one random ()
is as good as another random ()
. I'll give random, from here on, this type:
random :: Random Integer
Which is all well and fine, but not very useful. You may expect to be able to write expressions like random + 42
, but you can't, because it won't typecheck. You can't do anything with random variables, yet.
This raises an interesting question. What functions should exist to manipulate random variables?
This function can't exist:
bad :: Random a -> a
in any useful way, because then you could write:
badRandom :: Integer
badRandom = bad random
Which introduces an inconsistency. badRandom is supposedly a value, but it is also a random number; a contradiction.
Maybe we should add this function:
randomAdd :: Integer -> Random Integer -> Random Integer
But this just a special case of a more general pattern. You should be able to apply any function to random thing in order to get other random things like so:
randomMap :: (a -> b) -> Random a -> Random b
Instead of writing random + 42
, we can now write randomMap (+42) random
.
If all you had was randomMap, you wouldn't be able to combine random variables together. You couldn't write this function for instance:
randomCombine :: Random a -> Random b -> Random (a, b)
You might try to write it like this:
randomCombine a b = randomMap (\a' -> randomMap (\b' -> (a', b')) b) a
But it has the wrong type. Instead of ending up with a Random (a, b)
, we end up with a Random (Random (a, b))
This can be fixed by adding another function:
randomJoin :: Random (Random a) -> Random a
But, for reasons that may eventually become clear, I'm not going to do that. Instead I'm going to add this:
randomBind :: Random a -> (a -> Random b) -> Random b
It's not immediately obvious that this actually solves the problem, but it does:
randomCombine a b = randomBind a (\a' -> randomMap (\b' -> (a', b')) b)
In fact, it's possible to write randomBind in terms of randomJoin and randomMap. It's also possible to write randomJoin in terms of randomBind. But, I'll leave doing this as an exercise.
We could simplify this a little. Allow me to define this function:
randomUnit :: a -> Random a
randomUnit turns a value into a random variable. This means that we can have random variables that aren't actually random. This was always the case though; we could have done randomMap (const 4) random
before. The reason defining randomUnit is a good idea is that now we can define randomMap in terms of randomUnit and randomBind:
randomMap :: (a -> b) -> Random a -> Random b
randomMap f x = randomBind x (randomUnit . f)
Ok, now we are getting somewhere. We have random variables that we can manipulate. However:
- It's not obvious how we might actually implement these functions,
- It's quite cumbersome.
Implementation
I'll tackle pseudo random numbers. It is possible implement these functions for real random numbers, but this answer is already getting quite long.
Essentially, the way this is going to work is that we are going to pass a seed value around everywhere. Whenever we generate a new random value, we will produce a new seed. At the end, when we're done constructing a random variable, we will want to sample from it using this function:
runRandom :: Seed -> Random a -> a
I'm going to define the Random type like this:
data Random a = Random (Seed -> (Seed, a))
Then, we just need to provide implementations of randomUnit, randomBind, runRandom and random which is quite straight-forward:
randomUnit :: a -> Random a
randomUnit x = Random (\seed -> (seed, x))
randomBind :: Random a -> (a -> Random b) -> Random b
randomBind (Random f) g =
Random (\seed ->
let (seed', x) = f seed
Random g' = g x in
g' seed')
runRandom :: Seed -> Random a -> a
runRandom seed (Random f) = (snd . f) seed
For random, I'm going to assume there's already a function of the type:
psuedoRandom :: Seed -> (Seed, Integer)
In which case random is just Random psuedoRandom
.
Making things less cumbersome
Haskell has syntactic sugar to make things like this nicer on the eyes. It's called do-notation and to use it all we have to do it create an instance of Monad for Random.
instance Monad Random where
return = randomUnit
(>>=) = randomBind
Done. randomCombine
from before could now be written:
randomCombine :: Random a -> Random b -> Random (a, b)
randomCombine a b = do
a' <- a
b' <- b
return (a', b')
If I was doing this for myself, I would even go one step further than this and create an instance of Applicative. (Don't worry if this makes no sense).
instance Functor Random where
fmap = liftM
instance Applicative Random where
pure = return
(<*>) = ap
Then randomCombine could be written:
randomCombine :: Random a -> Random b -> Random (a, b)
randomCombine a b = (,) <$> a <*> b
Now that we have these instances, we can use >>=
instead of randomBind, join instead of randomJoin, fmap instead of randomMap, return instead of randomUnit. We also get a whole load of functions for free.
Is it worth it? You could argue, that getting to this stage, where working with random numbers isn't completely horrendous was quite difficult and long-winded. What did we get in exchange for this effort?
The most immediate reward is that we can now see exactly which parts of our program are dependent on randomness and which parts are entirely deterministic. In my experience, forcing a strict separation like this simplifies things immensely.
We've assumed so far that we just want a single sample from each random variable that we generate, but if it turns out that in the future we'd actually like to see more of the distribution, this is trivial. You can just use runRandom lots of times on the same random variable with different seeds. This is, of course, possible in imperative languages, but in this case, we can be certain that we aren't going to perform unanticipated IO every time we sample a random variable and we don't have to be careful about initializing state.
Best Answer
Function chaining is known as function composition in the functional world.
A combinator however is a completely different thing, the composition operator is simply one combinator.
I wrote a blog attempting to explain combinators here it may help.
In it I try to explain a combinator is simply a function that follows a few rules in the general case, but in the common case "combinator" is used to refer to a function which abstracts a bit of common functionality that is very basic in form.
in SKI combinator calculus there are 3 combinators: Kxy = x Sxyz = xz(yz) Ix = x
These are combinators because they act only upon their terms, and they are generalized to not care what their terms are.
This seems useless but realize I can be defined in terms of S and K: SKKx = Kx(Kx) = x which is equivalent to Ix which = x, the benefit to combinators is they are fully generalized to whatever terms you pass them and therefore live at a higher level abstraction than other things.
The composition combinator is the B combinator from the B,C,K,W combinator system
Now to make combinators useful however, you have to understand when fully generalized functions like this are worth creating. Parsers are the perfect and most common example, because a parser as a concept has to deal with an extremely large number of possible inputs, therefore the ability to generalize the handling of their inputs comes in very handy.