偏序和全序是公里集合论中的概念。
首先你要知道什么是二元关系。
比如实数中的“大小”关系,集合的集合中的“包含”关系就是两种二元关系。
所谓偏序,即偏序关系,是一种二元关系。
所谓全序,即全序关系,自然也是一种二元关系。
全序是指,集合中的任两个元素之间都可以比较的关系。比如实数中的任两个数都可以比较大小,那么“大小”就是实数集的一个全序关系。
偏序是指,集合中只有部分元素之间可以比较的关系。比如复数集中并不是所有的数都可以比较大小,那么“大小”就是复数集的一个偏序关系。
显然,全序关系必是偏序关系。反之不成立。
题意:给定一个n的排列,给定询问区间[l,r],问在区间内有多少对倍数关系
思路:可以用前缀和的思想,处理出[1,x]位置的所有倍数关系。答案就是[1,r]的倍数关系数减去[1,l],同时还要减去其中一个点在l左边,一个点在l右边的情况
#include <iostream>
#include <algorithm>
#include <cstdio>
#include <queue>
#include <vector>
#include <set>
#include <map>
#include <unordered_map>
#include <cstring>
#include <string>
#include <cmath>
#define rep(i,a,b) for (int i=a; i<=b; ++i)
#define per(i,b,a) for (int i=b; i>=a; --i)
#define mes(a,b) memset(a,b,sizeof(a))
#define mp make_pair
#define ll long long
#define pb push_back
#define pii pair<int,int>
#define pll pair<ll,ll>
#define ls (rt<<1)
#define rs ((rt<<1)|1)
#define isZero(d) (abs(d) < 1e-8)
using namespace std;
const int maxn=1e5+5,INF=0x3f3f3f3f;
const int mod=1e9+7;
int n,m;
int C[maxn];
int pos[maxn];
vector<int> v[maxn];
int pref[maxn],ans[maxn];
struct query
{
int l,r,id;
bool operator<(const query & b) const
{
return l<b.l;
}
}q[maxn];
int lowbit(int x)
{
return x&(-x);
}
void add(int pos,int val)
{
for(int i=pos;i<=n;i+=lowbit(i))
C[i]+=val;
}
int getsum(int pos)
{
int ret=1;
for(int i=pos;i>0;i-=lowbit(i))
ret+=C[i];
return ret;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;++i)
{
int x;
scanf("%d",&x);
pos[x]=i;
}
for(int i=1;i<=m;++i)
scanf("%d%d",&q[i].l,&q[i].r),q[i].id=i;
sort(q+1,q+1+m);
for(int i=1;i<=n;++i)
{
for(int j=i+i;j<=n;j+=i)
{
int l=pos[i],r=pos[j];
if(l>r)
swap(l,r);
v[l].push_back(r);
pref[r]++;
}
}
for(int i=1;i<=n;++i)
pref[i]+=pref[i-1];
int now=1;
for(int i=1;i<=n;++i)
{
for(auto x : v[i-1])
add(x,1);
while(q[now].l==i&&now<=m)
{
ans[q[now].id]=getsum(q[now].r)-getsum(q[now].l-1);
now++;
}
}
for(int i=1;i<=m;++i)
ans[q[i].id]=pref[q[i].r]-pref[q[i].l-1]-ans[q[i].id];
for(int i=1;i<=m;++i)
printf("%d\n",ans[i]);
return 0;
}
题意:给定一个数组,给定一个区间,询问该区间不同的数字有多少个?
思路:用树状数组维护原数组,getsum(i)表示到[1,i]这个区间的不同数字的个数。对于 r 相同的询问来说,我们趋向于把值更新在右边
实现:记录每一个数字上一个出现的位置,因为在后面出现的时候,我们需要取消掉前面的,尽量更新到后面
可以参考大佬的博客https://www.luogu.org/blog/user3432/solution-p1972
#include <iostream>
#include <algorithm>
#include <cstdio>
#include <queue>
#include <vector>
#include <set>
#include <map>
#include <unordered_map>
#include <cstring>
#include <string>
#include <cmath>
#define rep(i,a,b) for (int i=a; i<=b; ++i)
#define per(i,b,a) for (int i=b; i>=a; --i)
#define mes(a,b) memset(a,b,sizeof(a))
#define mp make_pair
#define ll long long
#define pb push_back
#define pii pair<int,int>
#define pll pair<ll,ll>
#define ls (rt<<1)
#define rs ((rt<<1)|1)
#define isZero(d) (abs(d) < 1e-8)
using namespace std;
const int maxn=1e6+5,INF=0x3f3f3f3f;
const int mod=1e9+7;
int n,m,C[maxn],a[maxn];
int last[maxn],ans[maxn];
struct query
{
int l,r,id;
bool operator<(const query& b) const
{
return r<b.r;
}
}q[maxn];
int lowbit(int x)
{
return x&(-x);
}
void add(int pos,int val)
{
for(int i=pos;i<=n;i+=lowbit(i))
C[i]+=val;
}
int getsum(int pos)
{
int ret=1;
for(int i=pos;i>0;i-=lowbit(i))
ret+=C[i];
return ret;
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;++i)
scanf("%d",&a[i]);
scanf("%d",&m);
for(int i=1;i<=m;++i)
scanf("%d%d",&q[i].l,&q[i].r),q[i].id=i;
sort(q+1,q+1+m);
int now=1;
for(int i=1;i<=m;++i)
{
for(int j=now;j<=q[i].r;++j)
{
if(last[a[j]])
add(last[a[j]],-1);
add(j,1);
last[a[j]]=j;
}
ans[q[i].id]=getsum(q[i].r)-getsum(q[i].l-1);
now=q[i].r+1;
}
for(int i=1;i<=m;++i)
printf("%d\n",ans[i]);
return 0;
}