CF-192-diy-2 (三)

2014-11-23 22:30:47 ? 作者: ? 浏览: 23
ol[i][0],i);
continue;
}

}

return 0;
}


#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define eps 1e-6
#define INF 0x1f1f1f1f
#define PI acos(-1.0)
#define ll __int64
#define lson l,m,(rt<<1)
#define rson m+1,r,(rt<<1)|1
using namespace std;

/*
freopen("data.in","r",stdin);
freopen("data.out","w",stdout);
*/

bool rr[120],cc[120];
char save[120][120];
vectorrow[120];
vectorcol[120];

int main()
{
int n;

while(scanf("%d",&n)!=EOF)
{
memset(rr,false,sizeof(rr));
memset(cc,false,sizeof(cc));
for(int i=1;i<=n;i++)
{
row[i].clear();
col[i].clear();
}
for(int i=1;i<=n;i++)
{
scanf("%s",save[i]+1);
for(int j=1;j<=n;j++)
{
if(save[i][j]=='.')
{
rr[i]=true,cc[j]=true; //当前行和列能扫描到
row[i].push_back(j);
col[j].push_back(i);
}
}
}
bool r=true,c=true; //是否所有的行或列能扫描到

for(int i=1;i<=n;i++)
if(!rr[i])
{
r=false;
break;
}
for(int i=1;i<=n;i++)
if(!cc[i])
{
c=false;
break;
}
if(!r&&!c) //行或列都扫描不到
{
printf("-1\n");
continue;
}
if(r) //如果行都能扫描到,直接输出每一行的第一个可以放的位置
{
for(int i=1;i<=n;i++)
printf("%d %d\n",i,row[i][0]);
continue;
}
if(c) //对于列 同理
{
for(int i=1;i<=n;i++)
printf("%d %d\n",col[i][0],i);
continue;
}

}

return 0;
}

D. Biridian Forest


题目意思:

给一个n*m的矩阵,一个起始点和终点,矩阵中的数字表示该位置的魔鬼数量,当从起点到终点选择一条路径时,如果魔鬼能够在人之前或同时到达该路径某一位置,则人和魔鬼要打仗,选择一条路径,使得人与魔鬼打仗的个数最少。

解题思路:

该题主要是思维转换。bfs+贪心

假设人当前选择了一条路径C->D,终点为D,某处魔鬼B 能够和人C打仗,假设魔鬼在A位置与人打仗,则dis(B,A)<=dis(C,A) =>dis(B,A,D)<=dis(C,D) 如果选择D终点,只需使得Min(dis(B,D))<=dis(C,D) 而dis(B,A,D)>=Min(dis(B,D)) 所以对于某一特定路径,选择终点D打仗比其他任何一点都好。

如果选择从C-D的最短路,则能够打仗的魔鬼数量就最少。

代码:


[cpp]
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define eps 1e-6
#define INF 0x1f1f1f1f
#define PI acos(-1.0)
#define ll __int64
#define lson l,m,(rt<<1)
#define rson m+1,r,(rt<<1)|1
using namespace std;

/*
freopen("data.in","r",stdin);
freopen("data.out","w",stdout);
*/

int dp[1100][1100],n,m;
char save[1100][1100];
bool vis[1100][1100];
int dir[4][2]={{-1,0},{0,1},{1,0},{0,-1}};

struct PP
{
int x,y;
};
PP s,e;

void bfs() //找到终点到任何一点的最短距离
{
queuemyq;
myq.push(e);

while(!myq.empty())
{
PP f=myq.front();
myq.pop();

for(int i=0;i<4;i++)
{
int xx=f.x+dir[i][0],yy=f.y+dir[i][1];

if((save[xx][yy]=='S'||(save[xx][yy]>='0'&&save[xx][yy]<='9'))&&!vis[xx][yy])
{
//printf(":%d %d\n",xx,yy);
vis[xx][yy]=true;
dp[xx][yy]=dp[f.x][f.y]+1;
myq.push((PP){xx,yy});
} //不在方格里的点为空的,不会扫描的
}
}
return ;
}

int main()
{
while(scanf("%d%d",&n,&m)!=EOF)
{
memset(vis,false,sizeof(vis));
for(int i=1;i<=n;i++)
{
scanf("%s",save[i]+1);
for(int j=1;j<=m;j++)
{
if(save[i][j]=='S')
s.x=i,s.y=j;
if(save[i][j]=='E')
e.x=i,e.y=j,dp[i][j]=0,vis[i][j]=true;
}
}
// printf

-->

评论

帐  号: 密码: (新用户注册)
验 证 码:
表  情:
内  容: