Why you cant pass function as properties to server components and client components?

Ravi Chandola
Javarevisited
Published in
4 min readMay 21, 2024

In React, “server components” and “client components” refer to a pattern where some components are rendered on the server side and others on the client side. This distinction is part of the Server Components feature which was proposed for future versions of React as an experimental approach to improve performance by reducing the amount of JavaScript that needs to be loaded on the client side. The idea is that server components can render to a special format that can be streamed to the client and hydrated without including the full component logic.

One of the constraints of server components is that you cannot pass functions as properties (props) to them. This is because functions are inherently bound to the JavaScript environment they are created in, and they often capture local variables from their surrounding scope (this is called “closure”). This makes functions non-serializable, meaning they cannot be easily converted into a format that can be sent over the network from the server to the client.

Here are the main reasons why you can’t pass functions as props to server components:

  1. Serialization: Server components are designed to be serialized into a format that can be sent over the network. Functions, especially those with closures, are not serializable because they can reference variables and state that are not available outside of their original execution context.
  2. Hydration: When server-rendered HTML is sent to the client, it needs to be “hydrated” into a fully interactive React component tree. If functions were passed as props, the client would need the exact same function code to properly hydrate the component, which would defeat the purpose of reducing JavaScript bundle size.
  3. Statefulness: Server components are meant to be stateless, meaning they should not maintain any internal state that changes over time. Functions often encapsulate behavior that can modify state, so passing functions to server components could lead to state management issues and make components less predictable.
  4. Concurrency: The React team is exploring concurrent rendering, which allows multiple versions of the UI to be prepared at the same time. Functions with closures could have side effects or depend on mutable state, which could break assumptions about concurrency and lead to bugs.
  5. For client components, you can still pass functions as props because these components are fully rendered and executed in the client’s JavaScript environment. They are not subject to the same constraints as server components, as they don’t need to be serialized and sent over the network. However, even with client components, it’s important to be mindful of the potential performance implications of passing functions, especially if they are created inline during rendering, which can lead to unnecessary re-renders.

To work with server components, you would typically design your components so that any interactive functionality is handled by client components, and any data-fetching or computationally heavy operations are handled by server components without relying on passing functions as props.

Let’s simplify this concept with an analogy. Imagine you’re running a restaurant with two chefs: one who works in the kitchen (the server) and one who serves food at the table (the client). Your restaurant has a rule that the kitchen chef can only send out dishes that are fully cooked and ready to eat, while the table chef can add final touches like garnishes or sauces based on the diner’s preference.

In this analogy, the “dishes” are the components of your web application. The kitchen chef (server component) prepares most of the meal but doesn’t have the ability to make changes on the fly. The table chef (client component) can interact with the diners (users) directly and personalize the dish at the table.

Now, let’s say a diner (user) wants a special sauce (function) on their dish. The kitchen chef can’t send this special sauce over because it might not travel well, and it’s meant to be made fresh at the table. So, it’s the table chef’s job to prepare and add this sauce (function) as per the diner’s request.

Here’s a more technical explanation using components:

  • Server components are like the dishes prepared in the kitchen. They’re made ahead of time and don’t change once they leave the kitchen. You can’t give the kitchen chef a special request (function) because once the dish is out, it can’t be altered.
Server Components
  • Client components are like the tableside service. They can take special requests (functions) because they’re interacting directly with the diners (users) in real-time.
Client Components

Pros:

  • Reduced Load on Diners (Users): Just like diners don’t have to go to the kitchen to get their food, users don’t have to download lots of code. This makes the website faster to load.
  • Efficiency in the Kitchen (Server): By focusing on preparing dishes that don’t require last-minute changes, the kitchen can work more efficiently, just like server components can render pages more quickly without dealing with interactive parts.

Cons:

  • Less Flexibility in the Kitchen (Server): The kitchen chef can’t make changes once the dish is out, similar to how server components can’t handle dynamic functions that respond to user interactions.
  • Dependency on Table Chef (Client): If diners want something special, they rely entirely on the table chef, just like any interactive parts of a website must rely on client components.

Why React is Suggesting This Approach: React suggests this approach because it’s like having a specialized kitchen that does what it does best — preparing dishes efficiently — and a table chef who can personalize the dining experience. By splitting the work, the restaurant (web application) can serve meals (web pages) faster and make sure each diner (user) gets a personalized touch when needed, without overloading the kitchen (server) with special requests (functions).

--

--

Ravi Chandola
Javarevisited

As a technology enthusiast, I have a passion for learning and sharing my knowledge with others. https://www.linkedin.com/in/ravi-chandola-304522133