Any way to define getters for lazy variables in Javascript arrays?

后端 未结 5 1503
自闭症患者
自闭症患者 2021-01-04 05:50

I\'m trying to add elements to an array that are lazy-evaluated. This means that the value for them will not be calculated or known until they are accessed. This is like a p

相关标签:
5条回答
  • 2021-01-04 06:12

    I don't particularly like this answer, but could you store your "variable" as an expression string, and then eval() it when you need it? Not ideal, but it's compact...

    var x = 10, arr = [];
    arr.push("x + 10");
    alert(eval(arr[0]));
    x = 20;
    alert(eval(arr[0]));
    

    I've tested it, and it works, even if it's not exactly what you're looking for.

    0 讨论(0)
  • 2021-01-04 06:13

    Ok, unless I misunderstood your question, I believe I have found a simple solution which does not require a so-called "lazy_push" function.

    Following the method from my previous answer, you create a MyArray Class:

    function MyArray(){
         this.object = [];
     }
    
    MyArray.prototype.push = function(what){
         this.object.push(what);
    }
    

    Now the important part is the getter function, we will create a getIdx() function to grab the value out of the array. The function then uses the 'typeof' operator to determine if the returned value is a function. If it is, return the value returned from the function, if not return the original value.

    Code makes more sense:

    MyArray.prototype.getIdx = function(which){
         if(typeof this.object[which] == 'function'){
             alert("a function");
             //NOTICE THE '()' AT THE END OF THE NEXT LINE!!!
             return this.object[which]();
         }
         return this.object[which];
     }
    

    Hopefully if I didn't completely answer your question you can figure it out from here.

    <--------- My original post ------------->

    Not really an answer, but a few important points.

    1. There are no real arrays in Javascript, an Array is just an extended Object (as is everything in JS)

    2. You should ideally never add prototype functions to the native objects in JS, you might accidentally overwrite an existing property, or create confusing errors down the line. For instance, adding to the prototype of Object is going to add to every single Object in JS (which is everything), you need to make absolutely sure that you want every type in JS to have that property. This is just dangerous because if you accidentally overwrite the actual Array() or Object() functions, you will break javascript in the browser, period, a refresh of the page won't fix it.

    3. Rather than adding to the prototype of the Object you are modifying, create a new Object that extends it. For instance if you want to extend the Array class:

      //create the constructor, 
      //with an object variable that holds the object you wish to extend
      function MyArray(){
           this.object = [];
      }
      
      //create functions which reference the native functions of the object
      MyArray.prototype.push = function(what){
           this.object.push(what);
      }
      
      //Etc... Etc....  Etc.....
      

    Its not necessarily fun writing all the accessor methods for native Object functions, but it keeps the Javascript engine safe.

    0 讨论(0)
  • 2021-01-04 06:26

    Bleh. It's a major bummer you can't overload the indexer operator in JavaScript. Oh well. We'll just have to be creative and come up with a different solution. This is a good thing (and fun). :-)

    LLer, the solution you settle with is a damn good one. Kudos. It's refreshing to come across people who truly understand JavaScript to that degree.

    After reading this question I was struck with an epic idea and wrote up some code for the fun of it. My solution to the problem is very similar to yours and many others that have been done before. But, I feel I came up with something unique and very neat so I want to share it.

    So I'm hosting a project of mine on CodePlex where I use a very jQuery-esque technique for defining properties (self-contained getter/setter functions) very similar to the one you're using. For the solution I came up with I simply just extrapolated from this pre-existing code of mine. Here's my approach toward the lazy loading of array indexes. Starting from the beginning...

    Let's consider a property named "PageSize". Here's how the property would be used with my technique:

    var MyClass = function() { }; // MyClass constructor.
    
    var instance = new MyClass();
    instance.PageSize(5);
    
    alert(instance.PageSize());
    

    Take notice that the property is a single function where providing a value as the first parameter invokes the setter and leaving out the parameter invokes the getter. The "PageSize" property would be defined as part of the MyClass class like so:

    MyClass.prototype.PageSize = function(v) { return this.GetSetProperty("PageSize", v, myFunctionThatDoesLazyLoading); };
    

    The property function is merely a wrapper around a call to the utitily GetSetProperty method which does the actual getting and setting. Here's a snippet of the GetSetProperty function:

    Object.prototype.GetSetProperty = function(name, value, loadFunction) {
        if (!this.Properties)
        {
            this.Properties = {};
        }
    
    if (value)
    {
        this.Properties[name] = value;
    }
    else
    {       
        if (!this.Properties[name] && loadFunction)
        {
            this.Properties[name] = loadFunction();
        }
    
        return this.Properties[name]; // This will return "undefined" if property doesn't exist or the loadFunction wasn't provided.
    }
    };
    

    So that handles properties. But, to provide a means to access indexed values of a possible property of Array type I modify this code further like so:

    Object.prototype.GetSetProperty = function(name, value, loadFunction, index) {
      if (!this.Properties)
      {
        this.Properties = {};
      }
    
      if (typeof index === "number" && this.Properties[name] && this.Properties[name].constructor == Array)
      {
        return this.GetSetArrayProperty(name, index, value, loadFunction);
      }
      else 
      {
        value = index;
      }
    
      if (value)
      {
        this.Properties[name] = value;
      }
      else
      {    
        if (!this.Properties[name] && loadFunction)
        {
          this.Properties[name] = loadFunction();
        }
    
        return this.Properties[name]; // This will return "undefined" if property doesn't exist or the loadFunction wasn't provided.
      }
    };
    
    
    Object.prototype.GetSetArrayProperty = function(name, index, value, loadFunction) {
      if (value)
      {
        this.Properties[name][index] = value;
      }
      else
      {
        if (!this.Properties[name][index] && loadFunction)
        {
          this.Properties[name][index] = loadFunction();
        }
    
        return this.Properties[name][index];
      }
    };
    

    The prototype declaration would need to be modified like so:

    MyClass.prototype.PageSize = function(i, v) { return this.GetSetProperty("PageSize", v, myFunctionThatDoesLazyLoading, i); };
    

    Everyone reading this can access a working set of the code here: http://jsbin.com/ajawe/edit

    Here's a complete listing of the code with tests:

    Object.prototype.GetSetProperty = function(name, value, loadFunction, index) {
      if (!this.Properties)
      {
        this.Properties = {};
      }
    
      if (typeof index === "number" && this.Properties[name] && this.Properties[name].constructor == Array)
      {
        return this.GetSetArrayProperty(name, index, value, loadFunction);
      }
      else 
      {
        value = index;
      }
    
      if (value)
      {
        this.Properties[name] = value;
      }
      else
      {    
        if (!this.Properties[name] && loadFunction)
        {
          this.Properties[name] = loadFunction();
        }
    
        return this.Properties[name]; // This will return "undefined" if property doesn't exist or the loadFunction wasn't provided.
      }
    };
    
    
    Object.prototype.GetSetArrayProperty = function(name, index, value, loadFunction) {
      if (value)
      {
        this.Properties[name][index] = value;
      }
      else
      {
        if (!this.Properties[name][index] && loadFunction)
        {
          this.Properties[name][index] = loadFunction();
        }
    
        return this.Properties[name][index];
      }
    };
    
    
    // Here's a nifty function that declares the properties for you.
    Function.prototype.CreateProperty = function(propertyName, loadFunction) {
      eval("this.prototype['" + propertyName.toString() + "'] = function(i, v) { return this.GetSetProperty('" + propertyName.toString() + "', v, " + eval(loadFunction) + ", i); };");
    };
    
    
    
    
    var myFunctionThatDoesLazyLoading = function() {
      return "Ahoy!";
    };
    
    
    var MyClass = function() { }; // MyClass constructor.
    MyClass.prototype.PageSize = function(i, v) { return this.GetSetProperty("PageSize", v, myFunctionThatDoesLazyLoading, i); };
    
    var instance = new MyClass();
    alert(instance.PageSize()); // PageSize is lazy loaded.
    
    instance.PageSize(5); // PageSize is re-assigned.
    alert(instance.PageSize()); // Returns the new value.
    
    instance.PageSize([1, 2, 3]); // PageSize is re-assigned to have an Array value.
    alert(instance.PageSize(2)); // Returns the value at index 2 of the Array value.
    
    instance.PageSize(2, "foo"); // Re-assigns the value at index 2.
    alert(instance.PageSize(2)); // Returns the new value at index 2.
    
    MyClass.CreateProperty("NewProp", function() { return ["a", "b", "c"]; }); // Demo of the CreateProperty function.
    alert(instance.NewProp());
    alert(instance.NewProp(1));
    
    0 讨论(0)
  • 2021-01-04 06:28

    In 2019, you could use a Proxy.

    Here is an example which assumes any value which is a function should be evaluated when accessed.

    function LazyArray() {
      return new Proxy([], {
        get: (obj, prop) => {
          if (typeof obj[prop] === 'function') {
            // replace the function with the result
            obj[prop] = obj[prop]()
          }
          return obj[prop]
        },
      })
    }
    
    const a = LazyArray()
    
    a[0] = () => {
      console.log('calculating...')
      return 42
    }
    
    console.log(a[0]) // lazy evaluated
    console.log(a[0])
    
    console.log(a.length) // accessing other properties

    0 讨论(0)
  • 2021-01-04 06:30

    There is not. Unfortunately this is a big deal.

    0 讨论(0)
提交回复
热议问题