The 2022 ICPC Asia Regionals Online Contest (I)-D题题解(DFS暴搜+剪枝+打表去重+二分)

简介: 时间复杂度:O(227 + nlog(n) + T * log(n)),227是DFS打表的时间,nlog(n)是vertor排序的时间,T*log(n)是询问次数和二分搜答案的时间。(如果算错了,评论或者私信指出谢谢)

这次比赛很好,题也很好,就是下午人不太清醒,没做出来太菜了,回去才写出来的。

比赛原题已经上架到PTA了:点击跳转


题目


105875edd66443eaaaef4dbe4cc1379f.png


解释


题目意思其实很简单,就是给你一个[l, r]的区间你去找一个数,满足他后缀连续的0的个数等于前面的1的个数,举个例子68:二进制形式1000100,后缀连续的零的个数ctz(68)=2,前面1的个数popcount(68)=2,刚好相等所以他就是一个good的数。


思路


方法:DFS暴搜+剪枝+打表去重+二分


题目的意思不就是让你怎么去排列后缀零前面的1的位置和个数,直接用dfs暴搜出所有的good数,因为x只有1e9的范围,二进制也就是30位,我们从高往低去搜,当前位选1或者0,用两层for循环,外面那层我们用来枚举位数,里面枚举后缀0的个数。


dfs传递的是我们枚举的后缀0的个数now(需要多个1),当前到dfs哪一位step以及用了几个1(one)。


剪枝:我们当前位超过了后缀0的个数就return。


我们搜出now == one的时候就是good的数,用unordered_map和vector用来去重和保存。


dfs搜完后,把vector的数sort一下,后面求区间内的数用lowwer_bound(l)找就行。


代码


#include<bits/stdc++.h>
#define CL(a,b)  memset(a,b,sizeof(a))
using namespace std;
const int maxn = 1e5+10;
const int mod  = 1e9+7;
int T, l, r;
vector<int>num;
unordered_map<int,int>nums;
int tot;
void dfs(int now,int step,int need) {
  if(now==need) {
    if(!nums[tot])num.push_back(tot);
    nums[tot]=1;
    return;
  }
  if(step==now)return;
  for(int i=1; i>=0; i--) {
    if(i)need++, tot += 1 << step;
    step--;
    dfs(now,step,need);
    step++;
    if(i)need--, tot -= 1 << step;
  }
  return;
}
void solve() {
  for(int len = 30 ; len >= 2; len--) //构造的数的长度
    for(int p = 1; p <= len / 2 ; p++) { //构造末尾0
      tot = 1 << p;
      dfs(p,len-1,1);
    }
}
int getnum(int l) {
  return lower_bound(num.begin(),num.end(),l)-num.begin();
}
int main() {
  solve();
  sort(num.begin(),num.end());
  scanf("%d", &T);
    while(T--)
    {
        scanf("%d%d",&l,&r);
        int ans = num[getnum(l)];
        if(ans <= r)printf("%d\n", ans);
        else printf("-1\n");
    }
  return 0;
}

e829f6ec8b0d4bdd8189287612f744bb.png


复杂度分析


  • 时间复杂度:O(227 + nlog(n) + T * log(n)),227是DFS打表的时间,nlog(n)是vertor排序的时间,T*log(n)是询问次数和二分搜答案的时间。(如果算错了,评论或者私信指出谢谢)


  • 空间复杂度:O(n),n表示good的数的数量,我搜出来差不多有这么多


e5e2fd56c921473788b9d139b16e2661.png

目录
相关文章
|
IDE 开发工具 开发者
IDLE的运行方式
IDLE的运行方式
1618 3
|
监控 数据可视化 数据挖掘
【软件设计师备考 专题 】软件过程评估与能力成熟度评估的基本方法
【软件设计师备考 专题 】软件过程评估与能力成熟度评估的基本方法
547 0
|
Linux 网络安全 开发工具
Git学习笔记(一):基础与应用
本文档详细介绍了如何将本地项目关联到Gitee上的空仓库并上传代码,以及如何验证本机与Git服务器的SSH连接。同时,还概述了Git的基本概念、安装步骤、初始配置、常见命令及如何配置多个SSH-Key,适用于初学者快速上手Git操作。
337 51
Git学习笔记(一):基础与应用
|
11月前
|
分布式计算 DataWorks Serverless
通过函数计算节点实现GitHub实时数据分析与结果发送
开发人员在基于GitHub开源项目进行开发时会产生海量事件,GitHub会记录每次事件的类型、详情、开发者和代码仓库等信息,并开放其中的公开事件。DataWorks提供“Github十大热门编程语言”模板,通过对GitHub中公开数据集进行加工和分析,并将分析结果以邮箱的方式发送给指定用户。运行本案例后,您将得到Github中Top10编程语言每小时被提交的次数与排行。
283 10
|
存储 人工智能 小程序
比赛须知【2024 年睿抗机器人开发者大赛CAIP-编程技能赛(国赛)】
该文章是关于2024年睿抗机器人开发者大赛CAIP-编程技能赛(国赛)的参赛通知,强调了比赛时间、阅读比赛须知的重要性,并列举了多项比赛期间禁止的行为以确保比赛的公平性。
 比赛须知【2024 年睿抗机器人开发者大赛CAIP-编程技能赛(国赛)】
|
存储 消息中间件 NoSQL
深入了解Redis:高性能的内存数据库
深入了解Redis:高性能的内存数据库
|
Kubernetes Docker 容器
rancher docker k8s安装(二)
rancher docker k8s安装(二)
268 1
|
安全 Java 测试技术
避免低级错误:深入解析Java的ConcurrentModificationException异常
在软件开发中,我们常常会遇到各种错误和异常。其中有一类比较低级但又常见的错误就是`ConcurrentModificationException`异常。最近了我就写了个这种异常,这个异常通常发生在使用迭代器遍历集合时,同时对集合进行修改,从而导致迭代器检测到集合结构发生变化而抛出异常。在测试环境中可能因为数据量较小或者测试场景不充分未能显现问题,但一旦部署到生产环境,场景增多,并发操作增多,这个低级错误就会爆发。
477 0
避免低级错误:深入解析Java的ConcurrentModificationException异常
|
运维 监控 持续交付
"揭秘Puppet:自动化运维的超级英雄,一键驾驭复杂IT环境,让运维繁琐瞬间灰飞烟灭,引领未来运维新纪元!"
【8月更文挑战第9天】Puppet作为自动化运维的杰出代表,凭借其强大的配置管理和高度可定制性,助力IT运维实现自动化与智能化转型。通过定义资源模型与使用声明式语言描述系统状态,Puppet能自动调整系统至期望状态,实现标准化运维流程。其工作流程包括定义-应用-报告三步,支持从服务器配置到复杂网络、数据库管理等多种场景。示例代码展示了如何自动化部署Apache服务器,体现了Puppet在实际操作中的高效与便捷。随着技术演进与社区壮大,Puppet将持续推动运维领域的创新发展。
312 6
|
缓存 关系型数据库 MySQL
MySQL登录时出现Access denied for user ‘root‘@‘localhost‘ (using password: YES)无法打开的解决方法
MySQL登录时出现Access denied for user ‘root‘@‘localhost‘ (using password: YES)无法打开的解决方法
24060 0