(2022牛客多校四)D-Jobs (Easy Version)(三维前缀或)

题目:

样例输入:

3 5
2 1 1 2 2 1 2
3 1 3 2 2 3 1 2 3 3
2 2 3 1 3 2 1
191415411

样例输出:

34417749

题意:有n个公司,第i个公司有mi个工作,每个工作对三个能力分别有数值要求,必须三个能力都达标才能胜任这份工作。一个人只要能胜任一个公司的任意一份工作那么他都能进入这个公司。

q次询问,每次询问一个三元组,代表一个人三个能力的数值,求这个人可以去多少个公司工作,强制在线。

对于每个公司我们不可能都处理一遍前缀和,因为10*400*400*400肯定会超时,由于我们只需要知道能否胜任某个公司的某个工作而不是全部的工作都要胜任,而且我们要存储多个公司的信息,所以我们只需要用二进制维护一下就行,对于第t个公司我们用第t位二进制位去维护该信息,这样的话就可以使得公司与公司之间互补干扰。然后直接O(1)查询即可。

下面是代码:
 

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<queue>
#include <random>
using namespace std;
const int N=401,mod=998244353;
int f[N][N][N],s[2000005];
int lastans=0,seed;
inline int read()
{
    int x=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
    return x*f;
}
int main()
{
	int n,m;
	n=read();m=read();
	for(int i=1;i<=n;i++)
	{
		int t;
		t=read();
		for(int j=1;j<=t;j++)
		{
			int a,b,c;
			a=read(),b=read(),c=read();
			f[a][b][c]|=(1<<i);
		}
	}
	for(int i=1;i<=400;i++)
	for(int j=1;j<=400;j++)
	for(int k=1;k<=400;k++)
		f[i][j][k]|=f[i][j][k-1]|f[i][j-1][k]|f[i-1][j][k];
	seed=read();
	std::uniform_int_distribution<> u(1,400);
	std::mt19937 rng(seed);
	s[0]=1;
	for(int i=1;i<=m;i++)
		s[i]=1ll*s[i-1]*seed%mod;
	long long anss = 0;
	for(int i=1;i<=m;i++)
	{
	    int IQ=(u(rng)^lastans)%400+1;  // The IQ of the i-th friend
	    int EQ=(u(rng)^lastans)%400+1;  // The EQ of the i-th friend
	    int AQ=(u(rng)^lastans)%400+1;  // The AQ of the i-th friend
	    lastans=0;
		for(int i=1;i<=n;i++)
	    lastans+=f[IQ][EQ][AQ]>>i&1;
		anss = (anss+1ll*lastans*s[m-i]%mod)%mod;
	}
	printf("%lld\n", anss);
	return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值