UVa129 Krypton Factor

回溯法。

使用26个字母中前L个字母,要求输出一个不存在相同的相邻子串的序列,该序列应是所有合法序列中字典序第N小的。

DFS枚举所有可行序列,因为序列长度不定所以解答树中除根节点之外的每个结点都算合法序列,得到N个合法序列后停止枚举。每次填入字符后只需检查刚填入的字符是否形成了一对相同的相邻子串,因为填入前的序列已经被确认过是合法的。


#include <iostream>
using namespace std;
int n, l, len;
char a[99];

bool check(int p) {
    for(int k = 1; k * 2 <= p + 1; k++) {
        bool same = true;
        int i = p, j = p - k;
        for(int t = 0; t < k; t++)
            if(a[i--] != a[j--]) same = false;
        if(same) return false;
    }
    return true;
}

void dfs(int cur) {
    if(cur && --n == 0)
        len = cur;
    for(char c = 'A'; c < 'A' + l && n; c++) {
        a[cur] = c;
        if(!check(cur)) continue;
        dfs(cur + 1);
    }
}

void printAns() {
    cout << a[0];
    for(int i = 1; i < len; i++) {
        if(i % 64 == 0) cout << endl;
        else if(i % 4 == 0) cout << ' ';
        cout << a[i];
    }
    cout << endl << len << endl;
}

int main() {
    while(cin >> n >> l && n) {
        dfs(0);
        printAns();
    }
    return 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值