Markus Oberlehner

Tight Coupling vs. Loose Coupling in Vue.js


When talking about loose coupling and tight coupling, often, the impression arises that tight coupling is something we always have to avoid. But this is almost impossible. What’s essential is that we use loose coupling when bridging the gap between layers of our application. Within a layer, though, it is often no problem if there is tight coupling. In this article, we will look at when tight coupling is unproblematic, and when it is better to use loose coupling.

When tight coupling is ok

Tight coupling is ok when we want to do specific things generically, there is no variability depending on the use case, and we do not cross layers of our application. Let’s take a look at an example: we want to render a list of blog articles or comments. It’s always the same table view with a sort functionality. The sort function always does sort the entries alphabetically; it does not care if the entries are blog articles or comments. So in this example, our list component must be loosely coupled to its content (blog articles, comments,…), but there is no problem with tightly coupling the sorting function (or component).

<template>
  <div class="ListContent">
    <table>
      <thead>
        <th @click="sort">Label</th>
        <!-- ... -->
      </thead>
      <!-- Tabular representation of entries -->
    </table>
  </div>
</template>

<script>
// Importing `sortAlphabetically()` here means that this
// component is **tightly coupled** to exactly this
// implementation. But that's ok!
import { sortAlphabetically } from "../utils/sort-alphabetically";

export default {
  name: "ListContent",
  props: {
    // "Injecting" the `entries` as a prop is a simple yet
    // effective form of **loose coupling**. The component
    // does not care if the entries are blog articles or
    // comments.
    entries: {
      default: () => [],
      type: Array,
    },
  },
  // ...
  methods: {
    // Pseudo code!
    sort() {
      this.sortedEntries = sortAlphabetically(this.entries);
    },
  },
  // ...
};
</script>

sortAlphabetically() does one specific thing, and it does not care about which data it gets passed (as long as the entries adhere to an underlying interface). Also, there is no crossing of layers; sortAlphabetically() does not access an API or a database, for example, and it also is not concerned with any business logic.

But what about testing? Contrary to popular belief, loose coupling can make testing more difficult in some cases. Let’s imagine that we’ve used loose coupling to inject sortAlphabetically() into our application. For example, by using a Vue.js plugin, which makes this helper function globally available as this.$sortAlphabetically().

mount(ListContent, {
  // ...
  mocks: {
    $sortAlphabetically: (entries) => entries,
  },
  // ...
});

As we can see above, if we want to test our component, we now have to manually mock $sortAlphabetically(). Of course, we could write a helper method for mounting components with all the plugins we also have in the production app. But this makes our test code more complicated, slower, and harder to maintain. And this for virtually no benefit other than to not have to import sortAlphabetically() in our component.

But sometimes, as we will see in the next chapter, the benefits of loose coupling are much more significant, so it’s worth to accept the hassle of having to use mocking in our tests. Or we have to mock the dependency anyway (to avoid sending requests to a real API, for example).

When loose coupling is preferred

Let’s take a look at the following example where we slightly change the way our ListContent gets its contents.

<template>
  <div class="ListContent">
    <table>
      <thead>
        <th @click="sort">Label</th>
        <!-- ... -->
      </thead>
      <!-- Tabular representation of entries -->
    </table>
  </div>
</template>

<script>
import { sortAlphabetically } from '../utils/sort-alphabetically';

export default {
  name: 'ListContent',
  props: {
    // Here we "inject" a generic service object; this can be a blog article
    // service, a comment service,... you name it.
    service: {
      required: true,
      type: Object,
    },
  },
  data() {
    return { entries: [] },
  },
  created() {
    this.loadEntries();
  },
  methods: {
    async loadEntries() {
      this.entries = await this.service.list();
    },
    // ...
  },
  // ...
};
</script>

Instead of passing in the entries via a property, delegating the responsibility for fetching them from an API to the parent component, we pass in a service object so that this component can fetch the data itself. In this architecture, services adhere to the same interface, so in our ListContent component, we can rely on calling the list() method on our service object to get an array of entries. The service itself can either be an implementation to load blog articles or comments or anything else.

We loosely couple the ListContent component to the service by passing it via a prop. The critical difference between the service and sortAlphabetically() is that 1) we can pass in different services to get different results, and 2) services cross layers of our application because they typically access an API and they often contain business logic.

When it comes to testing, we must mock our service implementation anyway because we don’t want to make network requests in our unit tests. Additionally, because we pass in the service as a property, testing becomes a little bit easier as if we would import (tightly couple) a specific service implementation.


Do you want to learn more about advanced Vue.js techniques?

Register for the Newsletter of my upcoming book: Advanced Vue.js Application Architecture.


Different forms of (loose) coupling in Vue.js

If you want to learn more about different ways of how to use loose coupling in Vue.js, you can read one of my earlier articles about this very topic.

Wrapping it up

As we’ve seen, sometimes some tight coupling in the right place can improve your application because it makes it easier to test individual parts. As with so many things, there is no simple answer to whether tight coupling is right or wrong. It depends on the use case.