PAT Mooc datastructure 6-1

Saving James Bond - Hard Version

This time let us consider the situation in the movie "Live and Let Die" in which James Bond, the world's most famous spy, was captured by a group of drug dealers. He was sent to a small piece of land at the center of a lake filled with crocodiles. There he performed the most daring action to escape -- he jumped onto the head of the nearest crocodile! Before the animal realized what was happening, James jumped again onto the next big head... Finally he reached the bank before the last crocodile could bite him (actually the stunt man was caught by the big mouth and barely escaped with his extra thick boot).

Assume that the lake is a 100 by 100 square one. Assume that the center of the lake is at (0,0) and the northeast corner at (50,50). The central island is a disk centered at (0,0) with the diameter of 15. A number of crocodiles are in the lake at various positions. Given the coordinates of each crocodile and the distance that James could jump, you must tell him a shortest path to reach one of the banks. The length of a path is the number of jumps that James has to make.

Input Specification:

Each input file contains one test case. Each case starts with a line containing two positive integers N (<=100), the number of crocodiles, and D, the maximum distance that James could jump. Then N lines follow, each containing the (x, y) location of a crocodile. Note that no two crocodiles are staying at the same position.

Output Specification:

For each test case, if James can escape, output in one line the minimum number of jumps he must make. Then starting from the next line, output the position (x, y) of each crocodile on the path, each pair in one line, from the island to the bank. If it is impossible for James to escape that way, simply give him 0 as the number of jumps. If there are many shortest paths, just output the one with the minimum first jump, which is guaranteed to be unique.

Sample Input 1:

17 15

10 -21

10 21

-40 10

30 -50

20 40

35 10

0 -10

-25 22

40 -40

-30 30

-10 22

0 11

25 21

25 10

10 10

10 35

-30 10

Sample Output 1:

4

0 11

10 21

10 35
 
 

1.题目分析

    这个题目就是典型的最短路径问题~

    使用BFS对图进行层级遍历,只要在任一层发现可以跳出的话,那么就逆序打印出来全体路径。

    至于这个层次的问题,其实很好实现,因为在BFS入队的过程中,是将一个节点的链接节点依此入队。在入队的时候,只要将此节点的步点数加1写入下一层中即可完成层数的累积。

    同时,为了完成最后的打印工作,需要使用一个数组,完成类似于链表的工作。每次压入新的节点时,就要将新节点对应的父节点数组值设置为其父节点的index,这样在推出时逆序压入堆栈,再重新打印出来,即可完成逆序打印工作。

   有两个小点值得注意,一个是如果007非常牛逼,直接就可以跳出来的情况需要特殊处理一下。

   二是按照题目要求,如果存在相同步数的路径,需要选出第一跳最短的那一条路径。为了实现这一功能,我在将鳄鱼节点录入时,先按照距离原点的距离升序排序。这样,在进行BFS的时候,总是先从短距离向长距离遍历,保证了第一跳最短距离的哪条路径最先被发现。

 

