JavaScript 编程精解 中文第三版 二11、项目:技能分享网站

来源: ApacheCN『JavaScript 编程精解 中文第三版』翻译项目

原文:Project: Skill-Sharing Websitejavascript

译者:飞龙css

协议:CC BY-NC-SA 4.0html

自豪地采用谷歌翻译java

部分参考了《JavaScript 编程精解(第 2 版)》node

If you have knowledge, let others light their candles at it.git

Margaret Fullergithub

技能分享会是一个活动,其中兴趣相同的人聚在一块儿,针对他们所知的事情进行小型非正式的展现。在园艺技能分享会上,能够解释如何耕做芹菜。若是在编程技能分享小组中,你能够顺便给每一个人讲讲 Node.js。正则表达式

在计算机领域中,这类聚会每每名为用户小组,是开阔眼界、了解行业新动态或仅仅接触兴趣相同的人的好方法。许多大城市都会有 JavaScript 聚会。这类聚会每每是能够免费参加的,并且我发现我参加过的那些聚会都很是友好热情。apache

在最后的项目章节中,咱们的目标是创建网站,管理特定技能分享会的讨论内容。假设一个小组的人会在成员办公室中按期举办关于独轮车的聚会。上一个组织者搬到了另外一个城市,而且没人能够站出来接下来他的任务。咱们须要一个系统,让参与者能够在系统中发言并相互讨论,这样就不须要一个中心组织人员了。npm

就像上一章同样,本章中的一些代码是为 Node.js 编写的,而且直接在你正在查看的 HTML页面中运行它不太可行。 该项目的完整代码能够从eloquentjavascript.net/code/skillsharing.zip下载。

设计

本项目的服务器部分为 Node.js 编写,客户端部分则为浏览器编写。服务器存储系统数据并将其提供给客户端。它也提供实现客户端系统的文件。

服务器保存了为下次聚会提出的对话列表。每一个对话包括参与人员姓名、标题和该对话的相关评论。客户端容许用户提出新的对话(将对话添加到列表中)、删除对话和评论已存在的对话。每当用户作了修改时,客户端会向服务器发送关于更改的 HTTP 请求。

咱们建立应用来展现一个实时视图,来展现目前已经提出的对话和评论。每当某些人在某些地点提交了新的对话或添加新评论时,全部在浏览器中打开页面的人都应该当即看到变化。这个特性略有挑战,网络服务器没法创建到客户端的链接,也没有好方法来知道有哪些客户端如今在查看特定网站。

该问题的一个解决方案叫做长时间轮询,这恰巧是 Node 的设计动机之一。

长轮询

为了可以当即提示客户端某些信息发生了改变,咱们须要创建到客户端的链接。因为一般浏览器没法接受链接,并且客户端一般在路由后面,它不管如何都会拒绝这类链接,所以由服务器初始化链接是不切实际的。

咱们能够安排客户端来打开链接并保持该链接,所以服务器可使用该链接在必要时传送信息。

但 HTTP 请求只是简单的信息流:客户端发送请求,服务器返回一条响应,就是这样。有一种名为 WebSocket 的技术,受到现代浏览器的支持,是的咱们能够创建链接并进行任意的数据交换。但如何正确运用这项技术是较为复杂的。

本章咱们将会使用一种相对简单的技术:长轮询(Long Polling)。客户端会连续使用定时的 HTTP 请求向服务器询问新信息,而当没有新信息须要报告时服务器会简单地推迟响应。

只要客户端确保其能够持续不断地创建轮询请求,就能够在信息可用以后,从服务器快速地接收到信息。例如,若 Fatma 在浏览器中打开了技能分享程序,浏览器会发送请求询问是否有更新,且等待请求的响应。当 Iman 在本身的浏览器中提交了关于“极限降滑独轮车”的对话以后。服务器发现 Fatma 在等待更新请求,并将新的对话做为响应发送给待处理的请求。Fatma 的浏览器将会接收到数据并更新屏幕展现对话内容。

