[USACO14OPEN] Odometer S

news2025/1/11 2:55:22

洛谷[USACO14OPEN] Odometer S

题目大意

当一个数的每一位中有至少一半的数字相同,那么这个数就是一个有趣的数。求区间 [ L , R ] [L,R] [L,R]中有多少个有趣的数。

100 ≤ L ≤ R ≤ 1 0 18 100\leq L\leq R\leq 10^{18} 100LR1018


题解

这道题很容易能想到要用数位DP。

[ L , R ] [L,R] [L,R]分为 [ 1 , L − 1 ] [1,L-1] [1,L1] [ 1 , R ] [1,R] [1,R],分别求出答案后作差即可得到最终答案。也就是说,我们只要想办法求出区间 [ 1 , v ] [1,v] [1,v]中的有趣的数的数量即可。设 v v v的位数为 d d d

先枚举这相同的一半位置的数字 i i i,设 f j , k , 0 / 1 / 2 f_{j,k,0/1/2} fj,k,0/1/2表示 v v v的从低到高的前 j j j中有 k k k位于 i i i相同,这个数与 v v v的前 j j j位相比是小于、等于还是大于。然后对于每种情况考虑转移即可。

为了避免统计含有前导零的数字,我们需要用 g g g来存储 f f f中各个状态的数的数量,以便于统计最后的答案。

但是,这样可能会重复计算多个数,我们要将重复计算的贡献减去。因为满足题意的数需要有至少一半的数字相同,所以这样的数中只会恰好有两种数字。当位数小于 d d d时,用组合数求出重复计算的贡献即可;当位数等于 d d d时,因为有一些数字大于 v v v,没有计算贡献,但如果用组合数计算贡献的话,会把这些当做重复的贡献来减去。所以我们要用 d f s dfs dfs把所有可能的数求出,在这些数中找出小于等于 d d d的并减去其重复计算的贡献。因为这样的数中只会恰好有两种数字,所以每次 d f s dfs dfs只会查找 2 d 2^d 2d个数。

数位DP的时间复杂度为 O ( d 3 ) O(d^3) O(d3),减去重复贡献的时间复杂度为 O ( 10 ⋅ 2 d ) O(10\cdot 2^d) O(102d),所以总时间复杂度为 O ( d 3 + 10 ⋅ 2 d ) O(d^3+10\cdot 2^d) O(d3+102d)

这个方法跑得比较快,目前在洛谷上是最优解。

在这里插入图片描述

code

