`
iamweiming
  • 浏览: 41312 次
  • 性别: Icon_minigender_1
  • 来自: 苏州
社区版块
存档分类
最新评论

Why SeaJS

 
阅读更多

前言

本文主要面向刚接触 SeaJS 的同学。
文章会先提出传统 Javascript 开发上遇到的一些难以解决的问题(即“冲突”与“依赖”两节),然后介绍如何使用 SeaJS 来解决这些难点(即 “Why SeaJS” 一节)。
实际上,如果你想了解 RequireJS 等其他模块加载器,也可以阅读本文

冲突

我们从一个最简单的例子开始
以前我做项目时,常常会将一些通用的、底层的功能抽出来,独立成一个函数,比如

function print(str) {
    // 代码!
};

然后像模像样的将这个函数丢到 util.js 里面,最后告诉大家:你们想用 xx 功能时候,引入 util.js 就行啦
这是一个很好的习惯(嘻嘻),但是项目做久了,难免会遇到问题,比如常有人问我:

  • 你这个函数为啥叫 print 啊!正好我也刚写了个函数叫 print 啊!要不我改名叫 print2 吧!(我靠……)
  • 你这个函数为啥叫 print 啊!我引入了个开源的模块,里面也有个函数叫 print 啊!改人家的怕有问题,要不改你的吧!(我再靠……)

在没有命名空间的 Javascript 中,发生这样的情况一点都不奇怪,幸亏有补救方案,可以用对象模拟命名空间

var FocusTech = {};
FocusTech.print = function(str){
    // 代码!
}

这样需要调用这个函数时,得带上命名空间 FocusTech.print(‘我是字符串啊’) 
虽然麻烦一点,不过能降低函数命名冲突的概率,可惜也只是降低而已
比如这里我选择了用公司名字作为命名空间,但是我们公司几百号研发人员,跟我一样想法的人肯定大有人在
所以命名空间依然有可能会冲突。那么为了继续降低冲突概率,只好拉长命名空间,比如学 Java 用项目的网址做命名空间
下面这段代码节选自 Yahoo! 的一个开源项目

if (org.cometd.Utils.isString(response)) {
    return org.cometd.JSON.fromJSON(response);
}
if (org.cometd.Utils.isArray(response)) {
    return response;
}

看到这里我其实还是挺同情他们的,为了避免冲突用了那么长一个命名空间,对记忆和书写的负担实在太重了
好了,冲突的问题暂且放着,我们继续往下看

依赖

继续讲简单的例子,我开始编写一个通用的展示组件,提供给项目组使用,这样其他同学就不用重复造轮子了
我告诉大家:组件写在 componet_one.js 中,你们要用的时候引入一下就行啦
于是另一位同学就这么做了:

<script src="componet_one.js"></script>
<script>
  FocusTech.ComponetOne.init();
  // 代码!
</script>

看上去很好,可是报错了!
我赶紧查找原因,发现原来是我的组件中,调用了上面一个例子中提供的 print 方法,而页面中没有引入 util.js ,赶紧加上!

<script src="util.js"></script>
<script src="componet_one.js"></script>

很快就修复了,看上去好像挺简单的,不过让我们看看这个例子的后续发展:

  • 某天,我扩充了组件的功能,除了需要 util.js 之外,还需要 ooxx.js

这时候,项目中已经有 N 个地方用到了我的组件……
于是我只好全局搜索每一个调用的地方,都给页面加上对 ooxx.js 的引用

  • 某天,需求砍掉了组件上的一个交互效果

我修改了 componet_one.js 的代码,然后发现我不再需要 util.js 中提供的 print 方法了
于是我再次全局搜索每一个调用的地方,去掉对 util.js 的引用

  • 测试同学告诉我,改完之后,好多个页面报错!

我赶紧检查,发现有些页面上是这样的

<script src="ooxx.js"></script>
<script src="componet_one.js"></script>
<script src="componet_two.js"></script>

页面还引入了 componet_two.js ,而且 componet_two.js 中用到了 util.js 的 print 方法!
所以 BUG 原因找到了,我不能武断地把 util.js 全都去掉,而是需要仔细检查页面上的其他的 JS 文件或代码是否需要它!
崩溃……

  • 一段时间后,测试同学又来找我,说页面又报错了!

再次检查,发现原来是某人动过了 ooxx.js ,在里面调用了 util.js 的方法……害死人不偿命啊!
好吧,我只好再次全局搜索所有使用到 componet_one 组件的地方,给页面加上 util.js 的引用

小结

我已经不忍心再讲下去了,正所谓:看到哪句你哭了,不顶不是中国人!
相信做过大一点项目的同学,应该都遇到过上面的这种破事。
为什么会这么费神呢?因为 javascript 的语法中天生缺少引入其他 JS 文件的语法。
相比之下,我们的好战友设计师们就轻松很多了,比如:

@import url("base.css");

#test {...}
.classA {...}
.classB {...}

看到吧,人家只要用一个 @import 就解决了,页面中引入 css 时,只需要引入这个 css ,浏览器会自动去下载 base.css
css 文件的依赖能够实现自动管理,而不是像 js 一样,需要手动地去编写

OK,下面终于可以进入重点了

Why SeaJS

先不谈理论,直接来看看上面的这两个例子中如果引入 SeaJS 该怎么写
首先是 util.js ,我们改用 SeaJS 的 CMD 规范来书写

define(function (require, exports) {
    function print(str) {
        // ...代码
    }

    exports.print = print;
});

可以看到,其实改变并不大,主要是外部包裹了一层,再加最后多写了一行
最后这行很重要,通过它,文件对外提供一个叫做 print 方法的接口
另外大家发现没,这里我没有使用命名空间,为什么呢?看下面的 componet_one.js

define(function (require, exports) {
    var util = require('./util.js');

    var ComponetOne = {
        doSth : function() {
            util.print('我是字符串');
            // ...代码
        }
    }

    return ComponetOne;
});

高潮终于到了!
首先是 var util = require('./util.js'); 这句,有没有觉得很熟悉?是不是很像上面提到的 @import url("base.css"); ?
没错!这里的 require 可以认为是 SeaJS 给 Javascript 增加的一个关键词语法,就像 @import 这个关键词一样!

通过这个函数,SeaJS 赋予了 Javascript 直接加载 js 文件的功能,并且这个函数是 同步 的!
函数直接就有一个返回值,那么返回值是什么呢?相信你已经猜到了,就是 util.js 里面的 exports 对象
所以下面我们就可以使用 util.print() 来调用 util.js 提供的对外接口了

刚才那个问题有答案了:因为 require 函数的返回值赋值给哪个变量完全由 componet_one 决定,变量名可以随便起,与 util.js 毫无关系(这里为了方便变量也叫 util ),所以 util.js 里面自然就不需要命名空间了!

那么页面中该怎么引入呢?很简单

<script src="sea.js"></script>
<script>
seajs.use('./componet_one.js')
</script>

我们需要先引入 SeaJS 这个加载器,然后通过它提供的 API 来加载文件
而且最重要的是:因为有了 require ,这里我们只需要加载 componet_one.js 即可,util.js 会由 SeaJS 自动加载!
也就是说,因为有了 require ,我们获得了类似 css 的文件依赖自动管理机制!

小结

仔细琢磨一下这几行代码,我想大家应该能看出 SeaJS 带给 Javascript 开发的巨大好处:

  • 一般来说,不再需要冗长的命名空间了,也不再需要担心命名的冲突
  • 不需要手动管理 js 文件的依赖,依赖关系写在代码中,SeaJS 会自动处理

别小看这两点,对于稍微有些规模的项目来说它们非常重要,我想这也是 SeaJS 备受青睐的原因。
最后,SeaJS 的 CMD 规范非常简单易懂,SeaJS 的 API 也非常简洁优雅,相信有一定开发经验的同学很快就能上手
更多信息可以移步 SeaJS官网

 

转:http://chaoskeh.com/blog/why-seajs.html

分享到:
评论

相关推荐

    seajs-5分钟上手

    Seajs是中国开源社区推出的一款浏览器端的模块加载器,它借鉴了CommonJS的规范,但针对浏览器环境进行了优化,使得JavaScript在浏览器端也能实现模块化的开发。本教程将带你快速了解并掌握Seajs的使用,让你在5分钟...

    seajs js 模块加载器

    SeaJS 是一款专为Web端设计的JavaScript模块加载器,它的出现是为了解决JavaScript在浏览器环境中的组织和管理问题。随着Web应用的复杂度不断提升,JavaScript代码的组织和依赖管理变得至关重要,SeaJS 提供了一种...

    seajs官方小实例

    **Seajs 官方小实例详解** Seajs 是一个用于浏览器端的模块加载器,它遵循 CommonJS 规范,让 JavaScript 开发者能够在浏览器环境中实现模块化开发,提高代码的可维护性和复用性。Seajs 的核心理念是通过模块化解决...

    seajs下载以及规范

    Seajs是中国开源社区推出的一款基于模块化开发的前端加载器,它借鉴了CommonJS的模块化思想,但针对浏览器环境进行了优化。Seajs的核心理念是让JavaScript模块化变得简单,帮助开发者解决在大型Web项目中代码组织、...

    seajs-2.3.0.zip

    Seajs是中国开源社区非常受欢迎的一款JavaScript模块加载器,它的出现为Web开发引入了CommonJS规范,使得前端开发更加模块化,便于代码管理和维护。Seajs 2.3.0是该库的一个稳定版本,提供了丰富的功能和优化。 一...

    seajs手册,教程

    Seajs是中国开源社区发展出来的一款轻量级的前端模块化加载器,它的出现是为了应对JavaScript在浏览器端组织和管理代码的复杂性。本手册将详细阐述Seajs的核心概念、使用方法以及如何通过它来构建高效的前端应用。 ...

    seajs_2.3.0以及学习文档

    Seajs 是一个轻量级的前端模块加载器,它的出现是为了在浏览器端实现 CommonJS 规范,使得 JavaScript 的组织和开发变得更加模块化。Seajs 版本 2.3.0 是其稳定的一个版本,提供了更完善的特性和优化。 在...

    seajs-2.2.3

    Seajs-2.2.3 是一个开源的 JavaScript 模块加载器,旨在为开发者提供简单、极致的模块化开发体验。它强调的是简洁、自然的代码书写和组织方式,使得项目管理和协同工作变得更加高效。Seajs 在JavaScript社区中扮演着...

    seajs入门完整案例,适合新手学习

    Seajs 是一个基于浏览器的模块加载器,它遵循 CommonJS 规范,使得在浏览器环境中可以实现模块化开发。这个“seajs入门完整案例”针对初学者提供了学习资源,帮助他们快速掌握Seajs的基本用法和特性。 1. **模块化...

    seajs源码阅读

    **Seajs源码阅读** Seajs是一款轻量级的JavaScript模块加载器,它遵循CommonJS规范,使得在浏览器端也能实现模块化的开发。通过阅读Seajs的源码,我们可以深入理解JavaScript模块化的工作原理,以及如何实现一个...

    seajs-2.2.0源码

    Seajs是中国著名的前端模块加载器,它借鉴了CommonJS的规范,使得JavaScript在浏览器环境中也能实现模块化的开发。本文将深入解析Seajs 2.2.0版本的源码,探讨其核心机制、设计理念以及如何使用它来组织和管理前端...

    seajs相关模板

    Seajs是中国开源社区发展出来的一款模块化加载器,它的出现是为了应对JavaScript在浏览器环境中缺乏一个标准的模块化解决方案的问题。Seajs的设计理念是基于CMD(Common Module Definition)规范,它鼓励开发者采用...

    CMD规范之seajs

    CMD规范之Seajs Seajs是一款轻量级的前端模块加载器,它遵循CommonJS规范,使得JavaScript在浏览器端也能实现模块化的开发。这个规范主要包含模块定义、模块加载和模块化开发的思想,大大提高了代码的可维护性和...

    seaJS框架简介和完整例子

    SeaJS 是一个轻量级的前端模块加载器,它的出现是为了改善JavaScript在浏览器环境中的组织和加载方式。本文将深入探讨SeaJS的基本概念、工作原理以及如何通过一个完整的例子来理解其用法。 SeaJS的核心理念是遵循 ...

    seajs源代码

    Seajs是一款轻量级的JavaScript模块加载器,它遵循CommonJS规范,旨在解决浏览器环境中的模块化问题。在深入理解seajs源代码之前,我们首先需要了解模块化的基本概念和CommonJS规范。 模块化是软件开发中的一种组织...

    SeaJS与RequireJS区别

    **SeaJS 和 RequireJS 是两种广泛使用的 JavaScript 模块加载器和依赖管理工具,它们的主要目的是解决 JavaScript 在浏览器端的异步加载和模块化问题。** **SeaJS** SeaJS 是一个轻量级的模块加载器,由中国淘宝...

    前端开源库-grunt-react-seajs

    "grunt-react-seajs"就是一个这样的工具,它是一个专门为前端开发者设计的Grunt插件,旨在帮助开发者将React组件转换为SeaJS模块,使得在不支持ES6模块语法的环境中也能顺利地使用React组件。下面我们将深入探讨这个...

    前端项目-seajs.zip

    【前端项目-seajs.zip】是一个包含前端开发工具的压缩包,主要关注的是Seajs模块加载器,这是一个专为Web设计的模块化解决方案。Seajs在前端开发领域扮演着重要角色,因为它提供了一种组织和管理JavaScript代码的...

    seajs-css.js

    seajs-css.js 用于seajs加载css文件,上课用的课件资源,大家有需要的可以随时下载。版本是非常好用的,作为一个学习模块化我们必须的知识插件

    knockout seajs .net中实战应用

    总的来说,"knockout seajs .net中实战应用"意味着在.NET开发中,利用Knockout.js的强大数据绑定能力和Sea.js的模块化管理,可以构建出高效、响应式的前端应用。通过Ajax通信,前端和后端能够无缝协作,提供流畅的...

Global site tag (gtag.js) - Google Analytics