为了防止链接超时(由于链接必定时间不活跃后会被中断),长轮询技术经常为每一个请求设置一个最大等待时间,只要超过了这个时间,即便没人有任何须要报告的信息也会返回响应,在此以后,客户端会创建一个新的请求。按期从新发送请求也使得这种技术更具鲁棒性,容许客户端从临时的链接失败或服务器问题中恢复。

使用了长轮询技术的繁忙的服务器,能够有成百上千个等待的请求,所以也就有这么多个 TCP 链接处于打开状态。Node简化了多链接的管理工做,而不是创建单独线程来控制每一个链接,这对这样的系统是很是合适的。

HTTP 接口

在咱们设计服务器或客户端的代码以前,让咱们先来思考一下二者均会涉及的一点:双方通讯的 HTTP 接口。

咱们会使用 JSON 做为请求和响应正文的格式,就像第二十章中的文件服务器同样,咱们尝试充分利用 HTTP 方法。全部接口均以/talks路径为中心。不以/talks开头的路径则用于提供静态文件服务,即用于实现客户端系统的 HTML 和 JavaScript 代码。

访问/talksGET请求会返回以下所示的 JSON 文档。

[{"title": "Unituning",
  "presenter": "Jamal",
  "summary": "Modifying your cycle for extra style",
  "comment": []}]

咱们能够发送PUT请求到相似于/talks/Unituning之类的 URL 上来建立新对话,在第二个斜杠后的那部分是对话的名称。PUT请求正文应当包含一个 JSON 对象,其中有一个presenter属性和一个summary属性。

由于对话标题能够包含空格和其余没法正常出如今 URL 中的字符,所以咱们必须使用encodeURIComponent函数来编码标题字符串,并构建 URL。

console.log("/talks/" + encodeURIComponent("How to Idle"));
// → /talks/How%20to%20Idle

下面这个请求用于建立关于“空转”的对话。

PUT /talks/How%20to%20Idle HTTP/1.1
Content-Type: application/json
Content-Length: 92

{"presenter": "Maureen",
 "summary": "Standing still on a unicycle"}

咱们也可使用GET请求经过这些 URL 获取对话的 JSON 数据,或使用DELETE请求经过这些 URL 删除对话。

为了在对话中添加一条评论,能够向诸如/talks/Unituning/comments的 URL 发送POST请求,JSON 正文包含author属性和message属性。

POST /talks/Unituning/comments HTTP/1.1
Content-Type: application/json
Content-Length: 72

{"author": "Iman",
 "message": "Will you talk about raising a cycle?"}

为了支持长轮询,若是没有新的信息可用,发送到/talksGET请求可能会包含额外的标题,通知服务器延迟响应。 咱们将使用一般用于管理缓存的一对协议头:ETagIf-None-Match

服务器可能在响应中包含ETag(“实体标签”)协议头。 它的值是标识资源当前版本的字符串。 当客户稍后再次请求该资源时,能够经过包含一个If-None-Match头来进行条件请求,该头的值保存相同的字符串。 若是资源没有改变,服务器将响应状态码 304,这意味着“未修改”,告诉客户端它的缓存版本仍然是最新的。 当标签与服务器不匹配时,服务器正常响应。

咱们须要这样的东西,经过它客户端能够告诉服务器它有哪一个版本的对话列表,仅当列表发生变化时,服务器才会响应。 但服务器不是当即返回 304 响应,它应该中止响应,而且仅当有新东西的可用,或已通过去了给定的时间时才返回。 为了将长轮询请求与常规条件请求区分开来,咱们给他们另外一个标头Prefer: wait=90,告诉服务器客户端最多等待 90 秒的响应。

服务器将保留版本号,每次对话更改时更新,并将其用做ETag值。 客户端能够在对话变动时通知此类要求:

GET /talks HTTP/1.1
If-None-Match: "4"
Prefer: wait=90

(time passes)

