146 - ID Codes

题目:146 - ID Codes


题目大意:给出一个字符的全排列,字符可以重复,但是排列方式有先后顺序,就是从小到大的排,例如1个a, 2个b,排列:aab, aba, baa,题目问的是给出一个序列,看后面还有没序列,有输出紧接着的一个,如果是最后一个就输出No Successor。


思路:就是uva书上的7.2.2的改造,将重排范围缩小到从后向前找到的第一个反常的字符,后面的正常的字符满足的规律是当前的这个字符比他前一个要小。然后防止重复则是每次统计已经排列的集合里要准备排入的字符的个数,和这个字符在初始被列入排序的字符集合中的个数,如果前面的比后面的小,就说明还有可以用的这个字符,就不会多用,或没有的却用了。


#include<stdio.h>
#include<string.h>

const int N = 55;
char s[N], str[N];
int flag;

void build(int n, int c, int cur, char * A) {

	int i, j;
	if(cur != n) {

		for( i = 0; i < 26; i++){
			
			if(cur == n - c) {

				if((char)(i + 'a') <= s[n - c])
					continue;
			}
			int c1 , c2;
			c1 = c2 = 0;
			char m = (char)(i + 'a');
			for(j = n - c; j < cur; j++)
				if(A[j] == m)
						c1++;
			for(j = n - c; j < n; j++)
				if(s[j] == m)
						c2++;

			if(c1 < c2) {
						
				A[cur] = (char)(i + 'a');
				build(n, c, cur + 1, A);

			}
			if(flag)
				return;
		}
	}
	else {

			s[n - c] = '\0';
			strcat(s, A + n - c);
			printf("%s\n", s);
			flag = 1;	
	}
}
int main() {
	
	while(scanf("%s", s) && s[0] != '#') {
		
		int i, n;
		n = strlen(s);
		for(i = n - 1; i >= 1; i--)
			if(s[i] > s[i - 1])
				break;

		if(!i) {

			printf("No Successor\n");
			continue;
		}

		int count = n - i + 1;
		flag = 0;
		memset(str, 0, sizeof(str));
		build(n, count, i - 1,  str);

	}
	return 0;
}


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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值