#include<bits/stdc++.h>
using namespace std;
int v1,v[25];
long long L,R,ct,ansL,ansR,wt,jc[20],f[25][25][3],g[25][25][3];
void dd(long long x){
	v1=0;
	while(x){
		v[++v1]=x%10;x/=10;
	}
}
int gt(int i,int j){
	if(i<j) return 0;
	if(i==j) return 1;
	return 2;
}
void dfs(long long now,int w,int hv,long long bg){
	if(w==1){
		if(hv==0&&now<=bg) ++ct;
		return;
	}
	dfs(now*10+wt,w-1,hv-1,bg);
	dfs(now*10+v[v1],w-1,hv+1,bg);
}
int main()
{
	scanf("%lld%lld",&L,&R);
	jc[0]=1;
	for(int i=1;i<=18;i++) jc[i]=jc[i-1]*i;
	dd(L-1);
	for(int i=0;i<=9;i++){
		memset(f,0,sizeof(f));
		memset(g,0,sizeof(g));
		for(int j=0;j<=9;j++){
			++f[1][(i==j)][gt(j,v[1])];
			if(j>0) ++g[1][(i==j)][gt(j,v[1])];
		}
		for(int j=2;j<=v1;j++){
			for(int k=0;k<v1;k++){
				for(int p=0;p<=9;p++){
					if(p<v[j]){
						for(int o=0;o<=2;o++){
							f[j][k+(i==p)][0]+=f[j-1][k][o];
							if(p>0) g[j][k+(i==p)][0]+=f[j-1][k][o];
						}
					}
					else if(p==v[j]){
						for(int o=0;o<=2;o++){
							f[j][k+(i==p)][o]+=f[j-1][k][o];
							if(p>0) g[j][k+(i==p)][o]+=f[j-1][k][o];
						}
					}
					else{
						for(int o=0;o<=2;o++){
							f[j][k+(i==p)][2]+=f[j-1][k][o];
							if(p>0) g[j][k+(i==p)][2]+=f[j-1][k][o];
						}
					}
				}
			}
		}
		for(int j=1;j<=v1;j++){
			for(int k=(j+1)/2;k<=j;k++){
				ansL+=g[j][k][0]+g[j][k][1];
				if(j<v1) ansL+=g[j][k][2];
			}
		}
	}
	for(int o=2;o<v1;o+=2){
		for(int i=1;i<=9;i++){
			ansL-=jc[o-1]/jc[o/2]/jc[o/2-1];
			for(int j=1;j<i;j++){
				ansL-=jc[o]/jc[o/2]/jc[o/2];
			}
		}
	}
	if(v1%2==0){
		for(int o=1;o<v[v1];o++){
			for(int j=0;j<=9;j++){
				if(j==o) continue;
				ansL-=jc[v1-1]/jc[v1/2]/jc[v1/2-1];
			}
		}
		ct=0;
		for(wt=0;wt<=9;wt++){
			if(v[v1]==wt) continue;
			dfs(v[v1],v1,1,L-1);
		}
		ansL-=ct;
	}
	dd(R);
	for(int i=0;i<=9;i++){
		memset(f,0,sizeof(f));
		memset(g,0,sizeof(g));
		for(int j=0;j<=9;j++){
			++f[1][(i==j)][gt(j,v[1])];
			if(j>0) ++g[1][(i==j)][gt(j,v[1])];
		}
		for(int j=2;j<=v1;j++){
			for(int k=0;k<v1;k++){
				for(int p=0;p<=9;p++){
					if(p<v[j]){
						for(int o=0;o<=2;o++){
							f[j][k+(i==p)][0]+=f[j-1][k][o];
							if(p>0) g[j][k+(i==p)][0]+=f[j-1][k][o];
						}
					}
					else if(p==v[j]){
						for(int o=0;o<=2;o++){
							f[j][k+(i==p)][o]+=f[j-1][k][o];
							if(p>0) g[j][k+(i==p)][o]+=f[j-1][k][o];
						}
					}
					else{
						for(int o=0;o<=2;o++){
							f[j][k+(i==p)][2]+=f[j-1][k][o];
							if(p>0) g[j][k+(i==p)][2]+=f[j-1][k][o];
						}
					}
				}
			}
		}
		for(int j=1;j<=v1;j++){
			for(int k=(j+1)/2;k<=j;k++){
				ansR+=g[j][k][0]+g[j][k][1];
				if(j<v1) ansR+=g[j][k][2];
			}
		}
	}
	for(int o=2;o<v1;o+=2){
		for(int i=1;i<=9;i++){
			ansR-=jc[o-1]/jc[o/2]/jc[o/2-1];
			for(int j=1;j<i;j++){
				ansR-=jc[o]/jc[o/2]/jc[o/2];
			}
		}
	}
	if(v1%2==0){
		for(int o=1;o<v[v1];o++){
			for(int j=0;j<=9;j++){
				if(j==o) continue;
				ansR-=jc[v1-1]/jc[v1/2]/jc[v1/2-1];
			}
		}
		ct=0;
		for(wt=0;wt<=9;wt++){
			if(v[v1]==wt) continue;
			dfs(v[v1],v1,1,R);
		}
		ansR-=ct;
	}
	printf("%lld",ansR-ansL);
	return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

AcWing242. 一个简单的整数问题

输入样例&#xff1a; 10 5 1 2 3 4 5 6 7 8 9 10 Q 4 Q 1 Q 2 C 1 6 3 Q 2输出样例&#xff1a; 4 1 2 5 #include<bits/stdc.h> using namespace std; const int N1e55; int n,m,a[N],c[N],x,y,d; char ch; int lowbit(int x){return x&-x; } void add(int x,int…

pytest--allure报告中添加用例详情

前言 前面介绍了如何生成allure的报告&#xff0c;看着allure的页面非常好看&#xff0c;但是感觉少了一些内容&#xff0c;allure还可以增加一些用例详情内容&#xff0c;这样让我们的报告看着更加绚丽。 allure增加用例详情 我们可以在报告测试套件中增加用例详情内容。 …

Spring初识(四)

文章目录 前言一.Bean的作用域1.1 作用域例子1.2 Bean的作用域类型 二.Bean的生命周期 前言 在前面我们学习了spring简单的读取和存储对象之后,Spring 中 Bean 是最核心的操作资源&#xff0c;我们接下来会介绍Bean对象. 一.Bean的作用域 什么是Bean作用域呢? 限定程序中变…

[DDPM] Denoising Diffusion Probabilistic Models

直接看paper云里雾里&#xff0c;一些推荐的讲解&#xff1a; The Annotated Diffusion Model 生成扩散模型漫谈&#xff08;一&#xff09;&#xff1a;DDPM 拆楼 建楼 生成扩散模型漫谈&#xff08;二&#xff09;&#xff1a;DDPM 自回归式VAE 生成扩散模型漫谈&#xff…

SQL-每日一题【619.只出现一次的最大数字】

题目 MyNumbers 表&#xff1a; 单一数字 是在 MyNumbers 表中只出现一次的数字。 请你编写一个 SQL 查询来报告最大的 单一数字 。如果不存在 单一数字 &#xff0c;查询需报告 null 。 查询结果如下例所示。 示例 1&#xff1a; 示例 2&#xff1a; 解题思路 1.题目要求我…

MyBatis---多表查询,动态sql的详细介绍

目录 1.命名规则 1.resultMap&#xff08;对应类属性名称与数据库字段名称&#xff09; 2.多表查询&#xff08;ResultMap&#xff09; 1.创建articleInfo类 2.创建ArticleMapper.xml配置文件和ArticleMapper接口 ①&#xff1a;不建议使用 &#xff08;文件之间耦合严重…

十大排序算法详解

目录 1. 冒泡排序 a. 思路 b. code 2. 插入排序 a. 思路 b. code 3. 希尔排序【插入排序plus】 a. 思路 b. code 4. 选择排序 a. 思路 b. code 5. 基数排序 a. 前置知识 b. 思路 c. code 6. 计数排序 a. 思路 b. code 7. 桶排序&#xff08;计数排序plus &…

怎么在shell中查看python版本以及降低anaconda的python版本

输入命令 python --version 电脑anaconda的python版本为3.11&#xff0c;如何降低版本 &#xff0c;输入命令 conda install python3.9 当安装完anaconda后&#xff0c;ubuntu系统的shell命令行最前面会出现base字样&#xff0c;此时要退出&#xff0c;就输入命令 conda deacti…

如何开启QQ邮件的SMTP服务以及如何使用Python发送邮件

如何开启QQ邮件的SMTP服务以及如何使用Python发送邮件 &#x1f607;博主简介&#xff1a;我是一名正在攻读研究生学位的人工智能专业学生&#xff0c;我可以为计算机、人工智能相关本科生和研究生提供排忧解惑的服务。如果您有任何问题或困惑&#xff0c;欢迎随时来交流哦&…

Qt应用开发——Drag and Drop

目录 一、前言 二、相关事件和类 三、实例 拖动文件到编辑框 一、前言 在实际场景中&#xff0c;经常会有导入文件的需求。导入文件一般两种方式&#xff0c;第一种QFileDialog显示一个文件选择窗口&#xff0c;选择后处理和显示。第二种就是使用拖放机制实现。还有在一些制…

.faust加密勒索数据库恢复---惜分飞

有客户的win服务器被勒索病毒加密,里面运行有用友系统的Oracle数据库&#xff0c;加密提示为&#xff08;camry2020aol.com&#xff09;&#xff1a; 加密的数据文件类似&#xff08;.DBF.id[0E564ACA-3493].[camry2020aol.com].faust&#xff09;: 通过工具检测发现少量bl…

(2)前端控制器的扩展配置, 视图解析器类型以及MVC执行流程的概述

SpringMVC入门程序的扩展说明 注册前端控制器的细节 在web.xml文件注册SpringMVC的前端控制器DispatcherServlet时使用url-pattern标签中使用/和/*的区别 /可以匹配.html或.js或.css等方式的请求路径,但不匹配*.jsp的请求路径/*可以匹配所有请求(包括.jsp请求), 例如在过滤器…

Kotlin空安全类型之平台类型

前言 众所周知,Kotlin的空安全很方便,能避免绝大部分的空安全问题 一般类型分为: 可空类型如 String? 不可空类型 String其中可空类型的String?可以赋值null或String类型的值,在使用时必须显式的进行空处理 而不可空类型的String,只能赋值String类型,无法向其赋值null,使用时…

JAVA基础-Stream流

引言 Java 8 版本新增的Stream&#xff0c;配合同版本出现的Lambda &#xff0c;给我们操作集合&#xff08;Collection&#xff09;提供了极大的 便利。Stream流是JDK8新增的成员&#xff0c;允许以声明性方式处理数据集合&#xff0c;可以把Stream流看作是遍历数据集 合的一个…

【后端面经-Java】Java基本概念

【后端面经-Java】Java基本概念&#xff09; 1. 面向对象和面向过程1.1 概念1.2 优缺点对比 2. C和Java的区别2.1 "去其糟粕"——Java同学&#xff0c;你对C的这些东西有什么意见&#xff1f;2.2 改进之处————Java同学&#xff0c;你的这个论文&#xff0c;&…

伦敦金投资仓位控制的方法

留意本栏目过去的文章的朋友都会发现&#xff0c;其实小编认为资金管理很重要&#xff0c;甚至重要性超过技术分析找到入场机会。在资金管理中&#xff0c;关于仓位的控制是一门很大的学问&#xff0c;在伦敦金投资中&#xff0c;仓位的控制关系到我们盈亏的多少&#xff0c;甚…

利用百度地图做个亚洲范围地震可视化特效果

地图叠加等值面效果&#xff0c;绘制掩体模型 地震闪动特效 地震热力图效果

【人工智能】局部搜索算法、爬山法、模拟退火、局部剪枝、遗传算法

文章目录 局部搜索算法内存限制局部搜索算法示例:n-皇后爬山算法随机重启爬山模拟退火算法局部剪枝搜索遗传算法小结局部搜索算法 在某些规模太大的问题状态空间内,A*往往不够用 问题空间太大了无法访问 f 小于最优的所有状态通常,甚至无法储存整个边缘队列解决方案 设计选择…

vue-cli脚手架创建创建的项目打包后无法正常打开报 Failed to load resource: net::ERR_FILE_NOT_FOUND错误

亲爱的小伙伴们&#xff0c;你们最近是否有遇到用使用最新的脚手架打包项目后index.html文件无法正常打开&#xff0c;然后控制台报错的情况呢&#xff0c;不要担心&#xff0c;这个坑今天被我踩到了并且被我解决了&#xff0c;下边就让我来给大家分享一下经验吧&#xff01; …

树莓派4B通过wiringP库实现串口通信的实验

1.实现串口通信前的准备 树莓派4b实现与其它设备之间的串口通信需要做一些配置&#xff0c;可以点进这篇文章进行查看参考文章 wiringPi库串口相关的API参考文章链接wiringPI库串口API 2.实现单个字符的发送 #include <stdio.h> #include <wiringPi.h> #includ…