Angular4记账webApp练手项目之五(Angular4项目中创建service(服务)和使用http模块)

Angular4记账webApp练手项目之一(利用angular-cli构建Angular4.X项目)
Angular4记账webApp练手项目之二(在angular4项目中使用Angular WeUI)
Angular4记账webApp练手项目之三(在angular4项目中使用路由router)
Angular4记账webApp练手项目之四(在Angular4项目中用echarts绘制图表)
Angular4记账webApp练手项目之五(Angular4项目中创建service(服务)和使用http模块)
前台源码

后台源码
说明:后台代码是用asp.net编写的,和http://www.jianshu.com/p/e6ed43227840这篇文章很像。其中还包含了其他一些练手的东西。

前言

之前写了那么多,不过都是静态页面。现在使用http模块与后端通信,变可以让我们的应用活起来。
我把后台服务写成了可跨域请求的webapi,这样在node上面调试起来就方便多了。

创建服务模块

ng g service account

ng给我们创建的模块account.service.ts,内容如下。
有关@Injectable和@Component,都是angular中的关键字或者关键注解。通过注解来表明js文件的类型,以方便angular框架进行调用。
@Component表示该js文件所导出的类是组件。
@Injectable表示该js文件所导出的文件是服务,而服务是可以通过注入来创建的。
服务的注入,是angular中用来剥离controller和业务逻辑的方式。

import { Injectable } from '@angular/core';

@Injectable()
export class AccountService {

  constructor() { }

}

添加一个方法

  getBillTypes() {
    console.log('这是service里的方法');
  }

引用服务

在accounting.component.ts里引用

import {AccountService} from '../account.service';
@NgModule({
  providers: [
    AccountService
  ],
})

推荐使用依赖注入的方式

  constructor(private service: AccountService) {
      service.getBillTypes(); // 调用方法
  }

查看下效果,提示错误。

Unhandled Promise rejection: No provider for AccountService! ; Zone: angular ; Task: Promise.then ; Value:

原来是在app.module.ts 里面也要添加引用

import {AccountService} from './account.service';
  providers: [AccountService],

这下就完成了简单了例子。ng的编程风格越来越像我们使用的c#,java等的编程风格。当然编程思想也是越来越和我们后台开发相似了。


这里写图片描述

整理下我们的后台接口

添加一个Model文件夹,在下面添加一个model.url.ts文件来存储我们的接口信息


const host = 'http://127.0.0.1:8001';

export const Urls= {
  GetBillTypes: host + '/api/bill/getbilltypes', // 获取记账类型
  GetBills: host +  '/api/bill/GetBills', // 获取列表
  GetCount: host +  '/api/bill/GetCount', // 获取统计信息
  GetTotallCount: host +  '/api/bill/GetTotallCount', // 获取求和数据
  AddBills: host +  '/api/bill/AddBills', // 添加记账信息
  DeleteBill: host +  '/api/bill/DeleteBill', // 删除记账信息
};

在我们的service中引入

import {Urls} from './Model/model.url';

整理方法

export class AccountService {
  private urls = Urls;
  constructor() { }
  getBillTypes(): void {
    console.log(this.urls.GetBillTypes);
  }
  GetBills(): void {
    console.log(this.urls.GetBills);
  }
  GetCount(): void {
    console.log(this.urls.GetCount);
  }
  GetTotallCount(): void {
    console.log(this.urls.GetTotallCount);
  }
  AddBills(): void {
    console.log(this.urls.AddBills);
  }
  DeleteBill(): void {
    console.log(this.urls.DeleteBill);
  }
}

使用http模块

在我们的app.module.ts中已经引入了

import { HttpModule } from '@angular/http';

我们要在account.service.ts中引入

import { Http } from '@angular/http';
import 'rxjs/add/operator/toPromise';

构造函数中注入依赖

  constructor(private http: Http) { }

