自从 deno 1.0 发布以来,有关 deno 的文章很多,大多数都是在讨论怎么安装 deno 、deno 有哪些特点 、deno 和 node 有哪些异同、deno是不是 node 的替代品等。咱们今天不讨论这些,毕竟 Talk is cheap. Show me the code! 亲自体验一把 deno 开发带来的快感、用 deno 搞一个“企业级”应用:deno-supermarket,难道不香吗?

deno 常见的一些坑

在实战之前,还是先来介绍几个我在刚接触 deno 时遇到的小坑。

权限标志符位置的问题

我们都知道, deno 默认是安全的,就是导致了默认情况下是不允许访问网络、读写文件等。比如有个名为 index.ts 的文件内容如下:

import { serve } from "https://deno.land/std@0.50.0/http/server.ts";
const s = serve({ port: 8000 });
console.log("http://localhost:8000/");
for await (const req of s) {
req.respond({ body: "Hello World\n" });
}

如果直接执行 deno run index.ts, 会报错:

error: Uncaught PermissionDenied: network access to "0.0.0.0:8000", run again with the --allow-net flag

所以我们很自然的就会在启动命令的最后加上 --allow-net ,如下:

deno run index.ts --allow-net

但是,这样仍然会报错。查了资料才知道 ,--allow-net--allow-read 之类的标志是不可以放到文件名后面的,必须紧跟在 deno run 后面,比如,如下才是正确的:

deno run --alow-net index.ts

为什么调换了位置就不行呢? issue 上的回答是,如果 --allow-net 跟在文件名后面,是传给 js 脚本的,而不是传给 deno 的。想了解更多的,可以看这个 confused by order of cli option flags。反正,记住一点就行:权限标志一定要跟在 deno run 后面!

因为我们前端同学大多数平时很少写后台,不太清楚安全的重要性,为了避免遇到各种权限问题,我建议平时在写一些练手项目时,直接用 deno run -A 来启用全部的权限。(这只是方便调试,在生产环境中一定要慎用!)

不稳定的 API

因为实战过程中使用了 mongodb , 所以需要引入 Deno 的第三方模块 mongo,然而在启动项目会报错:

error: TS2339 [ERROR]: Property 'openPlugin' does not exist on type 'typeof Deno'.

查了一下,发现是因为 openPlugin 这个方法目前还不稳定。默认情况下,deno 只会提供稳定的 api。如果需要开启不稳定 api,可以添加 --stable 标志。比如:

deno run -A --unstable index.ts

可能有人会问,-A--unstable 的位置调换会不会有问题。这个亲测过不会有问题。只要标志符在文件名称之前就行了。

还有个问题,到底哪些是稳定 API,哪些是不稳定 API 呢?其实 deno 官方文档已经帮我们分好类的了,入口地址分别是:

如果你怀疑 --unstable 的作用,可以使用下面的方法打印出 Deno 上的所有成员:

console.log(Object.keys(Deno).length)

使用 deno run --unstable index.ts 输出的结果是 117,使用 deno run index.ts 输出的结果是 88 。说明稳定的 api 有 88 个,不稳定的有 29 个。

deno 的一些使用技巧

从 node 切换到 deno, 我们的开发思维也要随之转变。所以,我们再来看看 deno 的一些和 node 不一样的开发技巧。

如何管理版本?

刚开始我也很疑惑:没有了 package.json, 那怎么控制各依赖的版本呀?比如,我们有10个文件都依赖了 mongo@v0.0.6, 那每个文件都使用以下代码进行引入:

 import { init, MongoClient } from 'https://deno.land/x/mongo@v0.6.0/mod.ts'

可是有一天,我突然想把 0.6.0 升级到 0.7.0, 那怎么办呢?一个个文件的进行替换容易漏掉,当然也可以全局搜索批量替换 。但是这种效率都不是很高。

官方给出的推荐做法是,使用 deps.ts 文件来引入远程文件,并管理版本。(当然 ,文件名称不一定叫做 deps.ts, 你也可以改成其他的名称)。具体做法就是,把所有用到的远程依赖,都在 deps.ts 中引入 ,并且通过 Re-export 手段导出各依赖,然后其他文件就可以从 deps.ts 中拿到所需要的依赖了。

