fzu 2159 WuYou (中等难度)终于是AC了,太激动了。。。

1、http://acm.fzu.edu.cn/problem.php?pid=2159

这么一道中等难度的题目都做了好几天了,终于是AC了。。。

2、题目大意:

roblem 2159 WuYou

Accept: 59    Submit: 304
Time Limit: 1000 mSec    Memory Limit : 32768 KB

Problem Description

有两个正整数A和B,这两个数的位数相同且不含前缀0。A的一些位不能够确定,用‘?’代替。已知A是满足 A < B 的最大的A。求A 。

Input

第一行一个整数T(T<=1000),表示有T组数据。

每组数据两行,第一行为A,第二行为B(0 < A,B <= 10^10000)。

Output

对于每组数据,输出满足A<B的最大的A。如果不存在,输出-1。

Sample Input

319?8?111

Sample Output

17-1

3、解题思路:

首先这道题目题意很容易理解,仔细看题目就会发现情况特别多

1、如果a[i]==b[i]不处理

2、if(a[i]<b[i]) return 1;说明找到这样的数a满足a<b;对于a后边的?都换成9即可

3、if(a[i]>b[i]) 就要分情况讨论了

如果这之前没有出现过?,那么直接return 0就行,这已经说明a>b

如果之前出现过?,就要倒着走所有的?,看能不能比b小1

这时也要注意首位置不能为0,其余位置也要<=0;

4、AC代码:

#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
char a[1000005];
char b[1000005];
int c[1000005];
int k;
int compare(int l)
{
    int flag=0;//判断?前是否有a[i]<b[i]
    k=0;
    for(int i=0; i<l; i++)
    {
        if(a[i]==b[i])
            continue;
        if(a[i]!='?' && a[i]>b[i] && k==0)//第一个?前有a>b
            return 0;
        else if(a[i]!='?' && a[i]>b[i] && k!=0)//前边出现过?
        {
//            for(int j=k-1;j>=0;j--)
//            {
//                if(c[j]!=0 && a[c[j]]-'0'-1>=0)
//                {
//                    a[c[j]]=a[c[j]]-1;
//                    return 1;
//                }
//                else if(c[j]==0 && a[0]-'0'-1>0)
//                {
//                    a[0]=a[0]-1;
//                    return 1;
//                }
//                //???1 2000这组样例这样处理不对
////                else
////                {
////                    return 0;
////                }
//            }
            return -1;
        }
        else if(a[i]!='?' && a[i]<b[i])//比较出A<B,后边?改成9
        {
            return 1;
        }
        else if(a[i]=='?')
        {
            c[k++]=i;
            a[i]=b[i];
        }
    }
    return -1;
}
int main()
{
    int t;
    scanf("%d",&t);
    while(t--)
    {
        scanf("%s%s",a,b);
        int l=strlen(a);
        int flag=compare(l);
        if(flag==1)//找到A比B小
        {
            for(int i=0; i<l; i++)
            {
                if(a[i]!='?')
                    printf("%c",a[i]);
                else
                    printf("9");
            }
            printf("\n");
        }
        else if(flag==0)
        {
            printf("-1\n");
        }
        else
        {
            //printf("*****\n");
            int flag=0;
            for(int i=k-1; i>0; i--)
            {
                //printf("%c\n",a[c[i]]);
                if(a[c[i]]-'0'-1>=0)
                {
                    flag=1;
                    a[c[i]]=a[c[i]]-1;
                    break;
                }
                else
                {
                    //??? 910这组样例需要这样处理
                    for(int p=i; p>0; p--)
                    {
                        if(c[p]-1==c[p-1])
                        {
                            if(c[p-1]!=0 && a[c[p-1]]-'0'-1>=0)
                            {
                                a[c[p-1]]=a[c[p-1]]-1;
                                for(int q=i; q>p-1; q--)
                                {
                                    a[c[q]]='9';
                                }
                                flag=1;
                                break;
                            }
                            else if(c[p-1]==0 && a[c[p-1]]-'0'-1>0)
                            {
                                a[c[p-1]]=a[c[p-1]]-1;
                                for(int q=i; q>p-1; q--)
                                {
                                    a[c[q]]='9';
                                }
                                flag=1;
                                break;
                            }
                        }
                        else
                            break;
                    }

                }
                if(flag==1)
                    break;
            }
            if(flag==1)
            {
                for(int i=0; i<l; i++)
                {
                    if(a[i]=='?')
                        printf("9");
                    else
                        printf("%c",a[i]);

                }
                printf("\n");
                //printf("%s\n",a);
            }
            else
            {
                if(c[0]==0 && a[0]-'0'-1>0)
                {
                    a[0]=a[0]-1;
                    for(int i=0; i<l; i++)
                    {
                        if(a[i]=='?')
                            printf("9");
                        else
                            printf("%c",a[i]);

                    }
                    printf("\n");
                }
                else if(c[0]!=0)
                {
                    if(a[c[0]]-'0'-1>=0)
                    {
                        a[c[0]]=a[c[0]]-1;
                        //?2? 900 这个样例表明得分别打印,如果直接打印字符串a,中会有?
                        for(int i=0; i<l; i++)
                        {
                            if(a[i]=='?')
                                printf("9");
                            else
                                printf("%c",a[i]);

                        }
                        printf("\n");
                        //printf("%s\n",a);
                    }
                    else
                    {
                        printf("-1\n");
                    }
                }
                else
                {
                    printf("-1\n");
                }
            }
        }
    }
    return 0;
}
/*
10
???
800
???1
2000
?2?
900

*/
/*
3
???
910
????1?
111010
?10?1?
111010
*/
/*
3
1
9
?
8
?1
11


20
?123
2123
?123
1124
123?
1234
123?
1230
123?7??
1245678

10
123
123
124
123
?123?2
112302

10
?
1
?1
10

30
???
100
???
123
???
236
??3
124
?2?
131
?2?
133
???
101
??1
101

30
123?45?
1233442
?
2
???1
1000
???1
2000

*/


