详解Typescript中奇怪的赋值操作

 更新时间:2024年02月19日 15:45:09   作者:不如吃茶去  
这篇文章主要来和大家讨论一下typescript中一些奇怪的赋值语句,探索其背后原因,更深入的了解typescript作为一个结构化系统的特性,感兴趣的可以了解下

前言

本文主要讨论在typescript中一些奇怪的赋值语句,探索其背后原因,更深入的了解typescript作为一个结构化系统的特性。

我们先看这样一个例子:

// 定义一个Cat类
class Cat {
  name!: string
  age!: number
​
  constructor(name: string, age: number) {
    this.name = name
    this.age = age
  }
}
// 定义一个Dog接口类型
interface Dog {
  name: string
  age: number
}
​
let dog: Dog = {
  name: 'wuyue',
  age: 3
}
​
let cat: Cat
cat = new Cat('qiqi', 2)
cat = dog // is ok 
dog = cat // is ok 
cat = { name: 'jack', age: 12 } // is ok
  • 在上面的例子中,我们定义了一个Cat类,一个Dog接口类型, Cat类的实例对象与Dog类型的变量相互赋值,可以通过Typescript的类型检查,而将{ name: 'jack', age: 12 }分配给一个Cat类型的变量也是可以的。
  • 这是结构化类型系统的特性,Typescript就采用了这种结构类型。结构化类型系统的思想在于,名称不重要,重要的是它们是否具有类型的相同成员,如果是则是兼容的。 结构化类型系统也叫鸭子类型, 如果你看到一只鸟走起来像鸭子,游泳像鸭子,叫得也像鸭子,那么这只鸟就是鸭子
  • 上个例子中cat变量,dog变量,以及{ name: 'jack', age: 12 } 字面量他们都具有相同的结构,所以Typescript在类型检查时认为他们之间的赋值操作是合法的。

我们再来看另一个例子

// 定义一个Cat类
class Cat {
  name!: string
  age!: number
​
  constructor(name: string, age: number) {
    this.name = name
    this.age = age
  }
}
// 定义一个Dog接口类型
interface Dog {
  name: string
  age: number
}
let dog: Dog = {
  name: 'wuyue',
  age: 3
}
​
function getCat(cat: Cat) {}
​
getCat(new Cat('cc', 2)) // ok
getCat(dog) // ok
getCat({ name: 'jack', age: 12 }) // ok
const obj = { name: 'jack', age: 12, sex: 1 } // ok
getCat(obj) // ? ok

在这个例子中,我们定义的Cat类和Dog接口类型和上面的例子一致,紧接着我们定义了一个getCat的函数,接受一个Cat类型的形参。

根据上一个例子中我们学到的结构化类型系统的知识,可以很快的判断出来 new Cat('cc', 2)dog{ name: 'jack', age: 12 } 都可以分配给Cat类型的变量,那么最后一个可以吗? 我们在 typescript Playground中测试发现是可以的,typescript对上面的代码都通过了类型检查,但是你可能还有困惑,obj明显多了一个sex属性,而sex属性在Cat类型中并不存在,Typescript为什么也对其通过了类型检查?先放下这个疑问,我们在看这么一行代码

getCat({ name: 'jack', age: 12, sex: 1 }) 

这行代码竟然报错了,事情变得更加奇怪了。

再来看一个例子:

是不是很奇怪,同样的值,直接赋值/传参会报错,如果先定义一个变量然后将这个变量进行赋值/传参就不会报错。

通过查阅各种官方资料发现这么一个结论: 在typescript类型系统中,对对象进行类型检查时定义的方式会对结果产生影响,在创建一个对象字面量并直接分配给具有某个类型的变量时,typescript会最严谨的验证对象, 进行严格的属性检查(ECP),而当将对象变量分配给另一个变量,对这个变量进行类型检查时,typescript会进行兼容性判断,如果兼容则通过检查,如果不兼容则报错。

而这个兼容性判断是如何判断的呢? 请记住下面这句话:

假如x要分配给y,在TypeScript中,X需要更具体,即X要有和Y相同的属性或者更多。 可分配性指的就是兼容性。

