Humans are lazy and programmers are humans J . We like things which
are simplified and clean. So if you see the evolution of programming, we
initially started with functional programming, and evolved towards object
oriented programming.
So as functional programmers where we used to remember methods
and function names like “Add”, “Subtract”, “Multiply” and “Divide”, in OOP we
just need to remember class “Maths” and all functions just pop out when we
create the object of the class.
But as object oriented programmers we need to remember there
are two worlds in OOP, one who creates the class and second who consumes it by
creating the object.
So by following good OOP principles like abstraction,
encapsulation etc developer writes the below “Customer” class so that other
developers who consume this class should not feel it complicated.
public class Customer
{
private string _FullName;
public string FullName
{
get { return _FullName; }
set { _FullName = value; }
}
private DateTime _Dob;
public DateTime Dob
{
get { return _Dob; }
set { _Dob = value; }
}
private string _Address;
public string Address
{
get { return _Address; }
set { _Address = value; }
}
}
Now the consumer who want to consume our “Customer” class
will create the customer object , set properties and invoke methods. Below is the sample code for the same.
Customer customer = new
Customer();
customer.FullName =
"Shiv";
customer.Dob = Convert.ToDateTime("1/1/2008");
customer.Address =
"Mumbai";
Now let’s zoom on the above consumer code. Is it complicated?,
probably not if you are C# developer or a object oriented programmer.
But what if your consumer is a tester who really does understand
c# and would like to have more simplified interfaces to invoke your customer
class for UNIT testing purpose.
What if you are component seller and you would like to
surprise your component consumers with simplified interfaces. You would like to
stand different from your competitors.
Welcome to the concept of “Fluent interfaces”.
“Fluent interfaces simplify your object consumption code by making your code more simple, readable
and discoverable.”
So
if our component consumers can write object invocation code in simple English sentence
like below , that would “ROCK” right.
customer.NameOfCustomer("Shiv")
.Bornon("12/3/1075")
.StaysAt("Mumbai");
So now the next thing how to we implement “Fluent”
interfaces. That’s done by using “Method chaining”.
“Method chaining” is a
common technique where each method returns an object and all these methods can
be chained together to form a single statement.
So the above customer class we can wrap in another class (“CustomerFluent”)
which will implement method chaining and expose chained methods in a simplified
format.
So you can see in the below code methods “NameofCustomer” ,
“BornOn” accept input s and return backs “CustomerFluent” class.
public class CustomerFluent
{
private Customer obj = new Customer();
public CustomerFluent
NameOfCustomer(string Name)
{
obj.FullName = Name;
return this;
}
public CustomerFluent Bornon(string
Dob)
{
obj.Dob = Convert.ToDateTime(Dob);
return this;
}
public void StaysAt(string Address)
{
obj.Address = Address;
}
}
Now your client code is simple, nice and FLUENT as in the
below code.
customer.NameOfCustomer("Shiv")
.Bornon("12/3/1075")
.StaysAt("Mumbai");
I have seen four visible uses of fluent interfaces:-
LINQ Queries
var x = context.Users.Select(u
=> new { u.Id, u.Name });
Unit testing
Assert.AreEqual(obj.Expected,obj1.Actual);
Mock testing
Container.GetMock()
.Setup(s
=> s.Save(new Person()))
.Returns(true)
.Verifiable();
DSL (Domain specific language)
DSL is a language for simple users who do not understand
programming. So they can type something in simple English as shown below. This
will be parsed and checked for syntaxes. Later this sentence can be mapped to a
internal fluent interface statement.
task "warn if website is
not alive":
every 1.Hour
starting now
when WebSite("http://www.questpond.com ").IsAlive is
false
then:notify "admin@example.org", "site
down!"
Below is the code of the Fluent interface statement to which
the above DSL can map.
new FluentTask("alert if
site is down")
.Every( TimeSpan.FromMinutes(3) )
.StartingFrom( DateTime.Now )
.When(delegate
{
return WebSite("http://www.questpond.com").IsAlive
== false;
})
.Execute(delegate
{
Notify(“admin@example.org”,
“site down!”);
});
Are Fluent interfaces always good?
If you are creating fluent interfaces for developers then probably you are wasting time. Developers are consistent with creating objects with the new keyword and setting properties. Even if you create something simple for them they are consistent with certain consuming methodology and you will find you are adding more complexity than simplifying it.
Second thing we need to understand that we need to write
more amount of code to create fluent interfaces. As you saw in the previous
code I created a separate class for making interfaces simplified.
There are very few instances when you need fluent
interfaces, below are few of them:-
- During UNIT testing when the developers are not full fledged programmers.
- You want your code to be readable by non-programmers so that they can understand if the code is satisfies their domain logic.
- You are component seller and you want to stand out in the market as compared to the others by making your interfaces simpler.
- You are creating a DSL language and this language is mapped with fluent interface statements.