Являются ли строго типизированные функции возможными параметрами в 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);

Функция обратного вызова save не является безопасной для типов, я даю ей функцию обратного вызова, где параметр функции является строкой, но я передаю ей число и компилирую без ошибок. Можно ли сделать параметр результата в функции сохранения типа безопасной?

TL; версия DR: есть ли в TypeScript эквивалент делегата .NET?

Ответ 1

Конечно:

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

Если вы хотите, вы можете определить тип для инкапсуляции:

type NumberCallback = (n: number) => any;

class Foo {
    // Equivalent
    save(callback: NumberCallback) : void {
        callback(42);
    }
}

Ответ 2

Вот TypeScript эквиваленты некоторых общих делегатов .NET:

interface Action<T>
{
    (item: T): void;
}

interface Func<T,TResult>
{
    (item: T): TResult;
}

Ответ 3

Я понимаю, что этот пост старый, но есть более компактный подход, который немного отличается от того, что было задано, но может быть очень полезной альтернативой. Вы можете по существу объявить функцию в строке при вызове метода (Foo 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 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
    }
)

Подход multipleCallback() очень полезен для таких вещей, как сетевые вызовы, которые могут преуспеть или потерпеть неудачу. Опять же, предполагая пример сетевого вызова, когда вызывается multipleCallbacks(), поведение для успеха и неудачи может быть определено в одном месте, что придает большую ясность будущим читателям кода.

В целом, по моему опыту, этот подход позволяет быть более кратким, менее беспорядочным и большей ясностью в целом.

Удачи всем!

Ответ 4

type FunctionName = (n: inputType) => any;

class ClassName {
    save(callback: FunctionName) : void {
        callback(data);
    }
}

Это, безусловно, согласуется с парадигмой функционального программирования.

Ответ 5

В TS мы можем вводить функции следующими способами:

Типы функций/подписи

Используется для реальных реализаций функций/методов, имеет следующий синтаксис:

(arg1: Arg1type, arg2: Arg2type) : ReturnType

Пример:

function add(x: number, y: number): number {
    return x + y;
}

class Date {
  setTime(time: number): number {
   // ...
  }

}

Тип функции Литералы

Литералы типа функции - это еще один способ объявить тип функции. Они обычно применяются в сигнатуре функции высшего порядка. Функция более высокого порядка - это функция, которая принимает функции в качестве параметров или возвращает функцию. Он имеет следующий синтаксис:

(arg1: Arg1type, arg2: Arg2type) => ReturnType

Пример:

type FunctionType1 = (x: string, y: number) => number;

class Foo {
    save(callback: (str: string) => void) {
       // ...
    }

    doStuff(callback: FunctionType1) {
       // ...
    }

}

Ответ 6

Если вы сначала определите тип функции, то это будет выглядеть так

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);

Без типа функции с использованием синтаксиса простого свойства это будет:

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);

Если вы хотите использовать интерфейсную функцию, такую как обобщенные делегаты С#, это будет:

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);

Ответ 7

Помимо того, что сказали другие, общая проблема заключается в объявлении типов той же функции, которая перегружена. Типичным случаем является метод EventEmitter on(), который будет принимать несколько типов слушателей. Подобное может произойти при работе с редуксными действиями - и там вы используете тип действия в качестве литерала для обозначения перегрузки. В случае EventEmitters вы используете имя события  буквенный тип:

interface MyEmitter extends EventEmitter {
  on(name:'click', l: ClickListener):void
  on(name:'move', l: MoveListener):void
  on(name:'die', l: DieListener):void
  //and a generic one
  on(name:string, l:(...a:any[])=>any):void
}

type ClickListener = (e:ClickEvent)=>void
type MoveListener = (e:MoveEvent)=>void
... etc

// will type check the correct listener when writing something like:
myEmitter.on('click', e=>...<--- autocompletion