第一次错的代码:、wrong answer代码

#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
char a[10005];
char b[10005];
int c[10005];
int k;
int compare(int l)
{
    int flag=0;//判断?前是否有a[i]<b[i]
    k=0;
    for(int i=0;i<l;i++)
    {
        if(a[i]==b[i])
        continue;
        if(a[i]!='?' && a[i]>b[i] && k==0)//第一个?前有a>b
        return 0;
        else if(a[i]!='?' && a[i]>b[i] && k!=0)//前边出现过?
        {
            for(int j=k-1;j>=0;j--)
            {
                if(c[j]!=0 && a[c[j]]-'0'-1>=0)
                {
                    a[c[j]]=a[c[j]]-1;
                    return 1;
                }
                else if(c[j]==0 && a[0]-'0'-1>0)
                {
                    a[0]=a[0]-1;
                    return 1;
                }
                else
                {
                    return 0;
                }
            }
        }
        else if(a[i]!='?' && a[i]<b[i])//比较出A<B,后边?改成9
        {
            return 1;
        }
        else if(a[i]=='?')
        {
            c[k++]=i;
            a[i]=b[i];
        }
    }
    return -1;
}
int main()
{
    int t;
    scanf("%d",&t);
    while(t--)
    {
        scanf("%s%s",a,b);
        int l=strlen(a);
        int flag=compare(l);
        if(flag==1)//找到A比B小
        {
            for(int i=0;i<l;i++)
            {
                if(a[i]!='?')
                printf("%c",a[i]);
                else
                printf("9");
            }
            printf("\n");
        }
        else if(flag==0)
        {
            printf("-1\n");
        }
        else
        {
            int flag=0;
            for(int i=k-1;i>0;i--)
            {
                if(a[c[i]]-'0'-1>=0)
                {
                    flag=1;
                    a[c[i]]=a[c[i]]-1;
                    break;
                }
            }
            if(flag==1)
            {
                printf("%s\n",a);
            }
            else
            {
                if(c[0]==0 && a[0]-'0'-1>0)
                {
                    a[0]=a[0]-1;
                    printf("%s\n",a);
                }
                else if(c[0]!=0)
                {
                    if(a[c[0]]-'0'-1>=0)
                    {
                        a[c[0]]=a[c[0]]-1;
                        printf("%s\n",a);
                    }
                    else
                    {
                        printf("-1\n");
                    }
                }
                else
                {
                    printf("-1\n");
                }
            }
        }
    }
    return 0;
}
/*
3
1
9
?
8
?1
11


20
?123
2123
?123
1124
123?
1234
123?
1230
123?7??
1245678

10
123
123
124
123
?123?2
112302

10
?
1
?1
10

30
???
100
???
123
???
236
??3
124
?2?
131
?2?
133
???
101
??1
101

3
123?45?
1233442

*/


 通过

