自学内容网 自学内容网

2024 China Collegiate Programming Contest (CCPC) Zhengzhou Onsite 基础题题解

L. Z-order Curve

 思路:这题目说了,上面那一行,只有在偶数位才有可能存在1,那么一定存在这样的数,0 ,1,100, 10000,那么反之,我们的数列是行的二倍,因此会出现10,1000,100000这样的数,因此,就可以发现,其实组成的数也是由二进制数递推的,因此我们可以从高位到低位逐步去找,如果相同且为1就变成0,如果不同就直接结束,输出L即可

#include<bits/stdc++.h>
using namespace std;
#define int long long
int t;
int l,r;
void solve()
{
cin>>l>>r;
for(int i=61;i>=0;i--)
{
int bitl=(l>>i)&1;
int bitr=(r>>i)&1;
if(bitl==bitr&&bitl==1)
{
l-=(1LL<<i);
r-=(1LL<<i);
}
else if(bitl!=bitr)
{
cout<<l<<"\n";
return ;
}
}
}
signed main()
{
cin>>t;
while(t--)
{
solve();
}
return 0;
}

F. Infinite Loop

 思路:这题有一个比较恶心的地方,就是说从1小时开始,实际上是从0小时开始计算,然后我们去计算公差是多少,我们现将所有的bi加在一起为sum,然后取sum和k的较大值作为公差,然后去对题目进行分析,我们会发现,从第二天开始,就去进行等差数列了,因此我们只需要计算出来第一天和第二天在什么时候完成即可,还有一个特判就是要对小时特判,如果小时是0,那么天数-1,小时+k

#include<bits/stdc++.h>
using namespace std;
#define int long long
int n,k,q;
int a[200005];
int b[200005];
int ans1[200005];
int ans2[200005];
int d;
int flag,x;
signed main()
{
cin>>n>>k>>q;
for(int i=1;i<=n;i++)
{
cin>>a[i]>>b[i];
a[i]-=1;
d+=b[i];
}
d=max(d,k);
int t=0;
for(int i=1;i<=n;i++)
{
ans1[i]=max(t,a[i])+b[i];
t=ans1[i];
}
for(int i=1;i<=n;i++)
{
a[i]+=k;
}
for(int i=1;i<=n;i++)
{
ans2[i]=max(t,a[i])+b[i];
t=ans2[i];
}
for(int i=1;i<=q;i++)
{
cin>>flag>>x;
if(flag==1)
{
int day=ans1[x]/k;
int hour=ans1[x]%k; 
if(hour==0)
{
day-=1;
hour+=k;
}
cout<<day+1<<" "<<hour<<"\n";
}
else
{
int time=ans2[x]+(flag-2)*d;
int day=time/k;
int hour=time%k; 
if(hour==0)
{
day-=1;
hour+=k;
}
cout<<day+1<<" "<<hour<<"\n";
}
}
return 0;
}

B. Rolling Stones

思路:很板的一个广搜,只需要找到翻转之后,每个面上面是什么就可以了,同时要确保翻转的时候翻转过去的面,等于那个底面上的值

#include<bits/stdc++.h>
using namespace std;
#define int long long
int n;
int a[205][205];
struct node{
int x,y;
int qian;
int zuo;
int you;
int di;
int tmp;
};
deque<node> q;
int vis[205][205];
int ans[205][205];
int fx,fy;
void bfs()
{
while(!q.empty())
{
node test=q.front();
q.pop_front();
int x=test.x;
int y=test.y;
//cout<<x<<" "<<y<<"\n";
//cout<<test.qian<<" "<<test.zuo<<" "<<test.you<<" "<<test.di<<"\n";
if(y%2==0)
{
if(vis[x][y-1]==0&&y-1>=1&&y-1<=2*x-1&&a[x][y-1]==test.zuo)//向左移动
{
vis[x][y-1]=1;
q.push_back((node){x,y-1,test.you,test.qian,test.di,test.zuo,test.tmp+1});
ans[x][y-1]=test.tmp+1;
} 
if(vis[x][y+1]==0&&y+1>=1&&y+1<=2*x-1&&a[x][y+1]==test.you)//向右移动
{
vis[x][y+1]=1;
q.push_back((node){x,y+1,test.zuo,test.di,test.qian,test.you,test.tmp+1});
ans[x][y+1]=test.tmp+1;
} 
if(vis[x-1][y-1]==0&&y-1>=1&&y-1<=2*(x-1)-1&&x-1>=1&&x-1<=n&&a[x-1][y-1]==test.qian)//向上移动
{
vis[x-1][y-1]=1;
q.push_back((node){x-1,y-1,test.di,test.zuo,test.you,test.qian,test.tmp+1});
ans[x-1][y-1]=test.tmp+1;
}
}
else
{
if(vis[x][y-1]==0&&y-1>=1&&y-1<=2*x-1&&a[x][y-1]==test.zuo)//向左移动
{
vis[x][y-1]=1;
q.push_back((node){x,y-1,test.you,test.qian,test.di,test.zuo,test.tmp+1});
ans[x][y-1]=test.tmp+1;
} 
if(vis[x][y+1]==0&&y+1>=1&&y+1<=2*x-1&&a[x][y+1]==test.you)//向右移动
{
vis[x][y+1]=1;
q.push_back((node){x,y+1,test.zuo,test.di,test.qian,test.you,test.tmp+1});
ans[x][y+1]=test.tmp+1;
} 
if(vis[x+1][y+1]==0&&y+1>=1&&y+1<=2*(x+1)-1&&x+1>=1&&x+1<=n&&a[x+1][y+1]==test.qian)//向下移动
{
vis[x+1][y+1]=1;
q.push_back((node){x+1,y+1,test.di,test.zuo,test.you,test.qian,test.tmp+1});
ans[x+1][y+1]=test.tmp+1;
}
}
}
}
signed main()
{
cin>>n;
for(int i=1;i<=n;i++)
{
for(int j=1;j<=2*i-1;j++)
{
cin>>a[i][j];
}
}
cin>>fx>>fy;
if(fx==1&&fy==1)
{
cout<<0<<"\n";
return 0;
}
q.push_back((node){1,1,2,1,3,4,0});
vis[1][1]=1;
bfs();
if(ans[fx][fy]==0)
{
cout<<"-1\n";
}
else
{
cout<<ans[fx][fy]<<"\n";
}
return 0;
}

 M. Rejection Sampling

 思路:这题一开始看起来其实是有点儿乱的,不知道在说什么,而且也不知道到底要操作什么,但是仔细阅读后会发现,那个S的概率就是C(n,k)*pi^k+(1-pi)^(n-k),若想要满足题目中的S与ai乘正比的话,我们需要满足pi/(1-pi)与ai成正比,我们可以将比例系数C设为c,因此我们可以得到式子