HTTP/1.1 200 OK
Content-Type: application/json
ETag: "5"
Content-Length: 295

[....]

这里描述的协议并无任何访问控制。每一个人均可以评论、修改对话或删除对话。由于因特网中充满了流氓,所以将这类没有进一步保护的系统放在网络上最后可能并非很好。

服务器

让咱们开始构建程序的服务器部分。本节的代码能够在 Node.js 中执行。

路由

咱们的服务器会使用createServer来启动 HTTP 服务器。在处理新请求的函数中,咱们必须区分咱们支持的请求的类型(根据方法和路径肯定)。咱们可使用一长串的if语句完成该任务,但还存在一种更优雅的方式。

路由能够做为帮助把请求调度传给能处理该请求的函数。路径匹配正则表达式/^\/talks\/([^\/]+)$//talks/带着对话名称)的PUT请求,应当由指定函数处理。此外,路由能够帮助咱们提取路径中有意义的部分,在本例中会将对话的标题(包裹在正则表达式的括号之中)传递给处理器函数。

在 NPM 中有许多优秀的路由包,但这里咱们本身编写一个路由来展现其原理。

这里给出router.js,咱们随后将在服务器模块中使用require获取该模块。

const {parse} = require("url");

module.exports = class Router {
  constructor() {
    this.routes = [];
  }
  add(method, url, handler) {
    this.routes.push({method, url, handler});
  }
  resolve(context, request) {
    let path = parse(request.url).pathname;

    for (let {method, url, handler} of this.routes) {
      let match = url.exec(path);
      if (!match || request.method != method) continue;
      let urlParts = match.slice(1).map(decodeURIComponent);
      return handler(context, ...urlParts, request);
    }
    return null;
  }
};

该模块导出Router类。咱们可使用路由对象的add方法来注册一个新的处理器,并使用resolve方法解析请求。

找处处理器以后,后者会返回一个响应,不然为null。它会逐个尝试路由(根据定义顺序排序),当找到一个匹配的路由时返回true

路由会使用context值调用处理器函数(这里是服务器实例),将请求对象中的字符串,与已定义分组中的正则表达式匹配。传递给处理器的字符串必须进行 URL 解码,由于原始 URL 中可能包含%20风格的代码。

文件服务

当请求没法匹配路由中定义的任何请求类型时,服务器必须将其解释为请求位于public目录下的某个文件。服务器可使用第二十章中定义的文件服务器来提供文件服务,但咱们并不须要也不想对文件支持 PUT 和 DELETE 请求,且咱们想支持相似于缓存等高级特性。所以让咱们使用 NPM 中更为可靠且通过充分测试的静态文件服务器。

我选择了ecstatic。它并非 NPM 中惟一的此类服务,但它可以完美工做且符合咱们的意图。ecstatic模块导出了一个函数,咱们能够调用该函数,并传递一个配置对象来生成一个请求处理函数。咱们使用root选项告知服务器文件搜索位置。

const {createServer} = require("http");
const Router = require("./router");
const ecstatic = require("ecstatic");

const router = new Router();
const defaultHeaders = {"Content-Type": "text/plain"};

class SkillShareServer {
  constructor(talks) {
    this.talks = talks;
    this.version = 0;
    this.waiting = [];

    let fileServer = ecstatic({root: "./public"});
    this.server = createServer((request, response) => {
      let resolved = router.resolve(this, request);
      if (resolved) {
        resolved.catch(error => {
          if (error.status != null) return error;
          return {body: String(error), status: 500};
        }).then(({body,
                  status = 200,
                  headers = defaultHeaders}) => {
          response.writeHead(status, headers);
          response.end(body);
        });
      } else {
        fileServer(request, response);
      }
    });
  }
  start(port) {
    this.server.listen(port);
  }
  stop() {
    this.server.close();
  }
}

它使用上一章中的文件服务器的相似约定来处理响应 - 处理器返回Promise,可解析为描述响应的对象。 它将服务器包装在一个对象中,它也维护它的状态。

