【JavaScript】手写 Object.prototype.toString()

🔧 ​​一、核心实现原理​

  1. ​内部逻辑​

    • 访问对象的 Symbol.toStringTag 属性(ES6+)或内部 [[Class]] 属性(ES5 及之前)。
    • 若属性不存在,根据对象类型推断默认标签(如 "Object""Array")。
    • 返回拼接字符串 "[object " + tag + "]"
  2. ​关键特性​

    • ​不可直接调用​ :需通过 call/apply 绑定 this 指向目标对象。
    • ​优先级​Symbol.toStringTag > 内置 [[Class]] 属性。

✨ ​​二、完整代码实现​

复制代码
Object.defineProperty(Object.prototype, 'toString', {
  value: function() {
    // 1. 处理 null 和 undefined
    if (this === null) return '[object Null]';
    if (this === undefined) return '[object Undefined]';

    // 2. 获取 Symbol.toStringTag 或内部 [[Class]]
    const tag = 
      Symbol.toStringTag in Object(this) 
        ? this[Symbol.toStringTag] 
        : Object.prototype.toString.getInternalClass(this);

    // 3. 返回格式化字符串
    return `[object ${tag}]`;
  },
  writable: true,
  configurable: true
});

// 辅助方法:模拟内部 [[Class]] 检测
Object.prototype.toString.getInternalClass = (obj) => {
  if (Array.isArray(obj)) return 'Array';
  if (obj instanceof Date) return 'Date';
  if (obj instanceof RegExp) return 'RegExp';
  // 其他内置类型判断...
  return 'Object'; // 默认值
};

🧪 ​​三、功能验证​

1. ​​基本类型检测​
复制代码
console.log(Object.prototype.toString.call([])); // [object Array]
console.log(Object.prototype.toString.call(new Date())); // [object Date]
console.log(Object.prototype.toString.call(null)); // [object Null]
2. ​​自定义类型标签​
复制代码
class MyClass {
  get [Symbol.toStringTag]() { return 'MyClass'; }
}
console.log(Object.prototype.toString.call(new MyClass())); // [object MyClass]
3. ​​避免重写影响​
复制代码
const arr = [];
arr.toString = () => "Overridden!";
console.log(Object.prototype.toString.call(arr)); // 仍输出 [object Array]

⚠️ ​​四、注意事项​

  1. ​兼容性处理​

    • 旧版浏览器(如 IE)需手动实现 Symbol.toStringTagPolyfill
    • 内部 [[Class]] 在 ES5 后不再直接暴露,需通过类型推断模拟。
  2. ​性能优化​

    • 避免频繁调用:每次调用涉及原型链查找和字符串拼接,对性能敏感场景慎用。
  3. ​与重写方法的区别​

    • 直接调用 obj.toString() 可能被重写逻辑覆盖,而 Object.prototype.toString.call(obj) 始终访问原始方法。

💡 ​​五、应用场景​

  1. ​精准类型判断​

    替代 typeofinstanceof,统一处理基本类型和对象类型:

    复制代码
    function getType(obj) {
      return Object.prototype.toString.call(obj).slice(8, -1);
    }
    getType(new Map()); // "Map"
  2. ​自定义对象描述​

    通过 Symbol.toStringTag 控制调试信息输出:

    复制代码
    class NetworkError {
      get [Symbol.toStringTag]() { return 'NetworkError'; }
    }
    console.log(new NetworkError() + ""); // [object NetworkError]

相关推荐
一点一木1 天前
深度体验TRAE SOLO移动端7天:作为独立开发者,我把工作流揣进了兜里
前端·人工智能·trae
天外飞雨道沧桑1 天前
TypeScript 中 omit 和 record 用法
前端·javascript·typescript
Lee川1 天前
mini-cursor 揭秘:从 Tool 定义到 Agent 循环的完整实现
前端·人工智能·后端
canonical_entropy1 天前
从 Spec-Driven Development 到 Attractor-Guided Engineering
前端·aigc·ai编程
研☆香1 天前
聊聊前端页面的三种长度单位
前端
给钱,谢谢!1 天前
React + PixiJS 实现果园成长页:从状态机到浇水动画
前端·react.js·前端框架
暗冰ཏོ1 天前
VUE面试题大全
前端·javascript·vue.js·面试
次元工程师!1 天前
LangFlow开发(三)—Bundles组件架构设计(3W+字详细讲解)
java·前端·python·低代码·langflow
Bug-制造者1 天前
现代Web应用全栈开发:从架构设计到部署落地实战
前端
青春喂了后端1 天前
IntelliGit 前端状态层重构:把一个全局 Store 拆成清晰的状态边界
前端·重构·状态模式