Codeforces Round #632 (Div. 2) C(计数题)(D 构造题) E(思维+构造) F(数学推导题)

题目链接

C. Eugene and an array

题意:给你n长度的序列,问你有多少个子序列(下标是连续的)是 好 的子序列

一个好的子序列定位:该序列中的子序列(下标不连续)没有和为0的。

做法:总的减去不合法的。

不合法的求法:l表示左边的边界。用前缀和得到当前左坐标  L 和  右坐标 R 区间内和为零。 左坐标的左边和右做坐标的右边都是可以组合一下 都是不合法的。

此时我们把边界l移到L+1 的位置 。因为如果下次   你又找到一个新的L1  R1   L<=L1的话  就会有重复计算的。

#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=(b);++i)
#define mem(a,x) memset(a,x,sizeof(a))
#define pb push_back
#define pi pair<int, int>
#define mk make_pair
using namespace std;
typedef long long ll;
ll gcd(ll a,ll b) { return b?gcd(b,a%b):a;}
const int N=2e5+10;
ll n;
ll a[N],pre[N];
map<ll,ll>mp;
int main()
{

    scanf("%lld",&n);
    rep(i,1,n) scanf("%lld",&a[i]);
    ll ans=n*(n+1)/2;

    ll sum=0;
    mp[0]=1;
    int l=1;
    rep(i,1,n)
    {
        pre[i]=pre[i-1]+a[i];
        if(mp[pre[i]]>=l){
            sum+=(n-i+1)*(mp[pre[i]]-l+1);
            l=mp[pre[i]]+1;
        }
        mp[pre[i]]=i+1;
    }
    printf("%lld\n",ans-sum);


}

D. Challenges in school №41

读了半天发现还是读错了。

参考题意和题解链接:

题解:

代码:

#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=(b);++i)
#define mem(a,x) memset(a,x,sizeof(a))
#define pb push_back
#define pi pair<int, int>
#define mk make_pair
using namespace std;
typedef long long ll;
ll gcd(ll a,ll b) { return b?gcd(b,a%b):a;}
const int N=3e3+10;
char s[N];
vector<int>v[3000100];
int n,k,mink,maxk;
void print()
{
    int p1=0,p2=0;
    int kk=mink;
    while(kk<k){
        //kk++;
        printf("1 %d\n",v[p1][p2]);
        p2++;
        if(p2==v[p1].size()) p2=0,p1++;
        else kk++;
    }


    printf("%d ",v[p1].size()-p2);
    for(int i=p2;i<v[p1].size();++i) printf("%d ",v[p1][i]);puts("");


    for(int i=p1+1;i<mink;++i){
        printf("%d ",v[i].size());
        for(int val:v[i]) printf("%d ",val);
        puts("");
    }
}
void run(int id)
{
    for(int i=1;i<n;++i){
        if(s[i]=='R'&&s[i+1]=='L')
        {
            swap(s[i],s[i+1]);
            v[id].push_back(i);
            ++i;
        }
    }
}
int main()
{
    cin>>n>>k>>s+1;
    run(0);
    while(v[mink].size()&&mink<=k)
    {
        maxk+=v[mink].size();
        mink++;
        run(mink);
    }
    //printf("mink:%d maxk:%d k:%d\n",mink,maxk,k);

    if(mink<=k&&k<=maxk) print();
    else puts("-1");
}

E. Road to 1600

题意有点难懂

题意和解法参考:博客

题意:

n*n的棋盘上有1~n^2 的数,且互不相同,一开始车和皇后都在数字为1的单元上(单独的一张棋盘上移动),且数字为1的这个单元被访问过了,车和皇后棋子 有以下移动规则:

 

1.当前棋子前往 所有可以移动但尚未被访问的单元中,数值最小的单元;

(每次只能走当前能到达的 且 之前 未到达的  格子中 数值最小的)

2.所有可移动的单元都被访问过了,则被传送到数值最少的未被访问的单元,执行该步需要1的花费;

3.所有单元格都要被访问。

要求找到n*n的棋盘,满足车的花费严格小于皇后的花费(不能都不花费),如果不存在则打印 -1

解法:大佬的解法真妙啊

 

#include<bits/stdc++.h>
using namespace std;
const int N=5e2+10;
int a[N][N],n;

