console.log 的非易失性替代品
non-volatile substitute for console.log
我发现自己经常被 console.log()
绊倒,因为它是异步的。
我的意思是它倾向于不捕获特定时间点的变量值。
它适用于简单的值,因为对具有立即值的变量的赋值会将变量绑定到一个全新的对象,例如
var x = 3;
console.log(x); //prints 3
x=4;
但是,当您开始使用通过引用绑定的对象时,事情会变得……违反直觉。例如
var obj = {x: 3};
console.log(x); //prints 4!
obj.x = 4;
是否有其他一些我可以使用的日志记录功能,可以根据我的代码在调用时为我提供对象的状态?我正在寻找同步的或至少看起来来自它产生的结果的东西。
如果它能跨平台工作我会很高兴,但我很乐意得到一个在 chrome 中工作的平台。
应要求,总结一下:
对于简单的可序列化对象,console.log(JSON.stringify(x))
效果很好:
var x = { foo: 17, bar: "quux" };
console.log(JSON.stringify(x))
// => {"foo":17,"bar":"quux"}
对于 HTML 个元素,console.log(x.outerHTML)
效果很好。
var x = document.getElementsByTagName('p');
console.log(Array.prototype.map.call(x, function(e) {
return e.outerHTML;
}));
// => ["<p>foo</p>", "<p id="bar">bar</p>"]
<p>foo</p>
<p id="bar">bar</p>
如果对象不可序列化,您可能希望深入研究它们并提取可序列化的部分:
var x = { content: { foo: 17, bar: "quux" } };
x.self = x;
console.log(JSON.stringify(x.content));
// => {"foo":17,"bar":"quux"}
如果 none 这些技巧适用,您可能需要深度克隆对象(this answer 提供了一个非常好的 clone
函数,我在下面使用了它,还有许多注意事项) :
function clone(item) {
if (!item) { return item; } // null, undefined values check
var types = [ Number, String, Boolean ],
result;
// normalizing primitives if someone did new String('aaa'), or new Number('444');
types.forEach(function(type) {
if (item instanceof type) {
result = type( item );
}
});
if (typeof result == "undefined") {
if (Object.prototype.toString.call( item ) === "[object Array]") {
result = [];
item.forEach(function(child, index, array) {
result[index] = clone( child );
});
} else if (typeof item == "object") {
// testing that this is DOM
if (item.nodeType && typeof item.cloneNode == "function") {
var result = item.cloneNode( true );
} else if (!item.prototype) { // check that this is a literal
if (item instanceof Date) {
result = new Date(item);
} else {
// it is an object literal
result = {};
for (var i in item) {
result[i] = clone( item[i] );
}
}
} else {
// depending what you would like here,
// just keep the reference, or create new object
if (false && item.constructor) {
// would not advice to do that, reason? Read below
result = new item.constructor();
} else {
result = item;
}
}
} else {
result = item;
}
}
return result;
}
var el = document.querySelector('p');
x = { el: el, content: { foo: 17, bar: "quux" } };
console.log("el.attributes[0]: changed", x);
console.log("el.attributes: empty", clone(x));
el.setAttribute('changed', 'true');
<p>foo</p>
在最坏的情况下,你总是可以暂停执行:
var el = document.querySelector('p');
for (var i = 0; i < 1000; i++) {
el.textContent = "Iteration #" + i;
// check 458th iteration:
if (i == 458) {
console.log("Accurate:", el);
debugger;
console.log("Not accurate:", el);
}
}
<p></p>
使用 JSON
或深度复制进行序列化的另一种可能性是使用 Node 的 inspect
for serialising. I have made a port (inspect-x
),它 90% 的功能都可以在浏览器上运行(并非所有功能都可以在浏览器中实现)。
var inspect = returnExports;
var x = {
foo: 17,
bar: 'quux',
arr: [1,'2', undefined, null, false],
fum: function blah() {},
fee: new ArrayBuffer(4),
woo: new Date(),
wee: /match/gi,
wiz: 1,
poo: true,
pee: Object('hi'),
jqu: $(document.body),
ppp: document.getElementsByTagName('pre')
};
document.getElementById('out').appendChild(document.createTextNode(inspect(x, {showHidden: true})));
console.log(inspect(x, {showHidden: true}));
<script src="https://cdnjs.cloudflare.com/ajax/libs/es5-shim/4.3.1/es5-shim.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/es5-shim/4.3.1/es5-sham.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/json3/3.3.2/json3.min.js"></script>
<script src="https://rawgit.com/Xotic750/inspect-x/master/lib/inspect-x.min.js"></script>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js"></script>
<pre id="out"></pre>
您可以与 JSON.stringify
进行比较
var inspect = returnExports;
var x = {
foo: 17,
bar: 'quux',
arr: [1,'2', undefined, null, false],
fum: function blah() {},
fee: new ArrayBuffer(4),
woo: new Date(),
wee: /match/gi,
wiz: 1,
poo: true,
pee: Object('hi'),
jqu: $(document.body),
ppp: document.getElementsByTagName('pre')
};
document.getElementById('out').appendChild(document.createTextNode(JSON.stringify(x, null, 2)));
console.log(JSON.stringify(x, null, 2));
<script src="https://cdnjs.cloudflare.com/ajax/libs/es5-shim/4.3.1/es5-shim.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/es5-shim/4.3.1/es5-sham.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/json3/3.3.2/json3.min.js"></script>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js"></script>
<pre id="out"></pre>
如您所见,inspect
提供了更多信息,并且可以配置为更深的深度。
我发现自己经常被 console.log()
绊倒,因为它是异步的。
我的意思是它倾向于不捕获特定时间点的变量值。
它适用于简单的值,因为对具有立即值的变量的赋值会将变量绑定到一个全新的对象,例如
var x = 3;
console.log(x); //prints 3
x=4;
但是,当您开始使用通过引用绑定的对象时,事情会变得……违反直觉。例如
var obj = {x: 3};
console.log(x); //prints 4!
obj.x = 4;
是否有其他一些我可以使用的日志记录功能,可以根据我的代码在调用时为我提供对象的状态?我正在寻找同步的或至少看起来来自它产生的结果的东西。
如果它能跨平台工作我会很高兴,但我很乐意得到一个在 chrome 中工作的平台。
应要求,总结一下:
对于简单的可序列化对象,console.log(JSON.stringify(x))
效果很好:
var x = { foo: 17, bar: "quux" };
console.log(JSON.stringify(x))
// => {"foo":17,"bar":"quux"}
对于 HTML 个元素,console.log(x.outerHTML)
效果很好。
var x = document.getElementsByTagName('p');
console.log(Array.prototype.map.call(x, function(e) {
return e.outerHTML;
}));
// => ["<p>foo</p>", "<p id="bar">bar</p>"]
<p>foo</p>
<p id="bar">bar</p>
如果对象不可序列化,您可能希望深入研究它们并提取可序列化的部分:
var x = { content: { foo: 17, bar: "quux" } };
x.self = x;
console.log(JSON.stringify(x.content));
// => {"foo":17,"bar":"quux"}
如果 none 这些技巧适用,您可能需要深度克隆对象(this answer 提供了一个非常好的 clone
函数,我在下面使用了它,还有许多注意事项) :
function clone(item) {
if (!item) { return item; } // null, undefined values check
var types = [ Number, String, Boolean ],
result;
// normalizing primitives if someone did new String('aaa'), or new Number('444');
types.forEach(function(type) {
if (item instanceof type) {
result = type( item );
}
});
if (typeof result == "undefined") {
if (Object.prototype.toString.call( item ) === "[object Array]") {
result = [];
item.forEach(function(child, index, array) {
result[index] = clone( child );
});
} else if (typeof item == "object") {
// testing that this is DOM
if (item.nodeType && typeof item.cloneNode == "function") {
var result = item.cloneNode( true );
} else if (!item.prototype) { // check that this is a literal
if (item instanceof Date) {
result = new Date(item);
} else {
// it is an object literal
result = {};
for (var i in item) {
result[i] = clone( item[i] );
}
}
} else {
// depending what you would like here,
// just keep the reference, or create new object
if (false && item.constructor) {
// would not advice to do that, reason? Read below
result = new item.constructor();
} else {
result = item;
}
}
} else {
result = item;
}
}
return result;
}
var el = document.querySelector('p');
x = { el: el, content: { foo: 17, bar: "quux" } };
console.log("el.attributes[0]: changed", x);
console.log("el.attributes: empty", clone(x));
el.setAttribute('changed', 'true');
<p>foo</p>
在最坏的情况下,你总是可以暂停执行:
var el = document.querySelector('p');
for (var i = 0; i < 1000; i++) {
el.textContent = "Iteration #" + i;
// check 458th iteration:
if (i == 458) {
console.log("Accurate:", el);
debugger;
console.log("Not accurate:", el);
}
}
<p></p>
使用 JSON
或深度复制进行序列化的另一种可能性是使用 Node 的 inspect
for serialising. I have made a port (inspect-x
),它 90% 的功能都可以在浏览器上运行(并非所有功能都可以在浏览器中实现)。
var inspect = returnExports;
var x = {
foo: 17,
bar: 'quux',
arr: [1,'2', undefined, null, false],
fum: function blah() {},
fee: new ArrayBuffer(4),
woo: new Date(),
wee: /match/gi,
wiz: 1,
poo: true,
pee: Object('hi'),
jqu: $(document.body),
ppp: document.getElementsByTagName('pre')
};
document.getElementById('out').appendChild(document.createTextNode(inspect(x, {showHidden: true})));
console.log(inspect(x, {showHidden: true}));
<script src="https://cdnjs.cloudflare.com/ajax/libs/es5-shim/4.3.1/es5-shim.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/es5-shim/4.3.1/es5-sham.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/json3/3.3.2/json3.min.js"></script>
<script src="https://rawgit.com/Xotic750/inspect-x/master/lib/inspect-x.min.js"></script>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js"></script>
<pre id="out"></pre>
您可以与 JSON.stringify
var inspect = returnExports;
var x = {
foo: 17,
bar: 'quux',
arr: [1,'2', undefined, null, false],
fum: function blah() {},
fee: new ArrayBuffer(4),
woo: new Date(),
wee: /match/gi,
wiz: 1,
poo: true,
pee: Object('hi'),
jqu: $(document.body),
ppp: document.getElementsByTagName('pre')
};
document.getElementById('out').appendChild(document.createTextNode(JSON.stringify(x, null, 2)));
console.log(JSON.stringify(x, null, 2));
<script src="https://cdnjs.cloudflare.com/ajax/libs/es5-shim/4.3.1/es5-shim.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/es5-shim/4.3.1/es5-sham.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/json3/3.3.2/json3.min.js"></script>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js"></script>
<pre id="out"></pre>
如您所见,inspect
提供了更多信息,并且可以配置为更深的深度。