做为资源的对话

已提出的对话存储在服务器的talks属性中,这是一个对象,属性名称是对话标题。这些对话会展示为/talks/[title]下的 HTTP 资源,所以咱们须要将处理器添加咱们的路由中供客户端选择,来实现不一样的方法。

获取(GET)单个对话的请求处理器,必须查找对话并使用对话的 JSON 数据做为响应,若不存在则返回 404 错误响应码。

const talkPath = /^\/talks\/([^\/]+)$/;

router.add("GET", talkPath, async (server, title) => {
  if (title in server.talks) {
    return {body: JSON.stringify(server.talks[title]),
            headers: {"Content-Type": "application/json"}};
  } else {
    return {status: 404, body: `No talk '${title}' found`};
  }
});

删除对话时,将其从talks对象中删除便可。

router.add("DELETE", talkPath, async (server, title) => {
  if (title in server.talks) {
    delete server.talks[title];
    server.updated();
  }
  return {status: 204};
});

咱们将在稍后定义updated方法,它通知等待有关更改的长轮询请求。

为了获取请求正文的内容,咱们定义一个名为readStream的函数,从可读流中读取全部内容,并返回解析为字符串的Promise

function readStream(stream) {
  return new Promise((resolve, reject) => {
    let data = "";
    stream.on("error", reject);
    stream.on("data", chunk => data += chunk.toString());
    stream.on("end", () => resolve(data));
  });
}

须要读取响应正文的函数是PUT的处理器,用户使用它建立新对话。该函数须要检查数据中是否有presentersummary属性,这些属性都是字符串。任何来自外部的数据均可能是无心义的,咱们不但愿错误请求到达时会破坏咱们的内部数据模型,或者致使服务崩溃。

若数据看起来合法,处理器会将对话转化为对象,存储在talks对象中,若是有标题相同的对话存在则覆盖,并再次调用updated

router.add("PUT", talkPath,
           async (server, title, request) => {
  let requestBody = await readStream(request);
  let talk;
  try { talk = JSON.parse(requestBody); }
  catch (_) { return {status: 400, body: "Invalid JSON"}; }

  if (!talk ||
      typeof talk.presenter != "string" ||
      typeof talk.summary != "string") {
    return {status: 400, body: "Bad talk data"};
  }
  server.talks[title] = {title,
                         presenter: talk.presenter,
                         summary: talk.summary,
                         comments: []};
  server.updated();
  return {status: 204};
});

在对话中添加评论也是相似的。咱们使用readStream来获取请求内容,验证请求数据,若看上去合法,则将其存储为评论。

router.add("POST", /^\/talks\/([^\/]+)\/comments$/,
           async (server, title, request) => {
  let requestBody = await readStream(request);
  let comment;
  try { comment = JSON.parse(requestBody); }
  catch (_) { return {status: 400, body: "Invalid JSON"}; }

  if (!comment ||
      typeof comment.author != "string" ||
      typeof comment.message != "string") {
    return {status: 400, body: "Bad comment data"};
  } else if (title in server.talks) {
    server.talks[title].comments.push(comment);
    server.updated();
    return {status: 204};
  } else {
    return {status: 404, body: `No talk '${title}' found`};
  }
});

尝试向不存在的对话中添加评论会返回 404 错误。

长轮询支持

服务器中最值得探讨的方面是处理长轮询的部分代码。当 URL 为/talksGET请求到来时,它多是一个常规请求或一个长轮询请求。

咱们可能在不少地方,将对话列表发送给客户端,所以咱们首先定义一个简单的辅助函数,它构建这样一个数组,并在响应中包含ETag协议头。

SkillShareServer.prototype.talkResponse = function() {
  let talks = [];
  for (let title of Object.keys(this.talks)) {
    talks.push(this.talks[title]);
  }
  return {
    body: JSON.stringify(talks),
    headers: {"Content-Type": "application/json",
              "ETag": `"${this.version}"`}
  };
};

