首页 文章

如何在订阅中返回Observable

提问于
浏览
0

我想在一个observable中调用http请求,该observable从数据库进行select操作 . 我提供了两项服务, DbServiceBackendService .

BackendService发出http post请求并返回响应数据 . 在我的设计中,BackendService应该订阅DbService以获取url,之后发出http post请求然后返回响应数据 .

BackendService可以从DbService获取url并尝试发出http请求但不能 . 响应数据是(Json格式)

{"_isScalar":false,"source":{"_isScalar":false},"operator":{}}

我不明白这里发生了什么 . 我的服务和AppComponent文件如下 .

There is BackendService

import { Injectable } from "@angular/core";
import { getString, setString } from "application-settings";
import { Headers, Http, Response, RequestOptions } from "@angular/http";
import { Observable } from "rxjs/Observable";
import 'rxjs/add/observable/of';
import "rxjs/add/operator/do";
import "rxjs/add/operator/map";
import "rxjs/add/observable/throw";
import "rxjs/add/operator/catch";
import { DbService } from "./db.service";

@Injectable()
export class BackendService {
    static BaseUrl= "http://blabla.com"

    constructor(public http: Http, private db: DbService) {
    }

        sendPost(key: string, requestObj: Object):Observable<any>{
        console.log("sendPost: ");
        return new Observable(obs=> {
            let obs1 = this.db.getActionUrl(key);
            obs1.subscribe(value => {
                let url = BackendService.BaseUrl + value;

                console.log("key: ", key);
                console.log("url: ", url);
                var h = BackendService.getHeaders();
                obs.next(this.http.post(
                    url,
                    JSON.stringify(requestObj),
                    { headers: h }
                ).map((res: Response) => res.json()));
                // .catch((error: any) => Observable.throw(error.json())));
                obs.complete();
            }
            , error => {
                console.error("send post error: "+ error);
                obs.error(error);
            }
        );
        });
    }

    static getHeaders() {
        let headers = new Headers();
        headers.append("Content-Type", "application/json");
        headers.append("SESSION-ID", this.sessionId);
        // headers.append("Authorization", BackendService.appUserHeader);
        return headers;
    }
}

There is DbService

import { Injectable } from "@angular/core";
import { Observable } from "rxjs/Observable";
import 'rxjs/add/observable/of';
import "rxjs/add/operator/do";
import "rxjs/add/operator/map";
import "rxjs/add/observable/throw";
import "rxjs/add/operator/catch";
import 'rxjs/add/operator/toPromise';

var Sqlite = require("nativescript-sqlite");

@Injectable()
export class DbService {
    private tableActions = "actions";

    private columnActionName = "name";
    private columnActionUrl = "url";


    private database: any;
    constructor() {
        console.log("DbService Constructor");
        (new Sqlite("my_app.db")).then(db => {
            db.execSQL("CREATE TABLE IF NOT EXISTS " + this.tableActions + " (" + this.columnActionName + " TEXT PRIMARY KEY, " + this.columnActionUrl +" TEXT)").then(id => {
                this.database = db;
                console.log("DB SERVICE READY");
            }, error => {
                console.log("CREATE TABLE ERROR", error);
            });
        }, error => {
            console.log("OPEN DB ERROR", error);
        });
    }


    public getActionUrl(key: string):Observable<any>{
    return new Observable(observer => { 
        if (key === "getApiMap") {
            observer.next("/getApiMap");
            observer.complete();
            return;
        }
        console.log("getActionUrl :" + key);
        this.database.all("SELECT * FROM " + this.tableActions).then(
            rows => {
                console.log(rows);
                observer.next(rows[0][this.columnActionUrl]);
                observer.complete();
            }, error => {
                console.log("SELECT ERROR: getActionUrl: ", error);
                observer.error(error);
            })
    });
    }
}

And there is my AppComponent which makes http requests...

//some imports

export class AppComponent {
    public resp: Observable<ModelMainGetApiMapRes>;
    public constructor(private bs: BackendService, private db: DbService) {
let req = new ModelMainGetApiMapReq()
    bs.sendPost("getApiMap", req, false).subscribe(
        (res: ModelMainGetApiMapRes) => {
            console.log("getapimap response received!");
            console.log(JSON.stringify(res));
            console.log("apimap version:" + res.api_version);

        },
        err => {
             console.error("error!", err);
        }
    );
 }

//some functions
}

app.component的控制台输出是

CONSOLE LOG file:///app/shared/backend.service.js:61:20: sendPost:
CONSOLE LOG file:///app/shared/backend.service.js:66:28: key:  getApiMap
CONSOLE LOG file:///app/shared/backend.service.js:67:28: url:  http://blabla.com/getApiMap
CONSOLE LOG file:///app/app.component.js:55:36: getapimap response received!
CONSOLE LOG file:///app/app.component.js:56:36: {"_isScalar":false,"source":{"_isScalar":false},"operator":{}}
CONSOLE LOG file:///app/tns_modules/tns-core-modules/profiling/profiling.js:10:16: ANGULAR BOOTSTRAP DONE. 7805.849
CONSOLE ERROR file:///app/tns_modules/@angular/core/bundles/core.umd.js:1486:24: ERROR Error: Uncaught (in promise): TypeError: undefined is not an object (evaluating 'res.api_version')

1 回答

  • 2

    使用 BackendService.ts 中的实际代码:

    return new Observable(obs=> {
        let obs1 = this.db.getActionUrl(key);
        obs1.subscribe(value => {
            let url = BackendService.BaseUrl + value;
    
            console.log("key: ", key);
            console.log("url: ", url);
            var h = BackendService.getHeaders();
            obs.next(this.http.post(
                url,
                JSON.stringify(requestObj),
                { headers: h }
            ).map((res: Response) => res.json()));
             obs.complete();
            ...
        });
     });
    

    你发出了http observable

    this.http.post(
                        url,
                        JSON.stringify(requestObj),
                        { headers: h }
                    ).map((res: Response) => res.json())
    

    这就是为什么你得到: {"_isScalar":false,"source":{"_isScalar":false},"operator":{}} 订阅它时,它是一个可观察的 .

    The simplest solution with your code, 您可以在订阅第二个observable后发出数据,例如:

    return new Observable(obs=> {
        let obs1 = this.db.getActionUrl(key);
        obs1.subscribe(value => {
            let url = BackendService.BaseUrl + value;
            console.log("key: ", key);
            console.log("url: ", url);
            var h = BackendService.getHeaders();
            this.http.post(
                url,
                JSON.stringify(requestObj),
                { headers: h }
            ).map((res: Response) => res.json())
             .subscribe(data => obs.next(data));
        });
    });
    

    But the better solution 是通过使用switchMap运算符:(或任何其他xxxxMap运算符)

    import 'rxjs/add/operator/switchMap';
    import 'rxjs/add/operator/map';
    ...
    sendPost(key: string, requestObj: Object):Observable<any>{
        return this.db.getActionUrl(key)
                .map( value => BackendService.BaseUrl + value)
                .switchMap(url => this.http.post(
                    url,
                    JSON.stringify(requestObj),
                    { headers: h }
                )
                .map((res: Response) => res.json()))
    }
    

相关问题