name: angular-migration description: 使用混合模式、增量组件重写和依赖注入更新从 AngularJS 迁移到 Angular。适用于升级 AngularJS 应用程序、规划框架迁移或现代化遗留 Angular 代码。
Angular 迁移
掌握从 AngularJS 到 Angular 的迁移,包括混合应用、组件转换、依赖注入变化和路由迁移。
何时使用此技能
- 迁移 AngularJS (1.x) 应用程序到 Angular (2+)
- 运行混合的 AngularJS/Angular 应用程序
- 将指令转换为组件
- 现代化依赖注入
- 迁移路由系统
- 更新到最新 Angular 版本
- 实现 Angular 最佳实践
迁移策略
1. 大爆炸(完全重写)
- 用 Angular 完全重写整个应用
- 并行开发
- 一次性切换
- 最适合: 小型应用、新项目
2. 增量(混合方法)
- 同时运行 AngularJS 和 Angular
- 逐个功能迁移
- 使用 ngUpgrade 进行互操作
- 最适合: 大型应用、持续交付
3. 垂直切片
- 完全迁移一个功能
- 新功能用 Angular 开发,旧功能保持在 AngularJS
- 逐步替换
- 最适合: 中型应用、独立功能
混合应用设置
// main.ts - 引导混合应用
import { platformBrowserDynamic } from "@angular/platform-browser-dynamic";
import { UpgradeModule } from "@angular/upgrade/static";
import { AppModule } from "./app/app.module";
platformBrowserDynamic()
.bootstrapModule(AppModule)
.then((platformRef) => {
const upgrade = platformRef.injector.get(UpgradeModule);
// 引导 AngularJS
upgrade.bootstrap(document.body, ["myAngularJSApp"], { strictDi: true });
});
// app.module.ts
import { NgModule } from "@angular/core";
import { BrowserModule } from "@angular/platform-browser";
import { UpgradeModule } from "@angular/upgrade/static";
@NgModule({
imports: [BrowserModule, UpgradeModule],
})
export class AppModule {
constructor(private upgrade: UpgradeModule) {}
ngDoBootstrap() {
// 在 main.ts 中手动引导
}
}
组件迁移
AngularJS 控制器 → Angular 组件
// 之前:AngularJS 控制器
angular
.module("myApp")
.controller("UserController", function ($scope, UserService) {
$scope.user = {};
$scope.loadUser = function (id) {
UserService.getUser(id).then(function (user) {
$scope.user = user;
});
};
$scope.saveUser = function () {
UserService.saveUser($scope.user);
};
});
// 之后:Angular 组件
import { Component, OnInit } from "@angular/core";
import { UserService } from "./user.service";
@Component({
selector: "app-user",
template: `
<div>
<h2>{{ user.name }}</h2>
<button (click)="saveUser()">保存</button>
</div>
`,
})
export class UserComponent implements OnInit {
user: any = {};
constructor(private userService: UserService) {}
ngOnInit() {
this.loadUser(1);
}
loadUser(id: number) {
this.userService.getUser(id).subscribe((user) => {
this.user = user;
});
}
saveUser() {
this.userService.saveUser(this.user);
}
}
AngularJS 指令 → Angular 组件
// 之前:AngularJS 指令
angular.module("myApp").directive("userCard", function () {
return {
restrict: "E",
scope: {
user: "=",
onDelete: "&",
},
template: `
<div class="card">
<h3>{{ user.name }}</h3>
<button ng-click="onDelete()">删除</button>
</div>
`,
};
});
// 之后:Angular 组件
import { Component, Input, Output, EventEmitter } from "@angular/core";
@Component({
selector: "app-user-card",
template: `
<div class="card">
<h3>{{ user.name }}</h3>
<button (click)="delete.emit()">删除</button>
</div>
`,
})
export class UserCardComponent {
@Input() user: any;
@Output() delete = new EventEmitter<void>();
}
// 用法:<app-user-card [user]="user" (delete)="handleDelete()"></app-user-card>
服务迁移
// 之前:AngularJS 服务
angular.module("myApp").factory("UserService", function ($http) {
return {
getUser: function (id) {
return $http.get("/api/users/" + id);
},
saveUser: function (user) {
return $http.post("/api/users", user);
},
};
});
// 之后:Angular 服务
import { Injectable } from "@angular/core";
import { HttpClient } from "@angular/common/http";
import { Observable } from "rxjs";
@Injectable({
providedIn: "root",
})
export class UserService {
constructor(private http: HttpClient) {}
getUser(id: number): Observable<any> {
return this.http.get(`/api/users/${id}`);
}
saveUser(user: any): Observable<any> {
return this.http.post("/api/users", user);
}
}
依赖注入变化
降级 Angular → AngularJS
// Angular 服务
import { Injectable } from "@angular/core";
@Injectable({ providedIn: "root" })
export class NewService {
getData() {
return "来自 Angular 的数据";
}
}
// 提供给 AngularJS 使用
import { downgradeInjectable } from "@angular/upgrade/static";
angular.module("myApp").factory("newService", downgradeInjectable(NewService));
// 在 AngularJS 中使用
angular.module("myApp").controller("OldController", function (newService) {
console.log(newService.getData());
});
升级 AngularJS → Angular
// AngularJS 服务
angular.module('myApp').factory('oldService', function() {
return {
getData: function() {
return '来自 AngularJS 的数据';
}
};
});
// 提供给 Angular 使用
import { InjectionToken } from '@angular/core';
export const OLD_SERVICE = new InjectionToken<any>('oldService');
@NgModule({
providers: [
{
provide: OLD_SERVICE,
useFactory: (i: any) => i.get('oldService'),
deps: ['$injector']
}
]
})
// 在 Angular 中使用
@Component({...})
export class NewComponent {
constructor(@Inject(OLD_SERVICE) private oldService: any) {
console.log(this.oldService.getData());
}
}
路由迁移
// 之前:AngularJS 路由
angular.module("myApp").config(function ($routeProvider) {
$routeProvider
.when("/users", {
template: "<user-list></user-list>",
})
.when("/users/:id", {
template: "<user-detail></user-detail>",
});
});
// 之后:Angular 路由
import { NgModule } from "@angular/core";
import { RouterModule, Routes } from "@angular/router";
const routes: Routes = [
{ path: "users", component: UserListComponent },
{ path: "users/:id", component: UserDetailComponent },
];
@NgModule({
imports: [RouterModule.forRoot(routes)],
exports: [RouterModule],
})
export class AppRoutingModule {}
表单迁移
<!-- 之前:AngularJS -->
<form name="userForm" ng-submit="saveUser()">
<input type="text" ng-model="user.name" required />
<input type="email" ng-model="user.email" required />
<button ng-disabled="userForm.$invalid">保存</button>
</form>
// 之后:Angular(模板驱动)
@Component({
template: `
<form #userForm="ngForm" (ngSubmit)="saveUser()">
<input type="text" [(ngModel)]="user.name" name="name" required>
<input type="email" [(ngModel)]="user.email" name="email" required>
<button [disabled]="userForm.invalid">保存</button>
</form>
`
})
// 或反应式表单(推荐)
import { FormBuilder, FormGroup, Validators } from '@angular/forms';
@Component({
template: `
<form [formGroup]="userForm" (ngSubmit)="saveUser()">
<input formControlName="name">
<input formControlName="email">
<button [disabled]="userForm.invalid">保存</button>
</form>
`
})
export class UserFormComponent {
userForm: FormGroup;
constructor(private fb: FormBuilder) {
this.userForm = this.fb.group({
name: ['', Validators.required],
email: ['', [Validators.required, Validators.email]]
});
}
saveUser() {
console.log(this.userForm.value);
}
}
迁移时间线
阶段 1: 设置(1-2 周)
- 安装 Angular CLI
- 设置混合应用
- 配置构建工具
- 设置测试
阶段 2: 基础设施(2-4 周)
- 迁移服务
- 迁移工具
- 设置路由
- 迁移共享组件
阶段 3: 功能迁移(可变)
- 逐个功能迁移
- 彻底测试
- 增量部署
阶段 4: 清理(1-2 周)
- 删除 AngularJS 代码
- 删除 ngUpgrade
- 优化包
- 最终测试
资源
- references/hybrid-mode.md: 混合应用模式
- references/component-migration.md: 组件转换指南
- references/dependency-injection.md: DI 迁移策略
- references/routing.md: 路由迁移
- assets/hybrid-bootstrap.ts: 混合应用模板
- assets/migration-timeline.md: 项目规划
- scripts/analyze-angular-app.sh: 应用分析脚本
最佳实践
- 从服务开始: 先迁移服务(更容易)
- 增量方法: 逐个功能迁移
- 持续测试: 每个步骤都测试
- 使用 TypeScript: 早期迁移到 TypeScript
- 遵循风格指南: 从一开始就遵循 Angular 风格指南
- 稍后优化: 先让它工作,再优化
- 文档: 保留迁移笔记
常见陷阱
- 未正确设置混合应用
- 在逻辑之前迁移 UI
- 忽略变化检测差异
- 未正确处理作用域
- 混合模式(AngularJS + Angular)
- 测试不足