原文地址:http://www.sencha.com/blog/top-10-ext-js-development-practices-to-avoid/

作者:Sean Lanktree
Sean is an Ext JS Professional Services Lead at CNX Corporation.

CNX,尽管大多数的Ext JS开发工作需要从0开始创建新的应用程序,偶尔会有客户让我们帮他们解决内部工作上的性能问题、臭虫和结构性问题。我们以“清洁工”这种角色进行工作已经有很长一段时间了,在我们审查过的应用程序中,我们注意到,有一些共同的不明智的编码方法经常会出现。基于过去几年的审查工作,我们列出了十个我们建议的,在Ext JS应用程序中应该避免的开发方法。

1. 过多或不必要的组件嵌套

开发人员最常见的错误之一是没理由的嵌套组件。这样做,会影响性能和也会造成应用程序的不美观,如爽边框火意外的布局行为。在下面的示例1A,在面板内只包含了一个Grid。在这种情况下,该面板是不必要的。如示例1B所示,额外的面板可以取消。要记住的是,表单面板、树面板、标签面板和Grid面板都是从面板扩展的,隐藏,在使用这些组件的时候,应该特别注意不要的嵌套情况。

  1. items: [{
  2. xtype : 'panel',
  3. title: My Cool Grid’,
  4. layout: fit’,
  5. items : [{
  6. xtype : 'grid',
  7. store : 'MyStore',
  8. columns : [{...}]
  9. }]
  10. }]

示例1A  不好的:面板(panel)是不必要的

  1. layout: fit’,
  2. items: [{
  3. xtype : 'grid',
  4. title: My Cool Grid’,
  5. store : 'MyStore',
  6. columns : [{...}]
  7. }]

示例1B 好:Grid已经是面板,因而可以直接在Grid中使用任何面板属性

2. 清理未使用组件失败造成内存泄漏

许多开发人员不知道为什么他们的应用程序随着使用时间越长越来越慢。在用户浏览整个应用程序期间清理未使用组件失败是最大的一个原因。在下面的实例2A中,每次用户右键单击Grid的行,都会创建一个新的右键菜单。如果用户保持应用程序处于打开状态并右键单击行上百次,那么,就会有上百个永远不会被摧毁的右键菜单。对于开发人员和用户来说,应用程序“看上去”显示是争取的是因为只有最后一个被创建的菜单能显示在页面上,而且与的则是隐藏的。由于没有创建新菜单并没有清理旧的,应用程序的内存利用率就会不断增长,这最终将导致较慢的操作或浏览器崩溃。

示例2A就很好,由于右键菜单只在Grid初始化时创建一次,并在用户每次右键单击行时重复使用。不过,如果Grid被销毁,右键菜单一直存在,尽管它不再需要。最好的方式是示例2C,在Grid销毁的时候,把右键菜单也销毁。

  1. Ext.define('MyApp.view.MyGrid',{
  2. extend : 'Ext.grid.Panel',
  3. columns : [{...}],
  4. store: MyStore’,
  5. initComponent : function(){
  6. this.callParent(arguments);
  7. this.on({
  8. scope : this,
  9. itemcontextmenu : this.onItemContextMenu
  10. });
  11. },
  12.  
  13. onItemContextMenu : function(view,rec,item,index,event){
  14. event.stopEvent();
  15. Ext.create('Ext.menu.Menu',{
  16. items : [{
  17. text : 'Do Something'
  18. }]
  19. }).showAt(event.getXY());
  20.  
  21. }
  22. });

示例2A 不好:每一次右键单击都会创建菜单,且永远不会被销毁

  1. Ext.define('MyApp.view.MyGrid',{
  2. extend : 'Ext.grid.Panel',
  3. store : 'MyStore',
  4. columns : [{...}],
  5. initComponent : function(){
  6. this.menu = this.buildMenu();
  7. this.callParent(arguments);
  8. this.on({
  9. scope : this,
  10. itemcontextmenu : this.onItemContextMenu
  11. });
  12. },
  13.  
  14. buildMenu : function(){
  15. return Ext.create('Ext.menu.Menu',{
  16. items : [{
  17. text : 'Do Something'
  18. }]
  19. });
  20. },
  21.  
  22. onItemContextMenu : function(view,rec,item,index,event){
  23. event.stopEvent();
  24. this.menu.showAt(event.getXY());
  25. }
  26. });

