Angular1--Hello

最近有个小错误,因为最近还是在看thingsboard,最近终于看到前端的代码,突然发现怎么全是ts的文件,仔细一看原来并不是之前认为的AngularJS,而是Angular。。。我tm真的无语了,又要去重新学。。。

Angular的结构比起AngularJS真的复杂很多,以前还可以说是传统HTML+JS结构的扩展。新的版本真的大变了。

以前的AngularJS只要一个html就是开炫,现在是要一堆文件,就算摸清楚最小系统,也要折腾一番,唉,好吧。。。

1 环境配置

手动配置Angular的环境也是堪称折磨,尤其是package.json,tsconfig.json。所以一般都用自动配置。

首先是安装node.js,安装的原始命令是:

curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.39.7/install.sh | bash
因为众所周知的原因,这个命令很大概率要超时,必须换成。

curl -o- https://gitee.com/mirrors/nvm/raw/v0.39.7/install.sh | bash

之后source ~/.bashrc

然后升级nvm install --lts
# 然后全局安装 Angular CLI
npm i -g @angular/cli

后面用到的ng命令,就是Angular CLI工具。这个工具的帮助如下: 

# 创建项目(这一步 CLI 会自动生成配置和依赖),
ng new hello-angular --minimal --routing=false --style=css
cd hello-angular
ng serve -o      # 默认 http://localhost:4200

2 最简单的Hello world

最小的angular结构

.
├── angular.json
├── package.json
├── src
│   ├── index.html
│   ├── main.ts
│   └── styles.css
├── tsconfig.app.json
└── tsconfig.json2 directories, 7 files

其中4个json配置,css都可以用ng new自动生成。(自己配置确实有点麻烦,我搞了很久配置都要报错)

mian.ts

import {Component} from '@angular/core';
import {bootstrapApplication} from '@angular/platform-browser';@Component({selector: 'app-root',template: `Hello world!`,
})
export class Playground {}bootstrapApplication(Playground);

index.html

<!DOCTYPE html>
<html lang="en"><head><meta charset="utf-8" /><title>HelloAngular</title><base href="/" /></head><body><app-root></app-root></body>
</html>

可以看出,一个angluar的基本结构还是index.html开始。

由app-root寻址到第一个组件。组件就是angular的核心。

@Component({...})
export class MyComponent {// 逻辑代码
}

@Component是组件类的装饰器,这个以前玩angularJS的时候还真没看到过。。。现在的typescript也是越来越复杂了。

3 更典型的Angular

在上一步生成的代码基础上,做了一些修改。如下: 

 main.ts

import { bootstrapApplication } from '@angular/platform-browser';
import { App } from './app/app';
import { appConfig } from './app/app.config';bootstrapApplication(App, appConfig).catch((err) => console.error(err));

index.html

<!DOCTYPE html>
<html lang="en"><head><meta charset="utf-8" /><title>HelloAngular</title><base href="/" /></head><body><app-root></app-root>  <!-- 👈 Angular 根组件挂载点 --></body>
</html>

 app.ts

import { Component } from '@angular/core';
import { FormsModule } from '@angular/forms';
import { CommonModule } from '@angular/common'; // ✅ 加上这个!
import { TodoService, TodoItem } from './todo.service';@Component({selector: 'app-root',standalone: true,imports: [FormsModule, CommonModule],  // ✅ 把 CommonModule 加入 importstemplateUrl: './app.component.html',styleUrls: ['./app.component.css'],
})
export class App {newTitle = '';constructor(public todo: TodoService) {}add() {if (this.newTitle.trim()) {this.todo.add({ title: this.newTitle.trim(), done: false });this.newTitle = '';}}toggle(item: TodoItem) {this.todo.toggle(item);}remove(item: TodoItem) {this.todo.remove(item);}
}

todo.service.ts

