- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
考虑一下:
type N = never;
type A = 'A';
type B = 'A' | 'B';
type S = string;
type RN = Record<N, string>;
type RA = Record<A, string>;
type RB = Record<B, string>;
type RS = Record<S, string>;
declare let n : N;
declare let a : A;
declare let b : B;
declare let s : S;
s = b;
b = a;
a = n;
declare let rn : RN;
declare let ra : RA;
declare let rb : RB;
declare let rs : RS;
rn = rs;
rs = rn;
rs = ra;
ra = rb;
让<
是子类型运算符。显然,N < A < B < S
因为n
可分配给 a
可分配给 b
可分配给 s
.
所以,我期望RS < RB < RA < RN
.
但是,从示例中您可以看到RB < RA < RS
因为rb
可分配给 ra
可分配给 rs
。此外,RS
和RN
似乎是等效类型。
我假设 string
可以看作是所有 string
的联合类型文字类型。所以实际上RS
应等于 never
因为不可能有一个对象具有所有可能存在的字符串文字的属性(取无限空间)。将此称为完整对象。
但是看起来像RS
实际上相当于空( RN
)且不完整的对象。
为什么是string
表现得像 never
在Record
?
最佳答案
Mapped types像the Record<K, V>
utility type map string
和number
literal各个属性的键,因此 Record<"A" | "B", string>
相当于 {a: string; b: string}
.
但是宽的、非文字类型的键,例如 string
本身,或 number
,或模式模板文字类型,如 `foo${string}`
(如 microsoft/TypeScript#40598 中实现)映射到 index signatures 。来自索引签名的文档:
Sometimes you don’t know all the names of a type’s properties ahead of time, but you do know the shape of the values. In those cases you can use an index signature to describe the types of possible values.
因此索引签名并不真正代表具有相关类型的所有可能的键的“完整对象”,例如无限 intersection所有单键对象的数量 {a: string} & {b: string} & {c: string} & ... & {foo: string} & ... {blahblah: string} & ...
.
(旁白:你说一个完整的对象相当于 never
因为这是不可能的。但这并不准确。一个 Proxy
对象可以很容易地被制作成符合这种类型。即使它在 JavaScript 中是不可能的,你不希望类型系统将其视为 never
,而不具有某种关于无穷大的显式公理,这并不明显,然后你就可以必须弄清楚如何在不禁止递归数据类型的情况下做到这一点。)
无论如何,索引签名更像是属性上的约束。 {[k: IndexType]: ValType}
形式的索引签名表示“如果该对象具有 IndexType
类型的属性键,那么这样的属性将具有 ValType
类型的值”。从某种意义上来说,它更像是所有单键对象与 optional properties 的无限交集。 ,例如{a?: string} & {b?: string} & {c?: string} & ... & {foo?: string} & ... {blahblah?: string} & ...
当然,情况比这更复杂,因为编译器传统上并不以相同的方式对待索引签名和可选属性。
在 TypeScript 4.1 之前,索引签名始终允许您读取属性并获取值,即使我刚刚解释完它们如何更像可选属性。对此有很多提示,因此 TypeScript 4.1 引入了 the --noUncheckedIndexedAccess
compiler flag ,其中添加了 undefined
读取时到索引签名属性值的域,但写入时不到索引签名属性值的域。默认情况下,即使使用--strict
,它也不会启用。 ,因为虽然它的类型更安全,但在人们通过数组或对象进行索引的任何情况下,它都会很烦人......像 for (let i=0; i<arr.length; i++) {arr[i]}
这样的代码或Object.keys(obj).forEach(k => obj[k])
从技术上讲应该显示 arr[i]
和obj[k]
可能 undefined
,至少没有办法跟踪 i
的身份和k
而不仅仅是类型。
在 TypeScript 4.4 之前,可选属性被视为具有 undefined
在阅读和写作时都将其视为其领域的一部分。人们对此也提示很多,因此 TypeScript 4.4 引入了 the --exactOptionalPropertyTypes
compiler flag其中保留了undefined
读取,但拒绝写入 undefined
的属性。这也不包含在 --strict
中,因为类似 foo.bar = foo.bar
如果 bar
现在被视为错误是可选的。
如果您启用这两个编译器标志,则索引签名和可选属性具有类似的行为,尽管我确信存在更多边缘情况。
无论如何... Record<string, string>
相当于 {[k: string]: string}
)同时Record<never, string>
相当于 empty object type {}
。这些不是相同的类型,但由于与 microsoft/TypeScript#7029 中实现的隐式索引签名有关的规则,它们是相互兼容的。 .
那里还有很多东西需要解开,我们可以花很长一段时间来了解weak type detection。 , excess property checking ,以及索引签名和 interface
之间的交互类型(参见 microsoft/TypeScript#15300 )。不过,我现在要停下来,因为这个答案已经足够长了。
关于typescript - 字符串作为变量和映射键类型的行为不同,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/69596074/
请看一下我的代码。 int main () { Program* allcommand = new Program; allcommand->addCommand("add", new
因此,当我遇到调试断言时,我正在编写代码。现在我很想知道为什么这段代码不起作用: for(Model::MeshMap::iterator it = obj1->GetMeshes().begin()
这是我上一个问题的延续 Group, Sum byType then get diff using Java streams . 按照建议,我应该作为单独的线程发布,而不是更新原始线程。 因此,通过我
我正在实现一些非常适合 map 的代码。但是,我要迭代的列表中有大量对象,所以我的问题是哪种方法是解决此问题的最佳方法: var stuff = $.map(listOfMyObjects, some
我正在尝试创建一个包含不同类的成员函数指针的映射。成员函数都具有相同的签名。为了做到这一点,我所有的类都继承了一个 Object 类,它只有默认构造函数、虚拟析构函数和一个虚拟 ToString()
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: how do you make a heterogeneous boost::map? 有可能在 C++ 中
我有一个 Mysql 查询,请检查以下内容: SELECT `tbl_classSubjects`.`classID` , `tbl_classSubjects`.`sectionID` , `tbl
抱歉,这可能是一个基本问题。 JNA直接映射和接口(interface)映射有什么区别? 我的解释是否正确: 直接映射 : 直接使用库对象(如 Java 中的静态 main) 接口(interface
在 Twitter's Scala school collections section ,它们显示了一个带有偏函数作为值的 Map: // timesTwo() was defined earlie
很难说出这里问的是什么。这个问题是模棱两可的、模糊的、不完整的、过于宽泛的或修辞的,无法以目前的形式得到合理的回答。如需帮助澄清这个问题以便重新打开它,visit the help center .
据我了解,从 scala stdlib 声明一个映射并没有将其专门用于原始类型。我要的不是付出装箱/拆箱的代价,而是同时拥有scala map 的接口(interface)。一个明显的选择是使用 tr
如何为这样的 JSON 响应创建对象映射,它只是一个整数数组: [ 565195, 565309, 565261, 565515, 565292, 565281, 566346, 5
是否可以为 DTO 对象创建映射然后查询它们 而不是域?如果不解释为什么? 如果我需要几个 dtos 怎么办? DTos 是只读的 ID 由 NH 自动生成 将来这些 dtos 将设置映射到链接的 d
我有一个返回的函数(常规代码) [words: "one two", row: 23, col: 45] 在 Scala 中,我将上面更改为 Scala Map,但随后我被迫将其声明为 Map[Str
我有一组与 Vanilla 磅蛋糕烘焙相关的数据(200 行),具有 27 个特征,如下所示。标签caketaste是衡量烤蛋糕的好坏程度,由 bad(0) 定义, neutral(1) , good
我有试图映射到新代码的遗留代码。 OLD_PERSON pid sid name age NEW_PERSON pid sid fid age RESOLVE_PERSON pid fid statu
我有一个表,其中一个字段可以指向其他 3 个表之一中的外键,具体取决于鉴别器值是什么(Project、TimeKeep 或 CostCenter。通常这是用子类实现的,我想知道我有什么 注意子类名称与
我有一个类型 [ST s (Int, [Int])] 的绑定(bind)我正在尝试申请runST使用映射到每个元素,如下所示: name :: [ST s (Int, [Int])] --Of Cou
在我正在进行的项目中,我有以下实体:分析师、客户 和承包商。每个都继承自基类 User。 public abstract class User { public virtual int Id
我想知道是否可以在 Vim 中创建一个映射(对于普通模式),允许用户在映射执行之前输入。 我想为我最常用的 grep 命令创建一个快捷方式的映射。我希望命令允许输入我正在搜索的内容,然后在输入时执行。
我是一名优秀的程序员,十分优秀!