Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Code Challenge: Create a class Foo that tracks the number of total object instances

Tags:

javascript

I am trying to solve a code challenge for a job app, but I am stuck and would appreciate any help.

Question: Create a class Foo that has a method called refCount. Calling refCount on the class or any of its instances should return how many total instances exists.

Example:

var f1 = new Foo();

f1.refCount(); // should be 1
Foo.refCount(); // should be 1

var f2 = new Foo();

f1.refCount(); //should be 2
f2.refCount(); // should be 2
Foo.refCount(); // should be 2

I have something like this so far:

function Foo() {

  this.refCount = function() {
    ++Foo.prototype.refs;
    return Foo.prototype.refs;
  }

}

Foo.prototype.refs = 0;

I also tried using an IIFE to attach a method to the class itself, but then I was unable to figure out how to create new instances.

like image 846
user6317504 Avatar asked May 10 '16 22:05

user6317504


2 Answers

That cannot be done.

JavaScript is object-oriented, and classes in JavaScript are just an illusion, created by setting an object's prototype property to a certain prototype object.

The new Foo() syntax is syntactic sugar for setting that prototype. And your best option, as others have shown, is to do the refCounting in the constructor function, effectively counting the amount of times that it got called. You can then store the count variable in the prototype or as a property of the constructor function itself or in an IIFE (see below), and return its value in the refCount function.

But if people start dynamically changing the object's prototypes, the objects can change class and I can think of no way for the refCount function to know that that happened.

var Foo

(function(){
  var n = 0;
  Foo = function Foo() {
    this.refCount = Foo.refCount;
    n++;
  };
  Foo.refCount = function() {
    return n;
  }
})()

function Bar() {}

f = new Foo();
console.log("created f, refCount = 1", f instanceof Foo, f.refCount(), Foo.refCount());

g = new Foo();

console.log("created g, refCount = 2", f instanceof Foo, g instanceof Foo, f.refCount(), g.refCount(), Foo.refCount());

g.__proto__ = Bar.prototype;
console.log("turned g into a Bar", f instanceof Foo, g instanceof Foo)
console.log("but refCount still is 2", Foo.refCount());

h = Object.assign(f)
console.log("created h without calling the constructor", h instanceof Foo)
console.log("But refCount still is 2", h.refCount(), Foo.refCount())

See it on JSBin

See https://developer.mozilla.org/en/docs/Web/JavaScript/Inheritance_and_the_prototype_chain and How to set the prototype of a JavaScript object that has already been instantiated?

like image 104
flup Avatar answered Oct 20 '22 05:10

flup


function Foo() {
    Foo.__n++;
}
Foo.refCount = function() { return Foo.__n; }
Foo.__n = 0;
Foo.prototype.refCount = Foo.refCount;

Output:

> var f = new Foo(); console.log(f.refCount(), Foo.refCount());
1 1
> f = new Foo(); console.log(f.refCount(), Foo.refCount());
2 2
> f = new Foo(); console.log(f.refCount(), Foo.refCount());
3 3
like image 29
OneOfOne Avatar answered Oct 20 '22 04:10

OneOfOne