NOIP2012提高组

D1T1.Vigenère密码

模拟

#include<iostream> 
#include<cstdio> 
using namespace std; 
int main() 
{ 
    string a,b;cin>>b>>a; 
    for(int i=0;i<a.size();i++) 
    { 
        bool ok=0;if(a[i]>='a'&&a[i]<='z')a[i]=a[i]-'a'+'A',ok=1;int j=i%b.size(); 
        if(b[j]>='a'&&b[j]<='z')b[j]=b[j]-'a'+'A'; 
        int c=((a[i]-'A')-(b[j]-'A')+26)%26;if(ok==0)cout<<(char)(c+'A');else cout<<(char)(c+'a'); 
    } 
    return 0; 
} 
View Code

D1T2.国王游戏

贪心+高精度

我们知道,一个排序是经过好多次交换变成的,我们分析一下何时两个大臣需要交换,

设第一个大臣左右手数字为(a1,b1),第二个大臣左手数字为(a2,b2),第一个大臣前面的所有大臣和国王左手数字的乘积为x

那么交换前第一个大臣的奖赏为x/b1,第二个为(x*a1)/b2

交换后第一个大臣为x/b2,第二个为(x*a2)/b1

为了好看,我们设c1=x/b1,c2=(x*a1)/b2,c3=x/b2,c4=(x*a2)/b1

因为a>=1,b>=1,所以c4>=c1,c2>=c3

比较c2与c4的大小,可以发现,当a1/b2<a2/b1时,即a1*b1<a2*b2时,不交换是更优的,所以我们应该让a*b小的排在前面

记得写高精度!!!!!!!!!!!!!!!!!

#include<iostream>  
#include<cstdio>  
#include<cstring>  
#include<algorithm>  
using namespace std;  
struct HP{  
    int s,n[10001];  
}aa[1001];  
struct xxx{  
    int a,b;  
}x[1001];  
bool cmp(xxx a,xxx b)  
{  
    return a.a*a.b<b.a*b.b;  
}  
void Max(HP a,HP b,HP &c)  
{  
    memset(c.n,0,sizeof(c.n));  
    if(a.s>b.s) c=a;  
    if(a.s<b.s) c=b;  
    if(a.s==b.s)  
    {  
        for(int i=a.s;i>=1;i--)  
        {  
            if(a.n[i]>b.n[i]){c=a;break;}  
            if(a.n[i]<b.n[i]){c=b;break;}  
        }  
        if(c.s==0)c=a;  
    }  
}  
void cheng(HP a,HP b,HP &c)  
{  
    memset(c.n,0,sizeof(c.n));  
    for(int i=1;i<=a.s;i++)  
    for(int j=1;j<=b.s;j++)  
    {  
        c.n[i+j-1]+=(a.n[i]*b.n[j])%10000;  
        c.n[i+j]+=(a.n[i]*b.n[j])/10000;  
    }  
    if(c.n[a.s+b.s])c.s=a.s+b.s;  
    else c.s=a.s+b.s-1;  
}  
void chu(HP a,int b,HP &c)  
{  
    memset(c.n,0,sizeof(c.n));  
    int now=0;  
    for(int i=a.s;i>=1;i--)  
    {  
        now+=a.n[i];c.n[i]=now/b;now=now%b*10000;  
    }  
    if(c.n[a.s])c.s=a.s;else c.s=a.s-1;  
}  
int main()  
{  
    int n;scanf("%d",&n);for(int i=0;i<=n;i++)scanf("%d%d",&x[i].a,&x[i].b);  
    sort(x+1,x+n+1,cmp);  
    for(int i=0;i<=n;i++)aa[i].n[1]=x[i].a,aa[i].s=1;  
    HP MAX,tot=aa[0];chu(tot,x[1].b,MAX);cheng(tot,aa[1],tot);  
    for(int i=2;i<=n;i++)  
    {  
        HP v;chu(tot,x[i].b,v);Max(v,MAX,MAX);cheng(tot,aa[i],tot);  
    }  
    int d[5],dd=0;  
    while(MAX.n[MAX.s])  
    {  
        d[++dd]=MAX.n[MAX.s]%10;MAX.n[MAX.s]/=10;  
    }  
    for(;dd;dd--)printf("%d",d[dd]);  
    for(int i=MAX.s-1;i>=1;i--)printf("%04d",MAX.n[i]);  
    return 0;  
}  
View Code

 D1T3.开车旅行

