vcs*_*nes 491 typescript
在TypeScript中,我可以将函数的参数声明为Function类型.是否存在一种"类型安全"的方式来解决这个问题?例如,考虑一下:
class Foo {
save(callback: Function) : void {
//Do the save
var result : number = 42; //We get a number from the save operation
//Can I at compile-time ensure the callback accepts a single parameter of type number somehow?
callback(result);
}
}
var foo = new Foo();
var callback = (result: string) : void => {
alert(result);
}
foo.save(callback);
Run Code Online (Sandbox Code Playgroud)
保存回调不是类型安全的,我给它一个回调函数,其中函数的参数是一个字符串,但我传递一个数字,并编译没有错误.我可以在保存类型安全功能时创建结果参数吗?
TL; DR版本:在TypeScript中是否有等效的.NET委托?
Rya*_*ugh 725
当然:
class Foo {
save(callback: (n: number) => any) : void {
callback(42);
}
}
var foo = new Foo();
var strCallback = (result: string) : void => {
alert(result);
}
var numCallback = (result: number) : void => {
alert(result.toString());
}
foo.save(strCallback); // not OK
foo.save(numCallback); // OK
Run Code Online (Sandbox Code Playgroud)
如果需要,可以定义一个类型来封装它:
type NumberCallback = (n: number) => any;
class Foo {
// Equivalent
save(callback: NumberCallback) : void {
callback(42);
}
}
Run Code Online (Sandbox Code Playgroud)
Dre*_*kes 87
以下是一些常见.NET委托的TypeScript等价物:
interface Action<T>
{
(item: T): void;
}
interface Func<T,TResult>
{
(item: T): TResult;
}
Run Code Online (Sandbox Code Playgroud)
kbp*_*ius 16
我意识到这篇文章已经过时了,但是有一个更紧凑的方法,与所提出的方法略有不同,但可能是一个非常有用的选择.实际上,您可以在调用方法时在线声明函数(在本例中为Foo
s save()
).它看起来像这样:
class Foo {
save(callback: (n: number) => any) : void {
callback(42)
}
multipleCallbacks(firstCallback: (s: string) => void, secondCallback: (b: boolean) => boolean): void {
firstCallback("hello world")
let result: boolean = secondCallback(true)
console.log("Resulting boolean: " + result)
}
}
var foo = new Foo()
// Single callback example.
// Just like with @RyanCavanaugh's approach, ensure the parameter(s) and return
// types match the declared types above in the `save()` method definition.
foo.save((newNumber: number) => {
console.log("Some number: " + newNumber)
// This is optional, since "any" is the declared return type.
return newNumber
})
// Multiple callbacks example.
// Each call is on a separate line for clarity.
// Note that `firstCallback()` has a void return type, while the second is boolean.
foo.multipleCallbacks(
(s: string) => {
console.log("Some string: " + s)
},
(b: boolean) => {
console.log("Some boolean: " + b)
let result = b && false
return result
}
)
Run Code Online (Sandbox Code Playgroud)
该multipleCallback()
方法对于可能成功或失败的网络调用等非常有用.再次假设一个网络调用示例,当multipleCallbacks()
调用时,可以在一个位置定义成功和失败的行为,这有助于未来代码读取器的更清晰.
一般来说,根据我的经验,这种方法有助于使整体更简洁,更简洁,更清晰.
祝你好运!
Kri*_*wal 13
type FunctionName = (n: inputType) => any;
class ClassName {
save(callback: FunctionName) : void {
callback(data);
}
}
Run Code Online (Sandbox Code Playgroud)
这肯定与函数式编程范例一致.
在 TS 中,我们可以通过以下方式键入函数:
函数类型/签名
这用于函数/方法的实际实现,它具有以下语法:
(arg1: Arg1type, arg2: Arg2type) : ReturnType
Run Code Online (Sandbox Code Playgroud)
例子:
function add(x: number, y: number): number {
return x + y;
}
class Date {
setTime(time: number): number {
// ...
}
}
Run Code Online (Sandbox Code Playgroud)
函数类型文字
函数类型文字是另一种声明函数类型的方法。它们通常应用于高阶函数的函数签名。高阶函数是接受函数作为参数或返回函数的函数。它具有以下语法:
(arg1: Arg1type, arg2: Arg2type) => ReturnType
Run Code Online (Sandbox Code Playgroud)
例子:
type FunctionType1 = (x: string, y: number) => number;
class Foo {
save(callback: (str: string) => void) {
// ...
}
doStuff(callback: FunctionType1) {
// ...
}
}
Run Code Online (Sandbox Code Playgroud)
如果您首先定义函数类型,那么它看起来像
type Callback = (n: number) => void;
class Foo {
save(callback: Callback) : void {
callback(42);
}
}
var foo = new Foo();
var stringCallback = (result: string) : void => {
console.log(result);
}
var numberCallback = (result: number) : void => {
console.log(result);
}
foo.save(stringCallback); //--will be showing error
foo.save(numberCallback);
Run Code Online (Sandbox Code Playgroud)
如果没有使用普通属性语法的函数类型,它将是:
class Foo {
save(callback: (n: number) => void) : void {
callback(42);
}
}
var foo = new Foo();
var stringCallback = (result: string) : void => {
console.log(result);
}
var numberCallback = (result: number) : void => {
console.log(result);
}
foo.save(stringCallback); //--will be showing error
foo.save(numberCallback);
Run Code Online (Sandbox Code Playgroud)
如果您想使用像 c# 泛型委托这样的接口函数,那么:
interface CallBackFunc<T, U>
{
(input:T): U;
};
class Foo {
save(callback: CallBackFunc<number,void>) : void {
callback(42);
}
}
var foo = new Foo();
var stringCallback = (result: string) : void => {
console.log(result);
}
var numberCallback = (result: number) : void => {
console.log(result);
}
let strCBObj:CallBackFunc<string,void> = stringCallback;
let numberCBObj:CallBackFunc<number,void> = numberCallback;
foo.save(strCBObj); //--will be showing error
foo.save(numberCBObj);
Run Code Online (Sandbox Code Playgroud)
因为您无法轻松地将函数定义和另一种数据类型联合起来,所以我发现将这些类型组合起来对于强类型化很有用。基于德鲁的回答。
type Func<TArgs extends any[], TResult> = (...args: TArgs) => TResult;
//Syntax sugar
type Action<TArgs extends any[]> = Func<TArgs, undefined>;
Run Code Online (Sandbox Code Playgroud)
现在您可以强类型化每个参数和返回类型!这是一个比上面更多参数的示例。
save(callback: Func<[string, Object, boolean], number>): number
{
let str = "";
let obj = {};
let bool = true;
let result: number = callback(str, obj, bool);
return result;
}
Run Code Online (Sandbox Code Playgroud)
现在,您可以编写联合类型,例如对象或返回对象的函数,而无需创建可能需要导出或使用的全新类型。
//THIS DOESN'T WORK
let myVar1: boolean | (parameters: object) => boolean;
//This works, but requires a type be defined each time
type myBoolFunc = (parameters: object) => boolean;
let myVar1: boolean | myBoolFunc;
//This works, with a generic type that can be used anywhere
let myVar2: boolean | Func<[object], boolean>;
Run Code Online (Sandbox Code Playgroud)
归档时间: |
|
查看次数: |
236112 次 |
最近记录: |