2021寒——Acm学习总结3——暴搜

1.P1036 [NOIP2002 普及组] 选数
解题思路:暴力搜索

#include<bits/stdc++.h>
using namespace std;
int n,k;
int ans;
int used[25];
int pd(int now)
{
    if(now==1) return 0;
    for(int i=2;i<=sqrt(now);i++)
    {
        if(now%i==0)
        {
            return 0;
        }
    }
    return 1;
}
int a[25];
void dfs(int index,int u,int now)
{
    if(u==k)
    {
        if(pd(now)==1) ans++;
        return;
    }
    for(int i=index;i<=n;i++)
    {
        if(used[i]!=1)
        {
            used[i]=1;
            dfs(i+1,u+1,now+a[i]);
            used[i]=0;

        }
    }
}
 int main(){
cin>>n>>k;
for(int i=1;i<=n;i++)
    cin>>a[i];
    dfs(1,0,0);
    cout<<ans;

 }

2.P1679 神奇的四次方数

#include <bits/stdc++.h>//万能头文件
using namespace std;
int m;
int ans=9999999;
void dfs(int last,int k,int now)
{
    if(now>m) return ;
    if(k>ans) return ;
    if(now==m)
    {

       if(ans>k) ans=k;
        return ;
    }
    int i;
    for( i=last;i*i*i*i<=m-now;)i++;//优化
    for( ;i>=1;i--)
    {

        dfs(i,k+1,now+i*i*i*i);
    }
}
int main(){

cin>>m;
dfs(1,0,0);
cout<<ans;

}

3.P1387 最大正方形
这个题可以搜索,逐个点搜素(自左上到右下)

#include<bits/stdc++.h>
using namespace std;
int n,m;
int xx=0,yy=0;
int a[1010][1010];
int ans=-9999;
int maxn=-9999;
void dfs(int x,int y)
{


   for(int i=xx;i<=x;i++)
   {
     if(a[i][y]==0)
        return ;
   }
   for(int j=yy;j<=y;j++)
   {
       if(a[x][j]==0)
        return ;
   }
  ans++;
   dfs(x+1,y+1);
}
int main(){

cin>>n>>m;
for(int i=1;i<=n;i++)
    for(int j=1;j<=m;j++)
    cin>>a[i][j];
for(int i=1;i<=n;i++)
{
    for(int j=1;j<=m;j++)
    {
        if(a[i][j]==0)continue;
        else{
                ans=0;
                xx=i;
                yy=j;
                dfs(i,j);
                maxn=max(ans,maxn);
        }
    }
}
cout<<maxn;
}

也可以考虑自最左上到右下
也可以用动态规划,只需要判断左方、上方、左上方的最小值+1即可

 for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=m;j++)
        {
            if(a[i][j]==0)continue;               
            f[i][j]=min(min(f[i-1][j],f[i][j-1]),f[i-1][j-1])+1;             
        }
    }

4.P1123 取数游戏
逐行搜素的dfs

#include <iostream>
using namespace std;

int T, n, m, a[2018][2018];

int ans;

int u[2018][2018];

void dfs(int x, int y, int z)     //(x, y)表示当前搜到的坐标位置,z表示搜到了(x, y)时,所取得的最大分数
{
    if (x > n)         //搜完时,更新答案
    {
        ans = max(ans, z);
        return ;
    }
    int next_x = x, next_y = y + 1;   //下一步的坐标(next_x, next_y)
    if (next_y > m)           //如果这一行搜完了
    {
        next_y = 1;           //从下一行的第一列继续搜
        next_x = x + 1;
    }
    if (!u[x - 1][y - 1] && !u[x - 1][y] && !u[x - 1][y + 1] && !u[x][y - 1] && !u[x][y + 1] && !u[x + 1][y - 1] && !u[x + 1][y] && !u[x + 1][y + 1])
    {                        //如果周围的8个点都没有选择的话
        u[x][y] = 1;         //标记这个点已搜
        dfs(next_x, next_y, z + a[x][y]);  //继续搜下一个点
        u[x][y] = 0;
    }
    dfs(next_x, next_y, z);   //如果这个点他周围的8个点有使用过的,这个点就不能选,因此找一下个点,now的值不变
                              //即使周围都为0,也不选她
}

int main()
{
    std::ios::sync_with_stdio(false);   //这句话可以加快读入速度
    cin >> T;
    for (int sth = 1; sth <= T; ++sth)
    {
        ans = 0;                       //ans是答案
        cin >> n >> m;
        for (int i = 1; i <= n; ++i)   //输入输出不解释
            for (int j = 1; j <= m; ++j)
                cin >> a[i][j];
        dfs(1, 0, 0);                  //开始搜
        cout << ans << endl;
    }
    return 0;
}

5.母亲的cow

#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
int ta,tb,tc,k,ans[1020],vis[22][22][22];
void dfs(int nowa,int nowb,int nowc){
    if(vis[nowa][nowb][nowc])return;
    if(nowa==0)ans[++k]=nowc;
    vis[nowa][nowb][nowc]=1;//防止死循环
    if(nowc){
        if(nowa<ta)
        dfs(min(ta,nowa+nowc),nowb,nowc-(min(ta,nowa+nowc)-nowa));
        if(nowb<tb)
        dfs(nowa,min(nowb+nowc,tb),nowc-(min(tb,nowb+nowc)-nowb));
    }
    if(nowb){
        if(nowa<ta)
        dfs(min(ta,nowa+nowb),nowb-(min(ta,nowa+nowb)-nowa),nowc);
        if(nowc<tc)
        dfs(nowa,nowb-(min(tc,nowc+nowb)-nowc),min(tc,nowc+nowb));
    }
    if(nowa){
        if(nowb<tb)
        dfs(nowa-(min(tb,nowa+nowb)-nowb),min(tb,nowa+nowb),nowc);
        if(nowc<tc)
        dfs(nowa-(min(tc,nowa+nowc)-nowc),nowb,min(tc,nowc+nowa));
    }
    return;
}
int main(){
    scanf("%d%d%d",&ta,&tb,&tc);
    dfs(0,0,tc);
    sort(ans+1,ans+k+1);
    for(int i=1;i<=k;i++)printf("%d ",ans[i]);
    return 0;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

沉梦昂志️

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值