zoukankan      html  css  js  c++  java
  • 【题解】长度为素数的路径个数-C++

    Description
    对于正整数n (3≤n<20),可以画出n阶的回形矩阵。下面画出的分别是3阶的,4阶的和7阶的回形矩阵:
    在这里插入图片描述
    对于n阶回形矩阵,从左上角出发,每步可以向右或向下走一格,走2* n-2步,可以到达右下角。我们把这样的路
    径上所有格子中的数值之和,叫做该路径的长度。本题要求,对于给出n值,求出n阶回形矩阵有多少路径的长度为
    素数?如n=3时,路径及长度有:
    在这里插入图片描述
    因此说,3阶回形矩阵有2条路径的长度为素数。
    Input
    一个自然数n (3≤n<20,不必判错)。
    Output
    一个正整数,即n阶回形矩阵中长度为素数的路径的个数。
    Sample Input
    3
    Sample Output
    2

    这道题目第一个难点在于构造回形矩阵。
    说是回形矩阵,我们可以想象成一个nn的矩阵叠加(n-1)(n-1)的矩阵…然后就可以叠加成为一个回形矩阵,但是需要判断n的奇偶性。。。build函数如下:

    void build(int s1,int n1)
    {
    	if(s1==n1)
    	{
    		mp[s1][n1]=s1;
    		return;
    	}
    	else if(s1>n1)return;
    	for(int i=s1;i<=n1;i++)
    	{
    		for(int j=s1;j<=n1;j++)
    		{
    			mp[i][j]=s1;
    		}
    	}
    	build(s1+1,n1-1);
    	return;
    }
    

    顺便写出判断路径长度是否是质数的函数。。。

    bool IsPrime/*这绝逼是我自己打的*/(int num)
    {
    	 if(num==1)
    		return 0;
         if(num==2||num==3)
            return 1;
         if(num%6!=1&&num%6!=5)
            return 0;
         int tmp=sqrt(num);
         for(int i=5;i<=tmp;i+=6)
            if(num%i==0||num%(i+2)==0)
               return 0;
         return 1;
    }
    

    不懂的去翻我博客,有一篇专门讲这个的。
    然后主要是搜索过程。
    每个状态最多有两个拓展可能:
    往右或往下,只要不出界,矩阵随便跑
    然后注意判断是否出界就可以了,总体还不算难

    #include<bits/stdc++.h>
    using namespace std;
    int n,mp[220][220],ans;
    bool IsPrime/*这绝逼是我自己打的*/(int num)
    {
    	 if(num==1)
    		return 0;
         if(num==2||num==3)
            return 1;
         if(num%6!=1&&num%6!=5)
            return 0;
         int tmp=sqrt(num);
         for(int i=5;i<=tmp;i+=6)
            if(num%i==0||num%(i+2)==0)
               return 0;
         return 1;
    }
    void build(int s1,int n1)
    {
    	if(s1==n1)
    	{
    		mp[s1][n1]=s1;
    		return;
    	}
    	else if(s1>n1)return;
    	for(int i=s1;i<=n1;i++)
    	{
    		for(int j=s1;j<=n1;j++)
    		{
    			mp[i][j]=s1;
    		}
    	}
    	build(s1+1,n1-1);
    	return;
    }
    void dfs(int x,int y,int cnt)
    {
    	if(x==n&&y==n)
    	{
    		if(IsPrime(cnt))
    		{
    			ans++;
    		}
    		return;
    	}
    	if(x<n)
    	{
    		cnt+=mp[x+1][y];
    		dfs(x+1,y,cnt);
    		cnt-=mp[x+1][y];
    	}
    	if(y<n)
    	{
    		cnt+=mp[x][y+1];
    		dfs(x,y+1,cnt);
    		cnt-=mp[x][y+1];
    	}
    	return;
    }
    int main()
    {
    	cin>>n;
    	build(1,n);
    	dfs(1,1,1);
    	cout<<ans<<endl;
    }
    

    ov.

    个人博客地址: www.moyujiang.com 或 moyujiang.top
  • 相关阅读:
    Linux 文件及目录管理命令基础
    MHA高可用及读写分离
    MySQL的备份和回复
    mysql的主从复制
    MySQL索引管理及执行计划
    [LeetCode]Linked List Cycle II解法学习
    浅谈reverse_iterator的base()函数
    [LeetCode]LRU Cache有个问题,求大神解答【已解决】
    分享一篇不错的博文《写给准备参加秋招的学弟学妹们~一定要来看哦~》
    将博客搬至CSDN
  • 原文地址:https://www.cnblogs.com/moyujiang/p/11167733.html
Copyright © 2011-2022 走看看