JSRUN 用代码说话

路由、模板的执行、渲染顺序

编辑教程

路由、模板的执行、渲染顺序

在Ember中路由和模板的执行都是有一定顺序的,它们的顺序为:主路由->子路由1->子路由2->子路由3->……。模板渲染的顺序与路由执行顺序刚好相反,从最后一个模板开始解析渲染。

注意:模板的渲染是在所有路由执行完之后,从最后一个模板开始。关于这一点下面的代码会演示验证,官网教程有介绍,点击查看。

路由与模板是相对应的,所以模板的目录结构与路由的目录结构是一致的。 你有两种方式构建上述目录:

  • 手动创建
  • 使用命令,比如创建comment.js使用命令:ember generate route posts/detail/comments/comment,Ember CLI会自动为我们创建目录和文件。

创建好目录结构之后我们添加一些代码到每个文件。

按前面讲的路由执行顺序分别列出每个文件的内容。

//  app/routes/posts.js

import Ember from 'ember';

export default Ember.Route.extend({
    model: function() { 
        console.log('running in posts...');
        return { id: 1, routeName: 'The route is posts'};
        // return Ember.$.getJSON('https://api.github.com/repos/emberjs/ember.js/pulls');
    }

});
import Ember from 'ember';
export default Ember.Route.extend({

    model: function(params) {
        console.log('params id = ' + params.post_id);
        console.log('running in detail....');

        return { id: 1, routeName: 'The route is detail..' };
    }
});
//  app/routes/posts/detail.js

import Ember from 'ember';

export default Ember.Route.extend({

    model: function(params) {
        console.log('params id = ' + params.post_id);
        console.log('running in detail....');

        return { id: 1, routeName: 'The route is detail..' };
    }
});
//  app/routes/posts/detail/comments.js

import Ember from 'ember';

export default Ember.Route.extend({
    model: function() {
        console.log('running in comments...');
        return { id: 1, routName: 'The route is comments....'};
    }
});
//  app/routes/posts/detail/comments/comment.js

import Ember from 'ember';

export default Ember.Route.extend({
    model: function(params) {
        console.log('params id = ' + params.post_id);
        console.log('running in comment...');
        return { id: 1, routeName: 'The route is comment...'};
    }
});

下面是模板各个文件的内容。其列出才顺序与路由的顺序一致。

{{model.routeName}} >> {{outlet}}
{{model.routeName}} >> {{outlet}}
{{model.routeName}} >> {{outlet}}
{{model.routeName}} >> {{outlet}}

路由执行的顺序

  • 执行主路由(默认是application),此时进入到路由的model回调方法,并且返回了一个对象{ id: 1, routeName: 'The route is application...' },执行完回调之后继续转到子路由执行直到最后一个路由执行完毕,所有的路由执行完毕之后开始渲染页面。

  • 页面的渲染则是从最后一个路由对应的模板开始,并沿着最近的父模板往回渲染。 为了验证是否是这样的执行顺序,下面修改detail.js和 comments.js。在代码中加入一个模拟休眠的操作。

//  app/routes/posts/detail.js
import Ember from 'ember';

export default Ember.Route.extend({

model: function(params) {
    console.log('params id = ' + params.post_id);
    console.log('running in detail....');

    //  执行一个循环,模拟休眠
    for (var i = 0; i < 10000000000; i++) {

    }
    console.log('The comment route executed...');

    return { id: 1, routeName: 'The route is detail..' };
}
});
//  app/routes/posts/detail/comments.js

import Ember from 'ember';

export default Ember.Route.extend({
    model: function(params) {
        console.log('params id = ' + params.post_id);
        console.log('running in comment...'); 
        //  执行一个循环,模拟休眠
        for (var i = 0; i < 10000000000; i++) {

        }

        return { id: 1, routeName: 'The route is comment...'};
    }
});

刷新页面,注意查看控制台输出信息和页面显示的内容。

新开一个窗口,执行URL:http://localhost:4200/posts/2/comments。

控制台输出到这里时处理等待(执行for循环),此时已经执行了两个路由application和posts,并且正在执行detail,但是页面是空白的,没有任何HTML元素。

在detail路由执行完成之后转到路由comments。然后执行到for循环模拟休眠,此时页面仍然是没有任何HTML元素。然后等到所有route执行完毕之后,界面才显示model回调里设置的信息。

每个子路由设置的信息都会渲染到最近一个父路由对应模板的{{outlet}}上面。

  • 渲染comment 得到的内如为:“comment渲染完成”

  • 渲染comment最近的父模板comments 得到的内容为:“comment渲染完成 comments渲染完成”

  • 渲染comments最近的父模板detail 得到的内容为:“comment渲染完成 comments渲染完成 detail渲染完成”

  • 渲染detail最近的父模板posts 得到的内容为:“comment渲染完成 comments渲染完成 detail渲染完成 posts渲染完成”

  • 渲染posts最近的父模板application 得到的内容为:“comment渲染完成 comments渲染完成 detail渲染完成 posts渲染完成 application渲染完成”

只要记住一句话:子模板的都会渲染到父模板的{{outlet}}上,最终所有的模板都会被渲染到application的{{outlet}}上。

JSRUN闪电教程系统是国内最先开创的教程维护系统, 所有工程师都可以参与共同维护的闪电教程,让知识的积累变得统一完整、自成体系。 大家可以一起参与进共编,让零散的知识点帮助更多的人。
X
支付宝
9.99
无法付款,请点击这里
金额: 0
备注:
转账时请填写正确的金额和备注信息,到账由人工处理,可能需要较长时间
如有疑问请联系QQ:565830900
正在生成二维码, 此过程可能需要15秒钟