处理器自己须要查看请求头,来查看是否存在If-None-MatchPrefer标头。 Node 在其小写名称下存储协议头,根据规定其名称是不区分大小写的。

router.add("GET", /^\/talks$/, async (server, request) => {
  let tag = /"(.*)"/.exec(request.headers["if-none-match"]);
  let wait = /\bwait=(\d+)/.exec(request.headers["prefer"]);
  if (!tag || tag[1] != server.version) {
    return server.talkResponse();
  } else if (!wait) {
    return {status: 304};
  } else {
    return server.waitForChanges(Number(wait[1]));
  }
});

若是没有给出标签,或者给出的标签与服务器的当前版本不匹配,则处理器使用对话列表来响应。 若是请求是有条件的,而且对话没有变化,咱们查阅Prefer标题来查看,是否应该延迟响应或当即响应。

用于延迟请求的回调函数存储在服务器的waiting数组中,以便在发生事件时通知它们。 waitForChanges方法也会当即设置一个定时器,当请求等待了足够长时,以 304 状态来响应。

SkillShareServer.prototype.waitForChanges = function(time) {
  return new Promise(resolve => {
    this.waiting.push(resolve);
    setTimeout(() => {
      if (!this.waiting.includes(resolve)) return;
      this.waiting = this.waiting.filter(r => r != resolve);
      resolve({status: 304});
    }, time * 1000);
  });
};

使用updated注册一个更改,会增长version属性并唤醒全部等待的请求。

var changes = [];

SkillShareServer.prototype.updated = function() {
  this.version++;
  let response = this.talkResponse();
  this.waiting.forEach(resolve => resolve(response));
  this.waiting = [];
};

服务器代码这样就完成了。 若是咱们建立一个SkillShareServer的实例,并在端口 8000 上启动它,那么生成的 HTTP 服务器,将服务于public子目录中的文件,以及/ talksURL 下的一个对话管理界面。

new SkillShareServer(Object.create(null)).start(8000);

客户端

技能分享网站的客户端部分由三个文件组成:微型 HTML 页面、样式表以及 JavaScript 文件。

HTML

在网络服务器提供文件服务时,有一种广为使用的约定是:当请求直接访问与目录对应的路径时,返回名为index.html的文件。咱们使用的文件服务模块ecstatic就支持这种约定。当请求路径为/时,服务器会搜索文件./public/index.html./public是咱们赋予的根目录),若文件存在则返回文件。

所以,若咱们但愿浏览器指向咱们服务器时展现某个特定页面,咱们将其放在public/index.html中。这就是咱们的index文件。

<!doctype html>
<meta charset="utf-8">
<title>Skill Sharing</title>
<link rel="stylesheet" href="skillsharing.css">

<h1>Skill Sharing</h1>

<script src="skillsharing_client.js"></script>

它定义了文档标题并包含一个样式表,除了其它东西,它定义了几种样式,确保对话之间有必定的空间。

最后,它在页面顶部添加标题,并加载包含客户端应用的脚本。

动做

应用状态由对话列表和用户名称组成,咱们将它存储在一个{talks, user}对象中。 咱们不容许用户界面直接操做状态或发送 HTTP 请求。 反之,它可能会触发动做,它描述用户正在尝试作什么。

function handleAction(state, action) {
  if (action.type == "setUser") {
    localStorage.setItem("userName", action.user);
    return Object.assign({}, state, {user: action.user});
  } else if (action.type == "setTalks") {
    return Object.assign({}, state, {talks: action.talks});
  } else if (action.type == "newTalk") {
    fetchOK(talkURL(action.title), {
      method: "PUT",
      headers: {"Content-Type": "application/json"},
      body: JSON.stringify({
        presenter: state.user,
        summary: action.summary
      })
    }).catch(reportError);
  } else if (action.type == "deleteTalk") {
    fetchOK(talkURL(action.talk), {method: "DELETE"})
      .catch(reportError);
  } else if (action.type == "newComment") {
    fetchOK(talkURL(action.talk) + "/comments", {
      method: "POST",
      headers: {"Content-Type": "application/json"},
      body: JSON.stringify({
        author: state.user,
        message: action.message
      })
    }).catch(reportError);
  }
  return state;
}

