JS魔法堂:函数重载 之 获取变量的数据类型

简介:

Brief                              

  有时我们需要根据入参的数据类型来决定调用哪个函数实现,就是说所谓的函数重载(function overloading)。因为JS没有内置函数重载的特性,正好给机会我们思考和实现一套这样的机制。

使用方式:

function foo(){
  return dispatch(this, arguments)
}
foo["object,number"] = function(o, n){console.log(o.toString() + ":" + n)}
foo["string"] = function(s){console.log(s)}
foo["array"] = function(a){console.log(a[0])}
机制实现:
;(function(e/*xports*/){
  e.dispatch = function(thisValue, args){
    var rSignature = getSignature(args)
    for (var p in args.callee){
      if (rSignature.test(p)){
        return args.callee[p].apply(thisValue, args)
      }
    }
  }
  function getSignature(args){
    var arg, types = []
    for (var i = 0, len = args.length; i < len; ++i){
      arg = args[i]
      types.push(type(arg))
    }
    var rTypes = "^\\s*" + types.join("\\s*,\\s*") + "\\s*$"
    return RegExp(rTypes)
  }
  function type(val){
    // TODO
  }
}(window))

 那现在问题就落在type函数的实现上了!

 关于获取变量的数据类型有typeof、Object.prototype.toString.call和obj.constructor.name三种方式,下面我们一起来了解一下!

 

typeof Operator                        

  语法: typeof val 

  内部逻辑:

function typeof(val){
  var ret = Type(val)
  if (ret === "Reference" 
      && IsUnresolvableReference(val)) return "undefined"
  ret = GetValue(val)
  ret = Type(ret)
  return ret
}

  Type(val)抽象操作的逻辑:

  Undefined -> "undefined"

  Null -> "object"

  Boolean -> "boolean"

  Number -> "number"

  String -> "string"

  Objecjt,若对象为native object并且没有[[Call]]内置方法,则返回"object"

      若对象为native object或host object且有[[Call]]内置方法,则返回"function"

      若对象为host object并且没有[[Call]]内置方法,则返回除"undefined"、"boolean"、"number"和"string"外的数据类型字符串。

  native object,就是Math、{foo:1}、[]、new Object()和RegExp等JS规范中定义的对象,其中Math、RegExp等程序运行时立即被初始化的对象被称为built-in object。

  host object,就是宿主环境提供的对象,如浏览器的window和nodejs的global。

  从上述Type(val)抽象操作的逻辑得知:

    1. typeof能清晰区分Boolean、Number、String和Function的数据类型;

    2. 对于未声明和变量值为Undefined的变量无法区分,但对未声明的变量执行typeof操作不会报异常;

    3. typeof对Null、数组和对象是无能的。

  针对2、3点我们可以求助于 Object.prototype.toString.call(val) 。

 

Object.prototype.toString.call(val)              

  Object.prototype.toString.call(val)或({}).toString.call(val)均是获取val的内置属性[[Class]]属性值,并经过加工后返回。

  内部逻辑:

function Object.prototype.toString(){
  if (this === undefined) return "[object Undefined]"
  if (this === null) return "[object Null]"
  var o = ToObject(this)
  var clazz = o.[[Class]]
  return "[object " + clazz + "]"
}

  注意:1. 由于内部硬编码null返回"[object Null]",因此虽然null本应不属于Object类型,但JS中我们依然将其当作Object来使用(历史+避免破坏已有库的兼容性,导致后来无法修正该错误了);

           2. 即使入参为primitive value,但内部还是会对其进行装箱操作(通过ToObject抽象操作)。

  那现在我们就需要了解一下[[Class]]内部属性了。

  内部属性[[Class]]

  在构造对象时会根据对象的类型设置[[Class]]的值,而其值类型为字符串。对于native object而言,其值范围是:Arguments
Array、Boolean、Date、Error、Function、JSON、Math、Number、Object、RegExp、String。对于host object而言,则用HTMLElement、HTMLDocument等了。

      注意:[[Class]]是用于内部区分不同类型的对象。也就是仅支持JS语言规范和宿主环境提供的对象类型而已,而自定义的对象类型是无法存储在[[Class]]中。





function Foo(){}
var foo = new Foo()
console.log(({}).toString.call(foo)) // 显示[object Object]

  于是我们需要求助于constructor.name属性了。

 

obj.constructor.name                    




function Foo(){}
var foo = new Foo()
console.log(foo.constructor.name) // 显示Foo

那如果采用匿名函数表达式的方式定义构造函数呢?只能说直接没辙,要不在构造函数上添加个函数属性来保存(如Foo.className="Foo"),要不自己构建一个类工厂搞定。

 

