admin管理员组文章数量:1341397
As described here: Declare class type with TypeScript (pass class as parameter) i can pass a class type as parameter.
There is a problem:
export namespace customCommand {
export class Command {
public constructor(parameter?: any) {
}
}
export function register(mand: typeof Command) {
new mand()
}
}
When i do this
customCommand.register(Map)
or
customCommand.register(Object)
There is no errors. I know that typeof Command
and typeof Map
both return the same result.
But how can i protect this and pass only Command
type?
As described here: Declare class type with TypeScript (pass class as parameter) i can pass a class type as parameter.
There is a problem:
export namespace customCommand {
export class Command {
public constructor(parameter?: any) {
}
}
export function register(mand: typeof Command) {
new mand()
}
}
When i do this
customCommand.register(Map)
or
customCommand.register(Object)
There is no errors. I know that typeof Command
and typeof Map
both return the same result.
But how can i protect this and pass only Command
type?
-
2
Why does
register
need a parameter if there exists only one value that can be passed? – jabaa Commented Jan 29, 2022 at 14:56 -
Because i can do this:
Class OtherCommand extends Command
– benifest Commented Jan 29, 2022 at 15:03 -
1
Give your
Command
class some methods so that it doesn't have the empty interface (that bothObject
andMap
and really anything else implements). – Bergi Commented Jan 29, 2022 at 15:06 -
It's still weird though that the
Map
constructor can be assigned to{new (p: number): object}
despite accepting an iterable not a number – Bergi Commented Jan 29, 2022 at 15:23
1 Answer
Reset to default 13In TypeScript, you can construct a new type, something like:
type Class<T> = new (...args: any[]) => T
Which you can then use this way:
function register(mand: Class<Command>) {
new mand()
}
This should constrain TypeScript to ONLY accept classes which construct or extend the Command type.
EDIT:
Sorry, missed something when I evaluated your example. Your class Command
is generic enough that almost any constructable object will overlap. What it's missing is constraints.
- The constructor will take
any
first argument, undefined or not, so any class which has a constructor accepting zero or one arguments will match - The signature of the object that will be returned is empty, no methods or properties exist to identify it and so any methods/props will "fall-through" and be allowed as an "extension" of your class
We can over e either of these by modifying either properties or methods required for the Command
class, thus constraining derived types to that explicit signature
If we want to narrow the possible matches by adding a method or property specific to Command
, then we get the desired behavior you've outlined.
export namespace customCommand {
export type Class<T> = new (...args: any[]) => T
export class Command {
public constructor(parameter?: any) {
}
picard (): string {
return 'Make it so.'
}
}
export function register(mand: Class<Command>) {
new mand()
}
}
/*
Argument of type 'MapConstructor' is not assignable to
parameter of type 'Class<Command>'. Property 'picard'
is missing in type 'Map<any, any>' but required in
type 'Command'. ts(2345)
*/
customCommand.register(Map)
/*
Argument of type 'ObjectConstructor' is not assignable to
parameter of type 'Class<Command>'. The 'Object' type is
assignable to very few other types. Did you mean to use
the 'any' type instead? Property 'picard' is missing in
type 'Object' but required in type 'Command'. ts(2345)
*/
customCommand.register(Object)
/* No error */
customCommand.register(customCommand.Command)
本文标签: javascriptHow to pass a class type as parameterStack Overflow
版权声明:本文标题:javascript - How to pass a class type as parameter? - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743673596a2519938.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论