咱们将用户的名字存储在localStorage中,以便在页面加载时恢复。

须要涉及服务器的操做使用fetch,将网络请求发送到前面描述的 HTTP 接口。 咱们使用包装函数fetchOK,它确保当服务器返回错误代码时,拒绝返回的Promise

function fetchOK(url, options) {
  return fetch(url, options).then(response => {
    if (response.status < 400) return response;
    else throw new Error(response.statusText);
  });
}

这个辅助函数用于为某个对话,使用给定标题创建 URL。

function talkURL(title) {
  return "talks/" + encodeURIComponent(title);
}

当请求失败时,咱们不但愿咱们的页面丝绝不变,不给予任何提示。所以咱们定义一个函数,名为reportError,至少在发生错误时向用户展现一个对话框。

function reportError(error) {
  alert(String(error));
}

渲染组件

咱们将使用一个方法,相似于咱们在第十九章中所见,将应用拆分为组件。 但因为某些组件不须要更新,或者在更新时老是彻底从新绘制,因此咱们不将它们定义为类,而是直接返回 DOM 节点的函数。 例如,下面是一个组件,显示用户能够向它输入名称的字段的:

function renderUserField(name, dispatch) {
  return elt("label", {}, "Your name: ", elt("input", {
    type: "text",
    value: name,
    onchange(event) {
      dispatch({type: "setUser", user: event.target.value});
    }
  }));
}

用于构建 DOM 元素的elt函数是咱们在第十九章中使用的函数。

相似的函数用于渲染对话,包括评论列表和添加新评论的表单。

function renderTalk(talk, dispatch) {
  return elt(
    "section", {className: "talk"},
    elt("h2", null, talk.title, " ", elt("button", {
      type: "button",
      onclick() {
        dispatch({type: "deleteTalk", talk: talk.title});
      }
    }, "Delete")),
    elt("div", null, "by ",
        elt("strong", null, talk.presenter)),
    elt("p", null, talk.summary),
    ...talk.comments.map(renderComment),
    elt("form", {
      onsubmit(event) {
        event.preventDefault();
        let form = event.target;
        dispatch({type: "newComment",
                  talk: talk.title,
                  message: form.elements.comment.value});
        form.reset();
      }
    }, elt("input", {type: "text", name: "comment"}), " ",
       elt("button", {type: "submit"}, "Add comment")));
}

submit事件处理器调用form.reset,在建立"newComment"动做后清除表单的内容。

在建立适度复杂的 DOM 片断时,这种编程风格开始显得至关混乱。 有一个普遍使用的(非标准的)JavaScript 扩展叫作 JSX,它容许你直接在你的脚本中编写 HTML,这可使这样的代码更漂亮(取决于你认为漂亮是什么)。 在实际运行这种代码以前,必须在脚本上运行一个程序,将伪 HTML 转换为 JavaScript 函数调用,就像咱们在这里用的东西。

评论更容易渲染。

function renderComment(comment) {
  return elt("p", {className: "comment"},
             elt("strong", null, comment.author),
             ": ", comment.message);
}

最后,用户可使用表单建立新对话,它渲染为这样。

function renderTalkForm(dispatch) {
  let title = elt("input", {type: "text"});
  let summary = elt("input", {type: "text"});
  return elt("form", {
    onsubmit(event) {
      event.preventDefault();
      dispatch({type: "newTalk",
                title: title.value,
                summary: summary.value});
      event.target.reset();
    }
  }, elt("h3", null, "Submit a Talk"),
     elt("label", null, "Title: ", title),
     elt("label", null, "Summary: ", summary),
     elt("button", {type: "submit"}, "Submit"));
}

