“在AngularJS中思考”,如果我有jQuery背景?[关闭]

"Thinking in AngularJS" if I have a jQuery background? [closed]

提问人:Mark Rajcok 提问时间:2/21/2013 最后编辑:Mark Rajcok 更新时间:11/3/2018 访问量:853263

问:

8年前关闭。
已锁定。这个问题及其答案被锁定,因为这个问题偏离了主题,但具有历史意义。它目前不接受新的答案或交互。

假设我熟悉在jQuery中开发客户端应用程序,但现在我想开始使用AngularJS。您能描述一下必要的范式转变吗?以下是一些可能有助于您确定答案的问题:

  • 如何以不同的方式构建和设计客户端 Web 应用程序?最大的区别是什么?
  • 我应该停止做什么/使用什么;我应该开始做什么/使用什么?
  • 是否有任何服务器端注意事项/限制?

我不是在寻找 和 之间的详细比较。jQueryAngularJS

javascript jquery angularjs

评论


答:

406赞 Mark Rajcok 2/21/2013 #1

命令式→声明式

在jQuery中,选择器用于查找DOM元素,然后将事件处理程序绑定/注册到它们。当事件触发时,将执行该(命令性)代码以更新/更改 DOM。

在 AngularJS 中,你要考虑视图而不是 DOM 元素。视图是包含 AngularJS 指令的(声明性)HTML。指令在后台为我们设置事件处理程序,并为我们提供动态数据绑定。选择器很少使用,因此对 ID(以及某些类型的类)的需求大大减少。视图与模型绑定(通过示波器)。视图是模型的投影。事件会更改模型(即数据、范围属性),并且投影这些模型的视图会“自动”更新。

在 AngularJS 中,考虑模型,而不是保存数据的 jQuery 选择的 DOM 元素。将视图视为这些模型的投影,而不是注册回调来操作用户看到的内容。

关注点分离

jQuery采用不显眼的JavaScript - 行为(JavaScript)与结构(HTML)分开。

AngularJS 使用控制器和指令(每个控制器和指令都可以有自己的控制器和/或编译和链接函数)从视图/结构 (HTML) 中删除行为。Angular 还提供服务和过滤器来帮助分离/组织您的应用程序。

另请参阅 https://stackoverflow.com/a/14346528/215945

应用程序设计

设计 AngularJS 应用程序的一种方法:

  1. 想想你的模型。为这些模型创建服务或您自己的 JavaScript 对象。
  2. 想想你想如何展示你的模型——你的观点。为每个视图创建 HTML 模板,使用必要的指令来获取动态数据绑定。
  3. 将控制器附加到每个视图(使用 ng-view 和路由,或 ng-controller)。让控制器仅查找/获取视图完成其工作所需的任何模型数据。使控制器尽可能薄。

原型继承

你可以用jQuery做很多事情,而不了解JavaScript原型继承是如何工作的。在开发 AngularJS 应用程序时,如果您对 JavaScript 继承有很好的理解,您将避免一些常见的陷阱。推荐阅读:AngularJS 中范围原型/原型继承的细微差别是什么?

152赞 Ulises 2/21/2013 #2

您能描述一下必要的范式转变吗?

命令式与声明式

使用jQuery,您可以一步一步地告诉DOM需要发生什么。使用 AngularJS,您可以描述您想要的结果,而不是如何去做。更多相关内容请点击此处。另外,请查看 Mark Rajcok 的回答。

如何以不同的方式构建和设计客户端 Web 应用程序?

AngularJS 是一个使用 MVC 模式的整个客户端框架(查看它们的图形表示)。它非常注重关注点的分离。

最大的区别是什么?我应该停止做什么/使用什么;我应该开始做什么/使用什么?

jQuery是一个库

AngularJS 是一个漂亮的客户端框架,高度可测试,它结合了大量很酷的东西,例如 MVC、依赖注入、数据绑定等等。

它侧重于关注点和测试(单元测试和端到端测试)的分离,这有助于测试驱动的开发。

最好的开始方式是阅读他们很棒的教程。您可以在几个小时内完成这些步骤;但是,如果您想掌握幕后的概念,它们包括大量参考资料以供进一步阅读。

是否有任何服务器端注意事项/限制?

您可以在已经在使用纯 jQuery 的现有应用程序上使用它。但是,如果您想充分利用 AngularJS 功能,您可以考虑使用 RESTful 方法对服务器端进行编码。

这样做将允许您利用他们的资源工厂,该工厂创建了服务器端 RESTful API 的抽象,并使服务器端调用(获取、保存、删除等)变得非常容易。

7161赞 Josh David Miller 2/22/2013 #3

1.不要设计你的页面,然后用DOM操作来改变它

在jQuery中,你设计一个页面,然后使它动态化。这是因为jQuery是为增强而设计的,并且从这个简单的前提中发展得令人难以置信。

但是在 AngularJS 中,你必须从头开始考虑你的架构。与其一开始就想着“我有这块 DOM,我想让它做 X”,不如从你想要完成的事情开始,然后开始设计你的应用程序,最后开始设计你的视图。

2.不要用AngularJS增强jQuery

同样,不要一开始就认为 jQuery 会执行 X、Y 和 Z,所以我将在此基础上为模型和控制器添加 AngularJS。当你刚开始的时候,这真的很诱人,这就是为什么我总是建议新的AngularJS开发人员不要使用jQuery,至少在他们习惯于以“Angular方式”做事之前。

我在这里和邮件列表上看到许多开发人员使用 150 或 200 行代码的 jQuery 插件创建了这些精心设计的解决方案,然后他们用一系列令人困惑和复杂的回调和 s 将其粘合到 AngularJS 中;但他们最终让它工作!问题在于,在大多数情况下,jQuery插件可以在AngularJS中重写一小部分代码,突然间一切都变得易于理解和简单明了。$apply

底线是这样的:在解决问题时,首先“在 AngularJS 中思考”;如果你想不出解决方案,请询问社区;如果毕竟没有简单的解决方案,那么请随时使用jQuery。但不要让jQuery成为拐杖,否则你永远不会掌握AngularJS。

3. 始终从架构的角度思考

