Swift lazy observe and mutate pattern for arrays
$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)
swift cocoa
New contributor
$endgroup$
add a comment |
$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)
swift cocoa
New contributor
$endgroup$
add a comment |
$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)
swift cocoa
New contributor
$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
swift cocoa
New contributor
New contributor
edited 12 mins ago
Jamal♦
30.4k11121227
30.4k11121227
New contributor
asked 10 hours ago
GilesGiles
101
101
New contributor
New contributor
add a comment |
add a comment |
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.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
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.
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.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
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
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
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