map<int,pair<int,int> >vis;
int main()
{
    vis[1]=make_pair(3,2);
    vis[2]=make_pair(4,4);
    vis[3]=make_pair(1,2);
    vis[4]=make_pair(1,1);
    vis[5]=make_pair(2,2);
    vis[6]=make_pair(1,3);
    vis[7]=make_pair(2,1);
    vis[8]=make_pair(4,2);
    vis[9]=make_pair(2,3);
    vis[10]=make_pair(3,4);
    vis[11]=make_pair(3,3);
    vis[12]=make_pair(1,4);
    vis[13]=make_pair(4,1);
    vis[14]=make_pair(3,1);
    vis[15]=make_pair(2,4);
    vis[16]=make_pair(4,3);
    cin>>n;
    if(n<=2){
        puts("-1");
        return 0;
    }
    if(n==3){
        printf("%d %d %d\n",1,7,9);
        printf("%d %d %d\n",3,2,5);
        printf("%d %d %d\n",4,8,6);
        return 0;
    }
    else if(n==4){
        int now=0;
        for(int i=1;i<=16;++i) a[vis[i].first][vis[i].second]=++now;
        for(int i=1;i<=n;++i){
            for(int j=1;j<=n;++j) printf("%d ",a[i][j]);
            puts("");
        }
        return 0;
    }
    int now=0;
    if(n%2==0){
        for(int i=1;i<=n-4;++i){
            int x=i,y;
            if(i%2){y=1;while(y<=n) a[x][y++]=++now;}
            else{y=n;while(y>=1) a[x][y--]=++now;}
        }

    }
    else{
        for(int i=1;i<=n-4;++i){
            int x=i,y;
            if(i%2){y=n;while(y>=1)a[x][y--]=++now;}
            else{y=1;while(y<=n)a[x][y++]=++now;}
        }
    }
    int x=n-3;
    for(int i=1;i<=n-4;++i) a[x][i]=++now;
    x=n-2;
    for(int i=n-4;i>=1;--i) a[x][i]=++now;

    x=n-1;
    for(int i=1;i<=n-4-1;++i) a[x][i]=++now;
    x=n;
    for(int i=n-4-1;i>=1;--i) a[x][i]=++now;
    a[n][n-4]=++now;
    a[n-1][n-4]=++now;


    int base=n-4;
    for(int i=1;i<=16;++i) a[base+vis[i].first][base+vis[i].second]=++now;
    for(int i=1;i<=n;++i){
        for(int j=1;j<=n;++j) printf("%d ",a[i][j]);
        puts("");
    }

}

 

F. Kate and imperfection

题意:给你一个n   要你构造k长度的序列,序列从 1  到 n中取k个  。定义 这个序列中任意两个数的gcd的最大值为这个序列的等级。

现在要你求k  从2取到n  且每个序列的等级最小。

我的做法比较迷:2到n 每个数除以它的最小素因子排个序,输出这几个数就好了,推了几个样例得到的

有一个不错的解法:

#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=(b);++i)
#define mem(a,x) memset(a,x,sizeof(a))
#define pb push_back
#define pi pair<int, int>
#define mk make_pair
using namespace std;
typedef long long ll;
ll gcd(ll a,ll b) { return b?gcd(b,a%b):a;}
const int N=5e5+10;
int ans[N],v[N];

bool isPrime[N];
int prime[N];//保存素数
int num_prime;//记录素数个数
void init()
{
	num_prime=0;
	for(int i=2;i<N;i++)
	{
		if(isPrime[i]==0)	prime[num_prime++]=i,v[i]=i;

		for(int j=0;j<num_prime&&i*prime[j]<N;j++)
		{
			isPrime[i*prime[j]]=1;
			v[i*prime[j]]=prime[j];
			if(i%prime[j]==0)  break;
		}
	}

}
void cal()
{
    for(int i=0;i<num_prime;++i)
    {
        for(int j=1;j*prime[i]<N;++j){
            if(v[j*prime[i]]==0) v[j*prime[i]]=prime[i];
        }
    }
}
int main()
{
    init();
    //cal();

	int n;
	cin>>n;
	ans[1]=1;
	for(int i=2;i<=n;++i) ans[i]=i/v[i];
	sort(ans+1,ans+1+n);
	for(int i=2;i<=n;++i) printf("%d ",ans[i]);
	return 0;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

长沙大学ccsu_deer

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值