???

910这组样例改正后的代码,虽然样例过了,仍然是wrong answer

#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
char a[10005];
char b[10005];
int c[10005];
int k;
int compare(int l)
{
    int flag=0;//判断?前是否有a[i]<b[i]
    k=0;
    for(int i=0;i<l;i++)
    {
        if(a[i]==b[i])
        continue;
        if(a[i]!='?' && a[i]>b[i] && k==0)//第一个?前有a>b
        return 0;
        else if(a[i]!='?' && a[i]>b[i] && k!=0)//前边出现过?
        {
            for(int j=k-1;j>=0;j--)
            {
                if(c[j]!=0 && a[c[j]]-'0'-1>=0)
                {
                    a[c[j]]=a[c[j]]-1;
                    return 1;
                }
                else if(c[j]==0 && a[0]-'0'-1>0)
                {
                    a[0]=a[0]-1;
                    return 1;
                }
                else
                {
                    return 0;
                }
            }
        }
        else if(a[i]!='?' && a[i]<b[i])//比较出A<B,后边?改成9
        {
            return 1;
        }
        else if(a[i]=='?')
        {
            c[k++]=i;
            a[i]=b[i];
        }
    }
    return -1;
}
int main()
{
    int t;
    scanf("%d",&t);
    while(t--)
    {
        scanf("%s%s",a,b);
        int l=strlen(a);
        int flag=compare(l);
        if(flag==1)//找到A比B小
        {
            for(int i=0;i<l;i++)
            {
                if(a[i]!='?')
                printf("%c",a[i]);
                else
                printf("9");
            }
            printf("\n");
        }
        else if(flag==0)
        {
            printf("-1\n");
        }
        else
        {
            int flag=0;
            for(int i=k-1;i>0;i--)
            {
                //printf("%c\n",a[c[i]]);
                if(a[c[i]]-'0'-1>=0)
                {
                    flag=1;
                    a[c[i]]=a[c[i]]-1;
                    break;
                }
                else
                {
                    //??? 910这组样例需要这样处理
                    for(int p=i;p>1;p--)
                    {
                        if(c[p]-1==c[p-1])
                        {
                            if(a[c[p-1]]-'0'-1>=0)
                            {
                                a[c[p-1]]=a[c[p-1]]-1;
                                for(int q=i;q>p-1;q--)
                                {
                                    a[c[q]]='9';
                                }
                                flag=1;
                                break;
                            }
                        }
                        else
                        break;
                    }
                }
                if(flag==1)
                break;
            }
            if(flag==1)
            {
                printf("%s\n",a);
            }
            else
            {
                if(c[0]==0 && a[0]-'0'-1>0)
                {
                    a[0]=a[0]-1;
                    printf("%s\n",a);
                }
                else if(c[0]!=0)
                {
                    if(a[c[0]]-'0'-1>=0)
                    {
                        a[c[0]]=a[c[0]]-1;
                        printf("%s\n",a);
                    }
                    else
                    {
                        printf("-1\n");
                    }
                }
                else
                {
                    printf("-1\n");
                }
            }
        }
    }
    return 0;
}
/*
3
???
910
*/
/*
3
1
9
?
8
?1
11


20
?123
2123
?123
1124
123?
1234
123?
1230
123?7??
1245678

10
123
123
124
123
?123?2
112302

10
?
1
?1
10

30
???
100
???
123
???
236
??3
124
?2?
131
?2?
133
???
101
??1
101

3
123?45?
1233442

*/

又有几组测试用例检测出错误,改正后仍wrong 的代码:

