蓝桥杯23年第十四届省赛真题-填充|DFS,贪心

题目链接:

1.填充 - 蓝桥云课 (lanqiao.cn)

蓝桥杯2023年第十四届省赛真题-填充 - C语言网 (dotcpp.com)

说明: 

dfs就不再多说了,对于每个?都有0和1两个分支,数据范围是:

那么有m个 ?,时间复杂度就是 O(2^{m}),会超时。蓝桥杯官网可以过35%的数据,暴力先拿到一部分再说。

这里需要注意的是:对于?的字符的每层,最后要把它复原,恢复为?

DFS代码:

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N=1e6+10;
string s;
int mx=0;
int len;

//参数含义:当前搜索层数k,搜索到第k个字符(从0开始),cn:前面是否有一个单个的字符可以出现00或11串  
//num当前找到的不重叠的00或11子串 数量 
void dfs(int k,int cn,int num){
	
	//剩下所有字符都可以组成合法子串加上当前数量都比已经找到的数量小,不再搜索 
	if(num+(len-k)/2<mx) return;
	
	//k个字符都已经搜索过,递归出口 
	if(k==len){
		if(num>mx){
			mx=num;
		}
		
		return;
	}
	
	if(s[k]!='?'){
		
		//同时满足 前面是否有一个单个的字符,两个字符相同才计数 ,单个字符数计为0 
		if(cn==1&&s[k]==s[k-1]){
	 		dfs(k+1,0,num+1);
		 }
		else{
		//不满足,不计数,该字符是一个单一的字符 
			dfs(k+1,1,num);
		}
	}
	
	else{
		s[k]='0';
		if(cn==1&&s[k]==s[k-1])
		dfs(k+1,0,num+1);
		else
		dfs(k+1,1,num);
		
		
		s[k]='1';
		if(cn==1&&s[k]==s[k-1])
		dfs(k+1,0,num+1);
		else
		dfs(k+1,1,num);
		
	    //退出前 要还原成'?' 
	    s[k]='?'; 
		
	 }
} 

signed main() {
 
   ios::sync_with_stdio(0);
   cin.tie(0);
   cout.tie(0);
    
   cin>>s;
   len=s.size();
   int cnt=0;
   
   
   for(int i=0;i<len;i++){
   	
   	if(s[i]!='?'){
   		
   		//i为0时特别处理,防止i-1越界 
   		if(i==0){
   			cnt=1;
		   }
		else{
		   	if(cnt==1&&s[i]==s[i-1]){
		   		cnt=0;
		   		mx++;
			   }
			else{
			   cnt=1;	
		    }
		   }
   		
	   }
	   
	int n=mx;
   	if(s[i]=='?'){
   		
   		//i为0时特别处理,防止i-1越界 
   	if(i==0) 
	{
		s[i]='0';
		dfs(i+1,1,n);
		s[i]='1';
		dfs(i+1,1,n);
	}
   		
   	else{
   		dfs(i,cnt,n);
	   }	
   		
		   
		break;
	}
	   
	   
	   
   }

  cout<<mx;
  return 0;
}


/*
测试数据
01?1001???101?1?0000?0110????0?001?110?0?00?01?1??

答案:22 

*/




贪心:当前能组成一个合法子串的时候我就组合起来,因为如果我放弃这个组合,我最多让后面一个字符和它后面的字符组成一个合法子串,而浪费了我当前这个字符,组合更多的合法子串就是尽量利用每一个可以利用的字符。
再观察可得: 
 '00、11、0?、1?、?1、?0、??'都是正确的,01,10 不行 。可以组成合法子串时,这两个字符已经成对不用再看了,跳过这两个字符,不能成对,看下个字符,因为下个字符可能和他的下个字符成对,我们需要 利用每一个可以利用的字符成对 。

贪心代码

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N=1e6+10;

int len;
int mx=0;//最大值 

signed main() {
 
   ios::sync_with_stdio(0);
   cin.tie(0);
   cout.tie(0);
   string s;
   
   cin>>s;
   len=s.size();
   
   for(int i=0;i<len-1;){
   	//贪心,当前能组成一个合法子串的时候我就组合起来,因为如果我放弃这个组合
	//我最多让后面一个字符和它后面的字符组成一个合法子串,而浪费了我当前这个字符
	//组合更多的合法子串就是尽量利用每一个可以利用的字符
	//再观察可得: 
	//'00、11、0?、1?、?1、?0、??'都是正确的,01,10 不行 
	//可以组成合法子串时,这两个字符已经成对不用再看了,跳过这两个字符,不能成对,看下个字符,因为下个字符可能和他的下个字符成对
	//需要 利用每一个可以利用的字符成对 
	
	if(s[i]==s[i+1]||s[i]=='?'||s[i+1]=='?') {
		i+=2;
		mx++;
	}
   	else   i++;
   }

  cout<<mx;
  return 0;
}

 参考:蓝桥杯真题讲解:填充(贪心)-CSDN博客

的另一个版本 ,更好想更好梳理思路一些。

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N=1e6+10;

