迭代加深+双向dfs+IDA*

简介: 迭代加深+双向dfs+IDA*

1.迭代加深

顾名思义说明迭代的层数逐渐加深,这样做法有点像bfs的做法层层突出,符合的题型是答案在层数较低的那一层里


加成序列

170. 加成序列 - AcWing题库

#include<bits/stdc++.h>
using namespace std;
const int N=110;
int n;
int path[N];
bool dfs(int u,int depth)//第一个是元素个数,第二个是当前的深度
{
    if(u>depth) return  false;//假如元素大于深度,说明不可能了
    if(path[u-1]==n&&depth==u) return true;//假如元素个数达到深度,且最后一个是n,则说明符合条件
    bool st[N]={0};//用来标记那个数已经枚举过
    for(int i=u-1;i>=0;i--)//枚举前u个数,两两可以相加,相当于组合数
        for(int j=i;j>=0;j--)
        {
           int s=path[i]+path[j];//获取这两个数的值
           if(st[s]||s>n||s<=path[u-1]) continue;//假如已经枚举过或者和大于n或者不符合后一个数比前一个数小,则跳过
            st[s]=true;//标记这个数已经用过
            path[u]=s;//这个数放进数组里
           if(dfs(u+1,depth)) return true;//继续搜索下一个数
        }
    return false;
}
int main()
{
    path[0]=1;//规定第一个数是1
   while(cin>>n,n)
   {
       int depth=1;//枚举深度,大概知道答案在比较近的一层,所以不枚举宽度
       while(!dfs(1,depth)) depth++;//假如这个深度不符合,则加1
       for(int i=0;i<depth;i++) cout<<path[i]<<' ';//输出答案
       puts("");
   }
   return 0;
}


2.双向dfs

顾名思义就是分两段dfs,搜索前半段与后半段

送礼物

171. 送礼物 - AcWing题库

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=110;
int n,k;
int cnt=1;
ll w[N],m,weight[1<<25],ans;
void dfs1(int u,ll s)
{
    if(s>m) return;//假如已经大于了,则直接返回
    if(u==k)//假如搜到了最后一个
    {
        weight[cnt++]=s;//把这个值放进数组里来
        return;//这里不能少
    }
    dfs1(u+1,s);//假如不选这个数
    if(s+w[u]<=m) dfs1(u+1,s+w[u]);//假如满足条件选这个数
}
void dfs2(int u,ll s)
{
   if(s>m) return;//假如已经大于了,则直接返回
   if(u==n)//假如搜到了最后一个
   {
       //下面二分有左边界,因为找刚好小于m的最大数
      int l=0,r=cnt-1;
      while(l<r)
      {
          int mid=l+r+1>>1;
          if(weight[mid]+s<=m) l=mid;
          else r=mid-1;
      }
     ans=max(ans,s+weight[l]);//更新一下答案
     return;//这里不能少
   }
    dfs2(u+1,s);//假如不选这个数
    if(s+w[u]<=m) dfs2(u+1,s+w[u]);//假如符合条件选这个数
}
int main()
{
    cin>>m>>n;
    for(int i=0;i<n;i++) cin>>w[i];
    //从大到小排序
    sort(w,w+n);
    reverse(w,w+n);
    k=n/2+1;//对半分开搜索
    dfs1(0,0);//搜索前半段,打表出来存进数组里头
    //去重
    sort(weight,weight+cnt);
    cnt=unique(weight,weight+cnt)-weight;
    dfs2(k,0);//搜索后半段
    cout<<ans<<endl;
   return 0;
}


3.IDA*

找估计函数,在枚举步数,用估计函数+当前步数来剪枝,大大提高效率,前提是答案的层数小

1.排书

180. 排书 - AcWing题库