示例2B 较好:菜单会在Grid创建时被创建,且每次可重用

  1. Ext.define('MyApp.view.MyGrid',{
  2. extend : 'Ext.grid.Panel',
  3. store : 'MyStore',
  4. columns : [{...}],
  5. initComponent : function(){
  6. this.menu = this.buildMenu();
  7. this.callParent(arguments);
  8. this.on({
  9. scope : this,
  10. itemcontextmenu : this.onItemContextMenu
  11. });
  12. },
  13.  
  14. buildMenu : function(){
  15. return Ext.create('Ext.menu.Menu',{
  16. items : [{
  17. text : 'Do Something'
  18. }]
  19. });
  20. },
  21.  
  22. onDestroy : function(){
  23. this.menu.destroy();
  24. this.callParent(arguments);
  25. },
  26.  
  27. onItemContextMenu : function(view,rec,item,index,event){
  28. event.stopEvent();
  29. this.menu.showAt(event.getXY());
  30. }
  31. });

示例2C 最好:Grid被销毁时,右键菜单也被销毁

3.怪物控制器

当看到应用程序拥有一个上千行代码的超级控制器的时候,我们不知道震惊了多少次。我们更倾向于根据应用程序功能拆分控制器。例如,订单处理应用程序可能会有划分条目、出货量、客户查找等控制器。这将使导航和维护代码更容易。

一些开发人员喜欢根据视图拆分控制器。例如,如果一个应用程序有一个Grid和表单,这将会使用一个控制器来管理Grid和使用一个控制器来管理表单。并没有一个“正确”的方式来划分控制器逻辑,只要是一致的就行。要记住,控制器可以与其他控制器进行通信。在示例3A,可以看到如何检索其他的控制器并调用其中的方法。

  1. this.getController('SomeOtherController').runSomeFunction(myParm);

示例3A 获取另一个控制器的引用并调用它的方法。

作为替代,也可以触发任何控制器可以监听到的应用程序层事件。在示例3B和3C,可以看到如何在一个控制器触发一个应用程序层事件并在另外一个控制器监听它。

  1. MyApp.getApplication().fireEvent('myevent');

示例3B 触发一个应用程序层事件

  1. MyApp.getApplication().on({
  2. myevent : doSomething
  3. });

示例3C 在另一个控制器监听应用程序层事件

注意: 自从Ext JS 4.2开始,使用多个控制器变得更容易了——他们可以触发其他控制器可以直接监听的事件

4.源代码的文件夹结构差

这虽然不影响性能和操作,但会让找跟进应用程序结构变得困难。随着应用程序的增长,如果源代码很有组织,那么寻找源代码以及增加特性或功能就会很容易。我们常常看到许多开发人员会将所有视图(即使是很大的应用程序)如示例4A一样放在一个目录。我们建议如示例4B所示通过逻辑功能来组织视图。

示例4A 不好:所有视图都处于同样层次

示例4B 好:视图根据逻辑功能进行组织

5.全局变量的使用

即使全局变量是不好的已经广为人知,但仍然会在我们审查过的一些应用程序中看到他们的身影。使用全局变量的应用程序可能会有名称冲突等重大问题,并且很难去调试。不使用全局变量,可以在类中定义“属性”,并引用这些属性的getter和setter。

例如,假设应用程序需要记录最后选择的客户。一般情况下会如示例5A那样在应用程序中定义一个变量,这很容易且值可以很便利的被应用程序的其他部分使用。

  1. myLastCustomer = 123456;

示例5A 不好:创建全局变量来存储最后的客户编号

作为替代,好的做法是创建一个用来保存属性的类来代替全局变量。在当前情况下,可以创建一个名为Runtime.js来保存应用程序中要使用到的运行属性。示例5B显示了Runtime.js在源代码结构中的位置。

示例5B Runtime.js文件的位置

示例5C显示了Runtime.js文件的内容,而示例5D显示了如何在app.js中“请求(require)”它。在应用程序中,就可以如5E或5F那样在任何地方“设置(set)”或“获取(get)”属性。

  1. Ext.define(‘MyApp.config.Runtime’,{
  2. singleton : true,
  3. config : {
  4. myLastCustomer : 0 // initialize to 0
  5. },
  6. constructor : function(config){
  7. this.initConfig(config);
  8. }
  9. });

示例5C 用来为应用程序保存全局属性的Runtime.js文件示例

  1. Ext.application({
  2. name : MyApp’,
  3. requires : [‘MyApp.config.Runtime’],
  4. ...
  5. });

示例5D 在app.js文件中请求Runtime类

  1. MyApp.config.setMyLastCustomer(12345);

示例5E 设置最后客户的方式

  1. MyApp.config.getMyLastCustomer();