pi/(1-pi)=c*ai;

可以得到pi=c*ai/(1+c*ai),可知,pi关于c单调递增

c=pi/((1-pi)*ai),我们可以去二分c然后去判断pi的和是否是k

然后就解决了

#include <bits/stdc++.h>
using namespace std;
#define int long long
int n, k;
long double a[100005];  
bool check(long double c) 
{
    long double ans=0;
    long double x;
    for(int i=1;i<=n;i++) 
{
        x =(c*a[i])/(1.00+c*a[i]);
        ans+=x;
    }
    return ans<=k;
}
signed main() 
{
    cin>>n>>k;
    for (int i=1;i<=n;i++) 
{
        cin>>a[i];
    }
    long double l=0.0;
    long double r=1e15;
    for (int i=1;i<=200;i++) 
{
        long double mid=(l+r)/2;
        if (check(mid)) 
{
            l=mid;
        } 
else 
{
            r=mid;
        }
    }
    cout<<fixed<<setprecision(10);
    for (int i = 1;i<=n;i++) 
{
        cout<<(long double)(l*a[i])/(1.00+l*a[i])<<"\n";
    }
    return 0;
}

C. Middle Point

思路:我们通过对题目的分析,发现每次都是通过用S集合内的元素,去折中创造新的结点,我们对其拆分可以发现,可以将一个S集合拆分为两个集合,在引用结论之前,先来写一个定式

(u+v)/2=(U+V)/2(u属于U集合,v属于V集合)

因此我们也可以将S集合拆分成两个集合一个是S'集合一个是S集合,我们的

S‘集合由原来的S'集合和S集合累加再除2得到

因此我们可以通过递归来不断去判断原来的S‘集合中选的数是什么,然后不断去更新,知道S’集合中的数等于S集合中的数就停止

用一个map去存储S集合中出现的数,如果重复出现就立刻输出-1,exit(0)退出程序,

当然了需要加上一个特判,如果一上来的数就是S集合的数,那么就直接输出0

否则就存进一个栈,然后因为是递归,倒着推,所以栈刚好可以顺序输出

#include <bits/stdc++.h>  
using namespace std;  
#define int long long  
int A, B;  
int x, y;  
struct node  
{  
    int x1, y1, x2, y2;  
};   
void solve(int x, int y)  
{  
    node p;  
    if (x <= A / 2)  
    {  
        p.x1 = 2 * x;  
        p.x2 = 0;  
    }  
    else  
    {  
        p.x1 = 2 * x - A;  
        p.x2 = A;  
    }  
    if (y <= B / 2)  
    {  
        p.y1 = 2 * y;  
        p.y2 = 0;  
    }  
    else  
    {  
        p.y1 = 2 * y - B;  
        p.y2 = B;  
    }  
        s.push(p);  
        if(p.x1==A&&p.y1==B)
        return;
        if(p.x1==A&&p.y1==0)
        return;
        if(p.x1==0&&p.y1==B)
        return;
        if(p.x1==0&&p.y1==0)
        return;
        if(s.size()<=50)
        {
        solve(p.x1, p.y1);  
} 
        else
        {
        cout<<-1;
        exit(0);
}
   
    return;  
}  

signed main()  
{  
    cin >> A >> B;  
    cin >> x >> y;  
    if(x==A&&y==B)
    {
    cout<<0<<"\n";
    return 0;
}
    if(x==A&&y==0)
    {
    cout<<0<<"\n";
    return 0;
}
    if(x==0&&y==B)
    {
    cout<<0<<"\n";
    return 0;
}
    if(x==0&&y==0)
    {
    cout<<0<<"\n";
    return 0;
}  
    solve(x, y);  
    cout << s.size() << "\n";  
    while (!s.empty())  
    {  
        node p = s.top();  
        s.pop();  
        cout << p.x1 << " " << p.y1 << " " << p.x2 << " " << p.y2 << "\n";  
    }  
    return 0; 
}

 


原文地址:https://blog.csdn.net/2301_80475191/article/details/145063603

免责声明:本站文章内容转载自网络资源,如侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!