如: let v = { name: 'john'; age: 20 } 兼容{ name: string }, 类型兼容,通过检查。let v = {foo: 1, bar: 2} 兼容{foo: number} , 类型兼容,通过检查。而这个问题可以被看成是类型加宽。

对于上面的现象,有好处也有坏处,对此有人提出来typescript新加入一个精确类型的语法,引起了激烈的讨论:可以看下面这个链接。 #12936

结尾

在网上大多数的案例中,总是会出现这样的例子:

interface Person {
  name: string
  age: number
}

let p: Person = {
  name: 'jack',
  age: 12,
}

我们似乎认为p变量一定要精确的满足于Person的定义,如果多加了属性那么就会报错,这是typescript的基石,ts就应该做这样的事,其实不然,typescript作为一个结构化系统,多加了属性并不会报错, “多加了属性那么就会报错” 的功能其实是在typescript中是一种linter校验的功能,而且它仅适用于对象字面量。所以有时候我们也会说Typescript是Javascript一个超级强大的linter工具

到此这篇关于详解Typescript中奇怪的赋值操作的文章就介绍到这了,更多相关Typescript赋值操作内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 前端开发之CSS原理详解

    前端开发之CSS原理详解

    这篇文章主要介绍了前端开发之CSS原理详解的相关资料,需要的朋友可以参考下
    2017-03-03
  • 使用JavaScript平移和缩放图像的示例代码

    使用JavaScript平移和缩放图像的示例代码

    平移和缩放是查看图像时常用的功能,我们可以放大图像以查看更多细节,进行图像编辑,Dynamsoft Document Viewer是一个用于此目的的SDK,它为文档图像提供了一组查看器,在本文中,我们将演示如何使用它来平移和缩放图像,需要的朋友可以参考下
    2024-08-08
  • 用js判断页面刷新或关闭的方法(onbeforeunload与onunload事件)

    用js判断页面刷新或关闭的方法(onbeforeunload与onunload事件)

    Onunload,onbeforeunload都是在刷新或关闭时调用,可以在<script>脚本中通过window.onunload来指定或者在<body>里指定
    2012-06-06
  • JavaScript模拟实现"双11"限时秒杀效果

    JavaScript模拟实现"双11"限时秒杀效果

    每年的“双11”啊,都是大家的剁手节。大家都在晚上12点,捧着手机看着倒计时,在他倒数到0的时候疯狂点击下单。可是你有没想过限时秒杀是怎么实现的呢?本文将为你揭秘如何用JavaScript实现限时秒杀,快来了解一下吧
    2022-03-03
  • 使用JavaScript + HTML5 Canvas实现互动爱心雨完整代码

    使用JavaScript + HTML5 Canvas实现互动爱心雨完整代码

    canvas是HTML5中推出的新功能,可以在页面上生成一个画布,使用js可以在画布上绘制一些图形,这篇文章主要介绍了使用JavaScript + HTML5 Canvas实现互动爱心雨的相关资料,需要的朋友可以参考下
    2025-03-03
  • JS使用setInterval实现的简单计时器功能示例

    JS使用setInterval实现的简单计时器功能示例

    这篇文章主要介绍了JS使用setInterval实现的简单计时器功能,涉及javascript基于setInterval的定时触发与数值运算相关操作技巧,需要的朋友可以参考下
    2018-04-04
  • JavaScript解决Joseph问题

    JavaScript解决Joseph问题

    JavaScript解决Joseph问题...
    2007-02-02
  • JavaScript防抖案例讲解

    JavaScript防抖案例讲解

    这篇文章主要介绍了JavaScript防抖案例讲解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • Code Review 方法论与实践总结梳理

    Code Review 方法论与实践总结梳理

    这篇文章主要为大家介绍了Code Review 方法论与实践总结梳理详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-02-02
  • ES6顶层对象、global对象实例分析

    ES6顶层对象、global对象实例分析

    这篇文章主要介绍了ES6顶层对象、global对象,结合实例形式分析了ES6顶层对象与global对象的概念、原理、用法及相关操作注意事项,需要的朋友可以参考下
    2019-06-06

最新评论