Swift lazy observe and mutate pattern for arrays












0












$begingroup$


I am using the following pattern in my code.
It works. But, it feels as though it is on the edge of what should work!



Should I consider an alternative?



I have an array in my class that maps an underlying data array. This is the sort of thing for which I could use a lazy sequence. However, I don't want to re-evaluate every time an element is accessed. So, I lazily load my array, then track the data array with KVO. I also want my top-level array to be observable. I mutate that with a mutableArrayValue.



I have mocked up a simple working example of the pattern below.



mutableArrayValue would usually cause a lazy var to initialise, but it seems OK to call it during an initialisation.



Setting up an observation on a lazy var does not cause it to initialise. I don't get a call here even though the effective setting of the array comes after the observation is created.



import Cocoa

class LazyData: NSObject
{
@objc dynamic lazy var units: [Int] = {
print("Loading units")
return [1,2,3,4,5,6] }()
}

class Lazy: NSObject
{
let data: LazyData

init(with data: LazyData) {
self.data = data
}

var unitsObservation: NSKeyValueObservation?

func tenTimes(_ input: Int) -> Int { return 10 * input }

@objc dynamic lazy var tens: [Int] = {
print("Loading tens")

let tenProxy = mutableArrayValue(forKeyPath: #keyPath(Lazy.tens))

unitsObservation = data.observe(.units) { object, change in
if change.kind == .insertion {
for insertionIndex in change.indexes! {
let ten = self.tenTimes(object.units[insertionIndex])
tenProxy.insert(ten, at: insertionIndex)
}
}
}

return data.units.map { tenTimes($0) }
}()
}

var lazyData = LazyData()
var lazy = Lazy(with: lazyData)

print(lazy.tens)

let lazyDataProxy = lazyData.mutableArrayValue(forKeyPath: #keyPath(LazyData.units))
lazyDataProxy.insert(9, at: 0)

print(lazy.tens)









share|improve this question









New contributor




Giles is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.







$endgroup$

















    0












    $begingroup$


    I am using the following pattern in my code.
    It works. But, it feels as though it is on the edge of what should work!



    Should I consider an alternative?



    I have an array in my class that maps an underlying data array. This is the sort of thing for which I could use a lazy sequence. However, I don't want to re-evaluate every time an element is accessed. So, I lazily load my array, then track the data array with KVO. I also want my top-level array to be observable. I mutate that with a mutableArrayValue.



    I have mocked up a simple working example of the pattern below.



    mutableArrayValue would usually cause a lazy var to initialise, but it seems OK to call it during an initialisation.



    Setting up an observation on a lazy var does not cause it to initialise. I don't get a call here even though the effective setting of the array comes after the observation is created.



    import Cocoa

    class LazyData: NSObject
    {
    @objc dynamic lazy var units: [Int] = {
    print("Loading units")
    return [1,2,3,4,5,6] }()
    }

    class Lazy: NSObject
    {
    let data: LazyData

    init(with data: LazyData) {
    self.data = data
    }

    var unitsObservation: NSKeyValueObservation?

    func tenTimes(_ input: Int) -> Int { return 10 * input }

    @objc dynamic lazy var tens: [Int] = {
    print("Loading tens")

    let tenProxy = mutableArrayValue(forKeyPath: #keyPath(Lazy.tens))

    unitsObservation = data.observe(.units) { object, change in
    if change.kind == .insertion {
    for insertionIndex in change.indexes! {
    let ten = self.tenTimes(object.units[insertionIndex])
    tenProxy.insert(ten, at: insertionIndex)
    }
    }
    }

    return data.units.map { tenTimes($0) }
    }()
    }

    var lazyData = LazyData()
    var lazy = Lazy(with: lazyData)

    print(lazy.tens)

    let lazyDataProxy = lazyData.mutableArrayValue(forKeyPath: #keyPath(LazyData.units))
    lazyDataProxy.insert(9, at: 0)

    print(lazy.tens)









    share|improve this question









    New contributor




    Giles is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
    Check out our Code of Conduct.







    $endgroup$















      0












      0








      0





      $begingroup$


      I am using the following pattern in my code.
      It works. But, it feels as though it is on the edge of what should work!



      Should I consider an alternative?



      I have an array in my class that maps an underlying data array. This is the sort of thing for which I could use a lazy sequence. However, I don't want to re-evaluate every time an element is accessed. So, I lazily load my array, then track the data array with KVO. I also want my top-level array to be observable. I mutate that with a mutableArrayValue.



      I have mocked up a simple working example of the pattern below.



      mutableArrayValue would usually cause a lazy var to initialise, but it seems OK to call it during an initialisation.



      Setting up an observation on a lazy var does not cause it to initialise. I don't get a call here even though the effective setting of the array comes after the observation is created.



      import Cocoa

      class LazyData: NSObject
      {
      @objc dynamic lazy var units: [Int] = {
      print("Loading units")
      return [1,2,3,4,5,6] }()
      }

      class Lazy: NSObject
      {
      let data: LazyData

      init(with data: LazyData) {
      self.data = data
      }

      var unitsObservation: NSKeyValueObservation?

      func tenTimes(_ input: Int) -> Int { return 10 * input }

      @objc dynamic lazy var tens: [Int] = {
      print("Loading tens")

      let tenProxy = mutableArrayValue(forKeyPath: #keyPath(Lazy.tens))

      unitsObservation = data.observe(.units) { object, change in
      if change.kind == .insertion {
      for insertionIndex in change.indexes! {
      let ten = self.tenTimes(object.units[insertionIndex])
      tenProxy.insert(ten, at: insertionIndex)
      }
      }
      }

      return data.units.map { tenTimes($0) }
      }()
      }

      var lazyData = LazyData()
      var lazy = Lazy(with: lazyData)

      print(lazy.tens)

      let lazyDataProxy = lazyData.mutableArrayValue(forKeyPath: #keyPath(LazyData.units))
      lazyDataProxy.insert(9, at: 0)

      print(lazy.tens)









      share|improve this question









      New contributor




      Giles is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.







      $endgroup$




      I am using the following pattern in my code.
      It works. But, it feels as though it is on the edge of what should work!



      Should I consider an alternative?



      I have an array in my class that maps an underlying data array. This is the sort of thing for which I could use a lazy sequence. However, I don't want to re-evaluate every time an element is accessed. So, I lazily load my array, then track the data array with KVO. I also want my top-level array to be observable. I mutate that with a mutableArrayValue.



      I have mocked up a simple working example of the pattern below.



      mutableArrayValue would usually cause a lazy var to initialise, but it seems OK to call it during an initialisation.



      Setting up an observation on a lazy var does not cause it to initialise. I don't get a call here even though the effective setting of the array comes after the observation is created.



      import Cocoa

      class LazyData: NSObject
      {
      @objc dynamic lazy var units: [Int] = {
      print("Loading units")
      return [1,2,3,4,5,6] }()
      }

      class Lazy: NSObject
      {
      let data: LazyData

      init(with data: LazyData) {
      self.data = data
      }

      var unitsObservation: NSKeyValueObservation?

      func tenTimes(_ input: Int) -> Int { return 10 * input }

      @objc dynamic lazy var tens: [Int] = {
      print("Loading tens")

      let tenProxy = mutableArrayValue(forKeyPath: #keyPath(Lazy.tens))

      unitsObservation = data.observe(.units) { object, change in
      if change.kind == .insertion {
      for insertionIndex in change.indexes! {
      let ten = self.tenTimes(object.units[insertionIndex])
      tenProxy.insert(ten, at: insertionIndex)
      }
      }
      }

      return data.units.map { tenTimes($0) }
      }()
      }

      var lazyData = LazyData()
      var lazy = Lazy(with: lazyData)

      print(lazy.tens)

      let lazyDataProxy = lazyData.mutableArrayValue(forKeyPath: #keyPath(LazyData.units))
      lazyDataProxy.insert(9, at: 0)

      print(lazy.tens)






      swift cocoa






      share|improve this question









      New contributor




      Giles is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.











      share|improve this question









      New contributor




      Giles is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.









      share|improve this question




      share|improve this question








      edited 16 mins ago









      Jamal

      30.4k11121227




      30.4k11121227






      New contributor




      Giles is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.









      asked 10 hours ago









      GilesGiles

      101




      101




      New contributor




      Giles is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.





      New contributor





      Giles is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.






      Giles is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.






















          0






          active

          oldest

          votes











          Your Answer





          StackExchange.ifUsing("editor", function () {
          return StackExchange.using("mathjaxEditing", function () {
          StackExchange.MarkdownEditor.creationCallbacks.add(function (editor, postfix) {
          StackExchange.mathjaxEditing.prepareWmdForMathJax(editor, postfix, [["\$", "\$"]]);
          });
          });
          }, "mathjax-editing");

          StackExchange.ifUsing("editor", function () {
          StackExchange.using("externalEditor", function () {
          StackExchange.using("snippets", function () {
          StackExchange.snippets.init();
          });
          });
          }, "code-snippets");

          StackExchange.ready(function() {
          var channelOptions = {
          tags: "".split(" "),
          id: "196"
          };
          initTagRenderer("".split(" "), "".split(" "), channelOptions);

          StackExchange.using("externalEditor", function() {
          // Have to fire editor after snippets, if snippets enabled
          if (StackExchange.settings.snippets.snippetsEnabled) {
          StackExchange.using("snippets", function() {
          createEditor();
          });
          }
          else {
          createEditor();
          }
          });

          function createEditor() {
          StackExchange.prepareEditor({
          heartbeatType: 'answer',
          autoActivateHeartbeat: false,
          convertImagesToLinks: false,
          noModals: true,
          showLowRepImageUploadWarning: true,
          reputationToPostImages: null,
          bindNavPrevention: true,
          postfix: "",
          imageUploader: {
          brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
          contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
          allowUrls: true
          },
          onDemand: true,
          discardSelector: ".discard-answer"
          ,immediatelyShowMarkdownHelp:true
          });


          }
          });






          Giles is a new contributor. Be nice, and check out our Code of Conduct.










          draft saved

          draft discarded


















          StackExchange.ready(
          function () {
          StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f215769%2fswift-lazy-observe-and-mutate-pattern-for-arrays%23new-answer', 'question_page');
          }
          );

          Post as a guest















          Required, but never shown

























          0






          active

          oldest

          votes








          0






          active

          oldest

          votes









          active

          oldest

          votes






          active

          oldest

          votes








          Giles is a new contributor. Be nice, and check out our Code of Conduct.










          draft saved

          draft discarded


















          Giles is a new contributor. Be nice, and check out our Code of Conduct.













          Giles is a new contributor. Be nice, and check out our Code of Conduct.












          Giles is a new contributor. Be nice, and check out our Code of Conduct.
















          Thanks for contributing an answer to Code Review Stack Exchange!


          • Please be sure to answer the question. Provide details and share your research!

          But avoid



          • Asking for help, clarification, or responding to other answers.

          • Making statements based on opinion; back them up with references or personal experience.


          Use MathJax to format equations. MathJax reference.


          To learn more, see our tips on writing great answers.




          draft saved


          draft discarded














          StackExchange.ready(
          function () {
          StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f215769%2fswift-lazy-observe-and-mutate-pattern-for-arrays%23new-answer', 'question_page');
          }
          );

          Post as a guest















          Required, but never shown





















































          Required, but never shown














          Required, but never shown












          Required, but never shown







          Required, but never shown

































          Required, but never shown














          Required, but never shown












          Required, but never shown







          Required, but never shown







          Popular posts from this blog

          Сан-Квентин

          8-я гвардейская общевойсковая армия

          Алькесар