2.伪码实现

    P = 007StartPoint;

    if (Could Get Out at the StartPoint)

       return FirstGetOut;

    EnQueue(P,Queue)

    while(IsNotEmpty(Queue))

   {

          P = DeQueue(Queue);

           Find All the Point NextP that jump from P

          {

                if(NextP is not reached)

                {

                       Jumped(NextP) = Jumped(P) + 1

                       father(NextP) =   P

                       if(CouldGetOutFrom(NextP))

                             return NextP;

                       EnQueue(NextP, Queue)

                 }

           }

   }

   return CouldNotJumpOut

  3. 通过代码:

  1 #define NONE -1

  2 #define DISK 100000

  3 #define MISTAKE -11

  4 #define FIOUT 5000000

  5 

  6 #include <stdio.h>

  7 #include <stdlib.h>

  8 

  9 

 10 

 11 static int jumped[101][101];

 12 

 13 void InitialJumped()

 14 {

 15     int i;

 16     int j;

 17     for(i=0;i<=100;i++)

 18     {

 19         for(j=0;j<=100;j++)

 20         {

 21             jumped[i][j]=NONE;

 22         }

 23     }

 24 }

 25 

 26 int IsJumped(int x,int y)

 27 {

 28     if(jumped[x+50][y+50] != NONE)

 29     {

 30         return 1;

 31     }

 32     else

 33     {

 34         return NONE;

 35     }

 36 }

 37 

 38 void JumpOn(int x,int y,int jumpNum)

 39 {

 40     jumped[x+50][y+50]= jumpNum;

 41 }

 42 

 43 int GetJump(int x,int y)

 44 {

 45     return jumped[x+50][y+50];

 46 }

 47 

 48 /////////End of Visited /////////////

 49 

 50 //////////Begin of Croc//////////////

 51 

 52 typedef struct Croc{

 53     //location of this Croc

 54     int x;

 55     int y;

 56     int dis;

 57 }tCroc;

 58 

 59 

 60 void SetDis(tCroc* C)

 61 {

 62     C->dis = (C->x)*(C->x) + (C->y)*(C->y);

 63 }

 64 

 65 int GetDis(tCroc* C)

 66 {

 67     return C->dis;

 68 }

 69 

 70 //If bond could reach from ori to des.

 71 int Reachable(tCroc* ori,tCroc* des,int step)

 72 {

 73     int oriX;

 74     int oriY;

 75     int desX;

 76     int desY;

 77     int distanceSquare;

 78     int stepSquare;

 79     

 80     oriX = ori->x;

 81     oriY = ori->y;

 82     

 83     desX = des->x;

 84     desY = des->y;

 85     

 86     distanceSquare = (oriX-desX)*(oriX-desX)+(oriY-desY)*(oriY-desY);

 87     stepSquare = step*step;

 88     

 89     if(stepSquare >= distanceSquare)

 90     {

 91         return 1;

 92     }

 93     else

 94     {

 95         return 0;

 96     }

 97 }

 98 

 99 

100 int GetOut(tCroc *ori,int step)

101 {

102     if(ori->x + step >= 50)

103     {

104         return 1;

105     }

106     if(ori->x - step <= -50)

107     {

108         return 1;

109     }

110     if(ori->y + step >= 50)

111     {

112         return 1;

113     }

114     if(ori->y - step <= -50)

115     {

116         return 1;

117     }

118     return 0;

119 }

120 

121 //////////End of Croc////////////////

122 

123 /////////DFS of Croc/////////////////

124 

125 int DFSofCroc(tCroc *ori,tCroc list[],int numOfCroc,int step)

126 {

127     int i;

128     int localStep;

129     

130     JumpOn(ori->x,ori->y,1);

131     

132     if((ori->x == 0)&& (ori->y == 0))

133     {

134         localStep = step+8;

135     }

136     else

137     {

138         localStep = step;

139     }

140     

141     

142     if(GetOut(ori,localStep)==1)

143     {

144         return 1;

145     }

146     

147     for(i = 0;i<numOfCroc;i++)

148     {

149         if(Reachable(ori,&list[i],localStep)==1)

150         {

151             if(IsJumped(list[i].x,list[i].y)==NONE)

152             {

153                 int result;

154                 result =  DFSofCroc(&list[i],list,numOfCroc,step);

155                 if(result == 1)

156                 {

157                     return 1;

158                 }

159             }

160         }

161     }

162     return 0;

163 }

164 //////////////////End of DFS Croc without COUNT/////////////////

165 

166 /////////////////Begin of queue//////////////

167 

168 typedef struct queueNode{

169     tCroc * thisCroc;

170     struct queueNode * nextCroc;

171 }QNode;

172 

173 typedef struct CrocQueue{

174     QNode *head;

175     QNode *tail;

176 }tCrocQueue;

177 

178 tCrocQueue* InitialQueue()

179 {

180     tCrocQueue* temp = malloc(sizeof(tCrocQueue));

181     temp->head = NULL;

182     temp->tail = NULL;

183     return temp;

184 }

185 