回到刚才说10个文件都依赖到 mongo 的问题,如果改成 deps.ts 文件来统一管理是这样的:

export * from 'https://deno.land/x/mongo@v0.6.0/mod.ts'

然后那些需要用到 mongo 的文件,不要直接从远程引入,而是从 deps.ts 中引入,如下:

import { init, MongoClient } from '../pathTo/deps.ts';

如果需要升级的话,我们可以直接把 deps.ts 里面的 mongo 地址中的 0.6.0 改成 0.7.0 就行了。

另外,有一点和 npm 类似的是,如果没有指定版本号,即远程地址中没有指定版本,比如:

export * from 'https://deno.land/x/mongo/mod.ts'

就会默认安装最新版的依赖。

如何查找一些对我有用的 deno 库?

使用 node ,可以到 npm 上查找一些库。deno 也有类似的平台,目前分为两种库,一种是官方标准的,另外一种是第三方的。标准库可以到这里查找:Deno Standard Modules。第三方库可以到这里查找: Deno Third Party Modules

实战: 使用 deno 开发一个具备增删查改的商城系统

OK, 具备以上的知识点,现在可以实战了。首先,需要保证你的电脑安装了 deno 1.0 。另外,由于用到了 mongodb ,所以需要你的电脑也要安装 mongodb。

界面

先来看看我们的商城的界面:

麻雀虽小五脏俱全哈!具备添加商品、查询商品、删除商品、修改商品的功能。这是典型的 REST API 风格的系统 。

项目结构

然后再来看看项目结构:

  • .deno_plugins: 这是 mongo 模块所下载的动态链接库,不用关注它。
  • congig/db.ts: 这是连接 mongodb 的相关配置文件。目前写死的端口号是 27017 , 如果你的 mongodb 端口不是这个,可以在这个文件里面修改。
  • controllers/goods.ts: 这是实现增删查改的逻辑代码
  • public/index.html: 这是前端静态页面,跟 deno 无关的,我们只需要用 deno 来服务该目录就行。
  • deps.ts: 用来管理远程依赖库,然后 Re-export 出去给其他文件使用。
  • server.ts 入口文件,跟我们用 epress 或 koa 时的入口 文件 app.js 类似。

依赖模块的选择

因为该项目涉及到了前后端,如果使用 node 的话,一般会选择 express 或 koa。同样的,我们使用 deno 也要选择对应的框架 ,不然的话,http服务以及路由跳转等都不是那么容易处理的。deno 上的这类框架,比较多人 star 的是 oakabc,这里我们选择使用 abc。

另外,因为使用 mongodb , 所以还需要引入 mongo

结束语

好啦,对于 deno 初体验就写到这啦, 具体的代码这里不打算贴出来了,有兴趣的可以前往 github 查看:

有问题的可以一起交流学习哈~

