zoukankan      html  css  js  c++  java
  • 【NOIP2002提高组T2】字串变换-双向BFS或迭代加深搜索

    测试地址:字串变换

    做法:这道题...比较玄学,网上大多数人的题解都是双向BFS,这里就不赘述了。但我认为,还有一种迭代加深搜索的算法,但是肯定比双向BFS要慢...在Vijos上这两种方法都过了,放在这里让大家参考。

    以下是本人代码(本人水平不行,代码非常丑,凑合着看吧):

    双向BFS:

    #include <cstdio>
    #include <cstdlib>
    #include <cstring>
    #include <iostream>
    #include <algorithm>
    #include <queue>
    using namespace std;
    struct str {char s[30];} a[10],b[10];
    str q1[10010],q2[10010];
    int tot=0,step=0;
    bool f=0;
    
    bool cmp(str a,str b)
    {
      if (strlen(a.s)!=strlen(b.s)) return 0;
      else
      {
        int len=strlen(a.s);
        for(int i=0;i<len;i++)
    	  if (a.s[i]!=b.s[i]) return 0;
    	return 1;
      }
    }
    
    void double_bfs()
    {
      int h1=1,h2=1,t1=1,t2=1;
      q1[1]=a[0];q2[1]=b[0];
      int f1=1,f2=1,s1=1,e1=1,s2=1,e2=1;
      if (cmp(a[0],b[0])) {f=1;return;}
      while(h1<=t1&&h2<=t2&&step<=10)
      {
        if (f1<=f2)
    	{
    	  str now=q1[h1++];
    	  for(int i=s2;i<=e2;i++)
    	    if (cmp(now,q2[i])) {f=1;return;}
    	  int len=strlen(now.s);
    	  for(int i=1;i<=tot;i++)
    	    for(int j=0;j<len;j++)
    		  if (j+strlen(a[i].s)-1<len&&a[i].s[0]==now.s[j])
    		  {
    		    bool flag=1;
    		    for(int k=1;k<strlen(a[i].s);k++)
    			  if (a[i].s[k]!=now.s[j+k]) {flag=0;break;}
    			if (flag)
    			{
    			  str nxt;
    			  memset(nxt.s,0,sizeof(nxt.s));
    			  for(int k=0,l=0;k<=strlen(now.s);k++,l++)
    			  {
    			    if (k==j)
    				{
    				  int x;
    				  for(x=0;x<strlen(b[i].s);x++)
    				    nxt.s[l+x]=b[i].s[x];
    				  l=l+x-1;
    				  k=j+strlen(a[i].s)-1;
    				}
    				else nxt.s[l]=now.s[k];
    			  }
    			  q1[++t1]=nxt;
    			}
    		  }
    	  if (h1>e1)
    	  {
    	    step++;
    		f1=t1-e1;
    		s1=e1+1;
    	    e1=t1;
    	  }
    	}
    	else
    	{
    	  str now=q2[h2++];
    	  for(int i=s1;i<=e1;i++)
    	    if (cmp(now,q1[i])) {f=1;return;}
    	  int len=strlen(now.s);
    	  for(int i=1;i<=tot;i++)
    	    for(int j=0;j<len;j++)
    		  if (j+strlen(b[i].s)-1<len&&b[i].s[0]==now.s[j])
    		  {
    		    bool flag=1;
    		    for(int k=1;k<strlen(b[i].s);k++)
    			  if (b[i].s[k]!=now.s[j+k]) {flag=0;break;}
    			if (flag)
    			{
    			  str nxt;
    			  memset(nxt.s,0,sizeof(nxt.s));
    			  for(int k=0,l=0;k<=strlen(now.s);k++,l++)
    			  {
    			    if (k==j)
    				{
    				  int x;
    				  for(x=0;x<strlen(a[i].s);x++)
    				    nxt.s[l+x]=a[i].s[x];
    				  l=l+x-1;
    				  k=j+strlen(b[i].s)-1;
    				}
    				else nxt.s[l]=now.s[k];
    			  }
    			  q2[++t2]=nxt;
    			}
    		  }
    	  if (h2>e2)
    	  {
    	    step++;
    		f2=t2-e2;
    		s2=e2+1;
    	    e2=t2;
    	  }
    	}
      }
      f=0;
    }
    
    int main()
    {
      while(scanf("%s%s",a[tot].s,b[tot].s)!=EOF)
      {
        getchar();
    	tot++;
      }
      tot--;
      
      double_bfs();
      
      if (f) printf("%d",step);
      else printf("NO ANSWER!");
      
      return 0;
    }
    
    迭代加深搜索:

    #include <cstdio>
    #include <cstdlib>
    #include <cstring>
    #include <iostream>
    #include <algorithm>
    #include <map>
    using namespace std;
    struct str {char s[30];} a[10],b[10];
    int tot,depth;
    
    bool cmp(str a,str b)
    {
      if (strlen(a.s)!=strlen(b.s)) return 0;
      for(int i=0;i<strlen(a.s);i++)
        if (a.s[i]!=b.s[i]) return 0;
      return 1;
    }
    
    bool dfs(int step,str now)
    {
      if (step==depth) return cmp(now,b[0]);
      for(int i=1;i<=tot;i++)
        for(int j=0;j<strlen(now.s);j++)
    	  if (now.s[j]==a[i].s[0]&&j+strlen(a[i].s)-1<strlen(now.s))
    	  {
    	    bool flag=1;
    		for(int k=1;k<strlen(a[i].s);k++)
    		  if (now.s[j+k]!=a[i].s[k]) {flag=0;break;}
    		if (flag)
    		{
    		  str nxt;
    		  memset(nxt.s,0,sizeof(nxt.s));
    		  for(int k=0,l=0;k<strlen(now.s);k++,l++)
    		  {
    		    if (k==j)
    			{
    			  for(int x=0;x<strlen(b[i].s);x++)
    			    nxt.s[l+x]=b[i].s[x];
    			  l=l+strlen(b[i].s)-1;
    			  k=j+strlen(a[i].s)-1;
    			}
    			else nxt.s[l]=now.s[k];
    		  }
    		  if (dfs(step+1,nxt)) return 1;
    		}
    	  }
      return 0;
    }
    
    int main()
    {
      tot=0;
      while(scanf("%s%s",a[tot].s,b[tot].s)!=EOF)
      {
    	tot++;
    	getchar();
      }
      tot--;
      
      for(depth=0;depth<=10;depth++)
        if (dfs(0,a[0])) break;
      
      if (depth<=10) printf("%d",depth);
      else printf("NO ANSWER!");
      
      return 0;
    }
    



  • 相关阅读:
    第二十九课 循环链表的实现
    第二十八课 再论智能指针(下)
    第二十七课 再论智能指针(上)
    第二十六课 典型问题分析(Bugfix)
    普通new和placement new的重载
    leetcode 581. Shortest Unsorted Continuous Subarray
    leetcode 605. Can Place Flowers
    leetcode 219. Contains Duplicate II
    leetcode 283. Move Zeroes
    leetcode 217. Contains Duplicate
  • 原文地址:https://www.cnblogs.com/Maxwei-wzj/p/9793866.html
Copyright © 2011-2022 走看看