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
英文地址,請注明出處:http://en.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來實現夾…