http.request ()에서 올바르게 예외를 잡는 방법은 무엇입니까?


132

내 코드의 일부 :

import {Injectable} from 'angular2/core';
import {Http, Headers, Request, Response} from 'angular2/http';
import {Observable} from 'rxjs/Observable';
import 'rxjs/add/operator/map';

@Injectable()
export class myClass {

  constructor(protected http: Http) {}

  public myMethod() {
    let request = new Request({
      method: "GET",
      url: "http://my_url"
    });

    return this.http.request(request)
      .map(res => res.json())
      .catch(this.handleError); // Trouble line. 
                                // Without this line code works perfectly.
  }

  public handleError(error: Response) {
    console.error(error);
    return Observable.throw(error.json().error || 'Server error');
  }

}

myMethod() 브라우저 콘솔에서 예외를 생성합니다.

원래 예외 : TypeError : this.http.request (...). map (...). catch는 함수가 아닙니다.

답변:


213

아마도 수입품에 이것을 추가 할 수 있습니다 :

import 'rxjs/add/operator/catch';

당신은 또한 할 수 있습니다 :

return this.http.request(request)
  .map(res => res.json())
  .subscribe(
    data => console.log(data),
    err => console.log(err),
    () => console.log('yay')
  );

댓글 당 :

예외 : TypeError : Observable_1.Observable.throw는 함수가 아닙니다.

마찬가지로이를 위해 다음을 사용할 수 있습니다.

import 'rxjs/add/observable/throw';

2
도움을 주셔서 감사합니다. 작동합니다. 그 후 나는 throw()기능에 동일한 문제가 있습니다. import 'rxjs/Rx';대신 이 줄을 추가했습니다 . 이제 모든 운영자가 올바르게 작동합니다.
mnv

.catch실제로 작동하는지 확인하기 위해 오류를 시뮬레이션 했습니까 ? 그것은 .subscribe() 확실히 작동합니다.
acdcjunior

1
네, 두 번째 문제는 EXCEPTION: TypeError: Observable_1.Observable.throw is not a function입니다. 위에서 언급했듯이 @MattScarpino 답변으로 고정 되거나이 플런저에서 구할 수 있습니다. angular.io/resources/live-examples/server-communication/ts/…
mnv

16
가져 오기도 너무 많이 import 'rxjs/add/observable/throw';가져 와서 모든 것을 가져 오지 마십시오. 너무 큽니다.
dfsq

훌륭한 해결책, 매우 유용합니다. (err)은 Response
Mohammed Suez

77

HttpClientModule 및 RxJS v5.5.x 를 사용하도록 새 서비스가 업데이트되었습니다 .

import { Injectable }                    from '@angular/core';
import { HttpClient, HttpErrorResponse } from '@angular/common/http';
import { Observable }                    from 'rxjs/Observable';
import { catchError, tap }               from 'rxjs/operators';
import { SomeClassOrInterface}           from './interfaces';
import 'rxjs/add/observable/throw';