示例5F 获取最后客户的方式

6. id的使用

我们不建议在组件上使用id,因为每一个id必须是唯一的。而这样很容易导致不小时使用了相同的id,从而引起重复的DOM id(名称冲突)。相反,应该让框架来处理id的生成。使用Ext JS的组件查询,没有任何理由要为Ext JS组件指定一个id。示例6A显示了一个应用程序中的两段代码,在这里创建了两个不同的保存按钮,而这两个保存按钮的id都为“savebutton”,从而导致了名称冲突。显而易见,在下面的代码很容易找出名称冲突,但在一个大型应用程序中,要确定名称冲突会很困难。

  1. // here we define the first save button
  2. xtype : 'toolbar',
  3. items : [{
  4. text : Save Picture’,
  5. id : 'savebutton'
  6. }]
  7.  
  8. // somewhere else in the code we have another component with an id of ‘savebutton’
  9. xtype : 'toolbar',
  10. items : [{
  11. text : Save Order’,
  12. id : 'savebutton'
  13. }]

示例6A 不好:为组件分配了重复id将造成名称冲突

替代方法是,如果需要手动确定每一个组件的,可以如示例6B那样使用itemid代替id。这就可以解决命名冲突,并且仍然可以通过itemid来引用组件。通过itemid有许多方法来获取组件的引用。示例6C列出了部分方法。

  1. xtype : 'toolbar',
  2. itemId : picturetoolbar’,
  3. items : [{
  4. text : 'Save Picture',
  5. itemId : 'savebutton'
  6. }]
  7.  
  8. // somewhere else in the code we have another component with an itemId of ‘savebutton’
  9. xtype : 'toolbar',
  10. itemId: ordertoolbar’,
  11. items : [{
  12. text : Save Order’,
  13. itemId: savebutton
  14. }]

