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

这次比赛很好,题也很好,就是下午人不太清醒,没做出来太菜了,回去才写出来的。
比赛原题已经上架到PTA了:点击跳转

题目

在这里插入图片描述

解释

题目意思其实很简单,就是给你一个[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
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

lin钟一

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

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

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

打赏作者

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

抵扣说明:

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

余额充值