#include<bits/stdc++.h>
using namespace std;
const int N=20;
int n;
int q[N],w[5][N];
int f()//用来计算估计函数
{
    int totol=0;//算位置不符合的总个数
    for(int i=0;i<n-1;i++)
      if(q[i]!=q[i+1]-1)
         totol++;
    return (totol+2)/3;//因为每次交换能更改三个位置的后缀,我们算他改的都是正确的则就除以三,但是实际肯定是小了的
}
bool dfs(int depth,int max_depth)//depth是当前已执行的步数,max_depth是最大步数
{
    if(f()+depth>max_depth) return false;//假如当前步数加上估计函数大于最大步数了,说明不符合
    if(f()==0) return true;//假如估计函数为0,也就是正确答案了,则直接返回true
    for(int len=1;len<=n;len++)//枚举需要更改的长度
        for(int l=0;l+len-1<n;l++)//枚举左边界
        {
            int r=l+len-1;//右边界
            for(int k=r+1;k<n;k++)//枚举需要跟右边那个位置交换
            {
                memcpy(w[depth],q,sizeof q);//把上一层的状态复制过来
                int x=l;
                //下面进行这一段跟右边的位置交换
                for(int y=r+1;y<=k;y++,x++) q[x]=w[depth][y];
                for(int y=l;y<=r;y++,x++) q[x]=w[depth][y];
                //进行下一步,假如下一步可以符合答案,则链式返回true
                if(dfs(depth+1,max_depth)) return true;
                memcpy(q,w[depth],sizeof q);//把更改后的这一层q复制到w这一层中,供下一层使用
            }
        }
    return false;//反之返回失败
}
int main()
{
   int T;
   cin>>T;
   while(T--)
   {
       cin>>n;
       for(int i=0;i<n;i++) cin>>q[i];
       int depth=0;//枚举所用的步数
       while(depth<5&&!dfs(0,depth)) depth++;//假如所用步数不能排好书,则步数++
       if(depth<5) cout<<depth<<endl;
       else puts("5 or more");
   }
   return 0;
}


2.回转游戏

181. 回转游戏 - AcWing题库

#include<bits/stdc++.h>
using namespace std;
const int N=24;
int n;
int q[N],path[110];
//分别为八个方向的位置
int op[8][7]=
{
    {0,2,6,11,15,20,22},
    {1,3,8,12,17,21,23},
    {10,9,8,7,6,5,4},
    {19,18,17,16,15,14,13},
    {23,21,17,12,8,3,1},
    {22,20,15,11,6,2,0},
    {13,14,15,16,17,18,19},
    {4,5,6,7,8,9,10}
};
int unop[8]={5,4,7,6,1,0,3,2};//八个方向的反操作,避免往上拉了,又往下拉,等于没操作
int cenctr[8]={6,7,8,11,12,15,16,17};//中心的8个位置
int f()//算估计函数
{
    static int temp[4];//静态数组节省空间
    memset(temp,0,sizeof temp);//清空
    for(int i=0;i<8;i++) temp[q[cenctr[i]]]++;//记录中间那个数出现的最多
    int m=0;
    for(int i=1;i<=3;i++) m=max(m,temp[i]);//求一下出现最多的数的个数
    return 8-m;//返回最少需要操作的次数,也就是8-m
}
void operate(int x)//进行把数组第一个往上拉去最后一个,其他往前一个位置的操作
{
    int t=q[op[x][0]];
    for(int i=0;i<6;i++) q[op[x][i]]=q[op[x][i+1]];
    q[op[x][6]]=t;
}
bool dfs(int depth,int max_depth,int last)//depth是当前步数,max_depth是最大步数,last是上一步的操作
{
    if(depth+f()>max_depth) return false;//假如估计函数+目前步数已经大于最大步数了,则返回false
    if(f()==0) return true;//假如已经排好了,则返回true
    for(int i=0;i<8;i++)//枚举8个操作
        if(unop[i]!=last)//返回操作了上又操作下,等于没操作
        {
           operate(i);//操作i
           path[depth]=i;//把当前路径标记为i操作的
           if(dfs(depth+1,max_depth,i)) return true;//假如下一步操作是true,则返回true
           operate(unop[i]);//恢复现场,也就是反向操作一下
        }
  return false;//找不到合法答案
}
int main()
{
  while(cin>>q[0],q[0])
  {
      for(int i=1;i<N;i++) cin>>q[i];
      int depth=0;//枚举需要走的步数
      while(!dfs(0,depth,-1)) depth++;//假如改步数不符合,则步数++
      if(!depth) printf("No moves needed");
      else
      {
          for(int i=0;i<depth;i++) printf("%c",path[i]+'A');//输出路径
      }
      printf("\n%d\n",q[6]);//输出中心的数字
  }
   return 0;
}





