<?xml version="1.0" encoding="utf-8"?>
<feed xmlns="http://www.w3.org/2005/Atom">
  <title>想念熊学编程</title>
  
  <subtitle>人生苦短，我学前端</subtitle>
  <link href="/atom.xml" rel="self"/>
  
  <link href="http://wfy.netlify.com/"/>
  <updated>2018-08-24T09:17:47.170Z</updated>
  <id>http://wfy.netlify.com/</id>
  
  <author>
    <name>王丰圆</name>
    
  </author>
  
  <generator uri="http://hexo.io/">Hexo</generator>
  
  <entry>
    <title>Promise异步函数顺序执行的四种方法</title>
    <link href="http://wfy.netlify.com/2018/08/24/promise/"/>
    <id>http://wfy.netlify.com/2018/08/24/promise/</id>
    <published>2018-08-24T17:01:11.000Z</published>
    <updated>2018-08-24T09:17:47.170Z</updated>
    
    <content type="html"><![CDATA[<p>Promise异步函数顺序执行的四种方法</p><p>前段时间遇到一个编程题，要求控制promise顺序执行，今天总结了一下这个至少有好四种方法都可以实现，包括promise嵌套，通过一个promise串起来，generator，async实现，以下逐一介绍。<br>题目：<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br></pre></td><td class="code"><pre><span class="line">//实现mergePromise函数，把传进去的数组顺序先后执行，</span><br><span class="line">//并且把返回的数据先后放到数组data中</span><br><span class="line">const timeout = ms =&gt; new Promise((resolve, reject) =&gt; &#123;</span><br><span class="line">    setTimeout(() =&gt; &#123;</span><br><span class="line">        resolve();</span><br><span class="line">    &#125;, ms);</span><br><span class="line">&#125;);</span><br><span class="line"></span><br><span class="line">const ajax1 = () =&gt; timeout(2000).then(() =&gt; &#123;</span><br><span class="line">    console.log(&apos;1&apos;);</span><br><span class="line">    return 1;</span><br><span class="line">&#125;);</span><br><span class="line"></span><br><span class="line">const ajax2 = () =&gt; timeout(1000).then(() =&gt; &#123;</span><br><span class="line">    console.log(&apos;2&apos;);</span><br><span class="line">    return 2;</span><br><span class="line">&#125;);</span><br><span class="line"></span><br><span class="line">const ajax3 = () =&gt; timeout(2000).then(() =&gt; &#123;</span><br><span class="line">    console.log(&apos;3&apos;);</span><br><span class="line">    return 3;</span><br><span class="line">&#125;);</span><br><span class="line"></span><br><span class="line">function mergePromise(ajaxArray) &#123;</span><br><span class="line">//todo 补全函数</span><br><span class="line">&#125;</span><br><span class="line"></span><br><span class="line">mergePromise([ajax1, ajax2, ajax3]).then(data =&gt; &#123;</span><br><span class="line">    console.log(&apos;done&apos;);</span><br><span class="line">    console.log(data); // data 为 [1, 2, 3]</span><br><span class="line">&#125;);</span><br><span class="line"></span><br><span class="line">// 分别输出</span><br><span class="line">// 1</span><br><span class="line">// 2</span><br><span class="line">// 3</span><br><span class="line">// done</span><br><span class="line">// [1, 2, 3]</span><br></pre></td></tr></table></figure></p><h2 id="一-promise嵌套"><a href="#一-promise嵌套" class="headerlink" title="一. promise嵌套"></a>一. promise嵌套</h2><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br></pre></td><td class="code"><pre><span class="line">function mergePromise1(ajaxArray) &#123;</span><br><span class="line">  let arr = [];</span><br><span class="line">    return ajaxArray[0]().then(data=&gt;&#123;</span><br><span class="line">        arr.push(data);</span><br><span class="line">        return ajaxArray[1]();</span><br><span class="line">    &#125;).then(data=&gt;&#123;</span><br><span class="line">        arr.push(data);</span><br><span class="line">        return ajaxArray[2]();</span><br><span class="line">    &#125;).then(data=&gt;&#123;</span><br><span class="line">      arr.push(data);</span><br><span class="line">      return arr;</span><br><span class="line">    &#125;);</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h2 id="二-Promise-resolve将promise串连成一个任务队列"><a href="#二-Promise-resolve将promise串连成一个任务队列" class="headerlink" title="二. Promise.resolve将promise串连成一个任务队列"></a>二. Promise.resolve将promise串连成一个任务队列</h2><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br></pre></td><td class="code"><pre><span class="line">function mergePromise2(ajaxArray) &#123;</span><br><span class="line">  let p = Promise.resolve();</span><br><span class="line">  let arr = [];</span><br><span class="line">  ajaxArray.forEach(promise =&gt; &#123;</span><br><span class="line">    p = p.then(promise).then((data) =&gt; &#123;</span><br><span class="line">        arr.push(data);</span><br><span class="line">        return arr;</span><br><span class="line">    &#125;);</span><br><span class="line">  &#125;);</span><br><span class="line">  return p;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><p>此方法相对于上面的方法简单并且书写直观易懂，还有一种类似的任务队列，将数组按顺序从左边头部取出一个执行，执行完成后触发自定义next方法，next方法负责从数组中取出下一个任务执行。</p><h2 id="三-generator函数"><a href="#三-generator函数" class="headerlink" title="三. generator函数"></a>三. generator函数</h2><h3 id="1-原生generator函数"><a href="#1-原生generator函数" class="headerlink" title="1. 原生generator函数"></a>1. 原生generator函数</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br></pre></td><td class="code"><pre><span class="line">var mergePromise3 = function* (ajaxArray) &#123;</span><br><span class="line">  let p1 = yield ajaxArray[0]();</span><br><span class="line">  let p2 = yield ajaxArray[1]();</span><br><span class="line">  let p3 = yield ajaxArray[2]();</span><br><span class="line">  return Promise.resolve([p1,p2,p3]);</span><br><span class="line">&#125;</span><br><span class="line"></span><br><span class="line">//自动运行的run</span><br><span class="line">function run(fn) &#123;</span><br><span class="line">  return new Promise((resolve, reject) =&gt; &#123;</span><br><span class="line">    var g = fn;</span><br><span class="line">    let arr = [];</span><br><span class="line">    function next(preData) &#123;</span><br><span class="line">      if(preData) &#123; //如果有数则push进数组</span><br><span class="line">        arr.push(preData); </span><br><span class="line">      &#125;</span><br><span class="line">      let result = g.next(preData); //获取每一步执行结果，其中value为promise对象，done表示是否执行完成</span><br><span class="line">      if (result.done) &#123; //函数执行完毕则resolve数组</span><br><span class="line">        resolve(arr);</span><br><span class="line">      &#125;</span><br><span class="line">      else &#123; //函数没有执行完毕则递归执行</span><br><span class="line">          result.value.then(function(nowData) &#123;</span><br><span class="line">            next(nowData);</span><br><span class="line">          &#125;);</span><br><span class="line">      &#125;</span><br><span class="line">    &#125;</span><br><span class="line">    next();</span><br><span class="line">  &#125;);</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><p>使用这种方法需要修改mergePromise方法为：<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line">run(mergePromise3([ajax1, ajax2, ajax3])).then(data =&gt; &#123;</span><br><span class="line">  console.log(&apos;done&apos;);</span><br><span class="line">  console.log(data); // data 为 [1, 2, 3]</span><br><span class="line">&#125;);</span><br></pre></td></tr></table></figure></p><h3 id="2-利用co模块自动执行"><a href="#2-利用co模块自动执行" class="headerlink" title="2. 利用co模块自动执行"></a>2. 利用co模块自动执行</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br></pre></td><td class="code"><pre><span class="line">const co = require(&apos;co&apos;)</span><br><span class="line">  co(mergePromise3([ajax1, ajax2, ajax3])).then(data =&gt; &#123;</span><br><span class="line">  console.log(&apos;done&apos;);</span><br><span class="line">  console.log(data); // data 为 [1, 2, 3]</span><br><span class="line">&#125;);</span><br></pre></td></tr></table></figure><p>此方法原理和上面一样，只是使用已有的封装好的co模块来自动执行</p><h2 id="四-async函数"><a href="#四-async函数" class="headerlink" title="四. async函数"></a>四. async函数</h2><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br></pre></td><td class="code"><pre><span class="line">function mergePromise4(ajaxArray) &#123;</span><br><span class="line">  let arr = [];</span><br><span class="line">  async function run() &#123;</span><br><span class="line">      for(let p of ajaxArray) &#123;</span><br><span class="line">          let val = await p();</span><br><span class="line">          arr.push(val);</span><br><span class="line">      &#125;</span><br><span class="line">      return arr;</span><br><span class="line">  &#125;</span><br><span class="line">  return run();</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><p>以上列出了四种方法，具体使用那种方法也根据喜好而定，如果有其他的好的方法欢迎留言补充。</p>]]></content>
    
    <summary type="html">
    
      
      
        &lt;p&gt;Promise异步函数顺序执行的四种方法&lt;/p&gt;
&lt;p&gt;前段时间遇到一个编程题，要求控制promise顺序执行，今天总结了一下这个至少有好四种方法都可以实现，包括promise嵌套，通过一个promise串起来，generator，async实现，以下逐一介绍。&lt;br&gt;题目
      
    
    </summary>
    
    
      <category term="算法" scheme="http://wfy.netlify.com/tags/%E7%AE%97%E6%B3%95/"/>
    
      <category term="面试" scheme="http://wfy.netlify.com/tags/%E9%9D%A2%E8%AF%95/"/>
    
  </entry>
  
  <entry>
    <title>Javacript二叉树算法实现及快排求第K大值</title>
    <link href="http://wfy.netlify.com/2018/08/24/binaryTree/"/>
    <id>http://wfy.netlify.com/2018/08/24/binaryTree/</id>
    <published>2018-08-24T17:00:16.000Z</published>
    <updated>2018-08-24T09:17:47.158Z</updated>
    
    <content type="html"><![CDATA[<p><b>之前实习笔试的时候刷题一直用的java，也参考某篇文章写过java版的二叉树常见算法，因为马上要转正面试了，这几天都在准备面试，就把之前的翻出来用javascript重新写了一遍，二叉树基本都是递归处理的，也比较简单，就当做热身。用快排求前K大值，另外如果之前java版的作者看到的话可以留言，我会标明文章引用。</b></p><h2 id="Javacript二叉树常见算法实现"><a href="#Javacript二叉树常见算法实现" class="headerlink" title="Javacript二叉树常见算法实现"></a>Javacript二叉树常见算法实现</h2><h3 id="节点构造函数和二叉树构造函数"><a href="#节点构造函数和二叉树构造函数" class="headerlink" title="节点构造函数和二叉树构造函数"></a>节点构造函数和二叉树构造函数</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br></pre></td><td class="code"><pre><span class="line">function Node(key) &#123;</span><br><span class="line">    this.key = key;</span><br><span class="line">    this.left = null;</span><br><span class="line">    this.right = null;</span><br><span class="line">&#125;</span><br><span class="line">function binaryTree() &#123;</span><br><span class="line">    this.root = null;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="插入节点生成二叉树"><a href="#插入节点生成二叉树" class="headerlink" title="插入节点生成二叉树"></a>插入节点生成二叉树</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br></pre></td><td class="code"><pre><span class="line">binaryTree.prototype.insert = function(root, key) &#123;</span><br><span class="line">    var node = new Node(key);</span><br><span class="line">    if (root === null) &#123; //树根节点为空则将此节点作为根节点</span><br><span class="line">        this.root = node;</span><br><span class="line">    &#125; else if (node.key &lt; root.key) &#123; //小于左孩子节点则要么作为左子节点要么递归插入左部门</span><br><span class="line">        if (root.left === null) &#123;</span><br><span class="line">            root.left = node;</span><br><span class="line">        &#125; else &#123;</span><br><span class="line">            this.insert(root.left, key);</span><br><span class="line">        &#125;</span><br><span class="line">    &#125; else &#123; //大于右孩子节点则要么作为右子节点要么递归插入到右部分</span><br><span class="line">        if (root.right === null) &#123;</span><br><span class="line">            root.right = node;</span><br><span class="line">        &#125; else &#123;</span><br><span class="line">            this.insert(root.right, key);</span><br><span class="line">        &#125;</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="先序遍历"><a href="#先序遍历" class="headerlink" title="先序遍历"></a>先序遍历</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br></pre></td><td class="code"><pre><span class="line">//先序遍历递归方法</span><br><span class="line">binaryTree.prototype.preOrder = function(node) &#123;</span><br><span class="line">    if (node !== null) &#123;</span><br><span class="line">        console.log(node.key); //先打印当前结点</span><br><span class="line">        this.inOrder(node.left); //打印左结点</span><br><span class="line">        this.inOrder(node.right); //打印右结点</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br><span class="line"></span><br><span class="line">//先序遍历非递归方法</span><br><span class="line">//首先将根节点入栈，如果栈不为空，取出节点打印key值，然后依次取右节点和左节点入栈，依次重复</span><br><span class="line">binaryTree.prototype.preOrder2 = function(node) &#123;</span><br><span class="line">    let stack = [];</span><br><span class="line">    stack.push(node);</span><br><span class="line">    while (stack.length &gt; 0) &#123;</span><br><span class="line">        let n = stack.pop();</span><br><span class="line">        console.log(n.key);</span><br><span class="line">        if (n.right != null) &#123;</span><br><span class="line">            stack.push(n.right);</span><br><span class="line">        &#125;</span><br><span class="line">        if (n.left != null) &#123;</span><br><span class="line">            stack.push(n.left);</span><br><span class="line">        &#125;</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="中序遍历"><a href="#中序遍历" class="headerlink" title="中序遍历"></a>中序遍历</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br></pre></td><td class="code"><pre><span class="line">//中序遍历递归方法</span><br><span class="line">binaryTree.prototype.inOrder = function(node) &#123;</span><br><span class="line">    if (node !== null) &#123;</span><br><span class="line">        this.inOrder(node.left);</span><br><span class="line">        console.log(node.key);</span><br><span class="line">        this.inOrder(node.right);</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br><span class="line"></span><br><span class="line">//中序遍历非递归方法</span><br><span class="line">//依次取左节点入栈直到左下角节点入栈完毕，弹出节点打印key,如果该节点有右子节点，将其入栈</span><br><span class="line">binaryTree.prototype.inOrder2 = function(node) &#123;</span><br><span class="line">    let stack = [];</span><br><span class="line">    while (node != null || stack.length) &#123;</span><br><span class="line">        if (node != null) &#123;</span><br><span class="line">            stack.push(node);</span><br><span class="line">            node = node.left;</span><br><span class="line">        &#125; else &#123;</span><br><span class="line">            let n = stack.pop();</span><br><span class="line">            console.log(n.key);</span><br><span class="line">            node = n.right;</span><br><span class="line">        &#125;</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="后序遍历"><a href="#后序遍历" class="headerlink" title="后序遍历"></a>后序遍历</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line">binaryTree.prototype.postOrder = function(node) &#123;</span><br><span class="line">    if (node !== null) &#123;</span><br><span class="line">        this.inOrder(node.left);</span><br><span class="line">        this.inOrder(node.right);</span><br><span class="line">        console.log(node.key);</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="求树的深度"><a href="#求树的深度" class="headerlink" title="求树的深度"></a>求树的深度</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br></pre></td><td class="code"><pre><span class="line">binaryTree.prototype.treeDepth = function(node) &#123;</span><br><span class="line">    if (node === null) &#123;</span><br><span class="line">        return 0;</span><br><span class="line">    &#125;</span><br><span class="line">    let left = this.treeDepth(node.left);</span><br><span class="line">    let right = this.treeDepth(node.right);</span><br><span class="line">    return (left &gt; right) ? (left + 1) : (right + 1);</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="判断两棵树结构是否相同"><a href="#判断两棵树结构是否相同" class="headerlink" title="判断两棵树结构是否相同"></a>判断两棵树结构是否相同</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br></pre></td><td class="code"><pre><span class="line">binaryTree.prototype.structCmp = function(root1, root2) &#123;</span><br><span class="line">    if (root1 == null &amp;&amp; root2 == null) &#123; //根节点都为空返回true</span><br><span class="line">        return true;</span><br><span class="line">    &#125;</span><br><span class="line">    if (root1 == null || root2 == null) &#123; //根节点一个为空一个不为空返回false</span><br><span class="line">        return false;</span><br><span class="line">    &#125;</span><br><span class="line">    let a = this.structCmp(root1.left, root2.left); //都有孩子节点则递归判断左边是不是相同并且右边也相同</span><br><span class="line">    let b = this.structCmp(root1.right, root2.right);</span><br><span class="line">    return a &amp;&amp; b; //左子树相同并且右子树相同</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="得到第k层节点个数"><a href="#得到第k层节点个数" class="headerlink" title="得到第k层节点个数"></a>得到第k层节点个数</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br></pre></td><td class="code"><pre><span class="line">binaryTree.prototype.getLevelNum = function(root, k) &#123;</span><br><span class="line">    if (root == null || k &lt; 1) &#123;</span><br><span class="line">        return 0;</span><br><span class="line">    &#125;</span><br><span class="line">    if (k == 1) &#123;</span><br><span class="line">        return 1;</span><br><span class="line">    &#125;</span><br><span class="line">    return this.getLevelNum(root.left, k - 1) + this.getLevelNum(root.right, k - 1); //从左子树角度看，根节点第k层就是相对于左子树k-1层，把左子树右子树k-1层相加即可</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="求二叉树的镜像"><a href="#求二叉树的镜像" class="headerlink" title="求二叉树的镜像"></a>求二叉树的镜像</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br></pre></td><td class="code"><pre><span class="line">binaryTree.prototype.mirror = function(node) &#123;</span><br><span class="line">    if (node == null) return;</span><br><span class="line">    [node.left, node.right] = [node.right, node.left]; //交换左右子树并依次递归</span><br><span class="line">    this.mirror(node.left);</span><br><span class="line">    this.mirror(node.right);</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="最近公共祖先节点"><a href="#最近公共祖先节点" class="headerlink" title="最近公共祖先节点"></a>最近公共祖先节点</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br></pre></td><td class="code"><pre><span class="line">binaryTree.prototype.findLCA = function(node, target1, target2) &#123;</span><br><span class="line">    if (node == null) &#123;</span><br><span class="line">        return null;</span><br><span class="line">    &#125;</span><br><span class="line">    if (node.key == target1 || node.key == target2) &#123; //如果当前结点和其中一个节点相等则当前结点为公共祖先</span><br><span class="line">        return node;</span><br><span class="line">    &#125;</span><br><span class="line">    let left = this.findLCA(node.left, target1, target2);</span><br><span class="line">    let right = this.findLCA(node.right, target1, target2);</span><br><span class="line">    if (left != null &amp;&amp; right != null) &#123; //如果左右子树都没找到则目标节点分别在左右子树，根节点为其祖先</span><br><span class="line">        return node;</span><br><span class="line">    &#125;</span><br><span class="line">    return (left != null) ? left : right; // 找到的话返回</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="测试用"><a href="#测试用" class="headerlink" title="测试用"></a>测试用</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br></pre></td><td class="code"><pre><span class="line">var tree = new binaryTree();</span><br><span class="line">let arr = [45, 5, 13, 3, 23, 7, 111];</span><br><span class="line">arr.forEach((node) =&gt; &#123;</span><br><span class="line">    tree.insert(tree.root, node);</span><br><span class="line">&#125;);</span><br><span class="line"></span><br><span class="line">var tree2 = new binaryTree();</span><br><span class="line">let arr2 = [46, 6, 14, 4, 24, 8, 112];</span><br><span class="line">arr2.forEach((node) =&gt; &#123;</span><br><span class="line">    tree2.insert(tree2.root, node);</span><br><span class="line">&#125;);</span><br><span class="line"></span><br><span class="line">tree.preOrder(tree.root);</span><br><span class="line">tree.preOrder2(tree.root);</span><br><span class="line">tree2.inOrder(tree2.root);</span><br><span class="line">tree.inOrder2(tree.root);</span><br><span class="line">let depth = tree.treeDepth(tree.root);</span><br><span class="line">console.log(depth);</span><br><span class="line">let isstructCmp = tree2.structCmp(tree.root, tree2.root);</span><br><span class="line">console.log(isstructCmp);</span><br><span class="line">let num = tree.getLevelNum(tree.root, 4);</span><br><span class="line">console.log(num);</span><br><span class="line">tree.mirror(tree.root);</span><br><span class="line">tree.inOrder(tree.root);</span><br><span class="line">let n = tree.findLCA(tree.root, 111, 3);</span><br><span class="line">console.log(n);</span><br></pre></td></tr></table></figure><h2 id="快速排序求第K大值"><a href="#快速排序求第K大值" class="headerlink" title="快速排序求第K大值"></a>快速排序求第K大值</h2><h3 id="快速排序"><a href="#快速排序" class="headerlink" title="快速排序"></a>快速排序</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br></pre></td><td class="code"><pre><span class="line">function quickSort(array) &#123;</span><br><span class="line">  if(array.length &lt;= 1)&#123;</span><br><span class="line">      return array;</span><br><span class="line">  &#125;</span><br><span class="line">  let middle = Math.floor(array.length / 2)</span><br><span class="line">  let pivot = array.splice(middle, 1);</span><br><span class="line">  let left =[], right = [];</span><br><span class="line">  for(let i = 0; i &lt; array.length; i++) &#123;</span><br><span class="line">      if(array[i] &lt; pivot) &#123;</span><br><span class="line">          left.push(array[i]);</span><br><span class="line">      &#125; else &#123;</span><br><span class="line">          right.push(array[i]);</span><br><span class="line">      &#125;</span><br><span class="line">      </span><br><span class="line">  &#125;</span><br><span class="line">  return quickSort(left).concat(pivot, quickSort(right));</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure><h3 id="快速排序改进求第K大值"><a href="#快速排序改进求第K大值" class="headerlink" title="快速排序改进求第K大值"></a>快速排序改进求第K大值</h3><blockquote><p>思想是通过快排把数组切割成左中右三部分，将K与右边数组（当然选左边数组也可以）长度作比较，如果右边数组长度为K-1,则中间元素即为第K大值，如果右边数组长度大于等于K，则第K大元素肯定在右边，则只需要对右边数组递归求K大值，如果右边数组长度小于K-1，则第K大值在左边，在左数组求第k-1-right.length大值即可<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br></pre></td><td class="code"><pre><span class="line">function getK(array, k) &#123;</span><br><span class="line">  if(array.length &lt;= 1)&#123;</span><br><span class="line">      return array;</span><br><span class="line">  &#125;</span><br><span class="line">  let middle = Math.floor(array.length / 2)</span><br><span class="line">  let pivot = array.splice(middle, 1);</span><br><span class="line">  let left =[], right = [];</span><br><span class="line">  for(let i = 0; i &lt; array.length; i++) &#123;</span><br><span class="line">      if(array[i] &lt; pivot) &#123;</span><br><span class="line">          left.push(array[i]);</span><br><span class="line">      &#125; else &#123;</span><br><span class="line">          right.push(array[i]);</span><br><span class="line">      &#125;</span><br><span class="line">      </span><br><span class="line">  &#125;</span><br><span class="line">  if(right.length == k - 1)&#123;</span><br><span class="line">      return pivot;</span><br><span class="line">  &#125; else if (right.length &gt;= k) &#123;</span><br><span class="line">      return getK(right, k);</span><br><span class="line">  &#125; else &#123;</span><br><span class="line">      return getK(left, k-right.length-1);</span><br><span class="line">  &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure></p></blockquote><blockquote><p>另外此方法也不是最佳解法，还有一种比较好的解法是利用建立K个元素的最小堆，新元素替换堆顶元素并调整堆，最后得到的K个元素即为最大的K个元素，时间复杂度NlogK,大家有其他方法或改进意见欢迎留言。</p></blockquote>]]></content>
    
    <summary type="html">
    
      
      
        &lt;p&gt;&lt;b&gt;之前实习笔试的时候刷题一直用的java，也参考某篇文章写过java版的二叉树常见算法，因为马上要转正面试了，这几天都在准备面试，就把之前的翻出来用javascript重新写了一遍，二叉树基本都是递归处理的，也比较简单，就当做热身。用快排求前K大值，另外如果之前java
      
    
    </summary>
    
    
      <category term="算法" scheme="http://wfy.netlify.com/tags/%E7%AE%97%E6%B3%95/"/>
    
      <category term="面试" scheme="http://wfy.netlify.com/tags/%E9%9D%A2%E8%AF%95/"/>
    
  </entry>
  
  <entry>
    <title>vue+express图片上传并用js-xlxs插件将链接导出excel</title>
    <link href="http://wfy.netlify.com/2018/08/01/exportExcel/"/>
    <id>http://wfy.netlify.com/2018/08/01/exportExcel/</id>
    <published>2018-08-01T01:08:12.000Z</published>
    <updated>2018-08-24T09:17:47.158Z</updated>
    
    <content type="html"><![CDATA[<h2 id="vue-axios-express图片上传并利用js-xlxs插件将图片链接导出到excel里"><a href="#vue-axios-express图片上传并利用js-xlxs插件将图片链接导出到excel里" class="headerlink" title="vue+axios+express图片上传并利用js-xlxs插件将图片链接导出到excel里"></a>vue+axios+express图片上传并利用js-xlxs插件将图片链接导出到excel里</h2><h2 id="一、介绍"><a href="#一、介绍" class="headerlink" title="一、介绍"></a>一、介绍</h2><p>vue + express实现图片上传，使用createObjectURL方法在前端页面显示图片缩略图，使用ClipboardJS实现复制剪切功能，最后利用js-xlxs插件将上传后的服务器端图片资源url地址导出到excel里面。本项目只是演示用，如果你是组件式开发，直接安装导入相关插件即可，其他方法都一样，最后实现效果图如下</p><p><img src="/2018/08/01/exportExcel/excel.gif" alt="图片描述"></p><h2 id="二、关键功能实现"><a href="#二、关键功能实现" class="headerlink" title="二、关键功能实现"></a>二、关键功能实现</h2><h3 id="1-图片缩略图生成"><a href="#1-图片缩略图生成" class="headerlink" title="1. 图片缩略图生成"></a>1. 图片缩略图生成</h3><p>最开始我是将图片上传到node端后，后台返回上传后的图片地址，然后前端显示，但是查找资料后发现createObjectURL就可以轻松实现图片缩略图，这在<b><a href="https://developer.mozilla.org/zh-CN/docs/Web/API/File/Using_files_from_web_applications" target="_blank" rel="noopener">MDN官网</a></b>上就有这些应用举例，这个网站介绍了使用createObjectURL方法可以展示图片，视频，PDF的缩略图，及其方便。因此这里采用该方法。另外input的mutiple属性可以实现多文件上传。<br>相关代码如下<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br></pre></td><td class="code"><pre><span class="line">html部分</span><br><span class="line">&lt;h1&gt;文件上传&lt;/h1&gt;</span><br><span class="line">&lt;form action=&quot;/upload&quot; method=&quot;post&quot; enctype=&quot;multipart/form-data&quot;&gt;</span><br><span class="line">    &lt;input type=&quot;file&quot; name=&quot;pic&quot; multiple ref=&quot;pic&quot;&gt;</span><br><span class="line">    &lt;input type=&quot;button&quot; @click=&quot;uploadFile&quot; value=&quot;上传&quot;&gt;</span><br><span class="line">&lt;/form&gt;</span><br><span class="line"></span><br><span class="line">js部分</span><br><span class="line">uploadFile() &#123;</span><br><span class="line">    //拿到上传的图片</span><br><span class="line">    var imgs = this.$refs.pic.files;</span><br><span class="line">    for (let i = 0; i &lt; imgs.length; i++) &#123;</span><br><span class="line">        //逐个获取图片</span><br><span class="line">        let file = imgs[i];</span><br><span class="line">        //使用createObjectURL方法生成图片缩略图预览</span><br><span class="line">        let src = window.URL.createObjectURL(file);</span><br><span class="line">        //上传该图片到服务器端并拿到返回的服务器端图片地址</span><br><span class="line">        let url = await uploadImg(file, this.$data.urlList);</span><br><span class="line">        //构造数组</span><br><span class="line">        this.$data.urlList.push(&#123;</span><br><span class="line">            src: src,</span><br><span class="line">            name: file.name,</span><br><span class="line">            url: &quot;localhost/&quot; + url</span><br><span class="line">        &#125;);</span><br><span class="line">    &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure></p><h3 id="2-图片上传并存储"><a href="#2-图片上传并存储" class="headerlink" title="2. 图片上传并存储"></a>2. 图片上传并存储</h3><p>先贴上这部分代码<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br></pre></td><td class="code"><pre><span class="line">//上传一个图片文件</span><br><span class="line">async function uploadImg(file, urlList) &#123;</span><br><span class="line">    var formData = new FormData();</span><br><span class="line">    formData.append(&quot;pic&quot;, file);</span><br><span class="line">    let url = await new Promise((resolve, reject) =&gt; &#123;</span><br><span class="line">            axios.post(&apos;/upload&apos;, formData, &#123;</span><br><span class="line">                    headers: &#123;</span><br><span class="line">                        &quot;Content-Type&quot;: &quot;multipart/form-data&quot;</span><br><span class="line">                    &#125;</span><br><span class="line">                &#125;)</span><br><span class="line">                .then((res) =&gt; &#123;</span><br><span class="line">                    resolve(res.data);</span><br><span class="line">                &#125;)</span><br><span class="line">                .catch((err) =&gt; &#123;</span><br><span class="line">                    console.log(err);</span><br><span class="line">                &#125;);</span><br><span class="line">        &#125;)</span><br><span class="line">        //返回服务器上对应地址</span><br><span class="line">    return url;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure></p><p>上面里面的uploadImg函数就是实现图片上传，图片文件上传使用FormData格式,利用axios发送post请求，在这里使用ES6的async，结合上面的<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">let url = await uploadImg(file, this.$data.urlList);</span><br></pre></td></tr></table></figure></p><p>这一句中的await，使得请求成功并完成后获取到线上url地址。node端代码如下：<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br></pre></td><td class="code"><pre><span class="line">app.post(&quot;/upload&quot;, function(req, res) &#123;</span><br><span class="line"></span><br><span class="line">    var form = new formidable.IncomingForm();</span><br><span class="line"></span><br><span class="line">    form.parse(req, function(err, fields, files) &#123;</span><br><span class="line">        let imgPath = files.pic.path</span><br><span class="line">        let imgName = files.pic.name</span><br><span class="line">        // 同步读取文件</span><br><span class="line">        let data = fs.readFileSync(imgPath)</span><br><span class="line">        // 存储上传的图片，同时获取静态图片地址并返回客户端</span><br><span class="line">        fs.writeFile(&quot;static/&quot; + imgName, data, function(err) &#123;</span><br><span class="line">            if (err) &#123;</span><br><span class="line">                console.log(err)</span><br><span class="line">                return;</span><br><span class="line">            &#125;</span><br><span class="line">            let itemUrl = &#123;</span><br><span class="line">                &quot;path&quot;: &quot;static/&quot; + imgName</span><br><span class="line">            &#125;;</span><br><span class="line">            let url = &quot;static/&quot; + imgName;</span><br><span class="line">            res.send(url);</span><br><span class="line">        &#125;)</span><br><span class="line">    &#125;)</span><br><span class="line"></span><br><span class="line">&#125;);</span><br></pre></td></tr></table></figure></p><p>使用formidable解析图片，并同步写入图片到static文件夹，最后获取线上图片地址返回</p><h3 id="3-前端显示"><a href="#3-前端显示" class="headerlink" title="3. 前端显示"></a>3. 前端显示</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br></pre></td><td class="code"><pre><span class="line">&lt;h1&gt;图片列表&lt;/h1&gt;</span><br><span class="line">&lt;div class=&quot;img-wrapper&quot;&gt;</span><br><span class="line">    &lt;div class=&quot;uploading&quot; v-for=&quot;(item, index) in urlList&quot; :key=&quot;index&quot;&gt;</span><br><span class="line">        &lt;div&gt;</span><br><span class="line">            &lt;img class=&quot;uploading-image&quot; :src=&quot;item.src&quot; alt=&quot;&quot;&gt;</span><br><span class="line">        &lt;/div&gt;</span><br><span class="line">        &lt;div class=&quot;uploading-info&quot;&gt;</span><br><span class="line">            &lt;span class=&quot;uploading-name&quot;&gt;</span><br><span class="line">                &lt;a target=&quot;_blank&quot; :href=&quot;item.src&quot;&gt;&#123;&#123; item.name &#125;&#125;&lt;/a&gt;</span><br><span class="line">            &lt;/span&gt;</span><br><span class="line">            &lt;span class=&quot;copy&quot; :data-clipboard-text=&quot;item.url&quot;&gt;复制&lt;/span&gt;</span><br><span class="line">        &lt;/div&gt;</span><br><span class="line">    &lt;/div&gt;</span><br><span class="line">&lt;/div&gt;</span><br></pre></td></tr></table></figure><p>其中src属性是生成的缩略图URL，name是图片名称，url属性是线上图片地址。</p><h3 id="4-导出excel"><a href="#4-导出excel" class="headerlink" title="4. 导出excel"></a>4. 导出excel</h3><p>导出excel目前有很多插件，最开始我是使用excel-export插件，官网地址<a href="https://github.com/functionscope/Node-Excel-Export" target="_blank" rel="noopener">在这里</a>,这个插件api很简单也很方便，但是有点BUG，他说能设置单元格宽度，但是按照他的来我不能设置单元格宽度，之后又找到一个更强大的<a href="https://github.com/SheetJS/js-xlsx" target="_blank" rel="noopener">js-xlxs插件</a>，这个插件很强大，基本上需要的excel相关的需求都能满足要求吧。具体导出excel代码如下，注意生成表格的数据需要是二维数组。<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br></pre></td><td class="code"><pre><span class="line">downloadURL() &#123;</span><br><span class="line">    var imgURLS = [];</span><br><span class="line">    //设置excel标题</span><br><span class="line">    imgURLS.push([&quot;序号&quot;, &quot;图片名称&quot;, &quot;URL链接&quot;]);</span><br><span class="line">    //获取所有图片url链接信息</span><br><span class="line">    this.$data.urlList.forEach((item, index) =&gt; &#123;</span><br><span class="line">        //构建一个数组</span><br><span class="line">        var itemArray = [index + 1].concat(item.name, item.url);</span><br><span class="line">        //构建二维数组</span><br><span class="line">        imgURLS.push(itemArray);</span><br><span class="line">    &#125;);</span><br><span class="line">    //生成工作表结构</span><br><span class="line">    const ws = XLSX.utils.aoa_to_sheet(imgURLS);</span><br><span class="line">    //设置三列单元格宽度</span><br><span class="line">    var wscols = [&#123;</span><br><span class="line">        wch: 6</span><br><span class="line">    &#125;, &#123;</span><br><span class="line">        wch: 50</span><br><span class="line">    &#125;, &#123;</span><br><span class="line">        wch: 50</span><br><span class="line">    &#125;];</span><br><span class="line">    ws[&apos;!cols&apos;] = wscols;</span><br><span class="line">    //生成excel基本数据结构</span><br><span class="line">    const wb = XLSX.utils.book_new();</span><br><span class="line">    //生成表名字为SheetJS的excel工作区</span><br><span class="line">    XLSX.utils.book_append_sheet(wb, ws, &quot;SheetJS&quot;);</span><br><span class="line">    //导出excel</span><br><span class="line">    XLSX.writeFile(wb, &quot;export.xlsx&quot;);</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure></p><h2 id="三、运行项目"><a href="#三、运行项目" class="headerlink" title="三、运行项目"></a>三、运行项目</h2><p>到这里就实现了最开始的功能，如果需要全部代码可以在我的github上下载</p><blockquote><p>本文详细代码可以在github上下载，地址<a href="https://github.com/wangfengyuan/ExportExcel" target="_blank" rel="noopener">https://github.com/wangfengyuan/ExportExcel</a><br>运行代码<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line">npm install</span><br><span class="line">node app</span><br><span class="line">浏览器输入： localhost</span><br></pre></td></tr></table></figure></p></blockquote><h2 id="四、写在最后"><a href="#四、写在最后" class="headerlink" title="四、写在最后"></a>四、写在最后</h2><blockquote><p>我前端刚入门没多久，最近在公司实习，刚写文章不久，所以可能写的不太好，大家对文章和代码有什么建议或者有更好的想法欢迎提出来一起交流，谢谢！</p></blockquote>]]></content>
    
    <summary type="html">
    
      
      
        &lt;h2 id=&quot;vue-axios-express图片上传并利用js-xlxs插件将图片链接导出到excel里&quot;&gt;&lt;a href=&quot;#vue-axios-express图片上传并利用js-xlxs插件将图片链接导出到excel里&quot; class=&quot;headerlink&quot; title
      
    
    </summary>
    
    
      <category term="vue" scheme="http://wfy.netlify.com/tags/vue/"/>
    
      <category term="nodejs" scheme="http://wfy.netlify.com/tags/nodejs/"/>
    
      <category term="html" scheme="http://wfy.netlify.com/tags/html/"/>
    
  </entry>
  
  <entry>
    <title>微信网页授权并获取用户信息</title>
    <link href="http://wfy.netlify.com/2018/07/28/wxAuthorize/"/>
    <id>http://wfy.netlify.com/2018/07/28/wxAuthorize/</id>
    <published>2018-07-28T18:44:10.000Z</published>
    <updated>2018-08-24T09:17:47.170Z</updated>
    
    <content type="html"><![CDATA[<h2 id="介绍"><a href="#介绍" class="headerlink" title="介绍"></a>介绍</h2><blockquote><p>在很多微信H5应用里，当用户访问第三方应用时就需要进行微信网页授权，并且很多涉及安全的操作我们必须要先获取用户信息才能继续，本文章简单介绍了微信授权流程，并通过申请微信测试账号来模拟网页授权，用户在授权页点击确定登录后获取用户信息并显示在前端页面，最后效果如下图</p></blockquote><p><img src="/2018/07/28/wxAuthorize/个人信息.PNG" alt="图片描述"></p><h2 id="工具及开发准备"><a href="#工具及开发准备" class="headerlink" title="工具及开发准备"></a>工具及开发准备</h2><h3 id="1-微信开发者工具及微信测试号"><a href="#1-微信开发者工具及微信测试号" class="headerlink" title="1. 微信开发者工具及微信测试号"></a>1. 微信开发者工具及微信测试号</h3><p>因为是微信授权，所以必须要在微信环境下使用，首先我们要在<a href="https://developers.weixin.qq.com/miniprogram/dev/devtools/download.html" target="_blank" rel="noopener">这里</a>安装微信开发者工具，因为我们没有自己的应用，所以还需要在<a href="https://mp.weixin.qq.com/debug/cgi-bin/sandbox?t=sandbox/login" target="_blank" rel="noopener">微信公众平台</a>申请一个接口测试号,这个接口测试号就相当于我们的第三方应用。</p><h3 id="2-参数设置"><a href="#2-参数设置" class="headerlink" title="2. 参数设置"></a>2. 参数设置</h3><p>登陆测试号后可以查看到自己的appId和appsecret信息，将体验接口权限表里的网页服务的网页授权获取用户基本信息修改为127.0.0.1:8800，该地址就是用户确认授权后回调的地址即我们应用的后台处理地址，如下图</p><p><img src="/2018/07/28/wxAuthorize/修改回调页面域名.PNG" alt="图片描述"></p><p>最后拿出自己微信扫码关注该测试号即可,如下图所示</p><p><img src="/2018/07/28/wxAuthorize/关注测试号.PNG" alt="图片描述"></p><h2 id="微信授权流程介绍"><a href="#微信授权流程介绍" class="headerlink" title="微信授权流程介绍"></a>微信授权流程介绍</h2><p>具体流程及详细介绍大家可以到官网<a href="https://mp.weixin.qq.com/wiki?t=resource/res_main&amp;id=mp1421140842" target="_blank" rel="noopener">微信公众平台技术文档</a>查看,大致分为四步：</p><ol><li><p>引导用户进入授权页面同意授权，此时会调用微信api获取code </p></li><li><p>授权通过后会带上code参数请求回调地址</p></li><li><p>后台获取code,再次调用微信接口换取网页授权access_token和openid</p></li><li><p>通过网页授权access_token和openid获取用户基本信息（如果有unionid还会获取到unionid参数）</p></li></ol><h2 id="正式开始"><a href="#正式开始" class="headerlink" title="正式开始"></a>正式开始</h2><blockquote><p>详细代码可以在github上下载，地址<a href="https://github.com/wangfengyuan/wxAuthorize" target="_blank" rel="noopener">https://github.com/wangfengyuan/wxAuthorize</a></p></blockquote><h3 id="1-原始代码"><a href="#1-原始代码" class="headerlink" title="1. 原始代码"></a>1. 原始代码</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br><span class="line">39</span><br><span class="line">40</span><br><span class="line">41</span><br><span class="line">42</span><br><span class="line">43</span><br><span class="line">44</span><br><span class="line">45</span><br><span class="line">46</span><br><span class="line">47</span><br><span class="line">48</span><br><span class="line">49</span><br><span class="line">50</span><br><span class="line">51</span><br><span class="line">52</span><br><span class="line">53</span><br><span class="line">54</span><br><span class="line">55</span><br><span class="line">56</span><br><span class="line">57</span><br><span class="line">58</span><br><span class="line">59</span><br><span class="line">60</span><br><span class="line">61</span><br><span class="line">62</span><br><span class="line">63</span><br><span class="line">64</span><br><span class="line">65</span><br></pre></td><td class="code"><pre><span class="line">let express = require(&quot;express&quot;);</span><br><span class="line">const https = require(&apos;https&apos;);</span><br><span class="line"></span><br><span class="line">let app = express();</span><br><span class="line"></span><br><span class="line">//appID</span><br><span class="line">let appID = `wxec6fa9e9bc03d885`;</span><br><span class="line">//appsecret</span><br><span class="line">let appSerect = `4c8a0d14cff08959b4e17334cabf9cf0`;</span><br><span class="line">//点击授权后重定向url地址</span><br><span class="line">let redirectUrl = `/getUserInfo`;</span><br><span class="line">let host = `http://127.0.0.1:3000`;</span><br><span class="line">//微信授权api,接口返回code,点击授权后跳转到重定向地址并带上code参数</span><br><span class="line">let authorizeUrl = `https://open.weixin.qq.com/connect/oauth2/authorize?appid=$&#123;appID&#125;&amp;redirect_uri=` +</span><br><span class="line">    `$&#123;host&#125;$&#123;redirectUrl&#125;&amp;response_type=code&amp;scope=snsapi_userinfo&amp;state=STATE#wechat_redirect`</span><br><span class="line"></span><br><span class="line">app.get(&quot;/login&quot;, function(req, res) &#123;</span><br><span class="line">    res.sendFile(path.resolve(__dirname,&apos;login.html&apos;));</span><br><span class="line">&#125;);</span><br><span class="line"></span><br><span class="line">app.get(&quot;/auth&quot;, function(req, res) &#123;</span><br><span class="line">    res.writeHead(302, &#123;</span><br><span class="line">        &apos;Location&apos;: authorizeUrl</span><br><span class="line">    &#125;);</span><br><span class="line">    res.end();</span><br><span class="line">&#125;);</span><br><span class="line"></span><br><span class="line">app.get(&quot;/getUserInfo&quot;, function(req, res) &#123;</span><br><span class="line">    let code = req.query.code;</span><br><span class="line">    let getaccess = `https://api.weixin.qq.com/sns/oauth2/access_token?appid=` +</span><br><span class="line">        `$&#123;appID&#125;&amp;secret=$&#123;appSerect&#125;&amp;code=$&#123;code&#125;&amp;grant_type=authorization_code`;</span><br><span class="line">    //通过拿到的code和appID、app_serect获取access_token和open_id</span><br><span class="line">    https.get(getaccess, (resText) =&gt; &#123;</span><br><span class="line">        var ddd = &quot;&quot;;</span><br><span class="line">        resText.on(&apos;data&apos;, (d) =&gt; &#123;</span><br><span class="line">            ddd += d;</span><br><span class="line">        &#125;);</span><br><span class="line">        resText.on(&apos;end&apos;, () =&gt; &#123;</span><br><span class="line">            // console.log(ddd);</span><br><span class="line">            var obj = JSON.parse(ddd);</span><br><span class="line">            var access_token = obj.access_token;</span><br><span class="line">            var open_id = obj.openid;</span><br><span class="line">            //通过上一步获取的access_token和open_id获取userInfo即用户信息</span><br><span class="line">            let getUserUrl = `https://api.weixin.qq.com/sns/userinfo?access_token=$&#123;access_token&#125;&amp;openid=$&#123;open_id&#125;&amp;lang=zh_CN`;</span><br><span class="line">            https.get(getUserUrl, (resText) =&gt; &#123;</span><br><span class="line">                user = &quot;&quot;;</span><br><span class="line">                resText.on(&apos;data&apos;, (d) =&gt; &#123;</span><br><span class="line">                    user += d;</span><br><span class="line">                &#125;);</span><br><span class="line">                resText.on(&apos;end&apos;, () =&gt; &#123;</span><br><span class="line">                    console.log(user);</span><br><span class="line">                    var userobj = JSON.parse(user);</span><br><span class="line">                    res.send(userobj);</span><br><span class="line">                    console.log(userobj);</span><br><span class="line">                &#125;);</span><br><span class="line"></span><br><span class="line">            &#125;)</span><br><span class="line">        &#125;);</span><br><span class="line">    &#125;).on(&apos;error&apos;, (e) =&gt; &#123;</span><br><span class="line">        console.error(e);</span><br><span class="line">    &#125;);</span><br><span class="line">    // res.end();</span><br><span class="line">&#125;);</span><br><span class="line"></span><br><span class="line">app.listen(3000);</span><br></pre></td></tr></table></figure><p>具体使用时要将appID和appSerect换成你对应的参数即可，因为我们的请求是要按一定顺序的，但是node发送请求是异步的，所以我们的请求嵌套了三层，代码很难看，所以这里可以采用ES6的async和await解决异步回调地狱。</p><h3 id="2-使用ES6的async和await的改进代码"><a href="#2-使用ES6的async和await的改进代码" class="headerlink" title="2. 使用ES6的async和await的改进代码"></a>2. 使用ES6的async和await的改进代码</h3><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br><span class="line">39</span><br><span class="line">40</span><br><span class="line">41</span><br><span class="line">42</span><br><span class="line">43</span><br><span class="line">44</span><br><span class="line">45</span><br><span class="line">46</span><br><span class="line">47</span><br><span class="line">48</span><br><span class="line">49</span><br><span class="line">50</span><br><span class="line">51</span><br><span class="line">52</span><br><span class="line">53</span><br><span class="line">54</span><br><span class="line">55</span><br><span class="line">56</span><br><span class="line">57</span><br></pre></td><td class="code"><pre><span class="line">async function wxAuth(req, res) &#123;</span><br><span class="line">    //解析querystring获取URL中的code值</span><br><span class="line">    let code = req.query.code;</span><br><span class="line">    //通过拿到的code和appID、app_serect获取返回信息</span><br><span class="line">    let resObj = await getAccessToken(code);</span><br><span class="line">    //解析得到access_token和open_id</span><br><span class="line">    let access_token = resObj.access_token;</span><br><span class="line">    let open_id = resObj.openid;</span><br><span class="line">    //通过上一步获取的access_token和open_id获取userInfo即用户信息</span><br><span class="line">    let userObj = await getUserInfo(access_token, open_id);</span><br><span class="line">    console.log(userObj);</span><br><span class="line">    res.render(path.resolve(__dirname,&apos;userInfo.ejs&apos;), &#123;userObj: userObj&#125;);</span><br><span class="line">    // res.send(userObj);</span><br><span class="line">&#125;</span><br><span class="line"></span><br><span class="line"></span><br><span class="line">//通过拿到的code和appID、app_serect获取access_token和open_id</span><br><span class="line">function getAccessToken(code) &#123;</span><br><span class="line">    return new Promise( (resolve, reject) =&gt; &#123;</span><br><span class="line">        let getAccessUrl = `https://api.weixin.qq.com/sns/oauth2/access_token?appid=` +</span><br><span class="line">            `$&#123;appID&#125;&amp;secret=$&#123;appSerect&#125;&amp;code=$&#123;code&#125;&amp;grant_type=authorization_code`;</span><br><span class="line">        https.get(getAccessUrl, (res) =&gt; &#123;</span><br><span class="line">            var resText = &quot;&quot;;</span><br><span class="line">            res.on(&apos;data&apos;, (d) =&gt; &#123;</span><br><span class="line">                resText += d;</span><br><span class="line">            &#125;);</span><br><span class="line">            res.on(&apos;end&apos;, () =&gt; &#123;</span><br><span class="line">                var resObj = JSON.parse(resText);</span><br><span class="line">                resolve(resObj);</span><br><span class="line">            &#125;);</span><br><span class="line">        &#125;).on(&apos;error&apos;, (e) =&gt; &#123;</span><br><span class="line">            console.error(e);</span><br><span class="line">        &#125;);</span><br><span class="line">    &#125;);</span><br><span class="line">    </span><br><span class="line">&#125;</span><br><span class="line"></span><br><span class="line">//通过上一步获取的access_token和open_id获取userInfo即用户信息</span><br><span class="line">function getUserInfo(access_token, open_id) &#123;</span><br><span class="line">    return new Promise( (resolve, reject) =&gt; &#123;</span><br><span class="line">        let getUserUrl = `https://api.weixin.qq.com/sns/userinfo?access_token=$&#123;access_token&#125;&amp;openid=$&#123;open_id&#125;&amp;lang=zh_CN`;</span><br><span class="line">        https.get(getUserUrl, (res) =&gt; &#123;</span><br><span class="line">            var resText = &quot;&quot;;</span><br><span class="line">            res.on(&apos;data&apos;, (d) =&gt; &#123;</span><br><span class="line">                resText += d;</span><br><span class="line">            &#125;);</span><br><span class="line">            res.on(&apos;end&apos;, () =&gt; &#123;</span><br><span class="line">                var userObj = JSON.parse(resText);</span><br><span class="line">                resolve(userObj);</span><br><span class="line">            &#125;);</span><br><span class="line">        &#125;).on(&apos;error&apos;, (e) =&gt; &#123;</span><br><span class="line">            console.error(e);</span><br><span class="line">        &#125;);</span><br><span class="line">    &#125;)</span><br><span class="line">&#125;</span><br><span class="line"></span><br><span class="line">app.listen(8800);</span><br></pre></td></tr></table></figure><p><img src="/2018/07/28/wxAuthorize/微信授权页面.PNG" alt="图片描述"><br>修改后代码流程清晰了很多，点击上图确认登陆后最后将获取到的userObj通过ejs模板渲染在前端页面，就能看到文章最开始展现的效果图。</p><h2 id="写在最后"><a href="#写在最后" class="headerlink" title="写在最后"></a>写在最后</h2><blockquote><p>我前端刚入门没多久，最近在公司实习，受到身边同事影响，所以也开始写文章来记录自己的学习心得，这是我第一次写文章，所以可能写的不太好，大家对文章和代码有什么建议欢迎提出来一起交流，谢谢！</p></blockquote>]]></content>
    
    <summary type="html">
    
      
      
        &lt;h2 id=&quot;介绍&quot;&gt;&lt;a href=&quot;#介绍&quot; class=&quot;headerlink&quot; title=&quot;介绍&quot;&gt;&lt;/a&gt;介绍&lt;/h2&gt;&lt;blockquote&gt;
&lt;p&gt;在很多微信H5应用里，当用户访问第三方应用时就需要进行微信网页授权，并且很多涉及安全的操作我们必须要先获取用户信息
      
    
    </summary>
    
    
      <category term="nodejs" scheme="http://wfy.netlify.com/tags/nodejs/"/>
    
      <category term="微信H5" scheme="http://wfy.netlify.com/tags/%E5%BE%AE%E4%BF%A1H5/"/>
    
  </entry>
  
</feed>
