牛客 7.13 月赛(留 C逆元 Ddp)

B-最少剩几个?_牛客小白月赛98 (nowcoder.com)

思路

奇数+偶数 = 奇数;奇数*偶数 = 奇数

所以在既有奇数又有偶数时,两者结合可以同时删除

先分别统计奇数,偶数个数

若偶个数大于奇个数,答案是偶个数-奇个数

若奇个数大于偶个数,奇数个数减去偶个数再对2取模

ac代码
复制代码
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
#define IOS ios::sync_with_stdio(0),cin.tie(0),cout.tie(0)

int main()
{
    IOS;
    int n,ans;
		int ji=0,ou=0;
		cin>>n;
		vector<ll>a(n);
		for(int i=0;i<n;i++) 
		{
			cin>>a[i];
			if(a[i] & 1) ji++;
			else ou++; 
		}
		if(ou>ji)  ans=ou-ji;
		else{
			if((ji-ou)%2) ans=1;
			else ans=0;
		}

		cout<<ans<<endl;   
		return 0;
}

C-两个函数_牛客小白月赛98 (nowcoder.com)

(超时问题如何解决)

初始代码(超时)
复制代码
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
#define IOS ios::sync_with_stdio(0),cin.tie(0),cout.tie(0)

int main()
{
    IOS;
    ll q,a,x;
		cin>>q;
		for(int i=0;i<q;i++)
		{
			ll ans=0;
			cin>>a>>x;
			if(x==1) ans=(a*x)%998244353;
			else{
				for(int i=1;i<x;i++)
					{
						ans+=(a*(a%998244353*i)%998244353)%998244353;
						ans%=998244353;					
					}
			}
			ans%=998244353;
			cout<<ans<<endl;
		}
		return 0;
}
解决思路
  1. 快速幂

时间复杂度是 O(log n),相比于直接进行指数运算,大大提高了计算效率

快速幂代码

复制代码
int FastPow(int a,int x,int mod)
{
	int ans = 1;
	a%=mod;
	while(x)
	{
		if(x&1) ans=(ans*a)%mod;
		a= (a*a)%mod;
		x>>=1;
	}
	return ans;
}
  1. 递推式

因为是求和过程,可以用递推式

ac代码
复制代码
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
#define IOS ios::sync_with_stdio(0),cin.tie(0),cout.tie(0)
ll mod=998244353;

int main()
{
    IOS;
    ll q,a,x;
		cin>>q;
		for(int i=0;i<q;i++)
		{
			ll ans=0;
			cin>>a>>x;
			if(x==1) ans=a%mod;
            else 
            {
                a = a*a %mod;
                ans=(x-1)*x/2 %mod *a %mod;
            }
			cout<<ans<<endl;
		}
		return 0;
}

D-切割 01 串 2.0_牛客小白月赛98 (nowcoder.com)

思路:
  1. 前缀和

记录在索引的每一个位置处之前,0或1的个数

2.dp

dp[i][j] 表示考虑前 i 个字符时,最多可以进行多少次切割;对于每个长度 len,遍历所有可能的切割起点 l,使得 l + len - 1 不超过序列的长度;对于每个起点 l,计算可能的切割终点 r;

对于每个起点 l 和终点 r,遍历所有可能的切割分割点 k,使得 klr 之间;

动态规划过程的关键在于,通过递归地考虑所有可能的切割方式,并使用前缀和数组来快速计算分割点 k 两侧的子串中0和1的累计数量。通过这种方式,算法能够高效地找到满足条件的切割次数的最大值

代码
复制代码
#include<bits/stdc++.h>
#define IOS ios::sync_with_stdio(false);cin.tie(0);cout.tie(0)

using namespace std;

/*
算法:区间DP + 前缀和 O(N*N*N)
数据结构:s0,s1前缀和数组 + 二维dp[l][r]
*/

const int N = 510;

int s0[N], s1[N];
int dp[N][N];

void solve() {
    int n, L, R;
    string s;
    cin >> n >> L >> R >> s;
    
    s = " " + s;
    for (int i = 1; i <= n; i ++ )
        s0[i] = s0[i - 1] + (s[i] == '0'),
        s1[i] = s1[i - 1] + (s[i] == '1');
    
    for (int len = 1; len <= n; len ++ )
        for (int l = 1; l <= n; l ++ )
        {
            int r = l + len - 1;
            if (r > n) break;
            for (int k = l; k < r; k ++ )
            {
                int c0 = s0[k] - s0[l - 1];
                int c1 = s1[r] - s1[k];
                if (L <= abs(c0 - c1) && abs(c0 - c1) <= R) 
                    dp[l][r] = max(dp[l][r], 1 + dp[l][k] + dp[k + 1][r]);
            }
        }
    
    cout << dp[1][n] << '\n';
}

signed main() {
    IOS;
    int t = 1;
//    cin >> t;
    while (t--) {
        solve();
    }
    return 0;
}
相关推荐
爱炼丹的James1 分钟前
第三章 搜索和图论
数据结构·算法·图论
菜菜笔记2 分钟前
【无标题】
算法
咩咦6 分钟前
C++学习笔记21:日期类加减天数
c++·学习笔记·运算符重载·日期类·operator+·operator+=
努力努力再努力wz7 分钟前
【QT入门系列】QWidget 六大常用属性详解:windowOpacity、cursor、font、focus、toolTip 与 styleSheet
android·开发语言·数据结构·c++·qt·mysql·算法
少司府9 分钟前
Tools相关:深入浅出学Git
大数据·c++·git·gitee·github·仓库·分支
神仙别闹10 分钟前
基于MFC(C++)实现(界面)学委作业管理系统
开发语言·c++·mfc
艾莉丝努力练剑13 分钟前
【Linux网络】Linux 网络编程:HTTP(三)HTTP 协议原理
linux·运维·服务器·网络·c++·http
Gauss松鼠会14 分钟前
GaussDB(DWS) 资源监控Topsql
java·网络·数据库·算法·oracle·性能优化·gaussdb
夏日听雨眠15 分钟前
数据结构(快速排序)
java·数据结构·算法
薇茗16 分钟前
【初阶数据结构】 升沉有序的平仄 排序 3
c语言·开发语言·数据结构·算法·排序算法·文件归并排序