Safari iOS ReferenceError: Can't find variable: xxx

掘金没有故记录之。

问题

以下代码有什么问题(注意在没有打包工具的环境下)?

js 复制代码
{
  const bar1 = 1

  function foo1() {
    console.log('bar1', bar1)
  }

  foo1();
}

答案:点击展开 Safari 或 iOS 移动端将报错『ReferenceError: Can't find variable: bar1』。其他浏览器正常。

为什么报错?

如果是下面这样倒还能理解,但是我们已经将 bar1 的定义放到了函数调用甚至申明之前为什么还是报错?

js 复制代码
{
  function foo1() {
    console.log('bar1', bar1);
  }

  foo1();

  const bar1 = 1;
}

正式解答: Safari 特有的兼容性问题,非严格模式下,各个浏览器实现各自不同,因为标准并没有规定(different browsers implemented it differently in non-strict mode)。

Safari 实现:

大括号生成块级作用域,bar1 变成局部变量,而在非严格模式下函数 foo1 是全局函数,故执行时找不到局部变量 bar1。相当于将函数 foo1 挪到了顶部。等价于:

js 复制代码
function foo2() {
  console.log('bar2', bar2);
}

{
  const bar2 = 1;

  foo2();
}

自然报错。

Chrome 实现

等价于:

js 复制代码
var foo1; // 意味着 globalThis.foo1 = undefined;

{
  const bar1 = 1

  foo1 = function () {
    console.log('bar1', bar1)
  }
  
  // 意味着 globalThis.foo1 = foo1

  foo1();
}

你可以尝试在 Safari、Chrome、Node.js 运行下面代码,验证我们的猜想是否正确:

js 复制代码
console.log('above scope foo1:', foo1, typeof foo1, 'foo1' in globalThis);

{
  const bar1 = 1

  function foo1() {
    console.log('bar1', bar1)
  }

  foo1();
}

console.log('below scope foo1:', foo1, typeof foo1, 'foo1' in globalThis);

Safari

javascript 复制代码
above scope foo1: 
function foo1() { console.log('bar1', bar1); }
"function"
true

[Error] ReferenceError: Can't find variable: bar1

Chrome

javascript 复制代码
above scope foo1: undefined undefined true
bar1 1
below scope foo1: ƒ foo1() {
    console.log('bar1', bar1);
  } function true
bar1 1

Node.js

❯ node reference-error-ios.js

javascript 复制代码
above scope foo1: undefined undefined false
bar1 1
below scope foo1: [Function: foo1] function false
bar1 1

但是有个怪异的地方,Node.js 和 Chrome 应该要一样,但是实际行为有少许不同:

'foo1' in globalThis Chrome 是 true,Node.js 是 false(各位有知道答案的吗?)改成 foo1 in globalThis 倒是变成 true 了,但是不对呀 in 的操作符应该是字符串。

验证字符串才是正确的 Chrome 验证:

如何解决

既然是作用域导致的报错,可以从作用域解决。

方法 1:去除大括号。让报错的变量也变成全局作用域

javascript 复制代码
const bar1 = 1

function foo1() {
console.log('bar1', bar1)
}

foo1();

但显然我们本意是利用 {} 大括号的块级作用域让变量不泄露到全局,故该方案并不是最佳的。

方法 2:将 const bar1 = 1 挪到括号外面的顶部。

和方法 1 同理,但好在保留了其他变量不被泄露,也不是最佳解决方案。

方法 3:使用 var

和方法 2 同理。

方法 4:IIFE

利用 IIFE 构造一个函数作用域,能保证变量不泄露。

js 复制代码
(function () {
  const bar1 = 1;

  function foo1() {
    console.log('bar1', bar1);
  }

  foo1();
})();

在 ES6 之前我们的最佳实践确实如此,但写起来太费劲了。

方法 5:use strict

开启严格模式

js 复制代码
'use strict';

{
  const bar1 = 1;

  function foo1() {
    console.log('bar1', bar1);
  }

  foo1();
}

还有一个问题,开启严格模式,是否可以把最外层的 {} 去掉?

答案:不能。如果去除,如果有同名的 bar1 则报错 cannot re-declare bar1

总结

严格模式 + {} 块级作用域才是最佳实践。

最后一个问题,为什么日常工作中,我们即使不写 use strict 也不会遇到这种问题?因为我们有打包工具(webpack)会自动在我们的代码顶部增加 use strict 指令,以及将编译后的代码用 IIFE 打包成一个个 module(即模块化)。

相关推荐
DisonTangor11 小时前
苹果发布iOS 18.2首个公测版:Siri接入ChatGPT、iPhone 16拍照按钮有用了
ios·chatgpt·iphone
- 羊羊不超越 -11 小时前
App渠道来源追踪方案全面分析(iOS/Android/鸿蒙)
android·ios·harmonyos
2401_865854881 天前
iOS应用想要下载到手机上只能苹果签名吗?
后端·ios·iphone
HackerTom2 天前
iOS用rime且导入自制输入方案
ios·iphone·rime
良技漫谈2 天前
Rust移动开发:Rust在iOS端集成使用介绍
后端·程序人生·ios·rust·objective-c·swift
2401_852403552 天前
高效管理iPhone存储:苹果手机怎么删除相似照片
ios·智能手机·iphone
星际码仔2 天前
【动画图解】是怎样的方法,能被称作是 Flutter Widget 系统的核心?
android·flutter·ios
emperinter2 天前
WordCloudStudio:AI生成模版为您的文字云创意赋能 !
图像处理·人工智能·macos·ios·信息可视化·iphone
关键帧Keyframe2 天前
音视频面试题集锦第 8 期
ios·音视频开发·客户端
pb82 天前
引入最新fluwx2.5.4的时候报错
flutter·ios