三大前端技术(React,Vue,Angular)探密

举报
hw82939365 发表于 2020/05/27 11:20:55 2020/05/27
【摘要】 本文对于当前顶级的前端技术做了较为详尽的探索,前端技术一个大的方向是单页应用 (SPA,Single Page Application),我们在选取针对本业务的前端技术时需要结合如下几个方面来考虑:成员当前技能、 可利用的学习时间、能否保证项目的复杂度最低。

【引言】

前段时间写过一篇关于前端技术的概括性文章《前端技术的选择](http://3ms.huawei.com/km/blogs/details/7971337),本文就对于当下顶级的前端开发技术做个相对详尽的探究,目的是为了解开这些前端技术的面纱,看看各自的庐山真面目。

 

React

React(也被称为React.jsReactJS)是一个用于构建用户界面的JavaScript库。它由Facebook和一个由个人开发者和公司组成的社区来维护。

 

React可以作为开发单页或移动应用的基础。然而,React只关注向DOM渲染数据,因此创建React应用通常需要使用额外的库来进行状态管理和路由,ReduxReact Router分别是这类库的例子。

基本用法

下面是一个简单的ReactHTML中使用JSXJavaScript的例子。

 

<div id="myReactApp"></div>

 

<script type="text/babel">

  function Greeter(props) {

    return <h1>{props.greeting}</h1>;

  }

  var App = <Greeter greeting="Hello World!" />;

  ReactDOM.render(Appdocument.getElementById("myReactApp"));

</script>

 

 

Greeter函数是一个React组件,它接受一个属性问候语。变量AppGreeter组件的一个实例,其中问候语属性被设置为 "Hello World"。然后,ReactDOM.render方法将我们的Greeter组件渲染在DOM元素(id myReactApp)中。

 

web浏览器中显示时,结果将是:

 

<div id="myReactApp">

  <h1>Hello World!</h1>

</div>

 

 

显著特点

 

组件化

React代码由称为组件的实体组成。组件可以使用React DOM库渲染到DOM中的一个特定元素。当渲染一个组件时,可以传入被称为 "props "的值。

 

ReactDOM.render(<Greeter greeting="Hello World!" />, document.getElementById('myReactApp'));

 

 

React中声明组件的两种主要方式是通过功能函数组件和基于类的组件。

 

功能函数组件

 

功能组件是用一个函数声明,用来返回一些JSX

 

const Greeting = (props=> <div>Hello, {props.name}!</div>;

 

 

类组件

基于类的组件是使用ES6类来声明的。它们也被称为 "有状态 "组件,因为它们的状态可以在整个组件中保持,并且可以通过props传递给子组件。

class ParentComponent extends React.Component {

  state = { color: 'green' };

  render() {

    return (

      <ChildComponent color={this.state.color} />

    );

  }

}

 

虚拟 DOM

另一个值得注意的特点是React使用了虚拟文档对象模型,也就是虚拟DOMReact创建了一个内存中的数据结构缓存,计算得出变化差异,只渲染实际变化的子组件从而高效地更新浏览器显示的DOM

 

生命周期方法

生命周期方法是指在组件的生命周期内,允许在设定的点执行代码的hooks处理函数。

 

l  shouldComponentUpdate允许开发者在不需要渲染的情况下,通过返回false来防止不必要的重新渲染组件。

l  componentDidMount是在组件 "挂载 "后调用的(组件已经在用户界面中创建了,通常是通过将其与DOM节点关联起来)。这通常用于通过API从远程数据源触发数据加载。

l  componentWillUnmount是在组件被拆解或 "解挂 "之前立即调用的。这通常用于清除组件的资源依赖关系,这些依赖关系不会随着组件的卸载而简单地被移除(例如,移除任何与组件相关的setInterval()实例,或者因为组件的存在而在 "文档 "上设置的 "eventListener")。

l  render是最重要的生命周期方法,也是任何组件中唯一必须存在的方法。它通常在每次更新组件的状态时都会被调用。

 

JSX

JSX,即JavaScript XML,是对JavaScript语言语法的扩展。JSX在外观上类似于HTML,它提供了一种开发者熟悉的语法结构化组件渲染的方法。React组件通常是使用JSX编写的,尽管不一定非要使用JSX(组件也可以用纯JavaScript编写)。JSX类似于FacebookPHP创建的另一种名为XHP的扩展语法。

 

JSX代码的一个例子:

class App extends React.Component {

  render() {

    return (

      <div>

        <p>Header</p>

        <p>Content</p>

        <p>Footer</p>

      </div>

    );

  }

}

 

 

嵌套元素

同一层次上的多个元素需要被包裹在一个容器元素中,如上图中的<div>元素。

 

属性

JSX提供了一系列的元素属性,旨在对应HTML提供的属性。这些自定义的属性也可以传递给组件,所有的属性都会被组件作为props接收。

 

JavaScript表达式

JavaScript表达式(但不是语句)可以在JSX内部通过大括号{}使用。

<h1>{10+1}</h1>

 

上面代码的显示结果是:

<h1>11</h1>

 

条件语句

If-else语句不能在JSX中使用,但可以使用条件表达式来代替。下面的例子当i1 { i === 1 ? 'true' : 'false' } 呈现为字符串 'true'

 

class App extends React.Component {

  render() {

    const i = 1;

    return (

      <div>

        <h1>{ i === 1 ? 'true' : 'false' }</h1>

      </div>

    );

  }

}

 

结果会是:

<div>

  <h1>true</h1>

</div>

 

函数和JSX可以用于条件表达式中:

class App extends React.Component {

  render() {

    const sections = [123];

    return (

      <div>

        {sections.length > 0 && sections.map(n => (

            /* 'key'必须唯一 */

            <div key={"section-" + n}>Section {n}</div>

        ))}

      </div>

    );

  }

}

 

结果会是:

 

 

<div>

  <div>Section 1</div>

  <div>Section 2</div>

  <div>Section 3</div>

</div>

 

JSX编写的代码需要被Babel等工具进行转换以后才能被Web浏览器所理解,这种处理一般是在软件构建过程中进行的,然后再部署构建后的应用程序。

 

超越HTML的架构

React的基本架构不仅仅适用于在浏览器中渲染HTML。例如,Facebook有动态图表,可以渲染到<canvas>标签,而NetflixPayPal使用通用加载,在服务器和客户端上渲染相同的HTML

 

React Hooks

Hooks是让开发者从函数组件中 "钩入"React状态和生命周期特性的函数。它们使代码具有更强的可读性且更易理解。Hooks并不在类组件内工作,它的终极目标是在React中消除类组件的存在。

 

React提供了一些内置的Hooks,如useStateuseContextuseReduceruseEffect等。它们都在Hooks API参考书中做了说明。使用最多的是useStateuseEffect,分别在React组件中控制状态和检测状态变化。

 

 

Hooks规则

 

Hooks也有一些规则,在使用Hooks之前必须遵循这些规则:

 

l  钩子只能在顶层调用(不能在循环或if语句中调用)。

l  钩子只能在React函数组件中调用,不能在普通函数或类组件中调用。

 

定制Hooks

构建自己的Hooks,也就是所谓的自定义Hooks,可以让你把组件逻辑提取到可重用的函数中。自定义钩子是一个名称以 "use "开头的JavaScript函数,它可以调用其他的钩子。钩子的规则也适用于它们。

 

常用术语

 

React并没有试图提供一个完整的 "应用程序库"。它是专门为构建用户界面而设计的,因此并不包括许多一些开发者认为构建应用程序所需的工具。这使得开发者可以选择任何一个库来完成诸如执行网络访问或本地数据存储等任务。这种情况也就决定了React技术在创建网页应用时标准无法统一。

 

Flux架构的使用

为了支持React的单向数据流的概念(与AngularJS/Angular的双向数据流形成对比),Flux架构是流行的模型-视图-控制器(MVC)架构的具有代表性的替代方案。Flux的特点是,数据动作通过中央调度器发送到一个存储仓库,而对存储仓库数据的变化会被传送回视图。当与React一起使用时,这种传送是通过组件属性完成的。

 

Flux可以被认为是观察者模式的一个变种。

 

Flux架构下的React组件不应该直接修改传递给它的任何props,而是应该传递回调函数,这些回调函数可以创建由调度器发送的数据动作来修改存储仓库。数据动作是一个对象,其职责是描述已经发生的事情:例如,一个数据动作描述的是一个用户 "follow"另一个用户。它可能包含如下数据:

用户ID

目标用户ID

以及USER_FOLLOWED_ANOTHER_USER枚举类型。

存储仓库,是一个数据模型,可以根据从调度器接收到的数据动作来改变自己。

这种模式有时被表述为 "属性向下流动,数据动作向上流动"。自Flux诞生以来,Flux的许多实现被创造出来,其中最著名的是Redux,它的特点是单一的存储仓库,通常被称为单一的数据真相源。

 

历史

React是由Facebook的软件工程师Jordan Walke创建的,受PHPHTML组件库XHP的启发,发布了React的早期原型,名为 "FaxJS",。它于2011年首次部署在FacebookNews Feed上,后来于2012年部署在Instagram上。20135月在美国JSConf大会上开源。

 

React Native20152月在FacebookReact Conf上宣布的,20153月开源的React Native,实现了原生的AndroidiOSUWP开发。

 

2017418日,Facebook宣布了React Fiber,这是React库的一个新的核心算法,用于构建用户界面,React Fiber将成为React库未来任何改进和功能开发的基础。

 

2017926日,React 16.0正式对外发布。

 

2019216日,React 16.8正式对外发布,该版本引入了React Hooks

 

常用命令

 

创建工程:

npx create-react-app my-app

开发环境运行:

npm start

 

生产环境打包:

npm run build

 

【官方网站】

http://reactjs.org/

 

【最新版本】

16.13.12020319

【授权】

MIT License

 

 

Angular

Angular(通常被称为 "Angular 2+" "Angular v2及以上版本")是一个基于TypeScript的开源Web应用框架,由GoogleAngular团队和由个人以及企业组成的社区领导。 Angular是由构建AngularJS的同一个团队从零开始重写的。

 

AngularAngularJS的区别

 

l  Angular没有 "Scope"或控制器的概念,相反,它使用组件的层次结构作为其主要的架构特征。

l  Angular有不同的表达式语法,重点是"[]"用于属性绑定,"() "用于事件绑定

l  模块化 - 许多核心功能已转移到模块上

l  Angular推荐使用微软的TypeScript语言,它引入了以下特性。

n  静态键入,包括Generics

n  注解

l  TypeScriptECMAScript 6 (ES6)的超集,向后兼容ECMAScript 5(即:JavaScript)。

l  动态加载

l  异步模板编译

l  RxJS提供的迭代回调。RxJS限制了状态的可见性和调试,但这些问题可以通过像ngReactngrx这样的反应式附加组件来解决。

l  支持Angular Universal,可以在服务器上运行Angular应用程序。

 

历史

命名

最初,AngularJS的重写被称为 "Angular 2",但这导致了开发人员的迷糊。为了澄清,团队宣布,每个框架使用不同的术语,其中 "AngularJS "指的是1.X版本, "Angular " 指的是2及以上版本。

 

版本2

Angular 2.020141022-23日的ng-Europe大会上宣布。2.0版本的剧烈变化在开发者中引起了相当大的争议。

 

2015430日,Angular开发者宣布Angular 2Alpha转为开发者预览版,201512Angular 2转为Beta版,20165月发布了第一个发布候选版本,2016914日发布了最终版本。

 

版本4

20161213Angular 4发布,跳过了3,避免了因路由器包的版本错位导致的混乱,当时已经发布的版本为v3.3.0。最终版本于2017323日发布,Angular 4向后兼容Angular 2

 

Angular 4.3版本是一个小版本,它是4.x.x版本的替换版本。

 

4.3版本的功能

 

l  介绍了HttpClient,一个更小、更容易使用、更强大的HTTP请求库。

l  为守护者和解析器提供了新的路由器生命周期事件。四个新事件。GuardsCheckStartGuardsCheckEndResolveStartResolveEnd加入了现有的NavigationStart等生命周期事件集。

l  有条件地禁用动画。

 

 

版本5

Angular 52017111日发布,Angular 5的主要改进包括支持渐进式Web应用、构建优化器以及与Material Design相关的改进。

版本6

Angular 6201854日发布。这个版本,关注的重点不在于底层框架,更多的是工具链,以及让Angular在未来的更新和升级更加容易,比如:ngupdateng addAngular元素、Angular Material+CDK组件、Angular Material入门组件、CLI工作区、库支持、树形摇动提供者、动画性能提升、RxJS v6

版本7

Angular 7已于20181018日发布。更新内容涉及到应用性能、Angular Material & CDK、虚拟滚动、Selects的可访问性改进、现在支持自定义元素使用Web标准的内容投影,以及关于Typescript 3.1RxJS 6.3Node 10(仍支持Node 8)的依赖性更新。

版本8

Angular 8已于2019528日发布。具有所有应用代码的差异化加载、惰性路由的动态导入、Web工作者、TypeScript 3.4支持、以及Angular Ivy作为预览版可配置使用。Angular Ivy预览包括:

 

l  生成的代码,在运行时更容易阅读和调试。

l  更快的重建时间

l  减少有效载荷

l  改进了模板类型检查

l  向后兼容

 

版本9

Angular 9已于202026日发布。第9版在默认情况下使用Ivy编译器。Angular可以与TypeScript 3.63.7兼容。除了数百个bug修复之外,Ivy编译器和运行时还提供了许多优势:

 

l  更小的软件包

l  更快的测试

l  更好的调试

l  改进的CSS类和样式绑定

l  改进的类型检查

l  改善了构建错误

l  改善了构建时间,默认开启AOT功能

l  提高国际化功能

 

特点

组件化

一个组件例子

Html部分

 

<h2>Products</h2>

 

<div *ngFor="let product of products">

 

  <h3>

      {{ product.name }}

  </h3>

 

</div>

 

 

Typescript部分

export class ProductListComponent {

  products = products;

}

 

 

路由

 

@NgModule({

  imports: [

    BrowserModule,

    ReactiveFormsModule,

    RouterModule.forRoot([

      { path: ''component: ProductListComponent },

      { path: 'products/:productId'component: ProductDetailsComponent },

    ])

  ],

 

数据管理

定义服务类

export class CartService {

  items = [];

 

  constructor(

    private httpHttpClient

  ) {}

 

  addToCart(product) {

    this.items.push(product);

  }

 

  getItems() {

    return this.items;

  }

 

  clearCart() {

    this.items = [];

    return this.items;

  }

 

  getShippingPrices() {

    return this.http.get('/assets/shipping.json');

  }

}

调用服务类

export class ShippingComponent implements OnInit {

  shippingCosts;

 

  constructor(

    private cartServiceCartService

  ) {

  }

 

  ngOnInit() {

    this.shippingCosts = this.cartService.getShippingPrices();

  }

 

}

常用命令

 

从终端上,全局安装Angular CLI

npm install -g @angular/cli

 

使用 ng new 命令创建一个新的 Angular CLI 工作区:

ng new my-project-name

 

开发环境运行:

ng serve

 

生产环境打包:

ng build --prod

 

【官方网站】

https://angular.io/

 

【最新版本】

9.1.22020415

【授权】

MIT License

 

 

Vue

 

Vue.js(通常被称为Vue;发音为/vjuː/,类似于 "view")是一个开源的Model-view-viewmodel JavaScript框架,用于构建用户界面和单页面应用程序。它由Evan You创建,由他和来自NetlifyNetguru等多家公司的核心成员维护。

 

概述

 

Vue.js的特点是,它采用了一个渐进式的架构,专注于声明式渲染和组件合成。复杂应用所需的高级功能,如路由、状态管理和构建工具等,都是通过官方维护的支持库和包提供的,其中Nuxt.js是最受欢迎的解决方案之一。

 

Vue.js可以让你用称为指令(directives)的HTML属性来扩展HTML

 

历史

Vue是由Evan You创建的。在Google工作期间,他使用AngularJS技术参与了多个项目的开发的,之后创建了Vue。他后来总结了自己的思考过程。"我想,如果我可以把AngularJS真正优秀的部分提取出来,然后构建一些轻量级的东西,会怎么样呢?项目的第一个版本源码提交日期是20137月,Vue20142月首次发布。

 

特点

组件化

Vue 组件扩展了基本的 HTML 元素来封装可重用的代码。从高层次的角度看,组件是Vue编译器附加行为的自定义元素。在Vue中,组件本质上就是一个带有预设选项的Vue实例。下面的代码片段包含了一个Vue组件的例子。该组件显示了一个按钮,并打印出按钮被点击的次数。

 

<div id="tuto">

    <button-clicked v-bind:initial-count="0"></button-clicked>

</div>

 

<script>

Vue.component('button-clicked', {

  props: [ "initialCount" ],

  data: () => ({

    count: 0,

  }),

  template: `<button v-on:click="onClick">Clicked {{ count }} times</button>`,

  computed: {

    countTimesTwo() {

      return this.count * 2;

    }

  },

  watch: {

    count(newValueoldValue) {

      console.log(`The value of count is changed from ${oldValue} to ${newValue}.`);

    }

  },

  methods: {

    onClick() {

        this.count += 1;

    }

  },

  mounted() {

    this.count = this.initialCount;

  }

});

 

new Vue({

  el: '#tuto',

});

</script>

 

模板

Vue使用基于HTML的模板语法,允许将渲染的DOM绑定到Vue实例的底层数据。所有 Vue 模板都是有效的 HTML,可以被符合规范的浏览器和 HTML 解析器解析。Vue 将模板编译成虚拟 DOM 渲染函数。虚拟文档对象模型(或 "DOM")允许Vue在更新浏览器之前在其内存中渲染组件。结合反应式系统,Vue能够计算出需要重新渲染的组件的最小数量,并在App状态发生变化时,启动最小量的DOM操作。

 

Vue用户可以使用模板语法,也可以选择使用JSX直接编写渲染函数,渲染函数允许从软件组件中构建应用程序。

 

反应式系统

Vue的特点是采用了反应式系统,它使用纯JavaScript对象和优化的重渲染。每个组件在渲染过程中都会跟踪其反应式的依赖关系,因此系统可以精确地知道什么时候重新渲染,以及哪些组件需要重新渲染。

 

变换效果

当从DOM中插入、更新或删除项目时,Vue提供了多种方法来部署变换效果。这包括了以下工具:

 

l  自动应用CSS变换和动画的类

l  集成第三方CSS动画库,如Animate.css等。

l  在变换hooks期间,使用JavaScript直接操作DOM

l  集成第三方JavaScript动画库,如Velocity.js等。

 

当在变换组件中的元素被插入或移除时,会出现这样的情况:

l  Vue会自动检测到目标元素是否应用了CSS变换或动画。如果有,CSS变换类将在适当的时间添加/删除。

l  如果变换组件提供了JavaScript hooks,这些hooks将在适当的时间被调用。

l  如果没有检测到CSS变换/动画,并且没有提供JavaScript hooks,那么插入和/或移除的DOM操作将在下一帧中立即执行。

 

 

路由

单页面应用程序(SPA)的一个传统缺点是无法分享到特定网页中的确切 " "页面的链接。由于SPA只向用户提供一个基于URL的服务器响应(它通常服务于index.htmlindex.vue),因此通常情况下,将某些屏幕作为书签或分享到特定部分的链接是很困难的,甚至是不可能的。为了解决这个问题,许多客户端路由器用 "hashbang"(#!)来划分动态URL,例如page.com/#!/。然而,在HTML5中,大多数现代浏览器都支持不使用hashbang的路由。

 

Vue提供了一个界面,可以根据当前的URL路径来改变页面上显示的内容 – 可以有多种方式(无论是通过电子邮件链接、刷新还是页面内链接)。此外,当某些浏览器事件(如点击)发生在按钮或链接上时,使用前端路由器可以有意识地转换浏览器路径。Vue本身并没有自带前端路由。但开源的 "vue-router "包提供了一个API来更新应用程序的URL,支持返回按钮(导航历史记录),并支持电子邮件密码重置或电子邮件验证链接的认证URL参数。它支持将嵌套路由映射到嵌套组件,并提供精细化的过渡控制。添加了vue-router后,组件只需映射到它们所属的路由,父/根路由必须指明子路由的渲染位置。

 

<div id="app">

  <router-view></router-view>

</div>

...

 

<script>

...

const User = {

  template: '<div>User {{ $route.params.id }}</div>'

}

 

const router = new VueRouter({

  routes: [

    { path: '/user/:id'component: User }

  ]

})

...

</script>

 

上面的代码:

 

l  websitename.com/user/<id>中设置一个前端路径。

l  这将在(const User...)中定义的User组件中呈现。

l  允许用户组件使用$route对象的params键输入用户的特定ID$route.params.id

l  这个模板(根据传递到路由器中的参数变化)将被渲染到DOMdiv#app里面的<router-view></router-view>

l  最后生成的HTML将是:websitename.com/user/1

 

<div id="app">

  <div>

    <div>User 1</div>

  </div>

</div>

 

生态系统

核心库自带的工具和库都是由核心团队和贡献者开发的。

 

官方工具

l  Devtools - 用于调试Vue.js应用程序的浏览器devtools扩展。

l  Vue CLI - 用于快速开发Vue.js的标准工具书

l  Vue Loader - 一个webpack加载器,允许以单文件组件(SFCs)的格式编写Vue组件。

 

官方程序库

 

l  Vue Router - Vue.js的官方路由器

l  Vuex – 基于 Flux模式的 Vue.js 的集中式状态管理。

l  Vue Server Renderer - 用于 Vue.js 的服务器端渲染。

 

常用命令

安装工具

npm install -g @vue/cli

 

创建工程:

vue create my-project

 

开发环境运行:

npm run serve

 

生产环境打包:

npm run build

 

【官方网站】

https://vuejs.org/

 

【最新版本】

2.6.120191213

【授权】

MIT License

                           

小结

本文对于当前顶级的前端技术做了较为详尽的探索,前端技术一个大的方向是单页应用 (SPASingle Page Application),我们在选取针对本业务的前端技术时需要结合如下几个方面来考虑:

 

1.     成员当前技能,这是一个很现实的问题,大多数程序员会选择自己比较熟悉的技术。这里要思考一下,目前自己熟悉的技术是不是最优选项?

2.     可利用的学习时间,如果发现要使用的技术需要一些时间学习,这个时间的开销到底会不会与开发进度有冲突?

3.     能否保证项目的复杂度最低,这个是比较关键的因素。先进技术之所以先进就是因为可以让开发者把时间和精力放在真正的业务开发上面来,如果要使用的技术需要进行很多与业务不相关的配置,就需要问一个问题,有没有更好的办法?

 

最后,希望本文对现有或者以后的业务开发有指导或者借鉴作用。

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。