int len;
int mx=0;//最大值 
bool st[N];//标记是否被匹配过 
signed main() {
 
   ios::sync_with_stdio(0);
   cin.tie(0);
   cout.tie(0);
   string s;
   
   cin>>s;
   len=s.size();
   
//   for(int i=0;i<len-1;){
   	//贪心,当前能组成一个合法子串的时候我就组合起来,因为如果我放弃这个组合
	//我最多让后面一个字符和它后面的字符组成一个合法子串,而浪费了我当前这个字符
	//组合更多的合法子串就是尽量利用每一个可以利用的字符
	//再观察可得: 
	//'00、11、0?、1?、?1、?0、??'都是正确的,01,10 不行 
	//可以组成合法子串时,这两个字符已经成对不用再看了,跳过这两个字符,不能成对,看下个字符,因为下个字符可能和他的下个字符成对
	//需要 利用每一个可以利用的字符成对 
	
//	if(s[i]==s[i+1]||s[i]=='?'||s[i+1]=='?') {
//		i+=2;
//		mx++;
//	}
//   	else   i++;
//   }
  
  //另一种角度的贪心,每个字符优先跟前一个字符匹配,不能,再跟后面的字符匹配
   for(int i=0;i<len;i++){
   	
   	//因为是一个一个挪的 首先要判断 当前 字符是否已经匹配过 
   	if(s[i]!='?'&&!st[i]){
   	if(i-1>=0&&(s[i-1]==s[i])&&!st[i-1])
   	{
   		st[i]=true;
   		st[i-1]=true;
   		mx++;
	}
   	else if(i+1<len&&(s[i+1]==s[i]))
   	{
   		st[i]=true;
   		st[i+1]=true;
   		mx++;
	}
			
	}
   	
   	
   	//?可以匹配 任意落单的字符   
	else{
		if(st[i]) continue;
		
		if(i-1>=0&&!st[i-1])
   	{
   		s[i]=s[i-1]; 
   		st[i]=true;
   		st[i-1]=true;
   		mx++;
	}
   	else if(i+1<len)
   	{
   		s[i]=s[i+1]; 
   		st[i]=true;
   		st[i+1]=true;
   		mx++;
	}
		
		
	}
	
	
   }
   

  cout<<mx;
  return 0;
}


/*
测试数据
01?1001???101?1?0000?0110????0?001?110?0?00?01?1??

答案:22 

*/




本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/499874.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

某物登录表单加密

之前分析过某物h5的以及小程序的搜索接口,就是一个aes,秘钥不固定,表单里把秘钥以及密文一起发过去,服务器解密后再把数据加密返回,客户端解密展示到页面上. 这期是关于app的登录,密码登录 声明 本文章中所有内容仅供学习交流使用&#xff0c;不用于其他任何目的&#xff0c;…

人工智能:数字化转型中的关键角色

人工智能&#xff1a;数字化转型中的关键角色 数字化转型是十多年来商业变革的核心&#xff0c;它本身也在人工智能的崛起中发生转变。人工智能有望极大地改变数字化转型&#xff0c;从企业设定的目标到从业者用来推进计划的工具。本文将深入探讨人工智能在数字化转型中的扮演的…

Rust编程(四)PackageCrateModule

这一部分的中文教程/文档都很混乱,翻译也五花八门,所以我建议直接看英文官方文档,对于一些名词不要进行翻译,翻译只会让事情更混乱,本篇从实战和实际需求出发,讲解几个名称的关系。 Module & Crate & Package & Workspace 英文中的意思: Cargo:货物 Crate:…

google浏览器网站不安全与网站的连接不安全怎么办?

使用google谷歌浏览器访问某些网站打开时google谷歌浏览器提示网站不安全,与网站的连接不安全,您之所以会看到此警告,是因为该网站不支持https造成的怎么办? 目录 1、打开谷歌google浏览器点击右上角【┇】找到设置

实战 | YOLOv8自定义数据集训练实现手势识别 (标注+训练+预测 保姆级教程)

导 读 本文将手把手教你用YoloV8训练自己的数据集并实现手势识别。 安装环境 【1】安装torch, torchvision对应版本,这里先下载好,直接安装 pip install torch-1.13.1+cu116-cp38-cp38-win_amd64.whlpip install torchvision-0.14.1+cu116-cp38-cp38-win_amd64.whl 安装好…

【JavaEE】Thread类中run和start的区别

文章目录 先说结论Run方法Start方法 先说结论 当你想要创建一个新的线程并执行某些任务时&#xff0c;你应该重写run方法以提供任务的具体实现&#xff0c;并通过调用start方法来启动新线程 run方法包含了线程应该执行的代码&#xff0c;但直接调用它并不会启动新的线程。 s…

RabbitMQ镜像集群搭建

RabbitMQ镜像集群搭建 前言环境服务器程序包 一、修改主机名和host二、安装RabbitMQ三、集群配置四、其他配置五、登录查看 博主 默语带您 Go to New World. ✍ 个人主页—— 默语 的博客&#x1f466;&#x1f3fb; 《java 面试题大全》 《java 专栏》 &#x1f369;惟余辈才…

