Arun Pandian M

Arun Pandian M

Android Dev | Full-Stack & AI Learner

Products — When Two Independent Things Become One Structure

In programming, we combine values all the time.

But rarely do we ask a deeper question:

Is there a correct way to combine things?

Not just a convenient way. Not just a common pattern. But a way that is universally right.

Category theory answers this using something called a universal construction. Instead of defining a structure directly, it defines it by how it relates to everything else.

And when we ask:

“What is the best way to combine two things?”

This perspective leads us to a fundamental idea:

The Product.

A product is not just a pair of values. It is the *most universal* way of combining them.

To make this concrete, consider something simple:

Latitude and longitude. Individually, they are just numbers. But when combined correctly, they represent a precise location on Earth. And when combined incorrectly… everything breaks.

So what does it mean to combine them correctly?

That’s exactly what the idea of a Product captures.

The Mathematical Idea

In category theory, a product of A and B is:

An object (A, B) with two projections:
π₁ : (A, B) → A  
π₂ : (A, B) → B

These just mean:

  • take first value
  • take second value
  • The Pattern (What really matters)

    We don’t define product by structure.

    We define it by behavior:

          c
         / \
        p   q
       /     \
      A       B
    https://storage.googleapis.com/lambdabricks-cd393.firebasestorage.app/product.svg?X-Goog-Algorithm=GOOG4-RSA-SHA256&X-Goog-Credential=firebase-adminsdk-fbsvc%40lambdabricks-cd393.iam.gserviceaccount.com%2F20260521%2Fauto%2Fstorage%2Fgoog4_request&X-Goog-Date=20260521T005023Z&X-Goog-Expires=3600&X-Goog-SignedHeaders=host&X-Goog-Signature=7317364f2590b90c4fd34083707ba2e483f5da0551a224af25521262471aef5c1b70ccf597eb544fa88ba32375acc5a90204469dada58549df1031f00277c6c01ed3752eeabaea313a083843a7cdd5bb6362392813314ccbb420718f95eb5de634fcf362bf7ebb123ec8e401af0b5df8746e4ab373ec4c2749fc2f6941507be7730ff977dd6ae3a0026a49249a44b53fd9f7d96a6681a1018664d61356460302d79d35014b5b05826a27e7c836dbd1979d02554eb8c3a6a9f8c3a5c71911671cd7bcadb2fd68734dce80829f00d947e29f7e8b528691c45e84a129c8555c6c90672f3915940b2475acb7916bec027e4abecb0f5145e3fcf21767a3013bd6a26d

    Key Idea

    If you have:

    p : C → A  
    q : C → B

    Then there must exist a unique function:

    m : C → (A, B)

    Factorizer (The Heart of Product)

    This function is called the factorizer:

    m(x) = (p(x), q(x))

    Kotlin Implementation

    fun <C, A, B> factorizer(
        p: (C) -> A,
        q: (C) -> B
    ): (C) -> Pair<A, B> = { x ->
        Pair(p(x), q(x))
    }

    Real Example — Latitude & Longitude

    Domain

    data class Location(
        val latitude: Double,
        val longitude: Double
    )

    Projections

    val getLatitude: (Location) -> Double = { it.latitude }
    val getLongitude: (Location) -> Double = { it.longitude }

    Type matches, meaning is wrong(Compiles!)

    val wrong = { loc: Location ->
        Pair(getLatitude(loc), getLatitude(loc)) // bug
    }

    Output:

    (12.97, 12.97)

    Here Longitude lost, Type system didn’t help

    Correct (Factorizer)

    val correct = factorizer(getLatitude, getLongitude)
    
    println(correct(Location(12.97, 77.59)))
    // (12.97, 77.59)

    Why Factorizer Matters

    It enforces laws:

    fst(m(x)) = p(x)  
    snd(m(x)) = q(x)

    Any function that breaks this is not a product

    Without ProductWith Product
    Many ways to combineOnly ONE valid way
    Easy bugsGuaranteed correctness
    No structureMathematical guarantee

    Real Use Case — Parallel Data Fetch

    A screen needs to show user info along with their posts.

    Define Combine

    fun <C, A, B> combine(
        f: (C) -> A,
        g: (C) -> B
    ): (C) -> Pair<A, B> = { x ->
        Pair(f(x), g(x))
    }

    Use combine

    val fetchAll = combine(
        ::fetchUser,
        ::fetchPosts
    )
    
    val result = fetchAll(userId)

    We are combining two independent computations into one.

    Parallel Version

    fun <C, A, B> combineAsync(
        f: suspend (C) -> A,
        g: suspend (C) -> B
    ): suspend (C) -> Pair<A, B> = { x ->
        coroutineScope {
            val a = async { f(x) }
            val b = async { g(x) }
            Pair(a.await(), b.await())
        }
    }
    val fetchAll = combineAsync(
        ::fetchUser,
        ::fetchPosts
    )
    
    val result = fetchAll(userId)

    When values are independent but needed together, they form a Product.

    https://storage.googleapis.com/lambdabricks-cd393.firebasestorage.app/product_factorization.svg?X-Goog-Algorithm=GOOG4-RSA-SHA256&X-Goog-Credential=firebase-adminsdk-fbsvc%40lambdabricks-cd393.iam.gserviceaccount.com%2F20260521%2Fauto%2Fstorage%2Fgoog4_request&X-Goog-Date=20260521T005023Z&X-Goog-Expires=3600&X-Goog-SignedHeaders=host&X-Goog-Signature=54f3635478580fe1001681d75e2fa700c69cb254a6fe0707aafa13cad0e89640fff01b42d2fe294de4223de3bef50e5a1de22454d02049c82dad4023936fc79729f429e6b088d34ca89f31a4c625fca96b75230bfdbe801163fee3d6371edc1e0458eff8e9b48846cb3b4405a423328d5f5057bbf3a4f5da0f2bcf425f0ef624f53b0146eee5ce569fbb1ccdfc9f7768446ec78e22b9776b2d77d5c8439163806fae457b9530bccf06fa82fac4fd082903895ae13b4ff17e098b37d1b5be2d31007874a593fb0e223167de24b7d5ddac88b00fa41e7bcde8a393e1e9a2433bdeb2c5d1909d5557a462bca5796c8767cf36572dc5a74a54839f17adb51af70d1e

    Final Takeaway

    A product is not just a pair.

    It is:

    A structure where **every valid combination must pass through one unique path**

    One-line Conclusion

    Product is not about combining values — it’s about guaranteeing the **only correct way to combine them**
    #TypeTheory#BuildInPublic#EngineeringMindset#ComputerScience#SoftwareDesign#CategoryTheory#DataModeling#FunctionalProgramming#KotlinFP#ProductType#MathForDevelopers#FPFoundations#ProgrammingConcepts#TheoreticalCS#LearnInPublic