Implementaion of type function              

  综上所述得到如下实现:

/*
 * 获取对象的数据类型
 * @method type
 * @param {Any} object - 获取数据类型的对象
 * @param {Function} [getClass] - 用户自定义获取数据类型的方法
 * @returns {String} 数据类型名称
 */
function type(o/*bject*/, g/*etClass*/){
  var t = typeof o
  if ("object" !== t) return t.replace(/^[a-z]/, function(l){return l.toUpperCase()})

  var rType = /\s*\[\s*object\s*([0-9a-z]+)\s*\]\s*/i
  t = ({}).toString.call(o)
  t = t.match(rType)[1]
  if ("Object" !== t) return t

  t = o.constructor.name
  if (!t && arguments.callee(g) === "Function"){
    t = g(o)
  }
  t = t || "Object"
  return t
}

目录
相关文章
|
4月前
|
存储 JavaScript 对象存储
js检测数据类型有那些方法
js检测数据类型有那些方法
146 59
|
4月前
|
JavaScript 前端开发
JavaScript如何判断变量undefined
JavaScript如何判断变量undefined
|
4月前
|
JavaScript 前端开发
JavaScript基础知识-变量的声明提前
关于JavaScript变量声明提前特性的基础知识介绍。
51 0
JavaScript基础知识-变量的声明提前
|
23天前
|
存储 JavaScript 前端开发
JavaScript中的数据类型以及存储上的差别
通过本文的介绍,希望您能够深入理解JavaScript中的数据类型及其存储差别,并在实际编程中灵活运用这些知识,以提高代码的性能和稳定性。
49 3
|
2月前
|
存储 JavaScript 前端开发
js中的数据类型
JavaScript 中的数据类型包括五种基本类型(String、Number、Undefined、Boolean、Null)和三种引用类型(Object、Array、Function,以及ES6新增的Symbol)。基本类型直接存储值,引用类型存储的是指向实际数据的内存地址。了解它们的区别对于掌握 JavaScript 的变量赋值和函数传参至关重要。
27 1
|
2月前
|
前端开发 JavaScript
如何在 JavaScript 中访问和修改 CSS 变量?
【10月更文挑战第28天】通过以上方法,可以在JavaScript中灵活地访问和修改CSS变量,从而实现根据用户交互、页面状态等动态地改变页面样式,为网页添加更多的交互性和动态效果。在实际应用中,可以根据具体的需求和场景选择合适的方法来操作CSS变量。
|
2月前
|
前端开发 JavaScript 数据处理
CSS 变量的作用域和 JavaScript 变量的作用域有什么不同?
【10月更文挑战第28天】CSS变量和JavaScript变量虽然都有各自的作用域概念,但由于它们所属的语言和应用场景不同,其作用域的定义、范围、覆盖规则以及与其他语言特性的交互方式等方面都存在明显的差异。理解这些差异有助于更好地在Web开发中分别运用它们来实现预期的页面效果和功能逻辑。
|
2月前
|
前端开发 JavaScript UED
如何使用 JavaScript 动态修改 CSS 变量的值?
【10月更文挑战第28天】使用JavaScript动态修改CSS变量的值可以为页面带来更丰富的交互效果和动态样式变化,根据不同的应用场景和需求,可以选择合适的方法来实现CSS变量的动态修改,从而提高页面的灵活性和用户体验。
|
3月前
|
存储 JavaScript 前端开发
JavaScript 数据类型详解:基本类型与引用类型的区别及其检测方法
JavaScript 数据类型分为基本数据类型和引用数据类型。基本数据类型(如 string、number 等)具有不可变性,按值访问,存储在栈内存中。引用数据类型(如 Object、Array 等)存储在堆内存中,按引用访问,值是可变的。本文深入探讨了这两种数据类型的特性、存储方式、以及检测数据类型的两种常用方法——typeof 和 instanceof,帮助开发者更好地理解 JavaScript 内存模型和类型检测机制。
137 0
JavaScript 数据类型详解:基本类型与引用类型的区别及其检测方法
|
4月前
|
存储 前端开发 JavaScript
前端基础(二)_JavaScript变量、JavaScript标识符、JavaScript获取元素、JavaScript的鼠标事件
本文介绍了JavaScript变量的声明和使用、标识符的命名规则、如何获取和操作HTML元素,以及JavaScript的鼠标事件处理,通过示例代码展示了这些基础知识点在实际开发中的应用。
52 2
前端基础(二)_JavaScript变量、JavaScript标识符、JavaScript获取元素、JavaScript的鼠标事件