相关文章
|
8月前
【每日一题Day335】LC1993树上的操作 | dfs
【每日一题Day335】LC1993树上的操作 | dfs
59 0
|
2月前
|
算法 定位技术 Python
震惊!Python 图结构竟然可以这样玩?DFS&BFS 遍历技巧大公开
在 Python 编程中,图是一种重要的数据结构,而深度优先搜索(DFS)和广度优先搜索(BFS)是遍历图的两种关键算法。本文将通过定义图的数据结构、实现 DFS 和 BFS 算法,并通过具体示例展示其应用,帮助读者深入理解这两种算法。DFS 适用于寻找路径和检查图连通性,而 BFS 适用于寻找最短路径。掌握这些技巧,可以更高效地解决与图相关的复杂问题。
31 2
|
8月前
LabVIEW使用VI脚本重新排列对象
LabVIEW使用VI脚本重新排列对象
42 1
|
6月前
|
存储 算法 Python
Python图论实战:从零基础到精通DFS与BFS遍历,轻松玩转复杂网络结构
【7月更文挑战第11天】图论在数据科学中扮演关键角色,用于解决复杂网络问题。Python因其易用性和库支持成为实现图算法的首选。本文通过问答形式介绍DFS和BFS,图是节点和边的数据结构,遍历用于搜索和分析。Python中图可表示为邻接表,DFS用递归遍历,BFS借助队列。DFS适用于深度探索,BFS则用于最短路径。提供的代码示例帮助理解如何在Python中应用这两种遍历算法。开始探索图论,解锁更多技术可能!
165 6
|
8月前
LabVIEW合并VI
LabVIEW合并VI
96 0
LabVIEW合并VI
|
存储 算法
深度优先遍历(DFS):探索图的奥秘
当进行深度优先遍历(DFS)时,我们需要按照一定的步骤来遍历图中的节点。 选择起始节点:选择图中的一个起始节点作为遍历的起点。 标记已访问:将起始节点标记为已访问,并将其放入数据结构中,比如栈或递归调用。 探索相邻节点:从起始节点开始,探索与其相邻的节点。这是通过查找邻接表来实现的,邻接表存储了每个节点的相邻节点信息。 深入探索:选择一个相邻节点,标记为已访问,并将其放入数据结构中。然后,从这个相邻节点出发,继续探索其相邻节点,形成一个深入的路径。这一步是递归的核心。 回溯:当没有未访问的相邻节点时,回溯到上一个节点,继续探索其他未访问的相邻节点。这可以通过从数据结构中弹出节点来实现,或者从递
318 0
|
运维 网络安全
巧用 nc 命令传输文件
今天在业务上云的时候,遇到了些问题。最终发现问题的根源不好排查,于是——把生产环境的全量配置文件,还有日志全量打包下载到开发机器分析!生产和开发机内网不通,都是走公网传输,但是速度特别慢……
208 3
|
存储 算法 前端开发
LIO-SAM回环检测模块代码解析
LIO-SAM回环检测模块代码解析
509 0
LIO-SAM回环检测模块代码解析
|
机器学习/深度学习 测试技术
|
存储 算法
双向DFS
复习acwing算法提高课的内容,本篇为讲解算法:双向DFS,关于时间复杂度:目前博主不太会计算,先鸽了,日后一定补上。
122 0
双向DFS

热门文章

最新文章