首先要知道,单页应用程序就是应用程序。它们不是网页。因此,除了像客户端开发人员一样思考之外,我们还需要像服务器端开发人员一样思考。我们必须考虑如何将应用程序划分为单独的、可扩展的、可测试的组件。

那么你是怎么做到的呢?你如何“在AngularJS中思考”?以下是一些与jQuery对比的一般原则。

该视图是“官方记录”

在jQuery中,我们以编程方式更改视图。我们可以将下拉菜单定义为这样:ul

<ul class="main-menu">
    <li class="active">
        <a href="#/home">Home</a>
    </li>
    <li>
        <a href="#/menu1">Menu 1</a>
        <ul>
            <li><a href="#/sm1">Submenu 1</a></li>
            <li><a href="#/sm2">Submenu 2</a></li>
            <li><a href="#/sm3">Submenu 3</a></li>
        </ul>
    </li>
    <li>
        <a href="#/home">Menu 2</a>
    </li>
</ul>

在jQuery中,在我们的应用程序逻辑中,我们将使用如下方式激活它:

$('.main-menu').dropdownMenu();

当我们只看视图时,这里没有任何功能并不明显。对于小型应用程序,这很好。但是对于重要的应用程序,事情很快就会变得混乱且难以维护。

但是,在 AngularJS 中,视图是基于视图的功能的官方记录。我们的声明将如下所示:ul

<ul class="main-menu" dropdown-menu>
    ...
</ul>

这两者做同样的事情,但在 AngularJS 版本中,任何查看模板的人都知道应该发生什么。每当开发团队的新成员加入时,她都可以看看这个,然后就知道有一个指令叫做操作它;她不需要凭直觉找到正确的答案或筛选任何代码。这个观点告诉我们应该发生什么。干净多了。dropdownMenu

刚接触 AngularJS 的开发人员经常会问这样的问题:我如何找到特定类型的所有链接并在其上添加指令。当我们回答时,开发人员总是大吃一惊:你没有。但你不这样做的原因是,这就像半jQuery,半AngularJS,而且不好。这里的问题是开发人员试图在 AngularJS 的上下文中“执行 jQuery”。这永远不会奏效。该视图官方记录。在指令之外(下面会详细介绍),你永远不会、永远、永远不要改变 DOM。并且指令在视图中应用,因此意图很明确。

记住:不要设计,然后标记。你必须进行架构设计,然后进行设计。

数据绑定

这是迄今为止 AngularJS 最令人敬畏的功能之一,并且消除了执行我在上一节中提到的各种 DOM 操作的大量需求。AngularJS 会自动更新您的视图,因此您不必这样做!在jQuery中,我们响应事件,然后更新内容。像这样:

$.ajax({
  url: '/myEndpoint.json',
  success: function ( data, status ) {
    $('ul#log').append('<li>Data Received!</li>');
  }
});

对于如下所示的视图:

<ul class="messages" id="log">
</ul>

除了混合关注之外,我们还面临着我之前提到的表示意图的相同问题。但更重要的是,我们必须手动引用和更新 DOM 节点。如果我们想删除一个日志条目,我们也必须为此对 DOM 进行编码。除了 DOM 之外,我们如何测试逻辑?如果我们想改变演示文稿怎么办?

这有点凌乱,有点脆弱。但是在 AngularJS 中,我们可以这样做:

$http( '/myEndpoint.json' ).then( function ( response ) {
    $scope.log.push( { msg: 'Data Received!' } );
});

我们的观点可以如下所示:

<ul class="messages">
    <li ng-repeat="entry in log">{{ entry.msg }}</li>
</ul>

但就此而言,我们的观点可能如下所示:

<div class="messages">
    <div class="alert" ng-repeat="entry in log">
        {{ entry.msg }}
    </div>
</div>

现在,我们不再使用无序列表,而是使用 Bootstrap 警报框。而且我们从来不需要更改控制器代码!但更重要的是,无论日志在何处如何更新,视图也会发生变化。自然而然。整洁!

虽然我没有在这里展示它,但数据绑定是双向的。因此,只需执行以下操作,即可在视图中编辑这些日志消息: .大家欢欣鼓舞。<input ng-model="entry.msg" />

不同的模型层

在jQuery中,DOM有点像模型。但是在 AngularJS 中,我们有一个单独的模型层,我们可以以任何我们想要的方式进行管理,完全独立于视图。这有助于上述数据绑定,保持关注点的分离,并引入更大的可测试性。其他答案都提到了这一点,所以我就不说了。

关注点分离

以上所有内容都与这个总体主题有关:将你的关注点分开。你的观点是应该发生的事情的官方记录(在大多数情况下);您的模型代表您的数据;您有一个服务层来执行可重用的任务;你做DOM操作,用指令来增强你的视图;然后用控制器将它们粘合在一起。其他答案中也提到了这一点,我唯一要补充的是可测试性,我将在下面的另一节中讨论。

依赖注入

为了帮助我们分离关注点,依赖注入 (DI)。如果你来自服务器端语言(从JavaPHP),你可能已经熟悉这个概念,但如果你是一个来自jQuery的客户端人,这个概念可能看起来很愚蠢,从多余到时髦。但事实并非如此。

从广义的角度来看,DI 意味着您可以非常自由地声明组件,然后从任何其他组件中请求它的实例,它就会被授予。您不必知道加载顺序、文件位置或类似的东西。力量可能不会立即显现出来,但我只举一个(常见的)例子:测试。

假设在我们的应用程序中,我们需要一个通过 REST API 实现服务器端存储的服务,并且根据应用程序状态,还需要本地存储。在控制器上运行测试时,我们不希望与服务器通信 - 毕竟,我们正在测试控制器。我们只需添加一个与原始组件同名的模拟服务,注入器将确保我们的控制器自动获得假服务——我们的控制器不知道也不需要知道其中的区别。

说到测试...

4. 测试驱动开发 - 始终

这确实是关于架构的第 3 节的一部分,但它非常重要,以至于我把它作为它自己的顶级部分。

在你见过、使用过或编写过的所有jQuery插件中,有多少有附带的测试套件?不是很多,因为jQuery不太适合。但 AngularJS 是。

