<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
  <channel>
    <title>老博客 on Code Now</title>
    <link>https://blog.0xnullpath.cc/tags/%E8%80%81%E5%8D%9A%E5%AE%A2/</link>
    <description>Recent content in 老博客 on Code Now</description>
    <generator>Hugo</generator>
    <language>zh-cn</language>
    <lastBuildDate>Tue, 10 Mar 2020 17:30:00 +0000</lastBuildDate>
    <atom:link href="https://blog.0xnullpath.cc/tags/%E8%80%81%E5%8D%9A%E5%AE%A2/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>树状数组（Binary Indexed）</title>
      <link>https://blog.0xnullpath.cc/posts/note-snippet-12-%E6%A0%91%E7%8A%B6%E6%95%B0%E7%BB%84binary-indexed/</link>
      <pubDate>Tue, 10 Mar 2020 17:30:00 +0000</pubDate>
      <guid>https://blog.0xnullpath.cc/posts/note-snippet-12-%E6%A0%91%E7%8A%B6%E6%95%B0%E7%BB%84binary-indexed/</guid>
      <description>&lt;blockquote&gt;&#xA;  &lt;p&gt;首先很感谢B站up主鹤翔万里的视频, 推荐看 &lt;a href=&#34;https://www.bilibili.com/video/av69667943?from=search&amp;amp;seid=10916758362943551299&#34;&gt;https://www.bilibili.com/video/av69667943?from=search&amp;amp;seid=10916758362943551299&lt;/a&gt; 本文为这个视频集合 算法笔记 写的总结&lt;/p&gt;&#xA;&lt;/blockquote&gt;&#xA;&lt;p&gt;问题引入： 给出了一个长度为&lt;code&gt;n&lt;/code&gt;的数组，完成以下两种操作&lt;/p&gt;&#xA;&lt;ol&gt;&#xA;&lt;li&gt;输出区间&lt;code&gt;[x, y]&lt;/code&gt;内每个数字的和&lt;/li&gt;&#xA;&lt;li&gt;将第&lt;code&gt;x&lt;/code&gt;个数加上&lt;code&gt;k&lt;/code&gt;&lt;/li&gt;&#xA;&lt;/ol&gt;&#xA;&lt;p&gt;最基础的算法：&lt;/p&gt;&#xA;&lt;ol&gt;&#xA;&lt;li&gt;维护一个&lt;code&gt;sum&lt;/code&gt;数组，其&lt;code&gt;sum[i]&lt;/code&gt;记录的为从0-i的和。其递推计算关系为&lt;code&gt;sum[i] = sum[i-1] + nums[i]&lt;/code&gt;其中&lt;code&gt;sum[0] = nums[0]&lt;/code&gt;。&lt;/li&gt;&#xA;&lt;li&gt;计算区间和即可使用&lt;code&gt;sum_xy = sum[y] - sum[x-1]&lt;/code&gt;。&lt;/li&gt;&#xA;&lt;li&gt;将第&lt;code&gt;x&lt;/code&gt;个数加上&lt;code&gt;v&lt;/code&gt;, 因为要更新&lt;code&gt;sum[x]...sum[n]&lt;/code&gt;所有值，因此其时间复杂度为$O(n)$&lt;/li&gt;&#xA;&lt;li&gt;如果进行&lt;code&gt;k&lt;/code&gt;次操作&lt;code&gt;add&lt;/code&gt;则时间复杂度为$O(kn)$，&lt;code&gt;k&lt;/code&gt;次求区间和的时间复杂度为$O(k)$&lt;/li&gt;&#xA;&lt;/ol&gt;&#xA;&lt;p&gt;在更新频繁的很多场景下$O(kn)$的时间复杂度是不可接受的。&lt;/p&gt;&#xA;&lt;p&gt;因此引入树状数组来解决此问题。&lt;/p&gt;&#xA;&lt;p&gt;树状数组(Binary Indexed Tree)：单点修改$O(logn)$，区间查询$O(logn)$，因此在区间内多次修改查询的速度为$O(klogn)$&lt;/p&gt;&#xA;&lt;h2 id=&#34;lowbit-运算&#34;&gt;LowBit 运算&lt;/h2&gt;&#xA;&lt;p&gt;非负证数n在二进制表示下最低为1及其后面的0构成的数值。&#xA;例如:&#xA;&lt;/p&gt;&#xA;$$lowbit(20) = lowbit(10100) = (100) = 4$$&lt;p&gt;如何操作呢:&lt;/p&gt;&#xA;&lt;p&gt;c++代码&lt;/p&gt;&#xA;&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;&#34;&gt;&lt;code class=&#34;language-c++&#34; data-lang=&#34;c++&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;lowbit&lt;/span&gt;(&lt;span style=&#34;color:#66d9ef&#34;&gt;unsigned&lt;/span&gt; &lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; n) {&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#75715e&#34;&gt;/**&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;     *  10100  20&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;     *  01100  20的补码&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;     * &amp;amp;------&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;     *  00100-&amp;gt;lowbit&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;    */&lt;/span&gt;&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;unsigned&lt;/span&gt; &lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; complement_n &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;~&lt;/span&gt;n &lt;span style=&#34;color:#f92672&#34;&gt;+&lt;/span&gt; &lt;span style=&#34;color:#ae81ff&#34;&gt;1&lt;/span&gt;; &lt;span style=&#34;color:#75715e&#34;&gt;// 求补码，如果是int的话直接 -n即可&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;&lt;/span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;return&lt;/span&gt; n &lt;span style=&#34;color:#f92672&#34;&gt;&amp;amp;&lt;/span&gt; complement_n; &#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;}&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;// 或者&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;&lt;/span&gt;&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;lowbit&lt;/span&gt;(&lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; n) {&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;return&lt;/span&gt; n &lt;span style=&#34;color:#f92672&#34;&gt;&amp;amp;&lt;/span&gt; (&lt;span style=&#34;color:#f92672&#34;&gt;-&lt;/span&gt;n)&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;}&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;python代码&lt;/p&gt;</description>
    </item>
  </channel>
</rss>
