Behaviours & Protocols

I have an intuitive understanding of behaviours and protocols of course, but after talking with a colleague today, I was surprised that behaviours and protocols are kind of the same thing.

Mainly because protocols are like type classes in Haskell, and if I squint, behaviours are also.

Which you know, is comforting, as I always like seeing links between data (protocols) and code / actions (behaviours).

But, where am I going wrong?

Can anyone explain these Elixir concepts in terms of Haskell, maybe a bit more precision will show me where I am going wrong…

Could I create a GenServer protocol? Or is that completely ridiculous?

I see a related link which is interesting: Why is Access a behaviour instead of a Protocol? When to use a Protocol instead of a Behaviour?

A “behaviour” defines an API that a module has to provide.

A “protocol” is implemented on top of behaviours and chooses from a set of modules to use when dispatching function calls with data, based on their “type”.

11 Likes

I see the intent behind either as being quite different. However historically OO has often conflated the notion of polymorphism and implementation inheritance - so it would be very easy to fall into the same trap.

  • Protocols are ultimately about polymorphism. One part is the interface that will give access to the same polymorphic behaviour. Then each data type wishing to expose that behaviour implements it separately.
  • Behaviours are about separating the generic implementation from the specific. A Behaviour captures the common implementation for reuse and uses callbacks to invoke the specific implementation (i.e. the part that varies - a Behaviour acts more like a framework).

Erlang OTP Design Principles: 1.2 Behaviours:

In OO the callback module would likely be implemented as a Strategy.

From a development perspective you are implementing an interface and you are implementing a callback module but that is focussing on the “what you are doing” and the “how you are doing it”.

The “why you are doing it” is quite different.

19 Likes

To complement a bit on what @peerreynders wrote. In OTP behaviours are generally coupled to processes in that it is around a process where the generic and specific are defined. The generic typically describes the basic process interface and the specific which callbacks are required and when and how they are called/used. So for example a GenServer is a process with a defined interface and a set of 6 callbacks.

8 Likes

I think it is also worth pointing out that Behaviours are inherited from Erlang, whereas Protocols are unique to Elixir. In my mind Behaviours are more comparable to interfaces in languages like Java and C#, and Protocols are more like Haskell type classes, except they cannot be polymorphic only in the return type like they can be in Haskell. I don’t think its possible to implement mempty for example.

3 Likes

This talk helped me a lot on the topic.


Maybe you’ll find it as useful as i did.

5 Likes

That is why something like Enum.into/1 takes an argument that defines the type of the return (and why not ‘merge’ it since it is basically free at that point anyway).

1 Like

Just back from holidays, thanks for all the replies! Each one thoughtful and thought provoking.

And why this place is one of the best places on the internet!

I found a nice blog post which explains Behaviours to me in terms of Java’s interfaces and Haskell’s records:

https://chris-martin.org/2017/interfaces-and-records

It’s an extremely circuitous way to understand Behaviours on an Elixir forum, but Haskell’s precision helps me.

4 Likes