树状数组与LIS(最长递增子序列)

求LIS的方法有很多,现在我知道的有三种,分别是dp, 贪心+二分,树状数组。这篇博客就重点写一点用树状数组解决LIS相关的题目。
预备知识:树状数组,离散化。
2019-11-7 想要写一篇树状数组,不知道会咕多久
一个板子题:题目
我就简单的说一下用树状数组的思路:首先数据很大,很自然的想到离散化。然后树状数组维护什么呢?[1-i](i为离散化后的相对大小)中以i结尾的LIS的长度,因为数据离散化成了[1-i]嘛,所以维护起来和普通的没什么两样。然后对于一个数a,离散化后它为i,那么我们就先求出 i-1 (想想为什么是i-1)结尾的LIS,再用这个结果+1去更新以 i 结尾的LIS的长度。
我把离散化所用的东西挑出来:

vector<int>v;
inline int getid(int x) {
  return lower_bound(v.begin(), v.end(), x) - v.begin() + 1;
}
  v.push_back(b[i]);
  sort(v.begin(), v.end());
  v.erase(unique(v.begin(), v.end()), v.end());
  getid(b[i]);

接下来是树状数组的更新和查询:

void update(int x, int y) {
  for(; x<=n; x += lowbit(x)) a[x] = max(a[x], y);
}
int query(int x) {
  int res = 0;
  for(; x; x -= lowbit(x)) res = max(res, a[x]);
  return res;
}

然后其实就结束了。

#include
using namespace std;
typedef long long LL;
const int INF = 0x3f3f3f3f;
namespace {
  template <typename T> inline void read(T &x) {
    x = 0; T f = 1;char s = getchar();
    for(; !isdigit(s); s = getchar()) if(s == '-') f = -1;
    for(;  isdigit(s); s = getchar()) x = (x << 3) + (x << 1) + (s ^ 48);
    x *= f;
  }
}
#define fio ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
#define _for(n,m,i) for (int i = (n); i < (m); i++)
#define _rep(n,m,i) for (int i = (n); i <= (m); i++)
#define lson rt << 1, l, mid
#define rson rt << 1 | 1, mid + 1, r
#define lowbit(x) x & (-x)
#define pii pair
int a[50010], n;
void update(int x, int y) {
  for(; x<=n; x += lowbit(x)) a[x] = max(a[x], y);
}
int query(int x) {
  int res = 0;
  for(; x; x -= lowbit(x)) res = max(res, a[x]);
  return res;
}
int b[50010];
vector<int>v;
inline int getid(int x) {
  return lower_bound(v.begin(), v.end(), x) - v.begin() + 1;
}
int main() {
  read(n);
  for (int i = 1; i <= n; i++) {
    read(b[i]);
    v.push_back(b[i]);
  }
  sort(v.begin(), v.end());
  v.erase(unique(v.begin(), v.end()), v.end());
  int ans = 0, num;
  for(int i = 1; i <= n; i++) {
    int id = getid(b[i]);
    num = query(id-1)+1;
    update(id, num);
    ans = max(ans, num);
  }
  cout << ans << endl;
}

提示,直接询问i求出来的是最长不下降子序列;
再来一个进阶题:题目
这题就是要求哪些数在LIS中一定会出现,那些数可能会出现。思路也很简单,求一遍最长递增子序列,再求一遍最长递减子序列,我们计f[i],g[i]为以i结尾的最长递增,递减子序列的长度。if(f[i] + g[i] == LIS + 1) 那么这个数就可能在LIS中出现,对于不同的i,如果有相同的f[i],那么些数在原来LIS中就是可能出现,反之的话就说明i对应的数是不可代替的,也就是一定会出现
ps:这题的数字都是大于0的,所以求递减子序列可以把每个数乘-1,再反着求最长递增子序列就可以了。
代码:

#include
using namespace std;
typedef long long LL;
const int INF = 0x3f3f3f3f;
namespace {
  template <typename T> inline void read(T &x) {
    x = 0; T f = 1;char s = getchar();
    for(; !isdigit(s); s = getchar()) if(s == '-') f = -1;
    for(;  isdigit(s); s = getchar()) x = (x << 3) + (x << 1) + (s ^ 48);
    x *= f;
  }
}
#define fio ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
#define _for(n,m,i) for (int i = (n); i < (m); i++)
#define _rep(n,m,i) for (int i = (n); i <= (m); i++)
#define lson rt << 1, l, mid
#define rson rt << 1 | 1, mid + 1, r
#define lowbit(x) x & (-x)
#define pii pair
const int N = 1e5+5;
int a[N], n, cnt;
void update(int x, int y) {
  for(; x<=cnt; x += lowbit(x)) a[x] = max(a[x], y);
}
int query(int x) {
  int res = 0;
  for(; x; x -= lowbit(x)) res = max(res, a[x]);
  return res;
}
int b[N>>1], f[N>>1], g[N>>1], Hash[N>>1];
vector<int>v;
inline int getid(int x) {
  return lower_bound(v.begin(), v.end(), x) - v.begin() + 1;
}
int main() {
  read(n);
  _rep(1,n,i) {
    read(b[i]);
    v.push_back(b[i]);
    v.push_back(-b[i]);
  }
  sort(v.begin(), v.end());
  v.erase(unique(v.begin(), v.end()), v.end());
  cnt = v.size();
  int ans = 0;
  _rep(1,n,i) {
    int id = getid(b[i]);
    f[i] = query(id-1)+1;
    update(id, f[i]);
    ans = max(ans, f[i]);
  }
  memset(a, 0, sizeof(a));
  for(int i = n; i >= 1; i--) {
    int id = getid(-b[i]);
    g[i] = query(id-1)+1;
    update(id, g[i]);
  }
  _rep(1,n,i) if(f[i] + g[i] == ans + 1) Hash[f[i]]++;
  printf("A:");
  _rep(1,n,i) if(f[i] + g[i] == ans + 1 && Hash[f[i]] > 1) printf("%d ", i);
  printf("\nB:");
  _rep(1,n,i) if(f[i] + g[i] == ans + 1 && Hash[f[i]] == 1) printf("%d ", i);
}

变形:求最长上升子序列的数量:题目
歇一歇,晚点填坑!
2019-10-24 今天想起来还有一个坑没有填。
貌似写的太劝退了。先构思一下怎么改好一点
update:
看到题目数据范围,还是基操,离散化。 然后关机,开始自闭
实际上我们需要维护两个变量,所以我就写成了pair,第一个是以当前数结尾的LIS的长度,另一个就是数量,我们通过比较长度来更新数量,第一个更新函数

void upd(pii &a, pii b) {
  if(a.fi > b.fi) return ;
  if(a.fi < b.fi) a = b;
  else a.se = (a.se + b.se) % Mod;
}

这个函数应该很好理解,如果a的长度大于b,那么它对答案是没有贡献的,所以直接返回,如果小于的话那就是a是没用的,因为我们只求最长的数量,相等的时候就直接加起来。(使用引用,优化代码)
其实上面的函数就是核心了,数状数组就是在维护最大的长度。
树状数组更新与查询:

void upd(int x, pii y) {
  for(; x <= cnt; x += lowbit(x)) upd(T[x], y);
}
pii qry(int x) {
  pii ret(0,1);
  for(; x; x -= lowbit(x)) upd(ret, T[x]);
  return ret;
}

用了重载,各位看官别看迷了
是不是感觉特别简单呢?
放出完整代码:

#include
using namespace std;
typedef long long LL;
const int INF = 0x3f3f3f3f;
namespace {
  template <typename T> inline void read(T &x) {
    x = 0; T f = 1;char s = getchar();
    for(; !isdigit(s); s = getchar()) if(s == '-') f = -1;
    for(;  isdigit(s); s = getchar()) x = (x << 3) + (x << 1) + (s ^ 48);
    x *= f;
  }
}
#define fio ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
#define _for(n,m,i) for (int i = (n); i < (m); i++)
#define _rep(n,m,i) for (int i = (n); i <= (m); i++)
#define lson rt << 1, l, mid
#define rson rt << 1 | 1, mid + 1, r
#define lowbit(x) x & (-x)
#define pii pair
#define fi first
#define se second
const int N = 5e4+10;
const int Mod = 1e9+7; 
int a[N], n, cnt;
pii T[N], f[N];
vector<int>v;
inline int getid(int x) {
  return lower_bound(v.begin(), v.end(), x) - v.begin() + 1;
}
void upd(pii &a, pii b) {
  if(a.fi > b.fi) return ;
  if(a.fi < b.fi) a = b;
  else a.se = (a.se + b.se) % Mod;
}
void upd(int x, pii y) {
  for(; x <= cnt; x += lowbit(x)) upd(T[x], y);
}
pii qry(int x) {
  pii ret(0,1);
  for(; x; x -= lowbit(x)) upd(ret, T[x]);
  return ret;
}
int main() {
  read(n);
  _rep(1,n,i) read(a[i]), v.push_back(a[i]);
  sort(v.begin(), v.end());
  v.erase(unique(v.begin(), v.end()), v.end());
  cnt = v.size();
  pii ans(0, 1);
  _rep(1,n,i) {
    int id = getid(a[i]);
    f[i] = qry(id - 1);
    f[i].fi++;
    upd(ans, f[i]);
    upd(id, f[i]);
  }
  printf("%d\n", ans.se);
} 

你可能感兴趣的:(树状数组)