Angular5如何使用Http Observable

时间:2018-05-21 23:52:06

标签: angular angular5 angular2-services angular-http angular2-observables

我对Angular5上的HTTP请求有问题。 我有两个组件(" LoginComponent "," WellcomeComponent ")和一个服务(" AuthService &#34)。

我正在使用auth servise来处理HTTP请求,而且这个服务也是两个组件之间的桥梁。

我可以从服务获取和发布请求,但由于Observable事情,无法正确使用结果。我不是一位经验丰富的Angular开发人员,因此无法找到解决方案。

所以现在,我正在逐步解释。我将在此处编写所有代码,所有步骤均由代码中的 .. Step _ 文本签署,因此您可以按照此标记执行所有步骤。

  

LoginComponent

import { Component } from '@angular/core';
import { Router } from '@angular/router';
import { AuthService } from '../../auth.service';

@Component({
  selector: 'app-login',
  templateUrl: './login.component.html'
})
export class LoginComponent {

    loginUserData = {}

    constructor(
        private _auth: AuthService,
        public router:Router ){
            //SOMETHING
        }

    //CLICK FORM SUBMIT BUTTON
    public onSubmit(values:Object):void {

        //SET USER DATA WITH FORM VALUES
        console.log("LoginStep_1 - Hello")

        this._auth.getToken(this.loginUserData)
        // CHECK TOKEN

        // IF VERIFIED
            console.log('LoginStep_2 - Token Verified');

            this._auth.setUserData(this.loginUserData['username']);
            console.log('LoginStep_3 - SetUserData Finished');

            this.router.navigate(['/wellcome'])
            console.log("LoginStep_4 - Navigated");
    }

}
  

AuthService

import { Injectable } from '@angular/core';
import { HttpClient } from '@angular/common/http';
import { Router } from '@angular/router';

@Injectable()
export class AuthService {

    private _tokenUrl = "http://localhost:8000/api/auth/token/";
    private getUserInfoUrl = "http://127.0.0.1:8000/api/userInfo/?username="
    private userInfo : any;

    constructor(
        private http: HttpClient,
        private _router: Router
    ) { }

    getToken(user){
        console.log("getToken_Step1 - Hello");

        this.http.post<any>(this._tokenUrl, user)
        .subscribe(
            res => {

                //getToken_Step2 : SET TOKEN SOMEWHERE
                console.log("getToken_Step2 - Token : ", res.token);

            },
            err => console.log(err)
        );

        console.log("getToken_Step3 - Bye");
    }

    setUserData(userName){
        console.log("setUserData_Step1 - Hello");

        this.http.get<any>(this.getUserInfoUrl+userName)
        .subscribe(
            res=>{

                //setUserData_Step2 : SET USERINFO SOMEWHERE
                console.log("setUserData_Step2 - User Info : ", res);

            },
            err=>{
                console.log(err);
            }
        )

        console.log("setUserData_Step3 - Bye");
    }

    getUserData(){
        console.log("getUserData_Step1 - Hello");
        return this.userInfo;
    }

}
  

WellcomeComponent

import { Component, OnInit } from '@angular/core';
import { AuthService } from '../../../auth.service';

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

    constructor(private _auth: AuthService){ }

    ngOnInit() {
        console.log("Wellcome_Step1 - Hello");
        let userInfo = this._auth.getUserData();
        console.log("Wellcome_Step2 - Show User Info");
    }

}

如果你看代码,有很多 console.log(&#34; Step .. ,我正在解释我想用它们做什么。

简而言之,我可以说我想在控制台屏幕上按此顺序查看它们:

  1. LoginStep_1 - 您好:此步骤可以点击表单上的提交按钮
  2. getToken_Step1 - 您好:转到授权服务
  3. getToken_Step2 - 令牌:从API获取令牌并将其写为localStorage
  4. getToken_Step3 - 再见
  5. LoginStep_2 - 令牌验证:然后返回并验证此令牌。
  6. setUserData_Step1 - Hello
  7. setUserData_Step2 - 用户信息:从API获取用户信息并将其写为localStorage
  8. setUserData_Step3 - 再见
  9. LoginStep_3 - SetUserData完成:我得到了令牌,验证了它,获得了用户信息并写了它。所以我现在可以去导航了。
  10. Wellcome_Step1 - 您好:从登录导航
  11. getUserData_Step1 - Hello:获取LoginComponent编写的所有信息
  12. Wellcome_Step2 - 显示用户信息:使用用户信息执行您想要的操作
  13. 这些步骤是我想在控制台日志屏幕上看到的,但我无法通过正确的顺序看到它们。我的控制台输出与我的预期非常不同。因为我不知道如何使用angular 5 observable,所以无法处理来自API的返回值。

    我的控制台输出类似的内容:

      
        
    1. LoginStep_1- Hello
    2.   
    3. LoginStep_3 - SetUserData已完成
    4.   
    5. getToken_Step1 - Hello
    6.   
    7. getToken_Step2 - 令牌
    8.   

    ... 在这,为什么这个订单?我该如何纠正?

    现在,我该怎么做?我想通过LoginComponent获取令牌和用户信息,并从WellCome组件中使用它们。

    正如我所说,我不是一位经验丰富的编码员,所以也许解决方案非常简单但无法找到。

    感谢您的帮助。

2 个答案:

答案 0 :(得分:0)

因为您对后端的请求是异步的,所以执行它需要时间。因此,Angular将在console.log('LoginStep_2 - Token Verified');console.log('LoginStep_3 - SetUserData Finished');完成之前继续执行其他任务:getToken()setUserData()

您可以移动订阅组件

在您的AuthService

    getToken(user){
        console.log("getToken_Step1 - Hello");

        return this.http.post<any>(this._tokenUrl, user)
        // subscribe moves to component
    }

在您的组件中

this._auth.getToken(this.loginUserData)
  .subscribe(
    res => {

      //getToken_Step2 : SET TOKEN SOMEWHERE
      console.log("getToken_Step2 - Token : ", res.token);
    },
    err => console.log(err)
  )

答案 1 :(得分:0)

基本上,您应该实现函数调用以返回promiseobservable(也称为异步)。 AuthService就像这样(改编自你的问题):

import {Injectable} from '@angular/core';
import {HttpClient, HttpHeaders} from '@angular/common/http';

import {Observable, of} from 'rxjs';

@Injectable()
export class AuthService {
  constructor(private http: HttpClient) {}

  // private _tokenUrl = "http://localhost:8000/api/auth/token/";

  // This is just a mock URL that returns a response (200 OK)
  private _tokenUrl =
      "https://bikewise.org:443/api/v2/incidents?page=1&proximity_square=100";
  // private getUserInfoUrl = "http://127.0.0.1:8000/api/userInfo/?username=";
  private userInfo: any;

  // can improve with getter/setter instead of exposing directly
  private isLoggedIn: boolean = false;

  isLoggedInObservable: Observable<boolean>;

  getToken(user): Promise<any> {
    console.log("getToken_Step1 - Hello");

    return new Promise(resolve => {

      // this.http.post<any>(this._tokenUrl, user)
      // fake request to make sure always get a response
      this.http.get<any>(this._tokenUrl)
          .subscribe(
              res => {

                // getToken_Step2 : SET TOKEN SOMEWHERE
                console.log("getToken_Step2 - Token : ", "token_1234");

                console.log("getToken_Step3 - Bye");
                // resolve(res.token);
                resolve("token1234");

              },
              err => {
                console.log(err);

                throw new Error('Unauthorized...');

              });
    })
  }

  setUserData(userName) {
    console.log("setUserData_Step1 - Hello");

    // If this is a real request, make sure to implement a promise/observable
    // return
    // this.http.get<any>(this.getUserInfoUrl + userName)

    // setUserData_Step2 : SET USERINFO SOMEWHERE
    console.log("setUserData_Step2 - User Info : ");

    console.log("setUserData_Step3 - Bye");
    this.isLoggedIn = true;
    this.isLoggedInObservable = of(true);
  }

  getUserData() {
    console.log("getUserData_Step1 - Hello");
    return this.userInfo;
  }

  logOut() {
    this.isLoggedInObservable = of(false);
    this.isLoggedIn = false;
  }

  isLogged(): boolean { return this.isLoggedIn; }
}

您还需要为路由设置AuthGuard以跟踪用户是否已登录。

这是一个工作示例stackblitz打开控制台以跟踪状态

此外,您始终可以在浏览器localStorage中缓存您的用户登录状态,以获得更好的用户体验。同样,这是根据您的代码改编的。您的代码可以改进很多东西。如果您有其他问题,请与我们联系。