javascript - 深度优先搜索的深拷贝对象
问题描述
我试图复制一个对象。我想用算法深度优先搜索来做到这一点。
function dfs(data) {
let stack = [];
let res = {};
for (let key in data) {
stack.push({ data: data[key], keyData: key });
while (stack.length !== 0) {
const first = stack.shift();
if (typeof first.data === "object") {
for (let key in first.data) {
if (typeof first.data[key] === "object") {
stack.push({ data: first.data[key], keyData: key });
} else {
res[first.parentKey] = first.keyData;
}
}
} else {
res[first.keyData] = first.data;
}
}
}
return res;
}
const data = {
a: 1,
b: 2,
c: {
d: 3,
g: {
e: 4,
r: {
y: 5,
},
},
},
};
const newData = dfs(data);
data.c.g.e = 5000;
data.c.g.d = 90000;
console.log("Original data", data);
console.log("Copied data", newData);
我创建了一个函数,它将复制我的对象,而没有旧对象上的链接。我有一个问题,我的函数没有正确计算结果。我在哪里犯错?
解决方案
深度优先搜索通常使用递归实现:
function dfs(data) {
if (typeof data === 'object') {
const result = {};
for (let key in data) {
result[key] = dfs(data[key]);
}
return result;
}
return data;
}
const data = {
a: 1,
b: 2,
c: {
d: 3,
g: {
e: 4,
r: {
y: 5,
},
},
},
};
const newData = dfs(data);
data.c.g.e = 5000;
data.c.g.d = 90000;
console.log("Original data", data);
console.log("Copied data", newData);
这是一种迭代方法:
function dfs(data) {
const result = {};
const source = [['ROOT', data]];
const target = [result];
while (source.length) {
const [k, v] = source.shift();
const current = target.shift();
const branch = typeof v === 'object';
current[k] = branch ? {} : v;
if (branch) {
const entries = Object.entries(v);
source.unshift(...entries);
target.unshift(...Array(entries.length).fill(current[k]));
}
}
return result.ROOT;
}
const data = {
a: 1,
b: 2,
c: {
d: 3,
g: {
e: 4,
r: {
y: 5,
},
},
},
};
const newData = dfs(data);
data.c.g.e = 5000;
data.c.g.d = 90000;
console.log("Original data", data);
console.log("Copied data", newData);
推荐阅读
- javascript - 在 PHP 中将数组分成三个(在 JS 中可以解决)
- python - AttributeError: 'float' 对象在生成生成器对象时没有属性 'split'
- java - Java 10 var 和捕获变量
- javascript - 我如何使用角度关联两个 excel 记录
- python - 访问 API 时在 Python 中获取请求错误 404
- php - 选中的复选框不会覆盖具有相同名称的隐藏输入
- python - Python解析网页链接计数器
- javascript - babel 7 和函数绑定运算符
- android - 通过蓝牙耳机 VoIP 接听电话
- ios - 如何在 iOS 上的 Swift 中创建类似于 Snapchat 的自定义过渡动画