186 void EnQueue(QNode *node,tCrocQueue *Q)

187 {

188     if(Q->head == NULL)

189     {

190         Q->head = node;

191         Q->tail = node;

192         return ;

193     }

194     else

195     {

196         Q->tail->nextCroc = node;

197         Q->tail = node;

198         return;

199     }

200 }

201 

202 QNode * DeQueue(tCrocQueue *Q)

203 {

204     if(Q->head == NULL)

205     {

206         return NULL;

207     }

208     else

209     {

210         QNode *temp = Q->head;

211         if(Q->head == Q->tail)

212         {

213             Q->head = NULL;

214             Q->tail = NULL;

215         }

216         else

217         {

218             Q->head = Q->head->nextCroc;

219         }

220         return temp;

221     }

222 }

223 

224 int IsQueueEmpty(tCrocQueue *Q)

225 {

226     if(Q->head == NULL)

227     {

228         return 1;

229     }

230     else

231     {

232         return 0;

233     }

234 }

235 ////////////////End of queue /////////////////

236 

237 ////////////////Begin of BFS/////////////////

238 

239 int BFS(tCroc *ori, tCroc list[], int Sum, int step, tCrocQueue *Q, int father[])

240 {

241     int count;

242     int myfather;

243     myfather = DISK;

244     count = 0;

245     JumpOn(ori->x,ori->y,count);

246     if(GetOut(ori,step+8)==1)

247     {

248         return FIOUT;

249     }

250     QNode * thisNode = malloc(sizeof(QNode));

251     thisNode->thisCroc = ori;

252     EnQueue(thisNode,Q);

253     while(IsQueueEmpty(Q)==0)

254     {

255         int i ;

256         int localStep;

257         QNode* temp = DeQueue(Q);

258         

259         count = GetJump(temp->thisCroc->x,temp->thisCroc->y);

260 

261         if((temp->thisCroc->x == 0)&&(temp->thisCroc->y==0))

262         {

263             localStep = step + 8;

264             myfather = DISK;

265         }

266         else

267         {

268             localStep = step;

269             for(i = 0; i < Sum ; i++)

270             {

271                 if((temp->thisCroc->x == list[i].x)&&(temp->thisCroc->y== list[i].y))

272                 {

273                     myfather = i;

274                     break;

275                 }

276             }

277         }

278         

279         

280         for(i = 0; i < Sum ;i++)

281         {

282             if(Reachable(temp->thisCroc,&list[i],localStep)==1)

283             {

284                 if(IsJumped(list[i].x,list[i].y)==NONE)

285                 {

286                     JumpOn(list[i].x,list[i].y,(count+1));

287                     father[i] = myfather;

288                     

289                     if(GetOut(&list[i],step)==1)

290                     {

291                         return i;

292                     }

293                     thisNode = malloc(sizeof(QNode));

294                     thisNode->thisCroc = &list[i];

295                     EnQueue(thisNode,Q);

296                 }

297             }

298         }

299     }

300     

301     return NONE;

302 }

303 

304 

305 

306 

307 

308 ////////////////End of BFS///////////////////

309 ////////////////Begin Stack//////////////////

310 typedef struct stackNode{

311     tCroc *thisNode;

312     struct stackNode * next;

313 }tStackNode;

314 

315 typedef struct CrocStack{

316     tStackNode* Top;

317     tStackNode* Bottom;

318 }tCrocStack;

319 

320 tCrocStack* InitialStack()

321 {

322     tCrocStack * temp = malloc(sizeof(tCrocStack));

323     temp->Top = NULL;

324     temp->Bottom = NULL;

325     return temp;

326 }

327 

328 void Push(tStackNode *node,tCrocStack * S)

329 {

330     if(S->Top == NULL)

331     {

332         S->Top = node;

333         S->Bottom = node;

334     }

335     else

336     {

337         node->next = S->Top;

338         S->Top = node;

339     }

340 }

341 

342 tStackNode *Pop(tCrocStack *S)

