其他分享
首页 > 其他分享> > #734 (Div. 3) 1551. D1,2. Domino (思维,构造easy/hard version)

#734 (Div. 3) 1551. D1,2. Domino (思维,构造easy/hard version)

作者:互联网

LINK

题意

现在必须恰好放下 k k k个水平的 1 ∗ 2 1*2 1∗2的长方块,和任意 2 ∗ 1 2*1 2∗1的垂直长方块

是否能填满 n ∗ m n*m n∗m的网格?


现特判掉 k = 0 k=0 k=0的情况

考虑当 n n n为奇数,那么填充完 k k k个水平方块后,所有列必须是偶数

也就是每一列必须被填充奇数个格子

Ⅰ.当 m m m为奇数无解,因为每次只能改变 2 2 2列的奇偶性,这样怎么填充都有一列和其他奇偶性质不同

Ⅱ.否则,我们先把第一行填满,也就是花费 m / 2 m/2 m/2个长方体,这样每列都是偶数了

于是余下的长方体 k − m / 2 k-m/2 k−m/2必须是偶数,因为每列必须填充偶数次才行

这样构造即可

考虑当 n n n为偶数

也就是每一列必须填充偶数个格子

Ⅰ.这种情况下,先计算 x = n ∗ ( m / 2 ) x=n*(m/2) x=n∗(m/2),最多能填充 x x x个水平长方体,和 k k k比一下看是否满足

Ⅱ.满足的话,考虑到每列都是偶数,那么 k k k必须为偶数,否则最后会出现两列奇数

Ⅲ.都满足的话就开始构造,也就是一直给 1 , 2 1,2 1,2列放 2 2 2个水平长方体直到放不下
再给 3 , 4 3,4 3,4列放 2 2 2个长方体…以此类推

这里只贴以下 D 1 D1 D1怎么判断,具体怎么构造上面大概已经说了

不放代码的原因是赛中根本没想这么多,直接模拟玄学过去的

#include <bits/stdc++.h>
using namespace std;
const int maxn = 2009;
int t,n,m,k,a[maxn];
int main()
{
	cin >> t;
	while( t-- )
	{
		cin >> n >> m >> k;
		if( k==0 )
		{
			if( n&1 )	cout << "NO\n";
			else	cout << "YES\n";
		}
		else if( n&1 )//奇数行,每列都要放奇数个 
		{
			if( m%2==0 && m/2<=k && (k-m/2)%2==0 )	cout << "YES\n";
			else	cout << "NO\n";	
		}
		else//偶数行,所以每列需要放偶数个
		{
			int x = n*(m/2);
			if( x>=k && k%2==0 )	cout << "YES\n";
			else	cout << "NO\n";
		} 
	}
}

标签:1551,填充,奇数,int,Domino,hard,偶数,每列,长方体
来源: https://blog.csdn.net/jziwjxjd/article/details/119053449