轮询

为了启动应用,咱们须要对话的当前列表。 因为初始加载与长轮询过程密切相关 -- 轮询时必须使用来自加载的ETag -- 咱们将编写一个函数来不断轮询服务器的/ talks,而且在新的对话集可用时,调用回调函数。

async function pollTalks(update) {
  let tag = undefined;
  for (;;) {
    let response;
    try {
      response = await fetchOK("/talks", {
        headers: tag && {"If-None-Match": tag,
                         "Prefer": "wait=90"}
      });
    } catch (e) {
      console.log("Request failed: " + e);
      await new Promise(resolve => setTimeout(resolve, 500));
      continue;
    }
    if (response.status == 304) continue;
    tag = response.headers.get("ETag");
    update(await response.json());
  }
}

这是一个async函数,所以循环和等待请求更容易。 它运行一个无限循环,每次迭代中,一般检索对话列表。或者,若是这不是第一个请求,则带有使其成为长轮询请求的协议头。

当请求失败时,函数会等待一下子,而后再次尝试。 这样,若是你的网络链接断了一段时间而后又恢复,应用能够恢复并继续更新。 经过setTimeout解析的Promise,是强制async函数等待的方法。

当服务器回复 304 响应时,这意味着长轮询请求超时,因此函数应该当即启动下一个请求。 若是响应是普通的 200 响应,它的正文将当作 JSON 而读取并传递给回调函数,而且它的ETag协议头的值为下一次迭代而存储。

应用

如下组件将整个用户界面结合在一块儿。

class SkillShareApp {
  constructor(state, dispatch) {
    this.dispatch = dispatch;
    this.talkDOM = elt("div", {className: "talks"});
    this.dom = elt("div", null,
                   renderUserField(state.user, dispatch),
                   this.talkDOM,
                   renderTalkForm(dispatch));
    this.setState(state);
  }

  setState(state) {
    if (state.talks != this.talks) {
      this.talkDOM.textContent = "";
      for (let talk of state.talks) {
        this.talkDOM.appendChild(
          renderTalk(talk, this.dispatch));
      }
      this.talks = state.talks;
    }
  }
}

当对话改变时,这个组件从新绘制全部这些组件。 这很简单,但也是浪费。 咱们将在练习中回顾一下。

咱们能够像这样启动应用:

function runApp() {
  let user = localStorage.getItem("userName") || "Anon";
  let state, app;
  function dispatch(action) {
    state = handleAction(state, action);
    app.setState(state);
  }

  pollTalks(talks => {
    if (!app) {
      state = {user, talks};
      app = new SkillShareApp(state, dispatch);
      document.body.appendChild(app.dom);
    } else {
      dispatch({type: "setTalks", talks});
    }
  }).catch(reportError);
}

runApp();

若你执行服务器并同时为localhost:8000/打开两个浏览器窗口,你能够看到在一个窗口中执行动做时,另外一个窗口中会当即作出反应。

习题

下面的习题涉及修改本章中定义的系统。为了使用该系统进行工做,请确保首先下载代码,安装了 Node,并使用npm install安装了项目的全部依赖。

磁盘持久化

技能分享服务只将数据存储在内存中。这就意味着当服务崩溃或觉得任何缘由重启时,全部的对话和评论都会丢失。

扩展服务使得其将对话数据存储到磁盘上,并在程序重启时自动从新加载数据。不要担忧效率,只要用最简单的代码让其能够工做便可。

重置评论字段

因为咱们经常没法在 DOM 节点中找到惟一替换的位置,所以整批地重绘对话是个很好的工做机制。但这里有个例外,若你开始在对话的评论字段中输入一些文字,而在另外一个窗口向同一条对话添加了一条评论,那么第一个窗口中的字段就会被重绘,会移除掉其内容和焦点。

在激烈的讨论中,多人同时添加评论,这将是很是烦人的。 你能想出办法解决它吗?

相关文章
相关标签/搜索