deno+mongo实战踩坑记的更多相关文章

  1. mongo java 踩坑记

    为什么会有这么多坑 1.  Java会把 id:String = "合法ObjectId"  好心好意的 转为  _id:ObjectId 类型. 2. 为了避免第1点, 我定义了 ...

  2. Vue + TypeScript + Element 搭建简洁时尚的博客网站及踩坑记

    前言 本文讲解如何在 Vue 项目中使用 TypeScript 来搭建并开发项目,并在此过程中踩过的坑 . TypeScript 具有类型系统,且是 JavaScript 的超集,TypeScript ...

  3. Spark踩坑记——Spark Streaming+Kafka

    [TOC] 前言 在WeTest舆情项目中,需要对每天千万级的游戏评论信息进行词频统计,在生产者一端,我们将数据按照每天的拉取时间存入了Kafka当中,而在消费者一端,我们利用了spark strea ...

  4. Spark踩坑记——数据库(Hbase+Mysql)

    [TOC] 前言 在使用Spark Streaming的过程中对于计算产生结果的进行持久化时,我们往往需要操作数据库,去统计或者改变一些值.最近一个实时消费者处理任务,在使用spark streami ...

  5. 【踩坑记】从HybridApp到ReactNative

    前言 随着移动互联网的兴起,Webapp开始大行其道.大概在15年下半年的时候我接触到了HybridApp.因为当时还没毕业嘛,所以并不清楚自己未来的方向,所以就投入了HybridApp的怀抱. Hy ...

  6. Spark踩坑记——共享变量

    [TOC] 前言 Spark踩坑记--初试 Spark踩坑记--数据库(Hbase+Mysql) Spark踩坑记--Spark Streaming+kafka应用及调优 在前面总结的几篇spark踩 ...

  7. Spark踩坑记——从RDD看集群调度

    [TOC] 前言 在Spark的使用中,性能的调优配置过程中,查阅了很多资料,之前自己总结过两篇小博文Spark踩坑记--初试和Spark踩坑记--数据库(Hbase+Mysql),第一篇概况的归纳了 ...

  8. djangorestframework+vue-cli+axios,为axios添加token作为headers踩坑记

    情况是这样的,项目用的restful规范,后端用的django+djangorestframework,前端用的vue-cli框架+webpack,前端与后端交互用的axios,然后再用户登录之后,a ...

  9. HttpWebRequest 改为 HttpClient 踩坑记-请求头设置

    HttpWebRequest 改为 HttpClient 踩坑记-请求头设置 Intro 这两天改了一个项目,原来的项目是.net framework 项目,里面处理 HTTP 请求使用的是 WebR ...

随机推荐

  1. Scala教程之:可变和不变集合

    文章目录 mutable HashMap immutable HashMap 集合在程序中是非常有用的,只有用好集合才能真正感受到该语言的魅力.在scala中集合主要在三个包里面:scala.coll ...

  2. mysql 5.7 MGR

    最近看了一下mysql5.7的MGR集群挺不错的,有单主和多主模式,于是乎搭建测试了一下效果还不错,我指的不错是搭建和维护方面都比较简单.网上绝大多数都是单主模式,当然我这里也是,为了加深印象,特意记 ...

  3. Spark学习笔记(一)

    概念: Spark是加州大学伯克利分校AMP实验室,开发的通用内存并行计算框架. 支持用scala.java和Python等语言编写应用程序.相较于Hdoop,往往有更好的运行效率. Spark包括了 ...

  4. 团队一致性的PHP开发环境之Docker

    docker php环境模型 docker 简介 Docker 是一个开源的应用容器引擎 让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的 Linux 机器上,也可以实现 ...

  5. 瑞银预计小扎的十年规划可获大回报 上调Facebook股票目标价

    瑞银认为马克·扎克伯格(Mark Zuckerberg)为Facebook定制的十年规划将带来丰厚回报,它已将Facebook股票的目标价由之前的155美元上调至165美元. Facebook首席执行 ...

  6. Vs Code在Vue项目中v-for指令提示错误的解决办法

    最近在做一个Vue项目,在其中用到v-for指令时,发现Vs Code报错,如下图(代码是没有任何问题的),在网上找了一下解决办法,希望能帮助到更多人. 解决方法: 打开    文件-首选项-设置 将 ...

  7. cms 环境搭建

    一 :安装 JDK: 1.在根目录下新建目录 JDK 2.通过 xftp 导入 jdk 压缩包到 JDK 目录中 jdk-8u144-linux-x64.tar.gz,解压 tar -zxvf jdk ...

  8. python(open 文件)

    一.open 文件 1.open('file','mode')打开一个文件 file 要打开的文件名,需加路径(除非是在当前目录) mode 文件打开的模式 需要手动关闭 close 2.with o ...

  9. Spring依赖注入—@Resource注解使用

    1.@Autowired默认按类型装配(这个注解是属业spring的),默认情况下必须要求依赖对象必须存在,如果要允许null 值,可以设置它的required属性为false,如:@Autowire ...

  10. 07 模型层 orm相关查询 F查询Q查询 django开启事务

    一.Django终端打印SQL语句 如果你想知道你对数据库进行操作时,Django内部到底是怎么执行它的sql语句时可以加下面的配置来查看 在Django项目的settings.py文件中,在最后复制 ...