javascript - is there a benefit to using classes in js vs creating standalone functions - Stack Overflow

admin2025-04-04  0

I do a lot of full-stack JS work, and I generally follow an approach like this when creating a file with encapsulated logic:

export const SOME_KEY_TO_STATE = 'some-key';
export const ANOTHER_KEY_TO_STATE = 'another-key';

let moduleState = {};

export function modifyState(someArg) {
  // ... do some logic
  // ... perhaps derive some new value based off of logic
  const newValue = derivedNewValue;
  moduleState[someKey] = newValue;
}

export function getSomeState(aKey) {
  // ... do some sanity checking?
  const initialValue = moduleState[aKey];

  // ... calculate value based on some conditions?
  const finalValue = calculatedValue;
  return moduleState
}

I've also used classes occasionally, which essentially provide the same structure, except the module state would be inside the class as an instance variable, and possibly static variables for those exported constants:

export default class SomeThing {
  static SOME_KEY = '';

  state = {};

  modifyState(arg) { ... }
  getSomeState() { ... }
}

My preferred approach is the first one, primarily because I can import only what I need in other parts of the code without having an entire object floating around (along with it's state and other methods that I may not use <-- is this statement accurate or am I pletely off?). Also, if I want to reference functions within a context, I can always do import { * as someName } from myModule

I'm curious though, is there a benefit that I'm not aware of to using a class rather than the first approach I outlined?

I do a lot of full-stack JS work, and I generally follow an approach like this when creating a file with encapsulated logic:

export const SOME_KEY_TO_STATE = 'some-key';
export const ANOTHER_KEY_TO_STATE = 'another-key';

let moduleState = {};

export function modifyState(someArg) {
  // ... do some logic
  // ... perhaps derive some new value based off of logic
  const newValue = derivedNewValue;
  moduleState[someKey] = newValue;
}

export function getSomeState(aKey) {
  // ... do some sanity checking?
  const initialValue = moduleState[aKey];

  // ... calculate value based on some conditions?
  const finalValue = calculatedValue;
  return moduleState
}

I've also used classes occasionally, which essentially provide the same structure, except the module state would be inside the class as an instance variable, and possibly static variables for those exported constants:

export default class SomeThing {
  static SOME_KEY = '';

  state = {};

  modifyState(arg) { ... }
  getSomeState() { ... }
}

My preferred approach is the first one, primarily because I can import only what I need in other parts of the code without having an entire object floating around (along with it's state and other methods that I may not use <-- is this statement accurate or am I pletely off?). Also, if I want to reference functions within a context, I can always do import { * as someName } from myModule

I'm curious though, is there a benefit that I'm not aware of to using a class rather than the first approach I outlined?

Share Improve this question asked Nov 22, 2019 at 20:56 Abdul AhmadAbdul Ahmad 10k16 gold badges74 silver badges136 bronze badges 5
  • 1 It is a personal opinion on why A is better than B – epascarello Commented Nov 22, 2019 at 21:16
  • 1 There are so many trade offs and benefits to both. There is not really a sure fire way to say one is better than the other unless you give a direct, concrete use case. One could preach the benefits of using classes and then have to write a whole serialization helper to use those same classes with Redux. At the same time someone else might write a factory that makes POJOs but then has to import a million helper functions that might serve better as methods or even mixins. Please narrow the focus of this question. – zero298 Commented Nov 22, 2019 at 21:23
  • I think it's worth mentioning that classes can be extended and polymorphic. Classes help the code to adhere to S.O.L.I.D. principles, particularly the "L" (Liskov substitution principle): "Objects in a program should be replaceable with instances of their subtypes without altering the correctness of that program." – benbotto Commented Nov 22, 2019 at 22:32
  • @epascarello yea, i agree, that's why i didnt ask which is better, rather if there are benefits to one over the other; and by benefits, i mean concrete/quantitative performance or optimization benefits – Abdul Ahmad Commented Nov 23, 2019 at 2:39
  • Does this answer your question? Differences between creating a new class to using export const – Heretic Monkey Commented May 6, 2021 at 14:11
Add a ment  | 

3 Answers 3

Reset to default 5
let moduleState = {};

This is global state, which is best to be avoided. While it might be encapsulated well in your module, it is a static variable, essentially a singleton - it exists only once in your entire application, being created when the module is loaded.

There are some cases where this might be appropriate, but in general don't use singletons!

A class solution has the clear advantage that you can instantiate it as many times as you want, in any place where you want (e.g. tests). So make sure to understand this difference, then choose the appropriate pattern.

I like the first one, primarily because I can import only what I need in other parts of the code without having an entire object floating around (along with it's state and other methods that I may not use)

This is pretty off indeed. You always have the plete module object and state and functions floating around in your application, it's not like they get optimised away. Sure, when you don't import them explicitly they're not around in your scope, but having a class instance available doesn't really hurt either. If you don't call all of its methods, it's just like not importing all functions from your module.

The syntactic difference between calling a plain function or invoking a method on an object really is a minor detail that should not affect your decision. You can always sidestep it anyway by namespace-importing the module or destructuring the object.

Well this is up to debate but I don't think there are any advantages. This question has been discussed quite a lot amongst other developers and it boils down to personal preferences.

The main reason for classes is to allow developers who work mainly with other languages to jump over to JS more easily.

As one of the developers on Quora has said:

Using functions, factory functions, old-fashioned JS classes, or ES6 classes will all achieve the same thing, and pliments of browser-level optimizations, many of these options pile down to the same code when run by the engine.

Here's a link to his answer.

Another thing is that OOP is an expression of design, it's all about structure.

Many of the modern JS frameworks such as Angular, React or Vue heavily empathise on well structured code. ReactJS even has class ponents containing a constructor which yet again leads to the idea of writing more OOP JS.

Even of you are talking about using classes when developing NodeJS applications, it es down to personal syntactic preferences because V8 does most of the heavy lifting for you. Classes in JS are just a syntactical sugar coating around objects (which pretty much everything is in JS) and the V8 engine translates those for you.

Here are some good reads on the topic:

  1. Should we use classes in JS
  2. How JS works inside the V8 engine.

I hope that helps a bit.

the below does not directly apply to my question because my specific example has loose functions with side-effects (aka, they mutate a state object). in this case, treeshaking will not happen even though the functions are named/loose functions instead of a class.


there is one advantage of using the first approach (loose functions) over the second (class/objects), and that's if you're using webpack and treeshaking.

with the second approach (class/object) webpack can't treeshake any part of the class and so the entire thing is imported (so a class with 5 or 10 methods will be imported fully with all 5-10 of its methods).

The loose functions on the other hand can be treeshaken as long as you only import what you need. So a file with 5 - 10 functions will not be fully imported if you only import the parts you need:

import { only, what, iNeed } from 'module';

there's obviously a little more to it than that, for more context: https://webpack.js/guides/tree-shaking/

转载请注明原文地址:http://conceptsofalgorithm.com/Algorithm/1743706792a215991.html

最新回复(0)