angular学习笔记
栏目: JavaScript · 发布时间: 6年前
# 使用 yarnpkg 只需要设置一次 ng config -g cli.packageManager yarn # 新建项目 有路由, 无测试, scss ng new $project --routing -S --style=scss # 添加 material yarnpkg add @angular/material @angular/cdk @angular/animations # 生成 component ng g c $name
模块
@NgModule({ declarations: [ 组件, 指令, 管道 ], imports: [ 依赖模块 ], providers: [ 服务 ], bootstrap: [ 主组件 ] })
组件
路由
app-routing.module.ts
const routes: Routes = [ // 路由重定向 { path: '', redirectTo: '/home', pathMath: 'full', }, // 基本路由配置 { path: 'home', component: HomeComponent, data: [ { isProd: true, }, ], }, { path: 'children', // 子路由 children: [ { path: '', component: Children1Component, }, { path: 'xxx/:id', component: Children2Component, }, ], }, // 默认路由 { path: '**', redirectTo: '', }, ];
html 中
<!-- routerLink数组, queryParams 对象 --> <a[routerLink]="['/product', id]"[queryParams]="{debug: 1}">XXXX</a> <!-- 在标签下面展示路由内容 --> <router-outlet></router-outlet>
component.ts
跳转
constructor(private router: Router){} this.router.navigate(["/", id], { queryParams: { debug: 1 } })
获取
constructor(private routeInfo: ActivatedRoute) { routeInfo.params.subscribe((params: Params) => this.productId = params["id"]); routeInfo.queryParams.subscribe((params: Params) => this.test = params["test"]); this.id = routeInfo.snapshot.params['id']; // 路由中配置 data: [{isprod: true}] this.isProd = routeInfo.snapshot.data[0]['isProd']; }
辅助路由
路由守卫
routes
{ path: 'product/:id', component: ProductComponent, // 在路由激活之前获取数据 resolve: { product: ProductResolveGuard }, // 处理导航到某个路由 canActivate: [LoginGuard], // 处理从当前路由离开 canDeactivate: [UnsavedGuard], data: [{ isProd: true }], }
canActivate
export class LoginGuard implements CanActivate { canActivate( route: ActivatedRouteSnapshot, state: RouterStateSnapshot ): Observable<boolean> | Promise<boolean> | boolean { let loggedIn: boolean = Math.random() < 0.5; if (!loggedIn) { console.log('LoginGuard:用户未登录' + new Date()); } return loggedIn; } }
CanDeactivate
export class UnsavedGuard implements CanDeactivate<ProductComponent> { canDeactivate() { return window.confirm('你还没有保存.确定要离开么?'); } }
Resolve
@Injectable() export class ProductResolveGuard implements Resolve<ProductComponent> { constructor(private router: Router) {} resolve( route: ActivatedRouteSnapshot, state: RouterStateSnapshot ): Observable<ProductComponent> | Promise<ProductComponent> | ProductComponent { let productId: number = route.params['id']; if (productId == 1) { return undefined; } else { this.router.navigate(['/home']); return undefined; } } }
依赖注入
providers: [ LoggerService, { provide: ProductService, // 单例, 只在第一次需要的时候初始化 useFactory: (logger: LoggerService, appConfig) => { if (appConfig.isDev) { return new ProductService(logger); } else { return new AnotherProductService(logger); } }, // 提供参数 deps: [LoggerService, 'APP_CONFIG'], }, { // 变量 provide: 'APP_CONFIG', useValue: { isDev: false, }, }, ];
数据绑定
插值表达式
<h1>{{title}}</h1>
属性表达式
<img[src]="imgUrl">
事件绑定
响应式编程
管道
@Pipe({ name: 'multiple', }) export class MultiplePipe implements PipeTransform { transform(value: number, args?: number): any { if (!args) { args = 1; } return value * args; } }
组件间通信
路由属性
输入属性
// 子组件 component 中 @Input() price: number;
<!-- 父组件html中 --> <child-component[price]="some-value"></child-componen>
输出属性
// 子组件 @Output('priceChange') lastPrice:EventEmitter<number> = new EventEmitter(); // some trigger this.lastPrice.emit(10);
<!-- 父组件中 --> <child-component(lastPrice)="priceHandle($event)"></child-component>
// 父组件中 priceHandle(event: number) { console.log('price: ', event) }
中间人模式(有共同的父组件)
// child1 组件 @Output('priceChange') lastPrice:EventEmitter<number> = new EventEmitter(); this.lastPrice.emit(10);
<!-- 父组件 --> <child1-component(lastPrice)="priceHandle($event)"></child1-component> <child2-component[lastPrice]="lastPrice"></child2-component>
组件生命周期
表单
数据模型
由 angular/form
模块中特定的类( FormControl
, FormGroup
, FormArray
)组成
两种表单比较
模板式表单 | 响应式表单 | |
---|---|---|
import | FormsModule |
ReactiveFormsModule |
如何构造 | 通过组件模板中相关指令 | 通过编写 typescript 代码 |
数据模型创建 | 由 angular 基于模板中的指令隐式创建 |
编码明确创建数据模型 |
能否直接访问数据模型 | 不能 | 能 |
HTML | 直接生成 | 自己编写绑定数据 |
模板式表单
-
NgFrom
会自动添加到<form>
表单上 - 手动添加
NgFrom
:<form #myForm="ngForm">
-
<form>
表单的提交不会被触发 - 实现
(ngSubmit)="onSubmit(myForm.value)"
- 在
NgForm
元素下 寻找标记为NgModel
属性的元素(需要指定name
属性才能绑定)
响应式表单
-
formGroup
和formControl
需要属性绑定语法 -
formGroupName
,formControlName
,formArrayName
不需要 属性绑定语法 -
formGroupName
,formControlName
,formArrayName
只能用在formGroup
指令覆盖的范围内
指令对照表
类名 | 模板式表单指令 | 响应式表单指令 |
---|---|---|
FormGroup | ngFrom ngModelGroup |
formGroup formGroupName |
FormControl | ngModel | formControl formControlName |
FormArray | formArrayName |
校验
-
formControl
校验器
function mobileValidator(control: FormControl):any{ let value = (control.value || '') + ''; var mobileReg = /^(((13[0-9]{1})|(15[0-9]{1})|(18[0-9]{1}))+\d{8})$/; let valid = mobileReg.test(value); return valid ? null : { mobile: true }; }
-
formGroup
校验器
function equalValidator(group: FormGroup):any{ let password: FormControl = group.get('password') as FormControl; let pconfirm: FormControl = group.get('pconfirm') as FormControl; let valid: boolean = false; if (password && pconfirm) { valid = password.value === pconfirm.value; } return valid ? null : { equal: { description: '密码和确认密码不匹配!' } }; }
- 异步校验器 (返回
observable
)
function mobileAsyncValidator(control: FormControl):any{ return Observable.of(valid ? null : { mobile: true }).delay(5000); }
- 在 响应式表单中 使用
constructor(fb: FormBuilder) { this.formModel = fb.group({ // 初始值 同步校验器 异步校验器 mobile: ['', mobileValidator, mobileAsyncValidator], // 多个同步校验器 username: ['', [Validators.required, Validators.minLength(5)]], passwordsGroup: fb.group( { password: ['', Validators.minLength(6)], pconfirm: [''], }, // formGroup 校验器 { validator: equalValidator } ), }) } onSubmit() { let isValid: boolean = this.formModel.get('username').valid; let errors: any = this.formModel.get('username').errors; if (this.formModel.valid) { console.log(this.formModel.value); } }
<form[formGroup]="formModel"(submit)="onSubmit()"> <!-- 同步校验 --> <div[hidden]="formModel.get('username').valid || formModel.get('username').untouched"> <!-- hasError的第一个参数是 校验器返回的对象的key, 不是校验器的名称 --> <div[hidden]="!formModel.hasError('required', 'username')"> 用户名是必填项 </div> <div[hidden]="!formModel.hasError('minlength', 'username')"> 用户名最小长度是5 </div> </div> <!-- 异步校验 --> <div> 手机号:<inputtype="number"formControlName="mobile"> </div> <div[hidden]="!formModel.get('mobile').pending"> 正在校验手机号合法性 </div> <div[hidden]="formModel.get('mobile').valid || formModel.get('mobile').pristine"> <div[hidden]="!formModel.hasError('mobile', 'mobile')"> 请输入正确的手机号 </div> </div> <!-- formGroup 校验 --> <divformGroupName="passwordsGroup"> <div>密码:<inputtype="password"formControlName="password"></div> <div[hidden]="!formModel.hasError('minlength', ['passwordsGroup','password'])"> 密码最小长度是6 </div> <div>确认密码:<inputtype="password"formControlName="pconfirm"></div> <div[hidden]="!formModel.hasError('equal', 'passwordsGroup')"> <!-- 从校验器中获取错误信息 --> {{formModel.getError('equal', 'passwordsGroup')?.description}} </div> </div> </form>
- 在 模板式表单 使用
// 先封装成一个 指令 @Directive({ // 中括号 括起来 selector: '[mobile]', // provide 是固定的 useValue 从 validator 引入 multi: true providers: [{ provide: NG_VALIDATORS, useValue: mobileValidator, multi: true }], }) export class MobileValidatorDirective { constructor() {} }
constructor() { } ngOnInit() {} onSubmit(value:any, valid:boolean){ console.log(valid); console.log(value); } usernameValid:boolean = true; usernameUntouched:boolean = true; // 模板式表单获取 状态 onUsernameInput(form:NgForm) { if(form) { this.usernameValid = form.form.get("username").valid; this.usernameUntouched = form.form.get("username").untouched; } }
<!-- 传递表单值 传递表单状态 --> <form#myForm="ngForm"(ngSubmit)="onSubmit(myForm.value, myForm.valid)"novalidate> <div>用户名:<inputngModelrequiredminlength="6"name="username"type="text" (input)="onUsernameInput(myForm)"> <!-- 模板式表单获取 状态 --> </div> <div[hidden]="usernameValid || usernameUntouched"> <!-- myForm.form --> <div[hidden]="!myForm.form.hasError('required','username')"> 用户名是必填项 </div> <div[hidden]="!myForm.form.hasError('minlength','username')"> 用户名最小长度是6 </div> </div> </form>
和服务器通讯
HttpClient
build
-
添加环境配置(如
staging
)编辑
angular.json
, 仿照projects.architect.build.configurations.staging
下添加环境变量 -
build
的时候加入--configuration
# 使用哪个 config 输出路径 ng build --prod --configuration=${env} --output-path dist
以上所述就是小编给大家介绍的《angular学习笔记》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
猜你喜欢:- 【每日笔记】【Go学习笔记】2019-01-04 Codis笔记
- 【每日笔记】【Go学习笔记】2019-01-02 Codis笔记
- 【每日笔记】【Go学习笔记】2019-01-07 Codis笔记
- Golang学习笔记-调度器学习
- Vue学习笔记(二)------axios学习
- 算法/NLP/深度学习/机器学习面试笔记
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
算法设计与分析导论
R.C.T.Lee (李家同)、S.S.Tseng、R.C.Chang、Y.T.Tsai / 王卫东 / 机械工业 / 2008-1 / 49.00元
本书在介绍算法时,重点介绍用干设计算法的策略.非常与众不同。书中介绍了剪枝搜索、分摊分析、随机算法、在线算法以及多项式近似方案等相对较新的思想和众多基于分摊分析新开发的算法,每个算法都与实例一起加以介绍,而且每个例子都利用图进行详细解释。此外,本书还提供了超过400幅图来帮助初学者理解。本书适合作为高等院校算法设计与分析课程的高年级本科生和低年级研究生的教材,也可供相美科技人员和专业人七参考使用。一起来看看 《算法设计与分析导论》 这本书的介绍吧!