HDU 5995 Kblack loves flag

问题描述
kblack喜欢旗帜(flag),他的口袋里有无穷无尽的旗帜。

某天,kblack得到了一个n*mnm的方格棋盘,他决定把kk面旗帜插到棋盘上。

每面旗帜的位置都由一个整数对\left(x,y \right)(x,y)来描述,表示该旗帜被插在了第xx行第yy列。

插完旗帜后,kblack突然对那些没有插过旗帜的行和列很不满,于是他想知道,有多少行、列上所有格子都没有被插过旗帜。

kblack还要把妹,于是就把这个问题丢给了你,请你帮他解决。
输入描述
由于本题输入数据较大,所以采取在程序内生成数据的方式。

随机数产生器中有个内部变量xx初始时为seedseedseedseed是我们提供的随机种子。每次请求生成一个\left[l,r \right][l,r]内的随机数时,它会将xx变为\left(50268147x+6082187\right)\ mod\ 100000007(50268147x+6082187) mod 100000007,然后返回x\ mod\ \left(r-l+1 \right)+lx mod (rl+1)+l。

输入包含多组数据。第一行有一个整数TT,表示测试数据的组数,对于每组数据:

输入一行3个整数nnmmkkseedseed分别表示棋盘的行数、列数、棋盘上旗帜的面数、随机种子。

接下来,你需要按顺序生成k面旗帜的位置信息。

对于每面旗帜,依次生成一个\left[1,n \right][1,n]内的随机数和一个\left[1,m \right][1,m]内的随机数,分别表示xxyy。

如果你无法理解数据生成的过程,你可以复制以下代码并调用Init函数来生成数据(限C++选手)。

`
const int _K=50268147,_B=6082187,_P=100000007;
int _X;
inline int get_rand(int _l,int _r){
	_X=((long long)_K*_X+_B)%_P;
	return _X%(_r-_l+1)+_l;
}
int n,m,k,seed;
int x[1000006],y[1000006];
void Init(){
	scanf("%d%d%d%d",&n,&m,&k,&seed);
	_X=seed;
	for (int i=1;i<=k;++i)
		x[i]=get_rand(1,n),
		y[i]=get_rand(1,m);
}
`

\left(1\leq T\leq 7 \right)(1T7)\left(1\leq n,m\leq 1000000 \right)(1n,m1000000)\left(0\leq k\leq 1000000 \right)(0k1000000)\left(0\leq seed<100000007 \right)(0seed<100000007)
输出描述
对于每组测试数据输出一行2个整数,分别表示没有被插过旗帜的行、列数目。
输入样例
2
4 2 3 233
3 4 4 2333
输出样例
2 1
1 0
Hint
第1组数据的旗帜的位置依次为:\left(4,2\right)(4,2),\left(1,2\right)(1,2),\left(1,2\right)(1,2)
简单题,搞两个数组弄一下就好了
#include<cmath>  
#include<queue>  
#include<cstdio>  
#include<cstring>  
#include<algorithm>  
using namespace std;
#define ms(x,y) memset(x,y,sizeof(x))  
#define rep(i,j,k) for(int i=j;i<=k;i++)  
#define loop(i,j,k) for (int i=j;i!=-1;i=k[i])  
#define inone(x) scanf("%d",&x)  
#define intwo(x,y) scanf("%d%d",&x,&y)  
#define inthr(x,y,z) scanf("%d%d%d",&x,&y,&z)  
#define lson x<<1,l,mid  
#define rson x<<1|1,mid+1,r  
typedef long long LL;
const int mod = 1e9 + 7;
const int N = 1e6 + 10;
int T;

const int _K = 50268147, _B = 6082187, _P = 100000007;
int _X;
inline int get_rand(int _l, int _r) {
    _X = ((long long)_K*_X + _B) % _P;
    return _X % (_r - _l + 1) + _l;
}
int n, m, k, seed;
int x[1000006], y[1000006];
void Init() {
    scanf("%d%d%d%d", &n, &m, &k, &seed);
    _X = seed;
    for (int i = 1; i <= k; ++i)
        x[i] = get_rand(1, n),
        y[i] = get_rand(1, m);
}

int f[N], g[N];

int main()
{
    inone(T);
    while (T--)
    {
        Init();
        ms(f, 0);
        ms(g, 0);
        for (int i = 1; i <= k; ++i)
        {
            if (!f[x[i]]++) n--;
            if (!g[y[i]]++) m--;
        }
        printf("%d %d\n", n, m);
    }
    return 0;
}


\left(3,2\right)(3,2)
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值