修改getBillTypes方法试试,看请求返回数据和http.get返回的是什么。

  getBillTypes() {
    console.log(this.urls.GetBillTypes);
    const data = this.http.get(this.urls.GetBillTypes)
      .toPromise()
      .then(response => console.log(response));
    console.log(data);
  }

http.get(url)(或者post put delete),访问服务器以后会返回一个observation对象,事实上是observation<服务器返回值>。通过toPromise转换成promise对象以后,就可以正常的使用then方法去处理返回值了。
通过promise的then方法,可以获得到服务器的返回值。个返回值都是json字符串,而在angular还是先按字符串处理。调用字符串的.json()方法转化为json数组或者json对象,继续调用关键字as将json数组或者json对象转化类,转化的方式是属性对应。


这里写图片描述

因此我们修改方法,在model文件夹下添加自定义的Result类型,

// 接口返回数据格式
export class Result {
  error: any; // 错误时返回的信息
  result: any; // 成功时返回的数据
  success: boolean; // 是否成功
}

在account.service.ts中引入并修改方法

import {Result} from './Model/model.result';
  getBillTypes(): Promise<Result> { // 获取记账类型
    return this.http.get(this.urls.GetBillTypes)
      .toPromise()
      .then(response =>  response.json() as Result)
      .catch(this.handleError);
  }

在accounting.component.ts中修改调用的方法

  constructor(private service: AccountService) {
      service.getBillTypes().then(r => { console.log(r); });
  }

这正是我们后台返回的数据且是json格式的。


这里写图片描述

这里我们用到了自定义类型Result的作用呢,看控制台打印的数据,对数据没什么影响,但是对我写代码是有帮助的。看下面:


这里写图片描述

对,会提示,如果使用了类型里没有的字段,还会报错。这活生生把一个弱类型语言变成了强类型的。当然如果不喜欢,我们可以不用自定义类。把自定义的Result换成any即可。
这里写图片描述

完善service

添加三个自定义类型

// 记账类型的数据结构
export class BillType {
  name: string;
  fontStyle: string;
  id: number;
}

// 记账的数据结构
export class Bill {
  id: number;
  creationTime: string;
  money: number;
  name: string;
  fontStyle: string;
  BillTypeId: number;
}

要细分就太多了,大致分成这几类吧,引入并完善我们的方法

export class AccountService {
  private urls = Urls;
  constructor(private http: Http) { }
  getBillTypes(): Promise<Result> { // 获取记账类型
    return this.get(this.urls.GetBillTypes);
  }
  GetBills(date, skipCount, user): Promise<Result> {
    const d = new URLSearchParams();
    d.set('date', date);
    d.set('skipCount', skipCount);
    d.set('user', user);
    return this.get(this.urls.GetBills, d);
  }
  GetCount(date: string, type: number, user: string, GroupBy = 0): Promise<Result> {
    const d = new URLSearchParams();
    d.set('date', date);
    d.set('type', type.toString());
    d.set('user', user);
    d.set('GroupBy', GroupBy.toString());
    return this.get(this.urls.GetCount, d);
  }
  GetTotallCount(user): Promise<Result> {
    return this.get(this.urls.GetTotallCount + '?user=' + user);
  }
  AddBills(data): Promise<Result> {
      return this.post(this.urls.AddBill, data);
  }
  DeleteBill(data: number): Promise<Result>  {
    return this.post(this.urls.DeleteBill, data);
  }
  // 对get请求进行封装
 private  get(url: string, data: URLSearchParams = null): Promise<Result>  {
    return this.http.get(url, { search: data} )
    .toPromise().then(r => r.json() as Result)
    .catch(this.handleError);
}
  // 对post请求进行封装
  private  post(url: string, data: any): Promise<Result>  {
    return this.http.post(url, data)
      .toPromise().then(r => r.json() as Result)
      .catch(this.handleError);
  }
  // 捕获异常并输出
  private handleError(error: any): Promise<any> {
    console.error('An error occurred', error);
    return Promise.reject(error.message || error);
  }
}

最后完善修结果如下:

这里写图片描述
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352