双向链表+倍增

先用双向链表求出每个点的小A和小B能到达的点,具体做法如下:

先排序,然后按数据在原数组的位置遍历,寻找每个数左边,左边的左边,右边,右边的右边的最小值和次小值

找完一个数,删一个数。把每个点小A开一次,小B开一次,算作一次

fa[i][j]表示从i点开2^j次到达的点,aa[i][j]表示从i点开2^j次小A开的距离,bb[i][j]表示从i点开2^j次小B开的距离

fa[i][j]=fa[fa[i][j-1]][j-1];
aa[i][j]=aa[i][j-1]+aa[fa[i][j-1]][j-1];
bb[i][j]=bb[i][j-1]+bb[fa[i][j-1]][j-1];

为了避免比值是double,所以a:b<c:d当ad<cb时,成立(c必须不等于0)

最后,如果ansb一直是0的话,应输出海拔最高的

#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;
int a[100011],b[100011],n,m,x[10011],s[10011],bh[100011];
int fa[100011][20];long long aa[100011][20],bb[100011][20];
struct xxx{
    int h,lnext,rnext,no;
}data[100011];
bool cmp(xxx a,xxx b){return a.h<b.h;}
void gx(int _h,int _no,int h,int i,int &Min)
{
    if(_h<=Min)
    {
        if(_h==Min)a[i]=_no;
        else {a[i]=b[i];Min=_h;b[i]=_no;}
    }
    else if(abs(h-data[bh[a[i]]].h)>_h)a[i]=_no;
}
void ab()
{
    data[0].lnext=0;data[0].rnext=0;data[0].h=2000000003;
    for(int i=1;i<=n;i++)data[i].lnext=i-1,data[i].rnext=i+1;
    data[n].rnext=0;
    for(int i=1;i<=n;i++)
    {
        int Min=1000000001;
        int no=data[bh[i]].no,h=data[bh[i]].h,l=data[bh[i]].lnext,r=data[bh[i]].rnext;
        int l2h,l1h,r1h,r2h,l2no,l1no,r1no,r2no,l2,l1,r1,r2;
        l1=data[l].lnext;l2=data[l1].lnext;
        r1=data[r].rnext;r2=data[r1].rnext;
        l1h=data[l].h;l2h=data[l1].h;
        r1h=data[r].h;r2h=data[r1].h;
        l1no=data[l].no;l2no=data[l1].no;
        r1no=data[r].no;r2no=data[r1].no;
        if(l1)gx(h-l2h,l2no,h,i,Min);
        if(l)gx(h-l1h,l1no,h,i,Min);
        if(r)gx(r1h-h,r1no,h,i,Min);
        if(r1)gx(r2h-h,r2no,h,i,Min);
        if(l)data[l].rnext=r;if(r)data[r].lnext=l;
    }
}
void st()
{
    for(int i=1;i<=n;i++)
    {
        if(a[i])aa[i][0]=abs(data[bh[i]].h-data[bh[a[i]]].h);
        if(b[a[i]])bb[i][0]=abs(data[bh[a[i]]].h-data[bh[b[a[i]]]].h);
        fa[i][0]=b[a[i]];
     } 
    for(int j=1;j<=16;j++)
    for(int i=1;i<=n;i++)
    {
        fa[i][j]=fa[fa[i][j-1]][j-1];
        aa[i][j]=aa[i][j-1]+aa[fa[i][j-1]][j-1];
        bb[i][j]=bb[i][j-1]+bb[fa[i][j-1]][j-1];
    }
}
void Ans(int u,int x,long long &ansa,long long &ansb)
{
    for(int i=16;i>=0;i--)
    {
        if(fa[u][i]&&aa[u][i]+bb[u][i]<=x)
        {
            ansa+=aa[u][i],ansb+=bb[u][i];
            x=x-aa[u][i]-bb[u][i];u=fa[u][i];
        }
    }
    if(aa[u][0]<=x)ansa+=aa[u][0];
}
int main()
{
    scanf("%d",&n);for(int i=1;i<=n;i++){scanf("%d",&data[i].h);data[i].no=i;}
    sort(data+1,data+n+1,cmp);for(int i=1;i<=n;i++)bh[data[i].no]=i;
    ab();st();int x;
    scanf("%d%d",&x,&m);
    long long ans=data[n].no,ansa=0ll,ansb=0ll;Ans(data[n].no,x,ansa,ansb);
    for(int i=n-1;i>=1;i--)
    {
        int u=data[i].no;
        long long aa=0ll,bb=0ll;Ans(u,x,aa,bb);
        if(ansa==0||aa*ansb<bb*ansa)ansa=aa,ansb=bb,ans=u;
    }
    if(ansb==0)ans=data[n].no;
    printf("%lld
",ans);
    for(int i=1;i<=m;i++)
    {
        int s,x;scanf("%d%d",&s,&x);
        long long aa=0ll,bb=0ll;Ans(s,x,aa,bb);
        printf("%lld %lld
",aa,bb);
    }
    return 0;
}
View Code

 D2T1.同余方程

ax ≡ 1 (mod b)意味着ax-by==1,求x

exgcd,ax-by==1可以看成ax+b(-y)==1

#include<iostream> 
#include<cmath> 
using namespace std; 
int x,y; 
void gcd(int a,int b) 
{ 
    if(b==0){x=1;y=0;} 
    else
    { 
        gcd(b,a%b); 
        int t=x;x=y;y=t-a/b*y; 
    } 
} 
int main() 
{ 
    int a,b; 
    cin>>a>>b; 
    gcd(a,b); 
    x=(x%b+b)%b; 
    cout<<x; 
    return 0; 
} 
View Code

D2T2.借教室

二分+差分

由于剩余的教室是非递增的,所以我们可以二分答案

差分数组cf[]来差分r[]数组,cf[i]=r[i]-r[i-1]

每次租借s到t的d间教室,只要cf[s]-=d;cf[t+1]+=d;即可

然后还原原数组,如果剩余教室个数<0,就借不到这份订单,反之可以

#include<iostream>  
#include<cstdio>  
#include<cstring>  
using namespace std;  
int cf[1000002],n,m,d[1000001],s[1000001],t[1000001],r[1000001];  
bool check(int ans)  
{  
    for(int i=1;i<=n+1;i++) 
    { 
        cf[i]=r[i]-r[i-1]; 
    } 
    for(int i=1;i<=ans;i++)  
    {  
        cf[s[i]]-=d[i];cf[t[i]+1]+=d[i];  
    }  
    int last=0;  
    for(int i=n;i>=1;i--)  
    {  
        int x=last-cf[i+1];last=x;if(x<0)return false;  
    } 
    return true;  
}  
int main()  
{  
    scanf("%d%d",&n,&m);for(int i=1;i<=n;i++)scanf("%d",&r[i]);  
    for(int i=1;i<=m;i++)scanf("%d%d%d",&d[i],&s[i],&t[i]);  
    int l=1,r=m+1,ans=0;  
    while(l<r){  
        int mid=(l+r)/2;  
        if(check(mid))l=mid+1;  
        else ans=mid,r=mid;  
    }  
    if(ans==0)cout<<0;  
    else cout<<-1<<endl<<ans;  
    return 0;  
}  
View Code
原文地址:https://www.cnblogs.com/lher/p/6841786.html