Noip2004 虫食算

题目传送门


暴搜都打不对 QAQ 调了好久


40分思路

显而易见,我们可以通过枚举全排列来找到一种合法的答案

#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
int n,ans[27];
string ss[4];
bool vis[27];
long long int k[4];
int main(){
    cin>>n;
    cin>>ss[1]>>ss[2]>>ss[3];
    for(int i=1;i<=n;i++)
      ans[i]=i-1;
      
    do{
        for(int i=1;i<=3;i++){
            k[i]=0;
            for(int j=0;j<n;j++)
              k[i]=k[i]*n+ans[ss[i][j]-'A'+1];
        }
        if(k[1]+k[2]==k[3]){
            for(int i=1;i<=n;i++)
              cout<<ans[i]<<" ";
            return 0;
        }
    }
    while(next_permutation(ans+1,ans+n+1));
    return 0;
}

这么暴力当然是行不通的,暴力也要暴力的优雅一些,使用\(next \text_permutation\)枚举全排列会难以剪枝,不如暴搜,而这道题目暴搜是可以过的(然而官方正解是高斯消元)

正解:高斯消元 暴搜+剪枝

剪枝:

  1. 搜索顺序从后向前,从上到下
  2. 搜索时要判断一下当前的算式是否合法
    别的……好像也没有什么了……
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
string add[4];
int a[21],b[21],c[21],num[110],n;
bool vis[21];
void judge(){  //搜出答案后判断是否合法
    bool flag=0,f=0;
    for(int i=n-1;i!=-1;i--){
        int x=(num[add[1][i]]+num[add[2][i]]+(int)flag)%n;
        if(x!=num[add[3][i]]){
            f=1; break;
        }
        flag=(num[add[1][i]]+num[add[2][i]]+(int)flag)/(n);
    }
    if(f) return ;
    else{
        for(int i='A';i<='A'+n-1;i++) printf("%d ",num[i]);
        exit(0);
    }
}
void dfs(int pos,int y,bool zzz){  //pos是第几列,y是第几行,zzz是当前是否要进位
    if(pos<0){
        judge(); return ;
    }
    bool flag=0;
    if(y==3){  //如果来到了答案行,直接根据答案行上方两个数+进位推出答案
        if(num[add[y][pos]]==-1)
            for(int i=0;i<=n-1;i++){
                if(!vis[i]&&(num[add[1][pos]]+num[add[2][pos]]+zzz)%n==i){
                    vis[i]=1; num[add[y][pos]]=i;
                    dfs(pos-1,1,(num[add[1][pos]]+num[add[2][pos]]+zzz)/n);
                    vis[i]=0; num[add[y][pos]]=-1;
                }
            }
        else{  //如果答案代表的数被其它字母占了,那么这组解一定不合法,也没有搜的必要了
            if((num[add[1][pos]]+num[add[2][pos]]+zzz)%n!=num[add[3][pos]]){
                return ;
            }
            else dfs(pos-1,1,(num[add[1][pos]]+num[add[2][pos]]+zzz)/n); //处理进位
        }
    }
    else if(num[add[y][pos]]==-1){
        for(int i=0;i<=n-1;i++){
            if(!vis[i]){
                vis[i]=1; num[add[y][pos]]=i;
                for(int j=0;j<=n-1;j++){  //从整体上扫一遍算式,判断合不合法
                    if(num[add[1][j]]!=-1&&num[add[2][j]]!=-1&&num[add[3][j]]!=-1){
                        if((num[add[1][j]]+num[add[2][j]])%n!=num[add[3][j]]&&(num[add[1][j]]+num[add[2][j]]+1)%n!=num[add[3][j]]){  //我懒,不想考虑进位了,直接进不进位都判断一下,应该会跑得慢一些
                            vis[i]=0; num[add[y][pos]]=-1; flag=1; break;
                        }
                    }
                }
                if(flag){
                    flag=0; continue;
                }
                dfs(pos,y+1,zzz);
                vis[i]=0; num[add[y][pos]]=-1;
            }
        }
    }
    else dfs(pos,y+1,zzz);
    
}
int main(){
    scanf("%d",&n);
    memset(num,-1,sizeof(num));
    cin>>add[1]>>add[2]>>add[3];
    dfs(n-1,1,0);
    return 0;
}

转载于:https://www.cnblogs.com/wxl-Ezio/p/9613199.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在信号处理领域,DOA(Direction of Arrival)估计是一项关键技术,主要用于确定多个信号源到达接收阵列的方向。本文将详细探讨三种ESPRIT(Estimation of Signal Parameters via Rotational Invariance Techniques)算法在DOA估计中的实现,以及它们在MATLAB环境中的具体应用。 ESPRIT算法是由Paul Kailath等人于1986年提出的,其核心思想是利用阵列数据的旋转不变性来估计信号源的角度。这种算法相比传统的 MUSIC(Multiple Signal Classification)算法具有较低的计算复杂度,且无需进行特征值分解,因此在实际应用中颇具优势。 1. 普通ESPRIT算法 普通ESPRIT算法分为两个主要步骤:构造等效旋转不变系统和估计角度。通过空间平移(如延时)构建两个子阵列,使得它们之间的关系具有旋转不变性。然后,通过对子阵列数据进行最小二乘拟合,可以得到信号源的角频率估计,进一步转换为DOA估计。 2. 常规ESPRIT算法实现 在描述中提到的`common_esprit_method1.m`和`common_esprit_method2.m`是两种不同的普通ESPRIT算法实现。它们可能在实现细节上略有差异,比如选择子阵列的方式、参数估计的策略等。MATLAB代码通常会包含预处理步骤(如数据归一化)、子阵列构造、旋转不变性矩阵的建立、最小二乘估计等部分。通过运行这两个文件,可以比较它们在估计精度和计算效率上的异同。 3. TLS_ESPRIT算法 TLS(Total Least Squares)ESPRIT是对普通ESPRIT的优化,它考虑了数据噪声的影响,提高了估计的稳健性。在TLS_ESPRIT算法中,不假设数据噪声是高斯白噪声,而是采用总最小二乘准则来拟合数据。这使得算法在噪声环境下表现更优。`TLS_esprit.m`文件应该包含了TLS_ESPRIT算法的完整实现,包括TLS估计的步骤和旋转不变性矩阵的改进处理。 在实际应用中,选择合适的ESPRIT变体取决于系统条件,例如噪声水平、信号质量以及计算资源。通过MATLAB实现,研究者和工程师可以方便地比较不同算法的效果,并根据需要进行调整和优化。同时,这些代码也为教学和学习DOA估计提供了一个直观的平台,有助于深入理解ESPRIT算法的工作原理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值