Organizing Modules in React Project — Low Coupling and High Cohesion
Design robust applications
data:image/s3,"s3://crabby-images/5bb2a/5bb2a30fdff4773605b50e99d1de66b1a5c862b5" alt="Coupling and Cohesion"
How do you organize modules in a React project you are working on? I hope you place elements related to each other together (module) and then set up connections between these groups of elements. Right?
The principle described above allows designing extensible applications. It sounds like the following:
Organize system elements in a way to be decoupled and cohesive. i.e., prefer low coupling and high cohesion.
You may have already heard this rule. But have you ever wondered what does it really mean and what real benefit could it provide? I suggest you dive deeper into these questions and gain a justification for following this rule.
Coupling
Coupling is something like connections between system elements. High coupling is when elements of a system have chaotic connections.
Technically, we can measure the value of coupling — this is just a number of connections between system elements (functions, objects, modules, etc.). The rule states that the system should have the lowest possible number of connections. In this case, we can be sure that there are only expected connections, and we are able to manage them effectively.
data:image/s3,"s3://crabby-images/ac0d2/ac0d2defea407393ec277be4c12846a8caadf441" alt="Coupling"
Actually, not sure that there is an easy way to calculate the number of connections. I even don’t think that this number could be really useful!
Let’s imagine that we’ve calculated a number of connections some way and it’s “100”. Is it much or not? How can we use this number to improve an existing system? Not sure what to answer here.
So, how to improve an existing system? The only adequate way is to follow the techniques we are going to discuss a bit later.
Cohesion
Cohesion is when elements of a system are grouped together by some criteria. Low cohesion means that app elements don’t have clear boundaries. This may look like a mess.
There are a lot of things we can call a “group”:
- module
- class
- React component (both class and functional)
- namespace
- service
- and so on…
data:image/s3,"s3://crabby-images/16f94/16f942df0f59ce44aa7eb648a2b9eaaf250dc630" alt="Cohesion"
But why is it useful? Because in a cohesive system there are much fewer connections between elements. And this leads to better management of these connections and we may be more sure that they are expected.
As you can see, “coupling” and “cohesion” terms always go together. In this case, supporting high cohesion is a better solution to decreasing the number of connections and making a system more manageable and extensible.
Coupling and Cohesion
Let’s explore different cases. Below you can find a diagram that describes connections between “coupling” and “cohesion”. This may be useful for analyzing your application and creating an improvement plan.
data:image/s3,"s3://crabby-images/c51ce/c51ced4c9c709acf466549de9e07977f7b7b79b7" alt="Coupling and Cohesion"
1. Ideal: low coupling, high cohesion
This is a case we need to support. Such a system is manageable and extensible. In most cases devs like working with similar systems as it may be divided into logical parts; this leads to lower mental pressure during the dev process.
Below you can see how connections between elements may look like in this case.
data:image/s3,"s3://crabby-images/c6417/c64174f54dfc127581f02a361845f7f33863879b" alt="Low Coupling, High Cohesion"
How to follow this rule? Not a complex one: organize related elements in groups (see above) and set up connections between groups.
This is how project structure may look in this case.
And now let’s see a contrived code example.
2. God Module: high coupling, high cohesion
For me, this is the worst case, and every team should avoid it. Such a system is hardly manageable and cannot be extended in an adequate manner.
data:image/s3,"s3://crabby-images/e14dd/e14dd794c256725a39e0643f4f4f230e3c42fd92" alt="High Coupling, High Cohesion"
In this situation, there is a mess of system elements. Let’s imagine how this may look in code structure.
And contrived code example.
The problem here is that there is data logic embedded into a shared component. This component should be reusable, and new data types should not affect it. Please, avoid such a design and keep all the specific logic outside the reusable component.
3. Wrong Boundaries: high coupling, low cohesion
Let’s explore the diagram and examples w/o any additional words.
data:image/s3,"s3://crabby-images/f5698/f569879e84327b7cb74dfae3596b6db42bd33bd0" alt="High Coupling, Low Cohesion"
Contrived project structure.
And contrived code example.
Hope it is clear to you that boundaries are wrong in this example. Such an app should be redesigned to be like in the first example.
4. Destructive Decoupling: low coupling, low cohesion
Let’s explore the diagram and examples w/o verbose explanations.
data:image/s3,"s3://crabby-images/a4aa6/a4aa62cb7aa27b62195902cad1d61436884c206f" alt="Low Coupling, Low Cohesion"
Contrived code structure.
And contrived code example.
The problem here is that there are a lot of elements thrown in the same pile. And it is rather hard to understand the usage scope of every module (file).
That’s it! Hope you enjoyed this content and this will help you design robust applications.