示例6B 好:使用itemid来创建组件

  1. var pictureSaveButton = Ext.ComponentQuery.query('#picturetoolbar > #savebutton')[0];
  2.  
  3. var orderSaveButton = Ext.ComponentQuery.query('#ordertoolbar > #savebutton')[0];
  4.  
  5. // assuming we have a reference to the “picturetoolbar” as picToolbar
  6. picToolbar.down(‘#savebutton’);

示例6C 好:使用itemid引用组件

7.不可靠的组件引用

有时候会看到代码利用组件位置来引用组件。应该避免出现这样的情况,因为有任何条目被增加、移除或嵌入不同的组件,就会导致错误。示例7A显示了几种常见情况。

  1. var mySaveButton = myToolbar.items.getAt(2);
  2.  
  3. var myWindow = myToolbar.ownerCt;

示例7A 不好:避免基于组件位置来获取组件引用

替代方法是,如示例7B那样使用组件查询或者最佳的up或down方法来返回引用。使用这种技术,代码就很少会因以后的结构或组件位置变化而导致错误。

  1. var mySaveButton = myToolbar.down(‘#savebutton’); // searching against itemId
  2.  
  3. var myWindow = myToolbar.up(‘window’);

示例7B  好:使用组件查询来返回相关引用

8. 不遵守大写/小写命名约定

Sencha在为组件、属性或xtype等等命名的时候,会遵循某些大写/小写标准。为了避免混淆,保持代码清洁,应对遵循相同的标准。示例8A显示几个不正确的情形。示例8B显示了在相同情况下,使用正确的大写/小写命名约定的情形。

  1. Ext.define(‘MyApp.view.customerlist’,{ // should be capitalized and then camelCase
  2. extend : Ext.grid.Panel’,
  3. alias : widget.Customerlist’, // should be lowercase
  4. MyCustomConfig : xyz’, // should be camelCase
  5. initComponent : function(){
  6. Ext.apply(this,{
  7. store : Customers’,
  8. ….
  9. });
  10. this.callParent(arguments);
  11. }
  12. });

示例8A 粗体显示的地方为不正确的大写/小写命名

  1. Ext.define(‘MyApp.view.CustomerList’,{
  2. extend : Ext.grid.Panel’,
  3. alias : widget.customerlist’,
  4. myCustomConfig : xyz’,
  5. initComponent : function(){
  6. Ext.apply(this,{
  7. store : Customers’,
  8. ….
  9. });
  10. this.callParent(arguments);
  11. }
  12. });

示例8B 粗体显示的地方为正确的大写/小写命名

另外,如果触发任何自定义事件,事件的名称应对是小写的。当然,不遵循这些约定,一切都仍然会工作,但为什么要迷失在标准之外并编写不太干净的代码?

9. 将组件约束在父组件的布局

在示例9A,面板总是会有“region:center”属性,因此,当想重用它的时候就可能行不通,例如将它放到“west”区域。

  1. Ext.define('MyApp.view.MyGrid',{
  2. extend : 'Ext.grid.Panel',
  3. initComponent : function(){
  4. Ext.apply(this,{
  5. store : MyStore’,
  6. region : 'center',
  7. ......
  8. });
  9. this.callParent(arguments);
  10. }
  11. });

示例9A 坏:“center”区域不应该放在这里

代替方法是,如示例8B那样,在创建组件的时候才指定布局配置。这样,就可以将组件重用到任何你希望的地方,切不受布局配置的约束。

  1. Ext.define('MyApp.view.MyGrid',{
  2. extend : 'Ext.grid.Panel',
  3. initComponent : function(){
  4. Ext.apply(this,{
  5. store : MyStore’,
  6. ......
  7. });
  8. }
  9. });
  10.  
  11. // specify the region when the component is created...
  12. Ext.create('MyApp.view.MyGrid',{
  13. region : 'center'
  14. });

示例9B 好:在创建组件的时候才知道区域

如示例9C所示,也可以为组件提供一个默认的区域,如果需要,可以重写它。

  1. Ext.define('MyApp.view.MyGrid',{
  2. extend : 'Ext.grid.Panel',
  3. region : 'center', // default region
  4. initComponent : function(){
  5. Ext.apply(this,{
  6. store : MyStore’,
  7. ......
  8. });
  9. }
  10. });
  11.  
  12. Ext.create(‘MyApp.view.MyGrid’,{
  13. region : north’, // overridden region
  14. height : 400
  15. });

示例9C 也很好:指定默认区域,在需要的时候重写

10. 使代码比所需的更复杂

有很多时候,我们会看到比所需更复杂的代码。这通常是由于对每个组件的可用方法不熟悉造成的。最常见的一种情况是,为每一个表单字段单独从数据记录中加载数据。示例10A就显示这种情况。

  1. // suppose the following fields exist within a form
  2. items : [{
  3. fieldLabel : User’,
  4. itemId : username
  5. },{
  6. fieldLabel : Email’,
  7. itemId : email
  8. },{
  9. fieldLabel : Home Address’,
  10. itemId : address
  11. }];
  12.  
  13. // you could load the values from a record into each form field individually
  14. myForm.down(‘#username’).setValue(record.get(‘UserName’));
  15. myForm.down(‘#email’).setValue(record.get(‘Email’));
  16. myForm.down(‘#address’).setValue(record.get(‘Address’));

示例10A 不好:为表单字段单独的从记录中加载数据

替代单独加载每一个值的方法是,使用loadRecord方法从记录中为所有字段的数据到表单字段,这只需要一行代码。如示例10B所示,这里的关键是确保表单字段的name属性和记录的字段名称是一样的。

  1. items : [{
  2. fieldLabel : User’,
  3. name : UserName
  4. },{
  5. fieldLabel : Email’,
  6. name : Email
  7. },{
  8. fieldLabel : Home Address’,
  9. name : Address
  10. }];
  11.  
  12. myForm.loadRecord(record);

示例10B 好:使用loadRecord方法只需要一行代码就可加载所有表单字段

这只是比必需的更复杂的代码的其中一个例子。而当中的重点是要审视组件的所有方法和和示例,以确保正在使用简单和适当的技术。

CNX公司是Sencha认证选择合作伙伴。Sencha合作伙伴网络是Sencha专业服务器团队的宝贵扩展。

自从1996年以来,CNX一直处于自定义商业应用程序开发的先行者。在2008年,CNX在Ext JS上对它的基于浏览器的用户界面开发进行了标准化,在2010年,又添加了Sencha Touch作为移动开发的标准。我们已经在世界各地,为教育、金融、食品、法律、物流、制造、出版和零售等许多行业的客户创建了一流的Web应用程序。我们的开发团队以芝加哥市中心的公司办公室为基地,可以处理任何规模的项目。CNX可以独立工作或与您的团队一起,以快速、经济高效的方式去实现项目目标。请查阅我们的网站http://www.cnxcorp.com。

【翻译】十大要避免的Ext JS开发方法的更多相关文章

  1. 十大要避免的Ext JS开发方法

    原文地址:http://www.sencha.com/blog/top-10-ext-js-development-practices-to-avoid/ 作者:Sean LanktreeSean i ...

  2. [ExtJS5学习笔记]第二十九节 sencha ext js 5.1.0中动态更换皮肤主题

    本文地址:http://blog.csdn.net/sushengmiyan/article/details/42016107 本文作者:sushengmiyan ------------------ ...

  3. Ext JS 6开发实例(一)

    很久没写文章了,主要原因和大家差不多,都要为生活奔忙,搞了两个小项目.这两个小项目很凑巧,都可以使用Ext JS来开发,这正是练习使用Ext JS 6的好机会,自然不会错过. 很多读者可能会问,为什么 ...

  4. 【翻译】Ext JS——高效的编码风格指南

    原文:ExtJS - Efficient coding style guide 作者:Raja 切勿使用"new"关键字:在Ext JS中,使用"new"关键字 ...

  5. Ext JS 6学习文档–第1章–ExtJS入门指南

    Ext JS 入门指南 前言 本来我是打算自己写一个系列的 ExtJS 6 学习笔记的,因为 ExtJS 6 目前的中文学习资料还很少.google 搜索资料时找到了一本国外牛人写的关于 ExtJS ...

  6. Ext JS 6 新特性和工具

    Ext JS 6 新特性和工具 Ext JS 6 带来很多新特性.工具和改进.以下是一些亮点: • 合并了 Ext JS & Sencha Touch - 在 Ext 6, 你可以访问 Ext ...

  7. Ext Js详解指南

    什么是Ext JS 走进Ext的世界 Ext JS是一款富客户端开发框架它基于javascript.HTML和CSS开发而成,无需安装任何插件即可在常用浏览器中创建出绚丽的页面效果. 个人总结Ext ...

  8. 谈谈Ext JS组件之引子

    Ext JS组件,对于Ext JS开发人员来说,应当不会陌生,毕竟做开发,都必须与它打交道.对于这样一个大家都熟悉的东西,为什么要用一个专题的形式来写呢?是否有这方面的需要?还不如去写点使用技巧? 确 ...

  9. Ext JS 5 beta版发布

    原文:Announcing Public Beta of Ext JS 5 我们非常高兴的宣布,Sencha Ext JS 5 beta版本开始进行公测了.这个beta版本可以让你.我们Sencha社 ...

随机推荐

  1. linux+nginx+mysql+php高性能服务器搭建

    1.安装基础包 yum -y install gcc gcc-c++ autoconf libjpeg libjpeg-devel libpng libpng-devel freetype freet ...

  2. 用来解析,格式化,存储和验证国际电话号码:libphonenumber

    用来解析,格式化,存储和验证国际电话号码:libphonenumber libphonenumber是Google的公共Java.C++和Javascript库用来解析,格式化,存储和验证国际电话号码 ...

  3. WCF技术剖析之二十三:服务实例(Service Instance)生命周期如何控制[下篇]

    原文:WCF技术剖析之二十三:服务实例(Service Instance)生命周期如何控制[下篇] 在[第2篇]中,我们深入剖析了单调(PerCall)模式下WCF对服务实例生命周期的控制,现在我们来 ...

  4. jsp页面中格式化为小数点两位

    <td align="center">  <% String avgnum = ""; if(request.getAttribute(&qu ...

  5. xmlns:android="http://schemas.android.com/apk/res/android的作用是

    xmlns:android="http://schemas.android.com/apk/res/android的作用是 这个是xml的命名空间,有了他,你就可以alt+/作为提示,提示你 ...

  6. mysql启动的四种方式

    mysql的四种启动方式: .mysqld 启动mysql服务器:./mysqld --defaults-file=/etc/my.cnf --user=root 客户端连接: mysql --def ...

  7. (诊断)为GitHub添加SSH key时出现“Could not open a connection to your authentication agent”错误的应对方案(转)

    在为windows 环境下的github账户添加SSH key时,需要在Git Bash执行如下命令: 第一步:检查已有的SSH keys $ ls -al ~/.ssh 第二步:生成新的SSH ke ...

  8. opencv实现连通域

    在本文中使用图像连通域统计使用opencv中的cvFloodFill方法,可是在cvFloodFill方法中CvConnectedComp參数无法返回详细点坐标位置信息,找了些资料.给CvSeq分配空 ...

  9. Windows下sass的安装

    sass依赖Ruby,所以,首先得先安装个Ruby 安装步骤: 1.安装Ruby的时候,勾上Add Ruby executables to your PATH(添加环境变量) 2.安装好Ruby之后, ...

  10. sql: sql developer tunnel转接

    Use putty tunnel instead of login terminal server 有时候本地直接ping不通sql 的server, 但另一个server能连上,这时就可以把端口和i ...