挖坑
------------------------
一 构造矩阵
SGU 409 Berland Flag
构造在一个边长为n*n(也就是说面积为n*n*n*n的)正方形。正方形中只包含' * '和' . '。
满足:
使得将这个正方形分隔成n*n个大小为n*n的正方形之后,每个小正方形里都刚好有k个 ' * '
每行每列都刚好只有k个' * '。
[solution]
根据样例推理构造。
#include <cstdio>
#include <cstring>
using namespace std;
const int maxn=1005;
int n,m;
bool a[maxn][maxn];
int x,y;
int main()
{
memset(a,0,sizeof(a));
scanf("%d%d",&n,&m);
for(int i=0;i<n;i++){
for(int j=0;j<n;j++){
x=i*n+j;
y=j*n+i;
for(int k=0;k<m;k++) a[x][(y+k)%(n*n)]=true;
}
}
for(int i=0;i<n*n;i++){
for(int j=0; j<n*n; ++j){
if(!a[i][j]) printf(".");
else printf("*");
}
printf("\n");
}
return 0;
}
------------------------
二 构造棋盘
SGU 361 National Flag
要求将n*m的矩阵染上蓝色
要求,任意一个2*3或3*2的矩阵里都恰好有两个蓝色,并且蓝色数量最少。
#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;
const int maxn=1005;
const int INF=0x3f3f3f3f;
bool a[maxn][maxn];
int n,m;
int main()
{
int Min=INF;
int p=0;
int sum=0;
cin>>n>>m;
for (int k=0;k<3;k++){
sum=0;
for (int i=0;i<n;i++){
for (int j=(i+k)%3;j<m;j+=3){
sum++;
}
}
if (sum<Min){
Min=sum;
p=k;
}
}
memset(a,0,sizeof(a));
for (int i=0;i<n;i++){
for (int j=(i+p)%3;j<m;j+=3){
a[i][j]=true;
}
}
for (int i=0;i<n;i++){
for (int j=0;j<m;j++){
if (a[i][j]) cout<<'#';
else cout<<'0';
}
cout<<endl;
}
return 0;
}
------------------------
三 电梯搬运
SGU 379 Elevator
一个n层高的大楼只有一个电梯。停车场坐落在一楼之下。楼层编号1~n。
第i层上有Ai个人想去停车场。电梯载人数不能超过C。电梯移动一层用时P分钟。
找到在T分钟内最多有多少人能到达停车场。电梯装载人不需要时间。
二分最大人数。
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long LL;
const int maxn=111;
int n,C,P,T;
int a[maxn];
int b[maxn];
bool Can(int ans){
LL cur=0,toPick;
int p=0;
for (int i=1;i<=n;i++){
cur+=a[i];
if (cur>=ans){
p=i;
toPick=ans-(cur-a[i]);
break;
}
}
LL ret=((toPick-1)/C+1)*p*2;
int capa=C-(toPick-1)%C-1;
for (int i=p-1;i>=1;i--){
if (a[i]<=capa){
capa-=a[i];
}
else{
toPick=a[i]-capa;
ret+=((toPick-1)/C+1)*i*2;
capa=C-(toPick-1)%C-1;
}
}
return ret*P<=T;
}
int main()
{
while (~scanf("%d%d%d%d",&n,&C,&P,&T)){
LL l=0,r=0;
for (int i=1;i<=n;i++){
scanf("%d",&a[i]);
r+=a[i];
}
LL ans=0;
while (l<=r){
LL mid=(l+r)>>1;
if (Can(mid)){
ans=mid;
l=mid+1;
}
else{
r=mid-1;
}
}
printf("%I64d\n",ans);
}
return 0;
}
------------------------
------------------------
------------------------