AdonisJS 入门篇

作者: 唐紫依 | 来源:发表于2017-03-03 00:32 被阅读1896次

    参考资料

    前言

    本篇文章仅对 AdonisJS 框架作简要介绍及简单语法提及,详细语法跟配置请参考官方文档。本篇的目的在于看完之后能够大体了解 AdonisJS 的架构,并实现简单的功能。

    零、简单介绍及安装

    1.Introduce

    • AdonisJS 官方介绍是 MVC Framework for Node.js to write webapps with less code,简单的说就是一个 服务端渲染 的 MVC 框架。如果熟悉 Laravel 框架(PHP 框架),你会发现这是 Laravel 的一个 NodeJS 版本。
    • 优点:AdonisJS 作为一个服务端渲染的框架,能够解决很多现在流行的前端框架无法实行的问题(比如兼容 IE8 等)

    2.Install

    • 既然 AdonisJS 是依赖于 NodeJS,那么安装自然离不开 NodeJS 和 npm。官方指定版本(当然,官方也建议 npm 版本 >= 6.1.x):
    node 版本 npm 版本
    • 创建 Adonis project 很简单,因为 Adonis 本身就是一个 npm package,可以运行以下代码启动一个示例 project:
    npm i -g adonis-cli           // 全局安装 adois
    adonis new awesome-project    // 当前目录下创建新工程
    cd awesome-project    
    // 然后你需要把 .env-example 文件改名为 .env
    npm run serve:dev    // 启动 project
    

    3.Directory structure

    文件夹结构图

    一、应用生命周期

    1.Http Server

    • Providers 和它们的参数被定义在 bootstrap/app.js
    • 自定义 Provider 在 providers/, provider 文件设置注册信息(注册为异步注册 ES2015 Generator)及依赖的模块
    • 设置参数是为了方便引用 Provider 时书写
    // bootstrap/app.js
    const provider = []
    const aceProviders = []
    const aliases = []
    const commads = []
    
    • bootstrap/http.js 文件引用 Providers 并注册在 IoC container
    // bootstrap/http.js
    
    // 引用 node_modules 提供的 providers
    const app = require('./app')
    const fold = require('adonis-fold')
    const path = require('path')
    const packageFile = path.join(__dirname, '../package.json')
    require('./extend')
    // 注册
    module.exports = function (callback) {
        ...
    }
    
    • 注册定义在 package.json 自动加载路径
    // 可以在此配置
    {
        "autoload": {
            "App": "./app"
        }
    }
    
    • 根据 bootstrap/http.js 定义的结果继续加载以下文件:

    • bootstrap/event.js: 注册监听事件

    • app/Http/kernel.js: Http server & 中间件

    • app/Http/routes.js: Http server & 路由

    • database/factory.js: Model 管理

    • 最后,启动 HTTP server 监听定义在 .env 文件的 hostport

    • 启动 HTTP server 之前,AdonisJS 提供了 Http.onStart 钩子,可用来自定义全局变量,如 View 的 全局变量过滤器 等。

    2.Http Request

    • HTTP request 是在给定时间处理一个或多个 HTTP请求的 动态流

    • 根据 public 文件夹里边的静态文件检查传入的请求 URL ,如果存在则提供静态文件。

    • 接下来,将根据请求的 URL ,搜索相应的路由,并且执行以下步骤:

    • 启动所有全局中间件

    • 启动所有路由中间件

    • 执行路由操作(可以是 控制器方法 或者 闭包

    • 如果以上两个步骤出错,将会抛出一个 404 HttpException。可以在 APP/httP.js 设置 error 时的 sendView(),以提高用户体验。

    二、MVC 设计模式

    本部分仅对 MVC 模式作简要介绍,后续将进一步介绍 View 和 Controller。

    1.概述

    • 根据上节生命周期的说明,我们大致可以知道整个架构的数据流程如果下图所示:
    MVC 设计模式示意图
    • 这便是我们常说的 MVC 设计模式。MVC 模式将应用分离成模型(Model),视图(View)和控制器(Controller)。AdonisJS 支持这三者且使其结合起来非常简单。而且,Adonis Router 也在处理 HTTP 请求并将其传递给控制器中起到重要作用。

    2.Model

    • Model 是从数据库(AdonisJS 使用的是 SQL)获取数据的数据层,为了使获取数据的过程简单和安全,Adonis 使用了很好的 ORM 架构(Lucid)。

    3.Controller

    • 控制器主要控制 HTTP 请求数据流,利用 Model 获取必要的数据(不仅数据库,第三方服务器的数据等都在控制器里边处理),并把获取的数据传递给 View 去渲染 HTML 页面。

    4.View

    • View 是整个 MVC 模式数据流的最后部分,主要是利用动态的数据(Nunjuck Template)去渲染(jinja 引擎) HTML 页面。

    三、Rounting

    本节简单介绍路由,以及常用的几个语法

    • 根据上节的数据流程所讲,接下来我们先介绍路由。路由的作用便是将浏览器的 HTTP 请求传递到相应的控制器中。当然,你也可以直接在路由部分作简单的逻辑处理,不过在这里比较建议将逻辑处理划分到 控制器 中,方便代码的管理。

    • 路由部分的代码书写在 app/Http/routes.js 中,简单的例子如下:

    const Route = use('Route')    // 类似 include,跟 require 不同
    
    Route.get('/', function * (request, response) {
        yield response.sendView('home')    // 这里使用 ES6 generator 语法达到异步请求数据的效果
    })
    
    • 路由参数:路由参数是 URL 中的动态数据段,我们可以通过自定义的 URL 接受动态数据。如下例子:
    Route.get('users/:id', function * (request, response) {
        const id = request.param('id')
        response.send('Profile for user with id ${id}')
    })
    
    • 路由群:通常,会存在一系列路由共享同一个模块的情况,这就可能有时候需要修改这个模块的一些共同参数,我们把这些路由“集合”到一起,就方便修改了。例如下面的例子:
    Route.group('version1', function () {
        Route.get('users', function * (request, response) {  // 路由是 /api/v1/users
            // ...
        })
    }).prefix('api/v1').middleware('auth')    // 该路由群都会加上 api/v1 的前缀,并经过用户登录中间件的检验
    
    • 路由命名:书写完整的路径有时过于麻烦,为方便使用可以给路由命名。如下例子:
    Route
        .get('users/:id', '与否.show')
        .as('profile') 
    

    四、Controller

    本节主要作控制器的简要介绍,以及 request 和 response 的常用语法使用。最后简单说说中间件。

    1.概述

    • 控制器的作用正如前面所说,就是用来处理数据并传递数据给 View 渲染 HTML 页面。处理的数据来源有来自浏览器的 HTTP 请求,也可以从数据库获取,也包括从第三方服务器请求的数据(需自行搭建服务器客户端环境)。控制器的代码结构大致如下:
    // 引入依赖文件
    const path = require('path')
    
    // 定义控制器
    class UsersController {
    
      // 依赖注入
      static get inject () {
        return ['App/Model/User']   
      }
    
      // 以参数形式接收依赖注入
      constructor (User) { 
        this.User = User 
      }
      
      // 自定义数据处理方法
      static _processUserInfo() {
        // ...
      }
      
      // 异步调用的方法
      * index () {
        const users = yield this.User.all()
        // 传递数据给视图渲染
        yield response.sendView('users',{
            userName,
            userInfo
        })
      }
    
    }
    
    module.exports = UserController
    

    2.request

    2.1 简单示例

    • AdonisJS 中,为了方便接收 HTTP 请求信息,所有的控制器方法和路由器闭包都会接收一个 Request 实例 。请看一下简单示例:
    // 路由闭包中调用 request ,response
    const Route = use('Route')
    
    Route.get('post', function * (request, response) {
      const body = request.all()
       
      // cherry picking fields
      const body = request.only('title', 'description', 'categories')
    })
    
    // 控制器中调用 request, response
    class IndexController {
      * index(requset, response) {
        // ...
      }
    }
    

    2.2 常用的 request 方法

    • request.all() :包括所有查询字符串和请求体:
    const data = request.all()
    // example.html?id=1&name=John 
    // <input name="gender" value="male" />
    // => {"id":1, "name": "John", "gender": "male" }
    
    • request.only():跟 request.all() 类似,但是只包含定义的 key 值:
    const data = request.only('name', 'email', 'age')
    /* returns
    {
      'name': '..',
      'email':'..',
      'age':'..'
    }
    */
    
    • request.input(key,[defaultValue]):返回 input 标签对应的键值对,如果 value 为空,则返回 defaultValue
    const name = request.input('name')
    const subscribe = request.input('subscribe', 'yes')
    
    • request.url() :获取请求的 URL(出去查询字符串):
    // url - http://foo.com/users?orderBy=desc&limit=10
    request.url()
    // returns - http://foo.com/users
    
    • request.param(key, [defaultValue]) :返回查询字符串:
    // url - http://foo.com/users?orderBy=desc&limit=10
    request.param('orderBy')
    // returns - desc
    
    // 返回所有的查询字符串
    request.params()
    
    • request.collect(key1, key2, ...):转化数据格式,直接看下面例子:
    <form method="POST" action="/users">
      <div class="row">
        <h2> User 1 </h2>
        <input type="email" name="email[]" />
        <input type="password" name="password[]" />
      </div>
    
      <div class="row">
        <h2> User 2 </h2>
        <input type="email" name="email[]" />
        <input type="password" name="password[]" />
      </div>
    
      <button type="submit"> Create Users </button>
    </form>
    
    request.only('email', 'password', )
    /* returns 
    {
      email: ['bar@foo.com', 'baz@foo.com'],
      password: ['secret', 'secret1']
    }
    */
    
    request.collect('email', 'password')
    /* returns
    [
      {
        email: 'bar@foo.com',
        password: 'secret'
      },
      {
        email: 'baz@foo.com',
        password: 'secret1'
      }
    ]
    */
    

    3.response

    • 为了方便尽快响应 HTTP 请求,AdonisJS 提供了 response 类给我们使用。可用于渲染 nunjucks views 或者格式化数据。以下是简单的示例代码:
    // Render Views
    Route
      .get('/', function *() {
        yield response.sendView('welcome')    // resources/views/welcome.njk
    })
    
    // JSON Response
    Route
      .get('user', function * () {
        const user = yield User.all()    // fetch users
        response.json(users)
    })
    
    • 常用的还有重定向(Redirects)
    response.location('/signup')
    // or
    response.location('back')
    

    4.middleware

    • 中间件的使用十分常见,比如验证用户是否登录。在 AdonisJS 中,中间件放在 app/Http/Middleware 文件夹下,以下是简单示例:
    'use strict'
    
    const geoip = use('geoip-lite')  // npm module
    
    class CountryDetector {
      
      * handle (request, response, next) {
        const ip = request.ip()
        request.country = geoip.lookup(ip).country
        yield next    // 如果想要把 req, res 传递给下个中间件或者路由行为,注意使用 yield next 
      }
    }
    

    五、View

    本节简要介绍 View 层以及 View 层常见的使用。

    1.概述

    • 正如前边的数据流图所述,View 便是整个数据流程的末端,即利用前边步骤传来的数据进行页面渲染。AdonisJS 的视图引擎是建立在 nunjucks 的基础上的,所以 nunjucks 有的模板语法,在 AdonisJS 中一样可以使用。另外,AdonisJS 也提供了一些特有的 过滤器(filter)

    • 简单示例:

    // Route
    const Route = use('Route')
    Route.get('/greet/:user', 'UserController.greet')
    
    
    // Controller
    class UserController {
      * greet (request, response) {
         const user = request.param('user')  // user: "John"
         yield response.sendView('greet', {user})
      }
    }
    
    // View
    <h2>Hello {{ user }}</h2>    // <h2>John</h2>
    
    

    2.常见使用

    • 条件语法
    {% if hungry %}
      I am hungry
    {% else %}
      I am good
    {% endif %}
    
    {{ username | capitalize }}
    // username = 'john' => output = John
    
    • 遍历语法
    {% for book in books %}
      book
    {% endfor %}
    
    • 模板继承
    // resources/views/master.njk
    <html>
      <body>
    
        <header class="header">
          {% block header %}
            Common Header
          {% endblock %}
        </header>
    
        <section class="sidebar">
          {% block sidebar %}
            Common Sidebar
          {% endblock %}
        </section>
    
        <section class="content">
          {% block content %}{% endblock %}
        </section>
    
      </body>
    </html>
    
    // resources/views/home.njk
    {% extends 'master' %}
    
    {% block content %}
      Here comes the content of the home page.
    {% endblock %}
    
    <html>
      <body>
    
        <header class="header">
          Common Header
        </header>
    
        <section class="sidebar">
          Common Sidebar
        </section>
    
        <section class="content">
          Here comes the content of the home page.
        </section>
    
      </body>
    </html>
    
    • 引入
    // resources/views/chat/message.njk
    <div class="chat__message">
      <h2> {{ message.from }} </h2>
      <p> {{ message.body }} </p>
    </div>
    
    // resources/views/chat/index.njk
    {% for message in messages %}
      {% include 'message' %}
    {% endfor %}
    
    • macro & import
    // resource/views/macros/button.njk
    {% macro button(value, style='default') %}
      <button type="button" class="btn btn-{{style}}"> {{ value }} </button>
    {% endmacro %}
    
    // resources/views/home.njk
    {% from 'macros.button' import button %}
    {{ button('Create User', 'primary') }}
    

    3.自定义全局变量和过滤器

    • app/Listeners/Http.js 中,提供了 Http.onStart 函数钩子,供我们定义全局变量和过滤器。下面是两个简单的例子:
    // 定义全局变量
    Http.onStart = function () {
      const View = use('View')
      View.global('time', function () {
        return new Date().getTime()
      })
    }
    
    // 定义过滤器
    const View = use('Adonis/Src/View')
    const accounting = use('accounting') // npm module
    
    View.filter('currency', function (amount, symbol) {
      return accounting.formatMoney(amount, {symbol})
    })
    
    // 使用过滤器
    {{ 1000 | currency('$') }}
    
    {# returns $1,000.00 #}
    

    六、IoC Container & Service Providers

    • Ioc Container 主要概念就是在容器内存储/绑定依赖关系,然后从容器中取回它们,而不是手动请求它们。这种方法主要有以下三种好处:

    • 对 End-user 隐藏对象的配置,提供简单明了的 API

    • 坚固支持依赖注入(DI),因为所有对象都是从单一的真实来源获取的

    • 易于编写第三方模块/插件,因为你可以从 IoC 容器获取依赖关系,而不是让 End-user 手动传递它们。

    • 绑定依赖关系示例代码如下:

    const Ioc = require('adonis-fold').Ioc
    const bugsnag = require('bugsnag')
    
    Ioc.bind('Adonis/Src/Bugsnag', function (app) { 
    
      const Config = app.use('Adonis/Src/Config') 
      const bugSnagConfig = Config.get('services.bugsnag') 
    
      bugsnag.register(bugSnagConfig.apiKey, bugSnagConfig.options) 
      return bugsnag 
    
    })
    
    • 绑定 Service Provider 的示例代码如下:
    const ServiceProvider = require('adonis-fold').ServiceProvider
    
    class BugSnagProvider extends ServiceProvider {
    
      * register () { 
        this.app.bind('Adonis/Addons/BugSnag', (app) => {
          const BugSnag = require('./BugSnag')
          const Config = app.use('Adonis/Src/Config')
          return new BugSnag(Config)
        })
      }
    
      * boot () { 
        // Everything is registered do some hard work
      }
    
    }
    

    注:本篇对 model 部分的简单略过。相关配置,Ace 开发工具等请参考官方文档。

    相关文章

      网友评论

      • 全栈之巅Johnny:hi 我做了个小程序 录了一些视频 欢迎扫我头像一起学习adonisjs
      • 全栈之巅Johnny:美女,有兴趣加入adonis China么:smile:
        唐紫依: @Johnny吴雪松 好的,非常感谢
        全栈之巅Johnny:@唐紫依 欢迎来 adonis-china.org 咱们一起宣传推广adonisjs。马上论坛上线,欢迎发文
        唐紫依: @Johnny吴雪松 联系方式?
      • 唐紫依:本篇尚未完结,因本框架仅有官方文档及GitHub issue供参考,有些地方仍存在疑惑,待操作验证后再更新文章

      本文标题:AdonisJS 入门篇

      本文链接:https://www.haomeiwen.com/subject/fzbzittx.html