This is the 4th day of my participation in the August More Text Challenge

preface

This series will be divided into a series of articles that will cover the “cold knowledge” of JS, meaning things that most developers don’t notice, or don’t expect. These knowledge points may not be very helpful or even of any use in the real interview or development, but I hope to make you laugh and expand your knowledge.

Most of the knowledge in this series comes from “Wangdoc.com” written by Yifeng Ruan. It is highly recommended that web front-end developers check out this document as both a starting point and a tool to expand their knowledge.

Any references to this series are indicated at the end of the article.

With statement

I’m sure most of you have never heard of the JS with statement, probably because few people or textbooks mention it, because its use has many and serious drawbacks.

Let’s see what “with” can do.

Here’s what MDN says.

The with statement extends the scope chain of a statement.

After reading this definition, probably no one will understand. For example, it is easy to understand.

Such as:

var obj = { 
  a: 1.print() {
    console.log(The value of 'a 'is:.this.a) 
  }
}
with(obj) {
  a = 2
  print()
}
Copy the code

After the above code runs, it changes the value of obj’s a property to 2 and runs obj’s print method to print the value of a. Obj. A = 2; obj. Print () in the with statement is the same as obj.print()

We can see that the with statement is probably designed to simplify object calls and omit obj. This prefix.

One more chestnut.

with ('abc') {
    console.log(toUpperCase()) // 'ABC'
}
Copy the code

The code above is equivalent to console.log(‘ ABC ‘.touppercase ()), capitalizing ABC and printing it.

Notice that the ‘ABC’ passed in is a string of primitive types, not an object, but the with statement is for objects. Why is a string used here?

This has nothing to do with the with statement. I’ll mention an extra point here: wrapping objects.

I’m sure many of you have written code like this.

var str = 'abc'
if (str.length === 3) {... }Copy the code

But few people have probably questioned why STR, a string of the primitive type, can call the attribute Length like an object.

That’s because the JS engine automatically converts the primitive type to the corresponding wrapper object in some cases. For example, the String STR above is converted to a temporary String object (destroyed immediately), which is equivalent to doing a new String(STR) operation. So the code str.length actually shows up in the engine as new String(STR).length, so the primitive String STR looks like it has an attribute length, but in fact STR doesn’t have any attributes or methods. See the chapter “Wrapping Objects – JavaScript Tutorial – Netpath (wangdoc.com)” for details.

Back to the point, the above code is equivalent to the following code.

with (new String('abc')) {
    console.log(toUpperCase()) // 'ABC'
}
Copy the code

advantages

Here is a direct quote from MDN.

The with statement reduces the length of a variable without incurring a performance penalty. It causes little additional computation. Using ‘with’ reduces unnecessary pointer path parsing. Note that in many cases it is possible to achieve the same effect by not using the with statement but using a temporary variable to hold the pointer.

disadvantages

The with statement seems a little useful, but it has a variety of drawbacks.

Accidentally creating a global variable

Let’s take a look at the following code.

with(obj) {
   a = 2
}
console.log(obj.a) / / output 1
console.log(a) 2 / / output
Copy the code

The results of “output 1” and “output 2” here depend on whether obj has an A attribute.

If obj has an attribute a, then the a attribute of obj is expected to be assigned the value of 2. Output 1 is 2, and output 2 will report an error a is not defined.

If obj does not have property A, we might expect the effect to be obj. A = 2. Obj will assign property A with value 2, but it will not. Instead, it will create a global variable a in the global scope with value 2, that is, The result of “output 1” is undefined, and the result of “output 2” is 2.

This feature contaminates the global scope by accidentally creating global variables.

Semantics is unknown

If we don’t know what properties and methods are in B, how do we know if a is a in B or an entry to function?

function f(a, b) {
  with (b) {
    console.log(a); }}Copy the code

Performance loss

The relevant interpretation of MDN is directly quoted here.

The with statement causes the program to first look up a variable value in the specified object. So variables that aren’t properties of this object are going to be slow to find. If performance is critical, the variables in the statement below ‘with’ should contain only the attributes of the specified object.

conclusion

The with statement is a useless statement and is explicitly not recommended by w3School and MDN. In js’s strict mode, the with statement is even completely disabled.

reference

  • Object – JavaScript Tutorial – Webpath (wangdoc.com)
  • with – JavaScript | MDN (mozilla.org)

“Js unpopular knowledge” series of articles recommended

  • Js Unpopular knowledge – labels
  • Js – null and undefined why both