web-dev-qa-db-fra.com

Comment créer un nouvel objet de type 'T' dans une classe générique en typographie?

J'ai le service suivant

@Injectable()
export class CollectionService<T> {    

    constructor(protected http: Http) {}

    factory<T>(item?: any): T {
        let type: new (item?: any) => T;
        return new type(item);
    }

    item(id): Observable<T> {
        return this.http.get(`${this.baseUrl}/${id}`)
            .map((resp: Response)=> this.factory(resp.json()))
            .catch((error: any) => {
                return Observable.throw(error);
            });
    }
}

La compilation passe avec succès, mais j'ai une erreur suivante dans la console du navigateur

TypeError: type is not a constructor
at PortalVideoService.webpackJsonp.../../../../../src/app/services/collection/collection.service.ts.CollectionService.factory (http://localhost:4200/main.bundle.js:1568:16)
at http://localhost:4200/main.bundle.js:1580:64
at Array.map (native)
at MapSubscriber.project (http://localhost:4200/main.bundle.js:1580:29)
at MapSubscriber.webpackJsonp.../../../../rxjs/operator/map.js.MapSubscriber._next (http://localhost:4200/vendor.bundle.js:24034:35)
at MapSubscriber.webpackJsonp.../../../../rxjs/Subscriber.js.Subscriber.next (http://localhost:4200/vendor.bundle.js:13455:18)
at XMLHttpRequest.onLoad (http://localhost:4200/vendor.bundle.js:101703:38)
at ZoneDelegate.webpackJsonp.../../../../zone.js/dist/zone.js.ZoneDelegate.invokeTask (http://localhost:4200/polyfills.bundle.js:2838:31)
at Object.onInvokeTask (http://localhost:4200/vendor.bundle.js:93420:37)
at ZoneDelegate.webpackJsonp.../../../../zone.js/dist/zone.js.ZoneDelegate.invokeTask (http://localhost:4200/polyfills.bundle.js:2837:36)

Comment créer un objet de type T?

5
Andrey K

Le nom T n'existe qu'au moment de la compilation. Vous pouvez l'utiliser pour la vérification de type, mais vous ne pouvez pas construire un objet de type T si vous n'avez pas accès au constructeur.

Modifiez la définition de factory pour prendre le constructeur de type à l'exécution en tant qu'argument:

factory<T>(type: {new(): T}, item?: any): T {
    return new type(item);
}

Maintenant, le seul problème sera de trouver l’argument de type; vous devrez probablement aussi le transmettre à la méthode item() pour que le compilateur sache quel type créer le Observable<T>.

6
Duncan

Il n'y a pas besoin de faire ce que vous faites, faites simplement

.map((resp: Response)=> resp.json())

et si vous voulez un type personnalisé pour la réponse do

.map((resp: Response)=> resp.json() as CustomResponse) 

CustomResponse est une interface.

1
Chrillewoodz

Vous pouvez passer le constructeur en tant que classe générique, au lieu de méthode.

export class CollectionService<T, CT extends { new(item?: any): T }> {    

    constructor(protected http: Http, private type: CT) {}

    factory(item?: any): T {
        return new this.type(item);
    }

    // ...
}

class CustomClass {
    constructor(private item?: any) {
    }
 };


let cs = new CollectionService<CustomClass, { new(): CustomClass }>(http, CustomClass);
console.dir(cs.factory(123));

// UPDATE
class CustomClassService extends CollectionService<CustomClass, { new(): CustomClass }> {
    constructor(protected http: Http) {
        super(http, CustomClass);
    }
}

let ccs = new CustomClassService(http);
console.dir(ccs.factory(456));
0
Rodris