在jQuery中,唯一的测试方法通常是使用示例/演示页面独立创建组件,我们的测试可以针对该页面执行DOM操作。因此,我们必须单独开发一个组件,然后将其集成到我们的应用程序中。多么不方便!很多时候,当使用jQuery进行开发时,我们选择迭代而不是测试驱动开发。谁能责怪我们?

但是由于我们分离了关注点,我们可以在 AngularJS 中迭代地进行测试驱动开发!例如,假设我们想要一个超级简单的指令来指示我们当前的路由是什么。我们可以在应用程序的视图中声明我们想要的东西:

<a href="/hello" when-active>Hello</a>

好了,现在我们可以为不存在的指令编写一个测试:when-active

it( 'should add "active" when the route changes', inject(function() {
    var elm = $compile( '<a href="/hello" when-active>Hello</a>' )( $scope );

    $location.path('/not-matching');
    expect( elm.hasClass('active') ).toBeFalsey();

    $location.path( '/hello' );
    expect( elm.hasClass('active') ).toBeTruthy();
}));

当我们运行测试时,我们可以确认它失败了。直到现在,我们才应该创建我们的指令:

.directive( 'whenActive', function ( $location ) {
    return {
        scope: true,
        link: function ( scope, element, attrs ) {
            scope.$on( '$routeChangeSuccess', function () {
                if ( $location.path() == element.attr( 'href' ) ) {
                    element.addClass( 'active' );
                }
                else {
                    element.removeClass( 'active' );
                }
            });
        }
    };
});

我们的测试现在通过了我们的菜单按要求执行。我们的开发是迭代的,也是测试驱动的。邪恶的酷。

5. 从概念上讲,指令不是打包的jQuery

你经常会听到“只在指令中做 DOM 操作”。这是必要的。以应有的尊重对待它!

但是,让我们更深入地了解一下......

有些指令只是装饰视图中已经存在的内容(思考),因此有时直接进行 DOM 操作,然后基本上就完成了。但是,如果指令就像一个“小部件”并且有一个模板,它也应该尊重关注点的分离。也就是说,模板也应该在很大程度上独立于其在链接和控制器功能中的实现。ngClass

AngularJS 附带了一整套工具,使这变得非常容易;使用我们可以动态更新类; 允许双向数据绑定; 并以编程方式显示或隐藏元素;还有更多 - 包括我们自己写的那些。换句话说,我们可以在没有 DOM 操作的情况下做各种令人敬畏的事情。DOM 操作越少,指令就越容易测试,越容易设置样式,将来越容易更改,并且它们的可重用性和可分发性就越高。ngClassngModelngShowngHide

我看到很多刚接触 AngularJS 的开发人员使用指令作为抛出一堆 jQuery 的地方。换句话说,他们认为“既然我不能在控制器中执行 DOM 操作,我就把这些代码放在指令中”。虽然这当然要好得多,但往往仍然是错误的

想想我们在第 3 节中编程的记录器。即使我们把它放在指令中,我们仍然希望以“Angular Way”的方式去做。它仍然不需要任何 DOM 操作!很多时候 DOM 操作是必要的,但它比你想象的要少得多!在应用程序中的任何位置执行 DOM 操作之前,问问自己是否真的需要这样做。也许有更好的方法。

