dfs

1 功夫传人 (25 分)

一门武功能否传承久远并被发扬光大,是要看缘分的。一般来说,师傅传授给徒弟的武功总要打个折扣,于是越往后传,弟子们的功夫就越弱…… 直到某一支的某一代突然出现一个天分特别高的弟子(或者是吃到了灵丹、挖到了特别的秘笈),会将功夫的威力一下子放大N倍 —— 我们称这种弟子为“得道者”。

这里我们来考察某一位祖师爷门下的徒子徒孙家谱:假设家谱中的每个人只有1位师傅(除了祖师爷没有师傅);每位师傅可以带很多徒弟;并且假设辈分严格有序,即祖师爷这门武功的每个第i代传人只能在第i-1代传人中拜1个师傅。我们假设已知祖师爷的功力值为Z,每向下传承一代,就会减弱r%,除非某一代弟子得道。现给出师门谱系关系,要求你算出所有得道者的功力总值。

输入格式:

输入在第一行给出3个正整数,分别是:N(≤10​5​​)——整个师门的总人数(于是每个人从0到N−1编号,祖师爷的编号为0);Z——祖师爷的功力值(不一定是整数,但起码是正数);r ——每传一代功夫所打的折扣百分比值(不超过100的正数)。接下来有N行,第i行(i=0,⋯,N−1)描述编号为i的人所传的徒弟,格式为:

K​i​​ ID[1] ID[2] ⋯ ID[K​i​​]

其中K​i​​是徒弟的个数,后面跟的是各位徒弟的编号,数字间以空格间隔。K​i​​为零表示这是一位得道者,这时后面跟的一个数字表示其武功被放大的倍数。

输出格式:

在一行中输出所有得道者的功力总值,只保留其整数部分。题目保证输入和正确的输出都不超过10​10​​。

输入样例:

10 18.0 1.00
3 2 3 5
1 9
1 4
1 7
0 7
2 6 1
1 8
0 9
0 4
0 3

输出样例:

404

结构体中潜逃数组时,如果数组太大,可用vector代替数组,节省空间,减少不必要的空间开辟

vector 加入元素时,用push_back();

注意:祖师爷可能会也没有徒弟

#include <iostream>
#include <vector>
using namespace std;


struct node
{
    int num;
    double b;
    vector<int>vec;
};
node name[100001];
double  sum=0;
double r;
void dfs(int x,double w)
{
    if(name[x].num==0) sum+=w*name[x].b;

    else
    {
        for(int i=0; i<name[x].num; i++)
       {
        int z=name[x].vec[i];
        dfs(z,w*r);
       }
    }

}
int main(int argc, char** argv)
{
    int n;
    double w;
    cin >>n>>w>>r;
    r=1.0-r/100;

    for(int i=0; i<n; i++)
    {
        cin>>name[i].num;
        if(name[i].num==0)
        {
            cin >>name[i].b;
        }
        else
        {
            int a;
            for(int j=0; j<name[i].num; j++)
            {
                cin >>a;
                name[i].vec.push_back(a);
            }
        }
    }
    dfs(0,w);
    cout <<(int)sum<<endl;
    return 0;
}

2 dfs 枚举思想   (得出结果的时间较长,适合填空)

在中国象棋中,马是走日字的。一个马的管辖范围指的是当前位置以及一步之内能走到的位置,下图的绿色旗子表示马能走到的位置。

如果一匹马的某个方向被蹩马脚,它就不能往这个方向跳了,如下图所示,海星的位置存在旗子,马就不能往上跳到那两个位置了:

那么问题来了,在一个 n×m 的棋盘内,如何用最少的马管辖住所有 n\times mn×m 个格子。比如 n=m=3 时,最少要用 5 只马才能管辖所有棋盘,一种可能的方案如下: 

当 n=m=5 时,请你求出用最少马管辖的 方案个数

 

#include <iostream>
#include <cstring>

using namespace std;
 
/* run this program using the console pauser or add your own getch, system("pause") or input loop */
int maps[6][6];
int fx[8][2]={{-2,-1},{-1,-2},{1,-2},{2,-1},{2,1},{1,2},{-1,2},{-2,1}};//马能跳向的方向 
int fx2[8][2]={{-1,0},{0,-1},{0,-1},{1,0},{1,0},{0,1},{0,1},{-1,0}};   //绊马腿上的情况 

int find(int n)
{
	memset(maps,0,sizeof(maps));
	for(int i=0;i<5;i++)         //输入每种情况 (二进制表示,0表示没有马,1表示有马) 
	 for(int j=0;j<5;j++)
	 {  
	 	if(n%2) maps[i][j]=1;
	 	else maps[i][j]=0;
	 	n=n>>1;
	 }

	 for(int i=0;i<5;i++)         //找到满足题目意思的情况 
	  for(int j=0;j<5;j++)
	  {  
	   
	  	if(maps[i][j]==1)      
	  	{ 
		   
	  	   	for(int k=0;k<8;k++)
	  	   	{
	  	   		int x=i+fx[k][0];
	  	   		int y=j+fx[k][1];
	  	   		if(x<0||y<0||y>=5||x>=5) continue;
				else 
				{  int xx=i+fx2[k][0];
				   int yy=j+fx2[k][1];
				   if(maps[x][y]==0&&maps[xx][yy]!=1)
				    {
				    	maps[x][y]=2;
					}	
				} 
			}
		}
	  }

	  
	  int sum=0;
	  for(int i=0;i<5;i++)
	    for(int j=0;j<5;j++)
	    {
	    	if(maps[i][j]==1)
	    	sum++;
	    	if(maps[i][j]==0)
	    	return 0;
		}
		//cout <<sum<<" ";
		return sum;
	  
}

int main(int argc, char** argv) 
{
    int DD[26]={0};
    int m=1<<25;
    cout <<m<<endl;
    for(int i=1;i<=m;i++)  //枚举所有情况 
    {
    	int k=find(i);
    	DD[k]++;
	}
	
	for(int i=1;i<26;i++)
	{
		if(DD[i])  
		{
		   cout <<DD[i]<<endl;
		   break; 
		}
		 
		
	} 
		
	return 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值