#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
char a[1000005];
char b[1000005];
int c[1000005];
int k;
int compare(int l)
{
    int flag=0;//判断?前是否有a[i]<b[i]
    k=0;
    for(int i=0;i<l;i++)
    {
        if(a[i]==b[i])
        continue;
        if(a[i]!='?' && a[i]>b[i] && k==0)//第一个?前有a>b
        return 0;
        else if(a[i]!='?' && a[i]>b[i] && k!=0)//前边出现过?
        {
//            for(int j=k-1;j>=0;j--)
//            {
//                if(c[j]!=0 && a[c[j]]-'0'-1>=0)
//                {
//                    a[c[j]]=a[c[j]]-1;
//                    return 1;
//                }
//                else if(c[j]==0 && a[0]-'0'-1>0)
//                {
//                    a[0]=a[0]-1;
//                    return 1;
//                }
//                //???1 2000这组样例这样处理不对
////                else
////                {
////                    return 0;
////                }
//            }
            return -1;
        }
        else if(a[i]!='?' && a[i]<b[i])//比较出A<B,后边?改成9
        {
            return 1;
        }
        else if(a[i]=='?')
        {
            c[k++]=i;
            a[i]=b[i];
        }
    }
    return -1;
}
int main()
{
    int t;
    scanf("%d",&t);
    while(t--)
    {
        scanf("%s%s",a,b);
        int l=strlen(a);
        int flag=compare(l);
        if(flag==1)//找到A比B小
        {
            for(int i=0;i<l;i++)
            {
                if(a[i]!='?')
                printf("%c",a[i]);
                else
                printf("9");
            }
            printf("\n");
        }
        else if(flag==0)
        {
            printf("-1\n");
        }
        else
        {
            //printf("*****\n");
            int flag=0;
            for(int i=k-1;i>0;i--)
            {
                //printf("%c\n",a[c[i]]);
                if(a[c[i]]-'0'-1>=0)
                {
                    flag=1;
                    a[c[i]]=a[c[i]]-1;
                    break;
                }
                else
                {
                    //??? 910这组样例需要这样处理
                    for(int p=i;p>0;p--)
                    {
                        if(c[p]-1==c[p-1])
                        {
                            if(c[p-1]!=0 && a[c[p-1]]-'0'-1>=0)
                            {
                                a[c[p-1]]=a[c[p-1]]-1;
                                for(int q=i;q>p-1;q--)
                                {
                                    a[c[q]]='9';
                                }
                                flag=1;
                                break;
                            }
                            else if(c[p-1]==0 && a[c[p-1]]-'0'-1>0)
                            {
                                a[c[p-1]]=a[c[p-1]]-1;
                                for(int q=i;q>p-1;q--)
                                {
                                    a[c[q]]='9';
                                }
                                flag=1;
                                break;
                            }
                        }
                        else
                        break;
                    }

                }
                if(flag==1)
                break;
            }
            if(flag==1)
            {
                printf("%s\n",a);
            }
            else
            {
                if(c[0]==0 && a[0]-'0'-1>0)
                {
                    a[0]=a[0]-1;
                    printf("%s\n",a);
                }
                else if(c[0]!=0)
                {
                    if(a[c[0]]-'0'-1>=0)
                    {
                        a[c[0]]=a[c[0]]-1;
                        printf("%s\n",a);
                    }
                    else
                    {
                        printf("-1\n");
                    }
                }
                else
                {
                    printf("-1\n");
                }
            }
        }
    }
    return 0;
}
/*
10
???
800
???1
2000
*/
/*
3
???
910
????1?
111010
?10?1?
111010
*/
/*
3
1
9
?
8
?1
11


20
?123
2123
?123
1124
123?
1234
123?
1230
123?7??
1245678

10
123
123
124
123
?123?2
112302

10
?
1
?1
10

30
???
100
???
123
???
236
??3
124
?2?
131
?2?
133
???
101
??1
101

30
123?45?
1233442
?
2
???1
1000
???1
2000

*/


你可能感兴趣的:(fzu 2159 WuYou (中等难度)终于是AC了,太激动了。。。)