import { Injectable } from '@angular/core';export interface TodoItem {title: string;done: boolean;
}@Injectable({ providedIn: 'root' })
export class TodoService {list: TodoItem[] = [];add(item: TodoItem) { this.list.push(item); }toggle(item: TodoItem) { item.done = !item.done; }remove(item: TodoItem) { this.list = this.list.filter(i => i !== item); }
}

app.config.ts

import { ApplicationConfig, provideBrowserGlobalErrorListeners, provideZoneChangeDetection } from '@angular/core';export const appConfig: ApplicationConfig = {providers: [provideBrowserGlobalErrorListeners(),provideZoneChangeDetection({ eventCoalescing: true }),]
};

app.component.html

<h1>📝 Angular Todo (standalone)</h1><inputplaceholder="输入待办事项"[(ngModel)]="newTitle"(keyup.enter)="add()"
/>
<button (click)="add()">添加</button><ul><li *ngFor="let item of todo.list"><input type="checkbox" [checked]="item.done" (change)="toggle(item)" /><span [class.done]="item.done">{{ item.title }}</span><button (click)="remove(item)">🗑</button></li>
</ul>

app.component.css

.done { text-decoration: line-through; color: #888; }
li   { margin: 4px 0; }

概念代码位置说明
组件 (Component)AppComponentUI 单元 + 逻辑
模板 (Template)app.component.htmlHTML + Angular 指令 (*ngFor, [(ngModel)])
服务 (Service)TodoService业务数据与方法,注入到组件
注入 (DI)constructor(public todo: TodoService)将服务注入组件
双向绑定[(ngModel)]="newTitle"表单输入 ↔ 组件字段
事件绑定(click)="add()"用户操作触发方法

在第二个程序中,组件Component的一部分被拆分出去了,作为Service,这个的原因是复用,这样其它的组件也可以使用这个Service。也相当于界面和业务分离。

下面是一个小类比。

Angular 元素类比于 Java作用
ComponentController / View处理 UI 展示
ServiceService / DAO处理业务逻辑
InterfaceJava interface / DTO定义数据结构

4 框架流程

4.1 组件定义和模板编译

首先是定义各种组件。一个组件就是显示,可以是一个页面,也可以是页面的一部分。

@Component({selector: 'app-root',standalone: true,imports: [FormsModule, CommonModule], templateUrl: './app.component.html',styleUrls: ['./app.component.css'],
})

这里的app-root就对应了html中的<app-root></app-root>

template和templateUrl就是模板

对于template或者templateUrl里面的内容,框架会再进行编译,并且在前端重新生成,

4.2 运行时加载

在框架加载后,会根据main.js重新对app-root字段处理,这个就是重新加载之后的。

4.3 代码流程 

在main.ts里面设置了启动的组件App

bootstrapApplication(App, appConfig).catch((err) => console.error(err));

在App中定义了Service,通过@Component和HTML关联。这部分在上面讲过了。

export class App {newTitle = '';constructor(public todo: TodoService) {}add() {if (this.newTitle.trim()) {this.todo.add({ title: this.newTitle.trim(), done: false });this.newTitle = '';}}toggle(item: TodoItem) {this.todo.toggle(item);}remove(item: TodoItem) {this.todo.remove(item);}
}

至于Service就是组件类,里面是业务的实现。

export interface TodoItem {title: string;done: boolean;
}@Injectable({ providedIn: 'root' })
export class TodoService {list: TodoItem[] = [];add(item: TodoItem) { this.list.push(item); }toggle(item: TodoItem) { item.done = !item.done; }remove(item: TodoItem) { this.list = this.list.filter(i => i !== item); }
}

现在的标签也改成ng开头了,比如ngModel,ngFor等等。

更详细的后面再讨论吧。。。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/910564.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/910564.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

什么是redission看门狗机制

Redisson 的看门狗机制(Watchdog Mechanism)是其实现可重入分布式锁时的一个核心特性,主要用于解决业务逻辑执行时间超过锁的过期时间(leaseTime)而导致锁提前释放,进而引发数据不一致的问题。它是一个自动的锁续期机制。 🔍 核心问题:为什么需要看门狗? 分布式锁的…

黑马程序员苍穹外卖DAY1

1. 前端页面能正常显示但无法登录&#xff08;一直转圈圈&#xff09; 找到下面路径的dev.yml port一定要跟自己本机的保持一致&#xff0c;&#xff0c;username和password也一定是自己主机的用户名和密码&#xff0c;不然连不上。 登录界面的密码为数据库表的密码&#xff0…

Frida Hook Android App 点击事件实战指南:从进程识别到成功注入

一、背景与目标 在逆向分析和自动化测试中&#xff0c;Hook Android 的点击事件是调试 UI 交互逻辑的重要手段之一。本文将以实际案例讲解如何通过 Frida Hook public void onClick(View view) 方法&#xff0c;并解决常见的 Hook 失败问题&#xff0c;最终实现对登录按钮的监…

Arduino Nano 33 BLE Sense Rev 2开发板使用指南之【环境搭建 / 点灯】

一、硬件介绍 1、产品特点 Arduino Nano 33 BLE Rev2&#xff0c;利用了nRF52840微控制器的先进功能。这款32位Arm Cortex-M4 CPU 64 MHz与MicroPython的兼容性增强了板子的灵活性&#xff0c;使其更容易被更广泛的开发者社区所接受。 该开发板的突出特点是其蓝牙低功耗&…

[QT]-宏使用

用宏,务必写清文档并用 do {…} while (0)为啥呢,示例 在 C/C++ 中,使用 do { … } while (0) 包裹宏定义是一种经典的最佳实践,主要用于解决宏展开后的语法和逻辑问题。以下是详细解释和示例: 一、为什么用 do { … } while (0) 包裹宏? 避免分号导致的语法错误 问题场…

python-property、反射

# ### property """ 可以把方法变成属性 : 可以动态的控制属性的获取,设置,删除相关操作 property 获取属性 方法名.setter 设置属性 方法名.deleter 删除属性 """ # 方法一 """是同一个方法名""" class MyCla…

【自动鼠标键盘控制器|支持图像识别】

[软件名称]: 电脑图像识别 [软件大小]: 57.2 MB [下载通道]: 夸克盘 | 迅雷盘 &#x1f3ae;【自动鼠标键盘控制器&#xff5c;支持图像识别】基于Python开发的智能自动化工具 轻量便捷的自动化操作工具&#xff0c;集成图像识别、鼠标控制、键盘模拟等功能&#xff0c;轻松…

ISO/IEC 8824规范实际应用案例

案例 1&#xff1a;X.509 数字证书&#xff08;互联网安全基石&#xff09; 标准依据&#xff1a;RFC 5280 (基于 ASN.1 定义) 核心应用&#xff1a; Certificate :: SEQUENCE {tbsCertificate TBSCertificate, -- 证书主体signatureAlgorithm AlgorithmIdentifier,…

QT6工程各种路径详解

一.当前工作目录 1.获取方法&#xff1a; #include <QDir> qDebug() << "当前工作目录&#xff1a;" << QDir::currentPath(); 打印结果&#xff1a; 当前工作目录&#xff1a; "D:/2.QT/test/test_console/build/QT6_8_2_64_MSVC-Release&…

1931. 用三种不同颜色为网格涂色

1931. 用三种不同颜色为网格涂色 mod_value 10**9 7 class Solution:def colorTheGrid(self, m: int, n: int) -> int:# 1、预处理所有合法的单行涂色方案# 存储 3^i&#xff0c;用于快速计算颜色编码的每一位&#xff08;类似位运算&#xff09;# [3^0, 3^1, 3^2, ...,…

整数的输入输出

整数的输入输出 两种形式输出&#xff1a;&#xff08;以int为界&#xff09; char、short、int都用 %dlong 和long long都用 %ld %d char、short、int%ld long long long%u unsignde%lu unsignde long long 整数的格式化输出示例 #include <stdio.h> int main(){cha…

【llm实战】Python打造BGE模型微调服务实战指南

1. 引言:为何需要BGE模型微调?定制化语义的力量 BGE(BAAI General Embedding)是由北京智源人工智能研究院(BAAI)发布的通用文本嵌入模型系列,因其在中英文任务上的优异表现而广受欢迎,尤其是在MTEB(Massive Text Embedding Benchmark)等权威榜单上名列前茅。 尽管通…

代码分析与自动化重构

PS&#xff1a;根据过去编写 Modernizing 相关的开源工具里&#xff0c;编写的《代码分析与自动化重构》指南。 遗留系统的现代化演进是一门艺术。在日常的软件开发里&#xff0c;我们经常会遇到一系列的问题&#xff1a; 如何解决人类智商不够的问题&#xff1f;模式、原则和…

【android bluetooth 框架分析 04】【bt-framework 层详解 8】【DeviceProperties介绍】

前面我们提到了 蓝牙协议栈中的 Properties &#xff0c; 这篇文章是 他的补充。 【android bluetooth 框架分析 04】【bt-framework 层详解 6】【Properties介绍】 1. 设计初衷与核心问题 1. 为什么要设计 DeviceProperties&#xff1f; 在 Android 蓝牙实际使用中&#x…

华为OD-2024年E卷-字母组合[200分] -- python

问题描述&#xff1a; 每个数字对应多个字母&#xff0c;对应关系如下&#xff1a; 0&#xff1a;a,b,c 1&#xff1a;d,e,f 2&#xff1a;g,h,i 3&#xff1a;j,k,l 4&#xff1a;m,n,o 5&#xff1a;p,q,r 6&#xff1a;s,t 7&#xff1a;u,v 8&#xff1a;w,x 9&#xff1…

机器学习竞赛中的“A榜”与“B榜”:机制解析与设计深意

在Kaggle、天池等主流机器学习竞赛平台上&#xff0c;“A榜”&#xff08;Public Leaderboard&#xff09;和“B榜”&#xff08;Private Leaderboard&#xff09;是选手们最关注的指标。但很多新人对两者的区别和设计意图感到困惑。本文将深入解析其差异及背后的逻辑。 &#…

云徙科技 OMS:让订单管理变得轻松又高效

在如今这个线上线下购物融合得越来越紧密的时代&#xff0c;企业要是想在竞争激烈的市场里站稳脚跟&#xff0c;订单管理这一块可得好好下功夫。云徙科技的 OMS&#xff08;订单管理系统&#xff09;就像是给企业量身打造的一把“金钥匙”&#xff0c;能帮企业把订单管理得井井…

qt常用控件--02

文章目录 qt常用控件--02toolTip属性focusPolicy属性styleSheet属性补充知识点按钮类控件QPushButton 结语 很高兴和大家见面&#xff0c;给生活加点impetus&#xff01;&#xff01;开启今天的编程之路&#xff01;&#xff01; 今天我们进一步c11中常见的新增表达 作者&…

P3258 [JLOI2014] 松鼠的新家

题目描述 松鼠的新家是一棵树&#xff0c;前几天刚刚装修了新家&#xff0c;新家有 n n n 个房间&#xff0c;并且有 n − 1 n-1 n−1 根树枝连接&#xff0c;每个房间都可以相互到达&#xff0c;且俩个房间之间的路线都是唯一的。天哪&#xff0c;他居然真的住在“树”上。 …

基于openfeign拦截器RequestInterceptor实现的微服务之间的夹带转发

需求&#xff1a; trade服务需要在下单后清空购物车 分析&#xff1a; 显然&#xff0c;清空购物车需要调用cart服务&#xff0c;也就是这个功能的实现涉及到了微服务之间的转发。 其次&#xff0c;清空购车还需要userId&#xff0c;所以需要使用RequestInterceptor来实现夹…