HTML input 实现回车切换到下一个输入框功能

前言 遇到需求&#xff0c;在客户填写单子时&#xff0c;有多个输入框&#xff0c;为了省事&#xff0c;不需要频繁移动光标填写。 实现效果 实现方式一 HTML <input type"text" name"serialNumber1" onkeydown"cursor(this);"/><in…

【Java多线程】4——特定场景解决办法

4 特定场景解决方法 ⭐⭐⭐⭐⭐⭐ Github主页&#x1f449;https://github.com/A-BigTree 笔记仓库&#x1f449;https://github.com/A-BigTree/tree-learning-notes 个人主页&#x1f449;https://www.abigtree.top ⭐⭐⭐⭐⭐⭐ 如果可以&#xff0c;麻烦各位看官顺手点个st…

Linux: 检测工具: monit: cpu占用率的一个例子

最近看到monit使用的CPU有时候会突然的增加,通过perf查看,可以看到是下面这个函数调用占用的比较高。 Overhead Comma Shared Object Symbol 29.72% monit [kernel.kallsyms] [k] __d_lookup__d_lookup29.65% d_lookup proc_fill_cache ` proc_readfd_common iterate_dir ksy…

FPGA芯片在通信基站中的作用

基站作为移动通信系统的核心组成部分&#xff0c;承担着信号的发送和接收任务&#xff0c;包括天线、射频前端、数字信号处理和控制等功能。 随着通信技术不断进步和网络容量的提升&#xff0c;基站功能日益复杂&#xff0c;数量也在增加。 与此同时&#xff0c;FPGA芯片被广…

JZ-7-201XMT跳位合位监视专用继电器 220VDC 板后接线,面板安装 JOSEF约瑟

系列型号&#xff1a; JZ-7Y-201XMT跳位合位监视继电器&#xff1b; JZ-7J-201XMT跳位合位监视继电器&#xff1b; JZ-7Y-203XMT跳位合位监视继电器&#xff1b; JZ-7J-203XMT跳位合位监视继电器&#xff1b; JZ-7Y-204XMT跳位合位监视继电器&#xff1b; JZ-7J-204XMT跳…

上采样技术在语义分割中的应用

目录 概要 一、概述 二、实现方法 1.转置卷积 2.反池化 3.双线性插值法 三、在经典网络中的的应用 1.U-Net 2.FCN 总结 概要 上采样是用于深度学习中提高语义分割精度的技术&#xff0c;可以实现图像放大和像素级别标注 一、概述 神经网络的基本结构为&#xff1a;…

【讲解下go和java的区别】

&#x1f525;博主&#xff1a;程序员不想YY啊&#x1f525; &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家&#x1f4ab; &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 &#x1f308;希望本文对您有所裨益&#xff0c;如有…

【深耕 Python】Data Science with Python 数据科学(2)jupyter-lab和numpy数组

关于数据科学环境的建立&#xff0c;可以参考我的博客&#xff1a;【深耕 Python】Data Science with Python 数据科学&#xff08;1&#xff09;环境搭建 Jupyter代码片段1&#xff1a;简单数组的定义和排序 import numpy as np np.array([1, 2, 3]) a np.array([9, 6, 2, …

vue3-pinia使用(末尾有彩蛋)

什么是 pinia Pinia 是 Vue 的专属状态管理库&#xff0c;它允许你跨组件或页面共享状态。 之前用的是 vuex&#xff0c;后面 vue 官方团队不维护了&#xff0c;推荐使用 pinia 安装 yarn add pinia # 或者使用 npm npm install piniapnpm install piniaStore 是什么&#xf…

2024 ccfcsp认证打卡 2022 06 01 归一化处理

import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner sc new Scanner(System.in);int n sc.nextInt(); // 输入数字的个数int[] a new int[1010]; // 创建一个数组来存储输入的数字double sum 0; // 用于计算所有输入数字的总…

COLMAP(Windows)实现SFM三维重建位姿估计

问题产生&#xff1a; Guassian splatting第一步用colmap进行位姿估计&#xff0c;图片匹配失败&#xff0c;输出图片全靠运气&#xff0c;最少的时候甚至一张都没匹配上&#xff0c;所以想到用colmap软件先进行匹配&#xff0c;再放入高斯训练。 colmap使用步骤&#xff1a;…

小米汽车引入革命性卫星通信技术:专利揭示直连卫星能力

小米汽车在近期的SU7发布会上&#xff0c;虽已展示了其运动轿跑车型的各项卓越性能&#xff0c;售价起于21.59万元&#xff0c;但其技术创新的深度远不止于此。一项最新公布的专利显示&#xff0c;小米汽车科技有限公司正在积极探索和开发车载卫星通信技术&#xff0c;该技术的…

后端常问面经之操作系统

请简要描述线程与进程的关系,区别及优缺点&#xff1f; 本质区别&#xff1a;进程是操作系统资源分配的基本单位&#xff0c;而线程是任务调度和执行的基本单位 在开销方面&#xff1a;每个进程都有独立的代码和数据空间&#xff08;程序上下文&#xff09;&#xff0c;程序之…