bitset小专题(待续)

bitset:一个01位如果用bool存的话需要 1byte,而用bitset只需要 1bit(=1/8 byte)
每次两个集合取并的时候可以除以一个大常数(32/64),从而优化复杂度

LOJ515
\(dp[i]\) 表示考虑到第 \(i\) 个区间,能形成的和的状态是0/1(该位为1代表可以出现这个数,否则不能)
转移就枚举一下当前区间 or 一下即可
注意这里不能用 1<<i 这种写法,因为会爆int

// by SkyRainWind
#include <bits/stdc++.h>
#define mpr make_pair
#define debug() cerr<<"Yoshino\n"
#define pii pair<int,int>
#define pb push_back

using namespace std;

typedef long long ll;
typedef long long LL;

const int inf = 1e9, INF = 0x3f3f3f3f, maxn = 105;

int n;
int a[maxn], b[maxn];
bitset<101 * 101 * 101>dp[2];

signed main() {
    ios::sync_with_stdio(false);
    cin >> n;

    for (int i = 1; i <= n; i++)
        cin >> a[i] >> b[i];

    for (int i = 1; i <= n; i++) {
        dp[i & 1].reset();

        for (int j = a[i]; j <= b[i]; j++) {
            dp[i & 1] |= (dp[i & 1 ^ 1] << (j * j));
        }

        if (i == 1)
            for (int j = a[i]; j <= b[i]; j++)
                dp[i & 1][j * j] = 1;
    }

    cout << dp[n & 1].count();

    return 0;
}

abc287_ex
帮助理解floyd本质的好题
floyd的代码一般是这样:

for(int k=1;k<=n;k++){
  for(int i=1;i<=n;i++){
    for(int j=1;j<=n;j++){
      dis[i][j] = min(dis[i][j], dis[i][k] + dis[k][j]);
    }   
  }
}

这里 \(k\) 的深层含义就是:\(i\rightarrow j\) 的路径上,(去除了起点终点)路径上的点的编号均不超过 \(k\) 时的最短距离
每次 \(k+1\) 的时候,就相当于 \((i,j)\) 的路径上可以出现 \(k+1\) 这个点,因此用 \(dis[i][k']+dis[k'][j]\) 更新之
这个题实际上就考察了floyd的本质。每次枚举一下路径上的最大点 \(k\),然后传递闭包即可
又因为我们只需要记录是否(路径上只利用 \(1..k\) 的点)可达即可,因此可以用bitset优化一下,时间复杂度 \(O(\frac{n^3+nQ}{32})\)

// by SkyRainWind
#include <bits/stdc++.h>
#define mpr make_pair
#define debug() cerr<<"Yoshino\n"
#define pii pair<int,int>
#define pb push_back

using namespace std;

typedef long long ll;
typedef long long LL;

const int inf = 1e9, INF = 0x3f3f3f3f;

int n,m;
bitset<2005>acc[2005];
int s[10005], t[10005], ans[10005];
signed main(){
	memset(ans,-1,sizeof ans);
	scanf("%d%d",&n,&m);
	for(int i=1;i<=m;i++){
		int x,y;scanf("%d%d",&x,&y);
		acc[x][y] = 1;
	}
	int Q;
	scanf("%d",&Q);
	for(int i=1;i<=Q;i++){
		scanf("%d%d",&s[i],&t[i]);
	}
	for(int k=1;k<=n;k++){
		acc[k][k] = 1;
		for(int i=1;i<=n;i++){
			if(acc[i][k])acc[i] |= acc[k];
		}
		for(int i=1;i<=Q;i++){
			int u=s[i], v=t[i];
			if(u<=k && v<=k && acc[u][v] && ans[i]==-1){
				ans[i] = k;
			}
		}
	}
	for(int i=1;i<=Q;i++)printf("%d\n",ans[i]);
	
	return 0;
}
posted @ 2023-02-01 22:07  SkyRainWind  阅读(15)  评论(0编辑  收藏  举报