下面是一个快速示例,显示了我最常看到的模式。我们想要一个可切换的按钮。(注意:这个例子有点做作,而且很啰嗦,以表示以完全相同的方式解决的更复杂的案例。

.directive( 'myDirective', function () {
    return {
        template: '<a class="btn">Toggle me!</a>',
        link: function ( scope, element, attrs ) {
            var on = false;

            $(element).click( function () {
                on = !on;
                $(element).toggleClass('active', on);
            });
        }
    };
});

这有几点不对劲:

  1. 首先,jQuery从来都不是必需的。我们在这里所做的任何事情都不需要jQuery!
  2. 其次,即使我们的页面上已经有jQuery,也没有理由在这里使用它;我们可以简单地使用,我们的组件在放入没有jQuery的项目中时仍然可以工作。angular.element
  3. 第三,即使假设此指令需要 jQuery 才能工作,jqLite() 将始终使用 jQuery(如果已加载)!所以我们不需要使用 - 我们可以只使用 .angular.element$angular.element
  4. 第四,与第三个密切相关的是,jqLite 元素不需要包装在里面 - 传递给函数的元素已经是一个 jQuery 元素!$elementlink
  5. 第五,我们在前面的章节中提到过,为什么要将模板的东西混合到我们的逻辑中?

这个指令可以被重写(即使是非常复杂的情况!)更简单地说:

.directive( 'myDirective', function () {
    return {
        scope: true,
        template: '<a class="btn" ng-class="{active: on}" ng-click="toggle()">Toggle me!</a>',
        link: function ( scope, element, attrs ) {
            scope.on = false;

            scope.toggle = function () {
                scope.on = !scope.on;
            };
        }
    };
});

同样,模板内容在模板中,因此您(或您的用户)可以轻松地将其换成满足任何必要样式的模板,并且无需触及逻辑。可重用性 - 繁荣!

还有所有其他好处,比如测试 - 这很容易!无论模板中的内容如何,该指令的内部 API 都不会被触及,因此重构很容易。您可以根据需要更改模板,而无需触及指令。无论您进行什么更改,您的测试仍然会通过。

W00T!

那么,如果指令不仅仅是类似jQuery的函数的集合,它们是什么呢?指令实际上是 HTML 的扩展。如果 HTML 没有做你需要它做的事情,你就写一个指令来为你做这件事,然后像使用 HTML 的一部分一样使用它。

换句话说,如果 AngularJS 没有做一些开箱即用的事情,想想团队将如何完成它以适应 , , 等人。ngClickngClass

总结

甚至不要使用jQuery。甚至不要包括它。它会让你退缩。当你遇到一个你认为你已经知道如何在jQuery中解决的问题时,在你到达之前,试着想想如何在AngularJS的范围内做到这一点。如果您不知道,请询问!20 次中有 19 次,最好的方法不需要 jQuery,尝试用 jQuery 解决它会给您带来更多的工作。$

34赞 Jin 8/14/2013 #4

它们是苹果和橙子。你不想比较它们。它们是两回事。AngularJs 已经内置了 jQuery lite,它允许您执行基本的 DOM 操作,甚至不包括完整的 jQuery 版本。

jQuery是关于DOM操作的。它解决了所有跨浏览器的痛苦,否则您将不得不处理,但它不是一个允许您将应用程序划分为组件(如 AngularJS)的框架。

AngularJs 的一个好处是它允许您在指令中分离/隔离 DOM 操作。有一些内置指令可供您使用,例如 ng-click。您可以创建自己的自定义指令,这些指令将包含所有视图逻辑或 DOM 操作,这样您就不会在应该处理业务逻辑的控制器或服务中混合 DOM 操作代码。

Angular 将您的应用程序分解为 -控制器 -服务业 -视图 -等。

还有一件事,那就是指令。这是一个可以附加到任何 DOM 元素的属性,您可以在其中疯狂地使用 jQuery,而不必担心您的 jQuery 会与 AngularJs 组件发生冲突或弄乱其架构。

我从我参加的一次聚会上听说,Angular 的一位创始人说他们非常努力地将 DOM 操作分离出来,所以不要试图将它们重新纳入其中。

30赞 Evan Zamir 9/13/2013 #5

我觉得这个问题很有趣,因为我第一次认真接触JavaScript编程是Node.js和AngularJS。我从来没有学过jQuery,我想这是一件好事,因为我不必忘记任何东西。事实上,我积极避免使用jQuery来解决我的问题,而只是寻找一种“AngularJS方式”来解决这些问题。所以,我想我对这个问题的回答基本上可以归结为,“像从未学过jQuery的人一样思考”,并避免任何直接合并jQuery的诱惑(显然AngularJS在某种程度上在幕后使用它)。

45赞 Anand 10/10/2013 #6

jQuery 是一个 DOM 操作库。

AngularJS 是一个 MV* 框架。

事实上,AngularJS 是为数不多的 JavaScript MV* 框架之一(许多 JavaScript MVC 工具仍属于类别库)。

作为一个框架,它托管您的代码,并决定调用什么以及何时调用!

AngularJS 本身包含一个 jQuery-lite 版本。因此,对于一些基本的 DOM 选择/操作,您真的不必包含 jQuery 库(它可以节省许多字节以在网络上运行)。

AngularJS 具有用于 DOM 操作和设计可重用 UI 组件的“指令”概念,因此每当您觉得需要做与 DOM 操作相关的事情时,您都应该使用它(指令仅在使用 AngularJS 时应该编写 jQuery 代码的地方)。

AngularJS涉及一些学习曲线(比jQuery多:-)。

-->对于任何具有jQuery背景的开发人员,我的第一个建议是“在跳到像AngularJS这样的丰富框架之前,先学习JavaScript作为一流的语言! 我以艰难的方式了解了上述事实。

祝你好运。

61赞 Samuel 11/3/2013 #7

jQuery:你对 DOM 元素的“QUERYing the DOM”和做一些事情想了很多。

AngularJS:模型是事实,你总是从这个角度思考。

例如,当您从服务器获取数据时,您打算在 DOM 中以某种格式显示,在 jQuery 中,您需要 '1.FIND' 在 DOM 中要放置此数据的位置,即 '2.UPDATE/APPEND' 通过创建一个新节点或仅设置其 innerHTML 来将其添加到那里。然后,当您想要更新此视图时,您将 '3.FIND“位置和”4.更新”。在从服务器获取和格式化数据的相同上下文中完成的这种查找和更新的循环在 AngularJS 中消失了。

使用 AngularJS,你有你的模型(你已经习惯的 JavaScript 对象),模型的值告诉你模型(显然)和视图,对模型的操作会自动传播到视图,所以你不必考虑它。你会发现自己在 AngularJS 中不再在 DOM 中找到东西。

换句话说,在jQuery中,你需要考虑CSS选择器,即具有类或属性的or在哪里,等等,这样我就可以得到它们的HTML或颜色或值,但是在AngularJS中,你会发现自己是这样想的:我在处理什么模型,我会将模型的值设置为true。您不必为反映此值的视图是复选框还是驻留在元素中而烦恼(在jQuery中经常需要考虑的细节)。divtdtd

通过 AngularJS 中的 DOM 操作,您会发现自己添加了指令和过滤器,您可以将其视为有效的 HTML 扩展。

在AngularJS中,你将体验到的另一件事:在jQuery中,你经常调用jQuery函数,在AngularJS中,AngularJS会调用你的函数,所以AngularJS会“告诉你如何做事”,但好处是值得的,所以学习AngularJS通常意味着学习AngularJS想要什么,或者AngularJS要求你呈现函数的方式,它会相应地调用它。这是使 AngularJS 成为框架而不是库的原因之一。

31赞 codevinsky 11/9/2013 #8

收听播客 JavaScript Jabber:第 #32 集,其中介绍了 AngularJS 的原始创建者:Misko Hevery 和 Igor Minar。他们谈论了很多关于从其他 JavaScript 背景(尤其是 jQuery)来到 AngularJS 的感觉。

关于你的问题,播客中提出的一个观点让我对很多事情产生了兴趣:

MISKO:[...]在 Angular 中,我们很少考虑的一件事是,我们如何提供大量的逃生舱口,以便你可以出去并基本上找到摆脱困境的方法。所以对我们来说,答案就是这个叫做“指令”的东西。有了指令,你基本上就变成了一个普通的小jQuery JavaScript,你可以做任何你想做的事。

IGOR:所以可以把指令看作是给编译器的指令,每当你在模板中遇到这个特定元素或这个CSS时,它就会告诉它,你保留这种代码,而该代码负责DOM树中元素和该元素下面的所有内容。

整个剧集的文字记录可在上面提供的链接中找到。

因此,直接回答您的问题:AngularJS 非常固执己见,是一个真正的 MV* 框架。但是,您仍然可以在指令中使用jQuery做所有您知道和喜欢的非常酷的事情。这不是“我如何做我以前在jQuery中做的事情?”的问题,而是“我如何用我以前在jQuery中做的所有事情来补充AngularJS?”的问题。

这真的是两种截然不同的心态。

69赞 Nick Manning 2/4/2014 #9

jQuery的

jQuery制作了长得离谱的JavaScript命令,例如较短的和跨浏览器的命令。getElementByHerpDerp

AngularJS的

AngularJS 允许您创建自己的 HTML 标签/属性,这些标签/属性可以很好地与动态 Web 应用程序配合使用(因为 HTML 是为静态页面设计的)。

编辑:

说“我有jQuery背景,我如何在AngularJS中思考?”就像说“我有HTML背景,我如何在JavaScript中思考?你问这个问题的事实表明你很可能不了解这两个资源的基本目的。这就是为什么我选择通过简单地指出根本区别来回答这个问题,而不是通过列表说“AngularJS 使用指令,而 jQuery 使用 CSS 选择器来制作一个 jQuery 对象,它执行这个和那个等等......”。这个问题不需要冗长的答案。

jQuery 是一种使浏览器中的 JavaScript 编程更容易的方法。更短的跨浏览器命令等。

AngularJS 扩展了 HTML,因此您不必为了制作应用程序而到处放东西。它使 HTML 实际上适用于应用程序,而不是它的设计目的,即静态的教育网页。它使用 JavaScript 以迂回的方式实现了这一点,但从根本上说,它是 HTML 的扩展,而不是 JavaScript。<div>

84赞 Scott Rippey 2/21/2014 #10

要描述“范式转变”,我认为一个简短的回答就足够了。

AngularJS 改变了你查找元素的方式

jQuery中,通常使用选择器来查找元素,然后将它们连接起来:
$('#id .class').click(doStuff);

AngularJS 中,您可以使用指令直接标记元素,并将它们连接起来:
<a ng-click="doStuff()">

AngularJS 不需要(或希望)你使用选择器查找元素 - AngularJS 的 jqLite 与成熟的 jQuery 之间的主要区别在于 jqLite 不支持选择器

因此,当人们说“根本不包含jQuery”时,主要是因为他们不希望你使用选择器;他们希望你学会使用指令。直接,而不是选择!

184赞 superluminary 5/12/2014 #11

AngularJS 与 jQuery

AngularJS 和 jQuery 采用了截然不同的意识形态。如果你来自jQuery,你可能会发现一些令人惊讶的差异。Angular 可能会让你生气。

这是正常的,你应该坚持下去。Angular 是值得的。

最大的区别 (TLDR)

jQuery为您提供了一个工具包,用于选择DOM的任意位并对其进行临时更改。你几乎可以一点一点地做任何你喜欢的事情。

相反,AngularJS 为您提供了一个编译器

这意味着 AngularJS 从上到下读取整个 DOM,并将其视为代码,从字面上看是编译器的指令。当它遍历 DOM 时,它会寻找特定的指令(编译器指令),这些指令告诉 AngularJS 编译器如何行为以及该做什么。指令是充满 JavaScript 的小对象,可以与属性、标签、类甚至注释进行匹配。

当 Angular 编译器确定 DOM 的一部分与特定指令匹配时,它会调用指令函数,向其传递 DOM 元素、任何属性、当前$scope(这是一个局部变量存储)和其他一些有用的位。这些属性可能包含可由指令解释的表达式,并告诉指令如何呈现,以及何时应重新绘制自身。

然后,指令可以反过来引入额外的 Angular 组件,例如控制器、服务等。编译器底部显示的是一个完全成型的 Web 应用程序,已连接并准备就绪。

这意味着 Angular 是模板驱动的。您的模板驱动 JavaScript,而不是相反。这是对角色的彻底反转,与我们在过去 10 年左右的时间里一直在编写的不显眼的 JavaScript 完全相反。这可能需要一些时间来适应。

如果这听起来像是过度规定和限制,那么事实就不远了。由于 AngularJS 将 HTML 视为代码,因此您可以在 Web 应用程序中获得 HTML 级别的粒度。一切皆有可能,一旦你在概念上做出一些飞跃,大多数事情都会出奇地容易。

让我们深入了解细节。

首先,Angular 不会取代 jQuery

Angular 和 jQuery 做不同的事情。AngularJS 为您提供了一组用于生成 Web 应用程序的工具。jQuery主要为您提供修改DOM的工具。如果您的页面上存在 jQuery,AngularJS 将自动使用它。如果不是,AngularJS 附带了 jQuery Lite,这是一个精简的版本,但仍然完全可用 jQuery。

Misko 喜欢 jQuery,并不反对你使用它。然而,你会发现,随着你的进步,你可以使用范围、模板和指令的组合来完成几乎所有的工作,你应该尽可能更喜欢这个工作流程,因为你的代码将更离散、更可配置、更 Angular。

如果你确实使用jQuery,你不应该把它洒到各处。在 AngularJS 中进行 DOM 操作的正确位置是在指令中。稍后会详细介绍这些内容。

带有选择器与声明性模板的不显眼的 JavaScript

jQuery通常以不显眼的方式应用。您的 JavaScript 代码链接在页眉(或页脚)中,这是唯一提到它的地方。我们使用选择器来挑选页面的部分,并编写插件来修改这些部分。

JavaScript 在控制之中。HTML 具有完全独立的存在。即使没有 JavaScript,您的 HTML 仍保持语义。Onclick 属性是非常糟糕的做法。

关于 AngularJS,您会注意到的第一件事是自定义属性无处不在。你的 HTML 将充斥着 ng 属性,这些属性本质上是类固醇上的 onClick 属性。这些是指令(编译器指令),是将模板挂接到模型的主要方式之一。

当你第一次看到这一点时,你可能会想把 AngularJS 写成老式的侵入式 JavaScript(就像我一开始所做的那样)。事实上,AngularJS并不遵循这些规则。在 AngularJS 中,您的 HTML5 是一个模板。它由 AngularJS 编译以生成您的网页。

这是第一个重大区别。对于jQuery来说,你的网页是一个要纵的DOM。对于 AngularJS,你的 HTML 是要编译的代码。AngularJS 读取您的整个网页,并使用其内置编译器将其编译为新网页。

你的模板应该是声明性的;只需阅读它,它的意思就应该很清楚了。我们使用具有有意义名称的自定义属性。我们再次使用有意义的名称来组成新的 HTML 元素。HTML知识最少且没有编码技能的设计师可以阅读您的AngularJS模板并了解它在做什么。他或她可以进行修改。这就是 Angular 的方式。

模板在驾驶座上。

在启动 AngularJS 并运行教程时,我问自己的第一个问题是“我的代码在哪里?我没有写过 JavaScript,但我有所有这些行为。答案是显而易见的。因为 AngularJS 编译 DOM,所以 AngularJS 将您的 HTML 视为代码。对于许多简单的情况,通常只需编写一个模板并让 AngularJS 将其编译成应用程序即可。

模板驱动应用程序。它被视为 DSL。您编写 AngularJS 组件,AngularJS 将负责拉入它们并根据模板的结构在正确的时间使它们可用。这与标准 MVC 模式非常不同,在标准 MVC 模式中,模板仅用于输出。

例如,它更类似于 XSLT 而不是 Ruby on Rails

这是一种彻底的控制反转,需要一些时间来适应。

不要再尝试从 JavaScript 驱动应用程序了。让模板驱动应用程序,让 AngularJS 负责将组件连接在一起。这也是 Angular 的方式。

语义 HTML 与语义模型

使用jQuery,您的HTML页面应包含语义有意义的内容。如果 JavaScript 被关闭(由用户或搜索引擎关闭),您的内容仍然可以访问。

因为 AngularJS 将您的 HTML 页面视为模板。模板不应该是语义的,因为你的内容通常存储在你的模型中,而模型最终来自你的 API。AngularJS 使用模型编译 DOM 以生成语义网页。

你的 HTML 源代码不再是语义的,相反,你的 API 和编译的 DOM 是语义的。

在 AngularJS 中,意义存在于模型中,HTML 只是一个模板,仅用于显示。

在这一点上,你可能有各种各样的关于SEO和可访问性的问题,这是正确的。这里有一些悬而未决的问题。大多数屏幕阅读器现在将解析 JavaScript。搜索引擎还可以索引 AJAXed 内容。尽管如此,您需要确保使用的是 pushstate URL,并且您有一个不错的站点地图。有关该问题的讨论,请参阅此处:https://stackoverflow.com/a/23245379/687677

关注点分离 (SOC) 与 MVC

关注点分离 (SOC) 是一种在多年的 Web 开发中成长起来的模式,原因有很多,包括 SEO、可访问性和浏览器不兼容。它看起来像这样:

  1. HTML - 语义含义。HTML 应该是独立的。
  2. CSS - 样式,没有 CSS,页面仍然是可读的。
  3. JavaScript - 行为,没有脚本,内容仍然存在。

同样,AngularJS不遵守他们的规则。总而言之,AngularJS摒弃了十年来的智慧,而是实现了一种MVC模式,在这种模式中,模板不再是语义的,甚至不再是一点点。

它看起来像这样:

  1. 模型 - 模型包含语义数据。模型通常是 JSON 对象。模型作为名为 $scope 的对象的属性存在。您还可以在$scope上存储方便的实用程序函数,然后您的模板可以访问这些函数。
  2. 视图 - 您的视图是用 HTML 编写的。视图通常不具有语义性,因为数据位于模型中。
  3. 控制器 - 控制器是一个 JavaScript 函数,用于将视图挂接到模型。它的功能是初始化$scope。根据您的应用程序,您可能需要也可能不需要创建控制器。一个页面上可以有多个控制器。

MVC 和 SOC 不在同一规模的两端,它们位于完全不同的轴上。SOC 在 AngularJS 上下文中没有意义。你必须忘记它并继续前进。

如果你像我一样经历过浏览器大战,你可能会觉得这个想法非常令人反感。克服它,这是值得的,我保证。

插件与指令

插件扩展了 jQuery。AngularJS 指令扩展了浏览器的功能。

在jQuery中,我们通过向jQuery.prototype添加函数来定义插件。然后,我们通过选择元素并在结果上调用插件来将它们挂接到 DOM 中。这个想法是扩展jQuery的功能。

例如,如果希望在页面上使用轮播,则可以定义一个无序列表,这些图形可能包装在导航元素中。然后,您可以编写一些 jQuery 来选择页面上的列表,并将其重新设置为具有超时的库以执行滑动动画。

在 AngularJS 中,我们定义指令。指令是返回 JSON 对象的函数。这个对象告诉 AngularJS 要查找哪些 DOM 元素,以及要对它们进行哪些更改。指令使用您发明的属性或元素挂接到模板中。这个想法是用新的属性和元素来扩展 HTML 的功能。

AngularJS 的方式是扩展原生 HTML 的功能。您应该编写看起来像 HTML 的 HTML,并使用自定义属性和元素进行扩展。

如果你想要一个轮播,只需使用一个元素,然后定义一个指令来拉入一个模板,并使这个吸盘工作。<carousel />

许多小指令与带有配置开关的大插件

jQuery的趋势是编写像灯箱这样的大型插件,然后我们通过传入大量值和选项来配置这些插件。

这是 AngularJS 中的一个错误。

以下拉列表为例。在编写下拉插件时,您可能会想在点击处理程序中编写代码,也许是添加向上或向下的 V 形函数,也许更改展开元素的类,显示隐藏菜单,所有有用的东西。

直到你想做一个小小的改变。

假设您有一个菜单,要在悬停时展开。好吧,现在我们有一个问题。我们的插件已经为我们连接了点击处理程序,我们需要添加一个配置选项,使其在这种特定情况下的行为不同。

在 AngularJS 中,我们编写较小的指令。我们的下拉指令会小得离谱。它可能会保持折叠状态,并提供 fold()、unfold() 或 toggle() 的方法。这些方法只会更新 $scope.menu.visible,这是一个保存状态的布尔值。

现在,在我们的模板中,我们可以将其连接起来:

<a ng-click="toggle()">Menu</a>
<ul ng-show="menu.visible">
  ...
</ul>

需要在鼠标悬停时更新吗?

<a ng-mouseenter="unfold()" ng-mouseleave="fold()">Menu</a>
<ul ng-show="menu.visible">
  ...
</ul>

模板驱动应用程序,因此我们获得了 HTML 级别的粒度。如果我们想逐个案例进行例外处理,该模板使这变得容易。

闭合与$scope

JQuery 插件是在闭包中创建的。隐私在该关闭范围内得到维护。由你来维护你的范围链在该闭合中。您实际上只能访问由 jQuery 传递到插件的 DOM 节点集,以及闭包中定义的任何局部变量和您定义的任何全局变量。这意味着插件是相当独立的。这是一件好事,但在创建整个应用程序时可能会受到限制。尝试在动态页面的各个部分之间传递数据成为一件苦差事。

AngularJS 有$scope对象。这些是由 AngularJS 创建和维护的特殊对象,您可以在其中存储模型。某些指令将生成一个新的$scope,默认情况下,该使用 JavaScript 原型继承继承其包装$scope继承。$scope对象可在控制器和视图中访问。

这是聪明的部分。由于$scope继承的结构大致遵循 DOM 的结构,因此元素可以无缝地访问自己的作用域和任何包含的作用域,一直到全局$scope(与全局作用域不同)。

这使得传递数据和在适当的级别存储数据变得更加容易。如果展开下拉列表,则只有下拉列表$scope需要知道它。如果用户更新其首选项,则可能需要更新全局$scope,并且将自动向任何侦听用户首选项的嵌套作用域发出警报。

这听起来可能很复杂,事实上,一旦你放松下来,就像飞行一样。您无需创建 $scope 对象,AngularJS 会根据您的模板层次结构正确且适当地为您实例化和配置它。然后,AngularJS 使用依赖注入的魔力将其提供给您的组件(稍后会详细介绍)。

手动 DOM 更改与数据绑定

在jQuery中,您可以手动进行所有DOM更改。以编程方式构造新的 DOM 元素。如果你有一个 JSON 数组,并且想把它放在 DOM 中,你必须编写一个函数来生成 HTML 并插入它。

在 AngularJS 中,您也可以这样做,但鼓励您使用数据绑定。更改模型,由于 DOM 通过模板绑定到它,因此您的 DOM 将自动更新,无需干预。

由于数据绑定是从模板完成的,使用属性或大括号语法,因此操作起来非常容易。与之相关的认知开销很小,因此您会发现自己一直在这样做。

<input ng-model="user.name" />

将输入元素绑定到 。更新输入将更新当前作用域中的值,反之亦然。$scope.user.name

同样:

<p>
  {{user.name}}
</p>

将在段落中输出用户名。这是一个实时绑定,因此如果更新了值,模板也会更新。$scope.user.name

阿贾克斯一直

在jQuery中,进行Ajax调用相当简单,但你仍然可能会三思而后行。需要考虑的复杂性增加了,并且需要维护相当多的脚本。

在 AngularJS 中,Ajax 是您默认的首选解决方案,它一直在发生,几乎您没有注意到。您可以使用 ng-include 包含模板。您可以使用最简单的自定义指令应用模板。您可以将 Ajax 调用包装在服务中,并为自己创建一个 GitHub 服务或 Flickr 服务,您可以非常轻松地访问这些服务。

服务对象与帮助程序函数

在jQuery中,如果我们想完成一个与dom无关的小任务,比如从API中提取一个feed,我们可以在闭包中写一个小函数来做到这一点。这是一个有效的解决方案,但是如果我们想经常访问该提要怎么办?如果我们想在另一个应用程序中重用该代码怎么办?

AngularJS 为我们提供了服务对象。

服务是包含函数和数据的简单对象。它们始终是单例,这意味着它们永远不会超过一个。假设我们想访问 Stack Overflow API,我们可以编写一个定义方法的方法。StackOverflowService

假设我们有一个购物车。我们可以定义一个 ShoppingCartService,它维护我们的购物车并包含用于添加和删除商品的方法。由于该服务是单一实例,并且由所有其他组件共享,因此任何需要的对象都可以写入购物车并从中提取数据。它总是同一个购物车。

服务对象是独立的 AngularJS 组件,我们可以根据需要使用和重用它们。它们是包含函数和数据的简单 JSON 对象。它们始终是单例的,因此,如果您将服务上的数据存储在一个地方,则只需请求相同的服务即可将该数据导出到其他地方。

依赖注入 (DI) 与 Instatiation - 又名去意大利面化

AngularJS 为您管理依赖项。如果你想要一个对象,只需引用它,AngularJS 就会为你获取它。

在你开始使用它之前,很难解释这是一个多么巨大的时间福音。jQuery中不存在类似AngularJS DI的东西。

DI 意味着您不是编写应用程序并将其连接在一起,而是定义一个组件库,每个组件都由一个字符串标识。

假设我有一个名为“FlickrService”的组件,它定义了从 Flickr 中提取 JSON 提要的方法。现在,如果我想编写一个可以访问 Flickr 的控制器,我只需要在声明控制器时按名称引用“FlickrService”。AngularJS 将负责实例化组件并将其提供给我的控制器。

例如,这里我定义了一个服务:

myApp.service('FlickrService', function() {
  return {
    getFeed: function() { // do something here }
  }
});

现在,当我想使用该服务时,我只需按名称引用它,如下所示:

myApp.controller('myController', ['FlickrService', function(FlickrService) {
  FlickrService.getFeed()
}]);

AngularJS 将识别出需要一个 FlickrService 对象来实例化控制器,并将为我们提供一个对象。

这使得将东西连接在一起变得非常容易,并且几乎消除了任何分页化的趋势。我们有一个组件的平面列表,AngularJS在我们需要它们时将它们一一交给我们。

模块化服务架构

jQuery很少提到你应该如何组织你的代码。AngularJS有意见。

AngularJS 为您提供了可以将代码放入其中的模块。例如,如果你正在编写一个与 Flickr 对话的脚本,你可能想要创建一个 Flickr 模块来包装所有与 Flickr 相关的功能。模块可以包括其他模块 (DI)。您的主应用程序通常是一个模块,这应该包括您的应用程序将依赖的所有其他模块。

您可以获得简单的代码重用,如果您想编写另一个基于 Flickr 的应用程序,您只需包含 Flickr 模块,瞧,您可以在新应用程序中访问所有与 Flickr 相关的功能。

模块包含 AngularJS 组件。当我们包含一个模块时,该模块中的所有组件都可以作为由其唯一字符串标识的简单列表。然后,我们可以使用 AngularJS 的依赖注入机制将这些组件相互注入。

总结一下

AngularJS 和 jQuery 不是敌人。在AngularJS中可以很好地使用jQuery。如果你很好地使用了 AngularJS(模板、数据绑定、$scope、指令等),你会发现你需要的 jQuery 比你可能需要的要少得多

要意识到的主要事情是你的模板驱动你的应用程序。不要再尝试编写无所不能的大插件了。相反,编写执行一件事的小指令,然后编写一个简单的模板将它们连接在一起。

少考虑不显眼的 JavaScript,而是考虑 HTML 扩展。

我的小本子

我对 AngularJS 感到非常兴奋,我写了一本关于它的短书,非常欢迎您 http://nicholasjohnson.com/angular-book/ 在线阅读。我希望它对您有所帮助。

46赞 Dan 5/17/2014 #12

这些是一些非常好但冗长的答案。

总结一下我的经验:

  1. 控制器和提供者(服务、工厂等)用于修改数据模型,而不是 HTML。
  2. HTML 和指令定义布局和与模型的绑定。
  3. 如果需要在控制器之间共享数据,请创建服务或工厂 - 它们是在应用程序中共享的单例。
  4. 如果您需要 HTML 小部件,请创建一个指令。
  5. 如果您有一些数据,现在正在尝试更新 HTML...停!更新模型,并确保 HTML 绑定到模型。
19赞 webketje 6/21/2014 #13

作为 JavaScript MV* 的初学者,并且只关注应用程序架构(而不是服务器/客户端问题),我当然会推荐以下资源(我很惊讶尚未提及):JavaScript Design Patterns,作者 Addy Osmani,作为对不同 JavaScript 设计模式的介绍。本答案中使用的术语摘自上面的链接文档。我不打算重复公认的答案中措辞非常好的内容。相反,这个答案与支持AngularJS(和其他库)的理论背景有关。

像我一样,你会很快意识到AngularJS(或Ember.js,Durandal和其他MV*框架)是一个复杂的框架,组装了许多不同的JavaScript设计模式。

我发现,在深入研究一个全局框架之前,分别测试这些模式的 (1) 原生 JavaScript 代码和 (2) 较小的库也更容易。这使我能够更好地理解框架解决了哪些关键问题(因为您个人面临着这个问题)。

例如:

  • JavaScript 面向对象编程(这是一个 Google 搜索链接)。它不是一个库,但肯定是任何应用程序编程的先决条件。它教会了我原型、构造函数、单例和装饰器模式的原生实现
  • jQuery/ 下划线表示外观模式(类似于用于操作 DOM 的所见即所得)
  • 原型/构造函数/混合模式原型.js
  • 模块模式需要JS/ Curl.js / AMD
  • KnockoutJS 用于可观察的发布/订阅模式

注意:这个列表不完整,也不是“最好的库”;它们恰好是我使用的库。这些库还包含更多模式,提到的只是它们的主要关注点或原始意图。如果您觉得此列表中缺少某些内容,请在评论中提及,我很乐意添加它。

12赞 Huy Tran 8/27/2014 #14

实际上,如果你使用的是AngularJS,你就不再需要jQuery了。AngularJS本身具有绑定和指令,这是您可以使用jQuery执行的大多数操作的非常好的“替代品”。

我通常使用 AngularJS 和 Cordova 开发移动应用程序。我唯一需要的jQuery是选择器。

通过谷歌搜索,我看到那里有一个独立的jQuery选择器模块。嘶嘶作响。

我决定制作一个小代码片段,帮助我使用 AngularJS 和 jQuery Selector(使用 Sizzle)的强大功能快速启动网站。

我在这里分享了我的代码:https://github.com/huytd/Sizzular

23赞 Sanjeev Singh 9/22/2014 #15

AngularJS 和 jQuery:

除了 JQLite 功能之外,AngularJs 和 JQuery 在每个级别上都完全不同,一旦你开始学习 AngularJs 的核心功能,你就会看到它(我在下面解释了)。

AngularJs 是一个客户端框架,用于构建独立的客户端应用程序。JQuery 是一个围绕 DOM 的客户端库。

AngularJs Cool Principle - 如果你想对你的 UI 进行一些更改,请从模型数据更改的角度考虑。更改数据后,UI 将重新呈现自身。你不需要每次都玩弄 DOM,除非并且直到它几乎不需要,这也应该通过 Angular 指令来处理。

为了回答这个问题,我想分享一下我在第一个使用 AngularJS 的企业应用程序上的经验。这些是 Angular 提供的最令人敬畏的功能,我们开始改变我们的 jQuery 思维方式,我们得到的 Angular 就像一个框架,而不是库。

双向数据绑定令人惊叹:我有一个包含所有功能的网格 UPDATE、DELTE、INSERT。我有一个数据对象,它使用 ng-repeat 绑定网格的模型。你只需要编写一行简单的 JavaScript 代码来删除和插入,仅此而已。网格会随着网格模型的即时变化而自动更新。更新功能是实时的,无需代码。 你感觉很神奇!!

可重用的指令是超级的:在一个地方编写指令,并在整个应用程序中使用它。我的天啊!!我将这些指令用于分页、正则表达式、验证等。真的很酷!

路由很强:这取决于你的实现,你想如何使用它,但它需要很少的代码行来路由请求以指定 HTML 和控制器 (JavaScript)

控制器很棒:控制器负责处理自己的 HTML,但这种分离适用于通用功能。如果要在主 HTML 上单击按钮时调用相同的函数,只需在每个控制器中编写相同的函数名称并编写单独的代码即可。

插件:还有许多其他类似的功能,例如在您的应用程序中显示叠加层。您不需要为它编写代码,只需使用一个作为 wc-overlay 提供的覆盖插件,这将自动处理所有 XMLHttpRequest (XHR) 请求。

RESTful 架构的理想选择:作为一个完整的框架,AngularJS 非常适合与 RESTful 架构一起使用。调用 REST CRUD API 非常容易,并且

服务:使用服务编写通用代码,在控制器中编写较少的代码。服务可用于在控制器之间共享通用功能。

可扩展性:Angular 使用 angular 指令扩展了 HTML 指令。在 html 中编写表达式并在运行时对其进行评估。创建您自己的指令和服务,并在另一个项目中使用它们,而无需任何额外的努力。