Как вызвать rest api при загрузке angular 2 приложения

Мы хотели бы вызвать rest api, когда приложение angular 2 загружается, я имею в виду, что первое, что он должен делать с приложением, - это вызвать этот api и получить некоторые данные, необходимые для приложения.

В любом случае, для этого? Я видел следующую статью, но предназначен для бета-версии angular 2

Пример кода на основе angular 2 beta или rc

Чтение данных до запуска приложения

Ответ 1

Вы можете использовать APP_INITIALIZER для вызова метода службы при загрузке. Вам потребуется определить provider для него в AppModule.

Вот пример того, как это сделать.

StartupService (startup.service.ts)

import { Injectable } from '@angular/core';
import { Http, Response } from '@angular/http';

import 'rxjs/add/operator/map';
import 'rxjs/add/operator/toPromise';

@Injectable()
export class StartupService {

    private _startupData: any;

    constructor(private http: Http) { }

    // This is the method you want to call at bootstrap
    // Important: It should return a Promise
    load(): Promise<any> {

        this._startupData = null;

        return this.http
            .get('REST_API_URL')
            .map((res: Response) => res.json())
            .toPromise()
            .then((data: any) => this._startupData = data)
            .catch((err: any) => Promise.resolve());
    }

    get startupData(): any {
        return this._startupData;
    }
}

AppModule (app.module.ts)

import { BrowserModule } from '@angular/platform-browser';
import { NgModule, APP_INITIALIZER } from '@angular/core';

import { StartupService } from './startup.service';

// ...
// Other imports that you may require
// ...


export function startupServiceFactory(startupService: StartupService): Function {
    return () => startupService.load();
}

@NgModule({
    declarations: [
        AppComponent,
        // ...
        // Other components & directives
    ],
    imports: [
        BrowserModule,
        // ..
        // Other modules
    ],
    providers: [
        StartupService,
        {
            // Provider for APP_INITIALIZER
            provide: APP_INITIALIZER,
            useFactory: startupServiceFactory,
            deps: [StartupService],
            multi: true
        }
    ],
    bootstrap: [AppComponent]
})
export class AppModule { }

РЕДАКТИРОВАТЬ (как справиться с сбоем службы запуска):

AppComponent (app.component.ts)

import { Component, OnInit } from '@angular/core';
import { Router } from '@angular/router';

import { StartupService } from './startup.service';

@Component({
    selector: 'app-root',
    templateUrl: './app.component.html',
    styleUrls: ['./app.component.scss']
})
export class AppComponent implements OnInit {

    constructor(private router: Router, private startup: StartupService ) { }

    ngOnInit() {

        // If there is no startup data received (maybe an error!)
        // navigate to error route
        if (!this.startup.startupData) {
            this.router.navigate(['error'], { replaceUrl: true });
        }
    }

}

Ответ 2

Одна вещь, с которой я столкнулся, используя ответ, предоставленный Santanu, заключается в том, что он не работает, если StartupService не определен в провайдерах до APP_INITIALIZER. Используя Angular 4.x, поэтому часть поставщиков выглядит так в моем случае:

  providers: [
    UserService,
  {
    provide: APP_INITIALIZER,
    useFactory: userServiceFactory,
    deps: [UserService],
    multi: true
  }

Ответ 3

Вот статья, которая объясняет APP_INITIALIZER гораздо более подробно, а также показывает, как ее угловое использование.