Does a matter whether a value is primitive or boxe

2019-02-08 05:40发布

One can use typeof to determine whether a value is primitive or boxed.

Consider:

typeof "foo"; // "string"
typeof new String("foo"); // "object"

In combination with Object.prototype.toString we could define the following two functions

var toString = Object.prototype.toString;

var is_primitive_string = function(s) {
  return toString.call(s) === "[object String]" && typeof s === "string";
};

var is_boxed_string = function(s) {
  return toString.call(s) === "[object String]" && typeof s === "object";
};

Are there any use cases for these two functions? (Or similar functions for Number, Boolean, etc).

The concept behind this question came from the following Comment by T.J.Crowder.

Should we ever care whether a value we have is primitive or boxed?

3条回答
女痞
2楼-- · 2019-02-08 05:44

I'd say there's virtually no point, you almost never care whether you're dealing with a string primitive or a String object.

There are edge cases. For instance, a String object is an actual object, you can add properties to it. This lets you do things like this:

function test(arg) {
    arg.foo = "bar";
}

If calling code passes in a string primitive:

var s1 = "str";
test(s1);

...arg gets promoted to a String object and gets a property added to it, but that String object isn't used by anything after test returns.

In contrast, if calling code passes in a String object:

var s2 = new String("str");
test(s2);

...then the property is added to that object and the calling code can see it. Consider (live copy):

var s1, s2;

s1 = "str";

display("[Before] typeof s1.foo = " + typeof s1.foo);
test(s1);
display("[After] typeof s1.foo = " + typeof s1.foo);

s2 = new String("str");

display("[Before] typeof s2.foo = " + typeof s2.foo);
test(s2);
display("[After] typeof s2.foo = " + typeof s2.foo);

function test(arg) {
  arg.foo = "bar";
}

Output:

[Before] typeof s1.foo = undefined
[After] typeof s1.foo = undefined
[Before] typeof s2.foo = undefined
[After] typeof s2.foo = string

Note that s2.foo is a string, but s1.foo isn't (because s1 was a string primitive, the object created when we promoted it in test has nothing to do with the calling code).

Is there any use case for this? Dunno. I'd say it would be an extremely edgy edge case if so.

查看更多
狗以群分
3楼-- · 2019-02-08 05:52

I use underscore.js methods to detect the type of the variable. Try to use: isEmpty, isElement, isArray, isArguments, isFunction, isString, isNumber, isBoolean, isDate, isRegExp isNaN, isNull, isUndefined

Described here: http://documentcloud.github.com/underscore/

查看更多
可以哭但决不认输i
4楼-- · 2019-02-08 05:54

All the toString stuff seems to be an attempt to workaround problems with cross-frame mixing of different builtin String constructors. That is unnecessary for checking whether something is a primitive string -- typeof is sufficient, so there is no use case for is_primitive_string.

I very rarely see arguments passed as String instances so I can't see why I would need to check whether something is a String instance cross-frame instead of just coercing to a String value via ("" + s) or String(s). The only time I've ever used a String value in production code was when I needed an empty string that was truthy in some highly optimized code.

As far as the others go, instances of the Boolean class don't behave as one might expect in conditions.

if (new Boolean(false)) {
  alert("WTF!");
} else {
  alert("OK");
}

Boolean.prototype.not = function () { return !this; };

if (new Boolean(false).not()) {
  alert("OK");
} else {
  alert("Really, WTF!");
}

if (false.not()) {  // Autoboxing
  alert("OK");
} else {
  alert("Cmon, WTF!");
}

!(false) is true, but when you use create an instance of the Boolean class, the ! operator applies to the object value, and object values are always truthy.

I believe EcmaScript 5 strict mode is changing the way this is presented so the last example (false.not()) will behave as one might naively expect when "use strict"; is added to the top of Boolean.prototype.not in a valid ES5 interpreter.

With Numbers, comparisons using < are OK and addition and other operators tend to work as expected. new Number(0) and new Number(NaN) have the same problems as new Boolean(false) around conditions, and of course

alert(NaN === NaN);  // false
var NAN = new Number(NaN);
alert(NAN === NAN);  // true

and === and !== compare by reference for all of String, Number, and Boolean.

查看更多
登录 后发表回答