343 {

344     if(S->Top == NULL)

345     {

346         return NULL;

347     }

348     else

349     {

350         tStackNode * temp = S->Top;

351         if(S->Top == S->Bottom)

352         {

353             S->Bottom = NULL;

354             S->Top = NULL;

355         }

356         else

357         {

358             S->Top = S->Top->next;

359         }

360         return temp;

361     }

362 }

363 

364 int IsStackEmpty(tCrocStack *S)

365 {

366     if(S->Top == NULL)

367     {

368         return 1;

369     }

370     else

371     {

372         return 0;

373     }

374 }

375 /////////////////End of Stack////////////////

376 

377 

378 

379 

380 

381 int main()

382 {

383     int numOfCrocs;

384     int step;

385     int i;

386     

387     InitialJumped();

388     scanf("%d %d",&numOfCrocs,&step);

389     

390     tCroc crocs[numOfCrocs];

391     int preCro[numOfCrocs];

392     

393     

394     //Put all the cordinates X,Y into array

395     for(i=0;i<numOfCrocs;i++)

396     {

397         int j;

398         int tempX;

399         int tempY;

400         scanf("%d %d",&tempX,&tempY);

401         if( tempX > 50 || tempX< -50 || tempY > 50 || tempY < -50 )

402         {

403             i--;

404             numOfCrocs--;

405             continue;

406         }

407         crocs[i].x = tempX;

408         crocs[i].y = tempY;

409         SetDis(&crocs[i]);

410         for(j=i-1;j>=0;j--)

411         {

412             if(crocs[j].x == crocs[i].x && crocs[j].y == crocs[i].y)

413             {

414                 i--;

415                 numOfCrocs--;

416                 j=MISTAKE;

417                 break;

418             }

419         }

420         if(j==MISTAKE)

421         {

422             continue;

423         }

424         

425         for(j=i;j>0;j--)

426         {

427             if(GetDis(&crocs[j])<GetDis(&crocs[j-1]))

428             {

429                 tCroc temp = crocs[j-1];

430                 crocs[j-1]=crocs[j];

431                 crocs[j]=temp;

432             }

433             else

434             {

435                 break;

436             }

437         }

438         preCro[i] = NONE;

439     }

440     

441 //    printf("---------------\n");

442 //    for(i=0;i<numOfCrocs;i++)

443 //    {

444 //        printf("%d %d %d\n",crocs[i].x,crocs[i].y,crocs[i].dis);

445 //    }

446 //    printf("---------------\n");

447         

448     tCroc *Zero = malloc(sizeof(tCroc));

449     Zero->x = 0;

450     Zero->y = 0;

451     

452     tCrocQueue * MyQueue = InitialQueue();

453     

454     int result;

455     

456     result = BFS(Zero,crocs,numOfCrocs,step,MyQueue,preCro);

457     

458     if(result == NONE)

459     {

460         printf("0");

461         return 0;

462     }

463     else if(result == FIOUT)

464     {

465         printf("1");

466     }

467     else

468     {

469         int Dis = GetJump(crocs[result].x,crocs[result].y);

470         printf("%d\n",Dis+1);

471     }

472     

473     tCrocStack * MyStack = InitialStack();

474     while(result != DISK)

475     {

476         tStackNode *temp = malloc(sizeof(tStackNode));

477         temp->thisNode = &crocs[result];

478         Push(temp,MyStack);

479         result = preCro[result];

480     }

481     

482     while(IsStackEmpty(MyStack)==0)

483     {

484         int printX;

485         int printY;

486         tStackNode *temp = Pop(MyStack);

487         

488         printX = temp->thisNode->x;

489         printY = temp->thisNode->y;

490         printf("%d %d\n",printX,printY);

491     }

492     

493 //    result = DFSofCroc(Zero,crocs,numOfCrocs,step);

494     

495 //    if(result == 1)

496 //    {

497 //        printf("Yes");

498 //    }

499 //    else

500 //    {

501 //        printf("No");

502 //    }

503     

504     return 0;

505 }

 

你可能感兴趣的:(struct)