@Injectable() 
export class MyService {
    url = 'http://my_url';
    constructor(private _http:HttpClient) {}
    private handleError(operation: String) {
        return (err: any) => {
            let errMsg = `error in ${operation}() retrieving ${this.url}`;
            console.log(`${errMsg}:`, err)
            if(err instanceof HttpErrorResponse) {
                // you could extract more info about the error if you want, e.g.:
                console.log(`status: ${err.status}, ${err.statusText}`);
                // errMsg = ...
            }
            return Observable.throw(errMsg);
        }
    }
    // public API
    public getData() : Observable<SomeClassOrInterface> {
        // HttpClient.get() returns the body of the response as an untyped JSON object.
        // We specify the type as SomeClassOrInterfaceto get a typed result.
        return this._http.get<SomeClassOrInterface>(this.url)
            .pipe(
                tap(data => console.log('server data:', data)), 
                catchError(this.handleError('getData'))
            );
    }

더 이상 사용되지 않는 HttpModule을 사용하는 이전 서비스 :

import {Injectable}              from 'angular2/core';
import {Http, Response, Request} from 'angular2/http';
import {Observable}              from 'rxjs/Observable';
import 'rxjs/add/observable/throw';
//import 'rxjs/Rx';  // use this line if you want to be lazy, otherwise:
import 'rxjs/add/operator/map';
import 'rxjs/add/operator/do';  // debug
import 'rxjs/add/operator/catch';

@Injectable()
export class MyService {
    constructor(private _http:Http) {}
    private _serverError(err: any) {
        console.log('sever error:', err);  // debug
        if(err instanceof Response) {
          return Observable.throw(err.json().error || 'backend server error');
          // if you're using lite-server, use the following line
          // instead of the line above:
          //return Observable.throw(err.text() || 'backend server error');
        }
        return Observable.throw(err || 'backend server error');
    }
    private _request = new Request({
        method: "GET",
        // change url to "./data/data.junk" to generate an error
        url: "./data/data.json"
    });
    // public API
    public getData() {
        return this._http.request(this._request)
          // modify file data.json to contain invalid JSON to have .json() raise an error
          .map(res => res.json())  // could raise an error if invalid JSON
          .do(data => console.log('server data:', data))  // debug
          .catch(this._serverError);
    }
}

디버깅에 .do()( now.tap() )를 사용 합니다.

서버 오류가 발생하면 사용중인 서버 body에서 얻은 Response객체 (lite-server)에 텍스트 만 포함되므로 err.text()위의 것이 아닌 위 의 이유를 사용 err.json().error합니다. 서버의 해당 라인을 조정해야 할 수도 있습니다.

경우 res.json()인상이 오류는 JSON 데이터를 분석 할 수 있기 때문에, _serverError받지 않습니다 Response오브젝트의에 따라서 이유를 instanceof확인합니다.

이러한면에서 plunker, 변경 url하는 ./data/data.junk오류를 생성 할 수 있습니다.


두 서비스 중 하나에 오류를 처리 할 수있는 코드가 있어야합니다.

@Component({
    selector: 'my-app',
    template: '<div>{{data}}</div> 
       <div>{{errorMsg}}</div>`
})
export class AppComponent {
    errorMsg: string;
    constructor(private _myService: MyService ) {}
    ngOnInit() {
        this._myService.getData()
            .subscribe(
                data => this.data = data,
                err  => this.errorMsg = <any>err
            );
    }
}

4

이를 수행하는 몇 가지 방법이 있습니다. 둘 다 매우 간단합니다. 각 예제는 훌륭하게 작동합니다. 프로젝트에 복사하여 테스트 할 수 있습니다.

첫 번째 방법이 바람직하고 두 번째 방법은 약간 구식이지만 지금까지도 작동합니다.

1) 해결책 1

// File - app.module.ts
import { BrowserModule } from '@angular/platform-browser';
import { NgModule } from '@angular/core';
import { HttpClientModule } from '@angular/common/http';

import { AppComponent } from './app.component';
import { ProductService } from './product.service';
import { ProductModule } from './product.module';

@NgModule({
  declarations: [
    AppComponent
  ],
  imports: [
    BrowserModule,
    HttpClientModule
  ],
  providers: [ProductService, ProductModule],
  bootstrap: [AppComponent]
})
export class AppModule { }



// File - product.service.ts
import { Injectable } from '@angular/core';
import { HttpClient } from '@angular/common/http';

// Importing rxjs
import 'rxjs/Rx';
import { Observable } from 'rxjs/Rx';
import { catchError, tap } from 'rxjs/operators'; // Important! Be sure to connect operators

// There may be your any object. For example, we will have a product object
import { ProductModule } from './product.module';

@Injectable()
export class ProductService{
    // Initialize the properties.
    constructor(private http: HttpClient, private product: ProductModule){}

    // If there are no errors, then the object will be returned with the product data.
    // And if there are errors, we will get into catchError and catch them.
    getProducts(): Observable<ProductModule[]>{
        const url = 'YOUR URL HERE';
        return this.http.get<ProductModule[]>(url).pipe(
            tap((data: any) => {
                console.log(data);
            }),
            catchError((err) => {
                throw 'Error in source. Details: ' + err; // Use console.log(err) for detail
            })
        );
    }
}

2) 해결 방법 2. 그것은 오래되었지만 여전히 작동합니다.

// File - app.module.ts
import { BrowserModule } from '@angular/platform-browser';
import { NgModule } from '@angular/core';
import { HttpModule } from '@angular/http';

import { AppComponent } from './app.component';
import { ProductService } from './product.service';
import { ProductModule } from './product.module';

@NgModule({
  declarations: [
    AppComponent
  ],
  imports: [
    BrowserModule,
    HttpModule
  ],
  providers: [ProductService, ProductModule],
  bootstrap: [AppComponent]
})
export class AppModule { }



// File - product.service.ts
import { Injectable } from '@angular/core';
import { Http, Response } from '@angular/http';

// Importing rxjs
import 'rxjs/Rx';
import { Observable } from 'rxjs/Rx';

@Injectable()
export class ProductService{
    // Initialize the properties.
    constructor(private http: Http){}

    // If there are no errors, then the object will be returned with the product data.
    // And if there are errors, we will to into catch section and catch error.
    getProducts(){
        const url = '';
        return this.http.get(url).map(
            (response: Response) => {
                const data = response.json();
                console.log(data);
                return data;
            }
        ).catch(
            (error: Response) => {
                console.log(error);
                return Observable.throw(error);
            }
        );
    }
}

-1

RxJS 함수는 특별히 가져와야합니다. 이 작업을 수행하는 쉬운 방법은 모든 기능을import * as Rx from "rxjs/Rx"

그런 다음로 Observable클래스 에 액세스하십시오 Rx.Observable.


15
Rxjs는 매우 큰 파일입니다. 만약 모든 것을 가져 오면 로딩 시간이 길어질 것입니다
Soumya Gangamwar

하나 또는 두 개의 연산자 만 필요한 경우 Rxjs에서 모든 것을 가져 와서는 안됩니다.
marcel-k

-4

최신 버전의 angular4에서

import { Observable } from 'rxjs/Rx'

필요한 모든 것을 가져옵니다.


20
이 작업을 수행하지 마십시오. 모든 Rxjs를 가져옵니다.
marcel-k

따라서 번들 크기가 증가합니다!
Tushar Walzade
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.