当前位置: 首页 > article >正文

xtu oj 数字

文章目录

  • 回顾
  • 代码
  • 思路

回顾

  • A+B III
  • 问题 H: 三角数
  • 问题 G: 3个数
  • 等式 数组下标查询,降低时间复杂度
  • 1405 问题 E: 世界杯
  • xtu 数码串
  • xtu oj 神经网络
  • xtu oj 1167 逆序数(大数据)
  • xtu oj 原根
  • xtu oj 不定方程的正整数解
  • xtu oj 最多的可变换字符串
  • xtu oj String I
  • xtu oj 字母序列
  • xtu oj 分段
  • xtu oj 完全平方数II
  • xtu oj 连接字符串
  • xtu oj 2021

代码

#include<stdio.h>
int main(){
   int t;
   scanf("%d",&t);
   while(t--){
   	int n,k;
   	scanf("%d%d",&n,&k);//输入 n 和 k
   	int cnt=0;//存操作的次数
   	int temp=n;//临时变量存 n ,其实直接用 n 也行
   	while(1){
   		if(temp<k){//满足条件就跳出循环,严格小于
   			break;
   		}
   		if(temp%10==0){//尾数是 0,操作次数只需要一次,可以把全部的零去掉
   			cnt++;
   			while(temp%10==0){
   				temp/=10;
   			}
   		}else{//尾数不是 0
   			for(int i=1;i<=9;i++){//枚举个位数
   				if((temp-i*k)%10==0&&i*k<=temp){//这里的限制条件需要注意一下
   					temp-=i*k;
   					cnt+=i;
   					break;
   				}
   			}
   			if((temp>=k)&&(temp%10!=0)){//整除降低时间复杂度
   				int hhh=temp/k;
   				cnt+=hhh;
   				temp-=hhh*k;
   			}
   		}
   	}
   	printf("%d %d\n",cnt,temp);
   }
   return 0;
}

思路

复习 SOA 复习不下去了,写一写算法题。首先写了一个超时的代码

#include<stdio.h>
int main(){
   int t;
   scanf("%d",&t);
   while(t--){
   	int n,k;
   	scanf("%d%d",&n,&k);//输入 n 和 k
   	int cnt=0;//存操作的次数
   	int temp=n;//临时变量存 n ,其实直接用 n 也行
   	int i=0;
   	while(1){
   		if(temp%10==0){//尾数是 0
   			cnt++;
   			while(temp%10==0){
   				temp/=10;
   			}
   		}else{//尾数不是 0
   			while((temp-k*i)%10!=0&&temp>=k){
   				i++;//应该就是这里超时了,因为 n 是 10^9 ,能接受的时间复杂度是 10^8 ,所以应该要优化一下
   			}
   			cnt+=i;
   			temp-=k*i;
   		}
   		if(temp<k){
   			break;
   		}
   	}
   	printf("%d %d\n",cnt,temp);
   }
   return 0;
}

比较直观,没想到直接就超时了。难道是看 n 能不能整除 k 吗,感觉优化的话,就是用除法这种去代替原来超时代码里面的 i++i++ 估计有差不多 10^9 的时间复杂度。我们经过多少次操作可以使得 n 可以被 10 整除,比如说 n 是一个很大的数字,999999999999 ,假设 k 是一个质数 7 ,会不会经过很多次减法,还是不能让 n 可以整除 10 ,那假设我们直接用除法,那时间复杂度可以降到 O(1) 这样子,n/k 但是无法保证这就是正解,比如说 n=333,k=3 ,直接就是 n/k==111 ,但是这个不是正解。正解是 5 次。

这个题放在贪心这个章节。应该是要一边维护 n ,一边判断,关键就是这个判断的标准是什么。我感觉是先看能不能整除,能整除的话就把 n 减小一些,然后看n 能不能整除 10 ,我试一试。这样子好像运行程序的时候直接卡住了。

不会写了,搜到了一篇题解,XTUOJ杂题烩。还得是题解的思路比较好。

不过为啥我按照这个思路写,只有 33 分呢。发现自己主要是没想到枚举个位数这一步,想到了整除降低时间复杂度。之前看网课,里面说其实思路大家都差不多,就整体的思路,就比如说一开始直接模拟就超时这种思路,比较直观的,然后一些细节才真正决定能不能过掉这题。或者拿尽可能高一点的分数,不过 c 语言考试没有 33 分这种,是 acm 模式,某个题要么 100 ,要么 0 。看了题解的代码,感觉差不多。奇怪了。有没有读者知道问题出在哪里。

#include<stdio.h>
int main(){
	int t;
	scanf("%d",&t);
	while(t--){
		int n,k;
		scanf("%d%d",&n,&k);//输入 n 和 k
		int cnt=0;//存操作的次数
		int temp=n;//临时变量存 n ,其实直接用 n 也行
		while(1){
			if(temp%10==0){//尾数是 0
				cnt++;
				while(temp%10==0){
					temp/=10;
				}
			}else{//尾数不是 0
				for(int i=1;i<=9;i++){//枚举个位数
					if((temp-i*k)%10==0){
						temp-=i*k;
						cnt+=i;
						break;
					}
				}
				if(temp>=k&&temp%10!=0){//整除降低时间复杂度
					int hhh=temp/k;
					cnt+=hhh;
					temp-=hhh*k;
				}
			}
			if(temp<k){
				break;
			}
		}
		printf("%d %d\n",cnt,temp);
	}
	return 0;
}

交了一下题解的做法,能过,为啥我这个就过不了呢。那能过 33 的数据点又是什么意思呢,这里也不存在超出数据范围的情况。不行,我一定要把这个搞清楚。

嗷嗷我知道了。循环里面的限制条件没有限制严格,可能会出现负数的情况,之前的博客也有讲过,就是计算机负数进行取余运算的时候和我们理解的数学上的不太一样。


http://www.kler.cn/a/389300.html

相关文章:

  • HTTP 配置与应用(局域网)
  • 【游戏设计原理】77 - 沙盒与导轨
  • 深入探索C#中Newtonsoft.Json库的高级进阶之路
  • 安装wxFormBuilder
  • CloudberryDB(四)并行执行
  • C#与AI的共同发展
  • pdf转excel;pdf中表格提取
  • Three.js 搭建3D隧道监测
  • 江西省补贴性线上职业技能培训管理平台(刷课系统)
  • HarmonyOS Next 实战卡片开发 02
  • 【微服务】不同微服务之间用户信息的获取和传递方案
  • 11 Oracle Golden Gate 高可用解决方案:Golden Gate 助力企业保障业务连续性
  • (RK3566驱动开发 - 1).pinctrl和gpio子系统
  • vscode中使用c/c++插件运行c代码后占用c盘空间问题的解决
  • 呼叫中心系统监控预警功能的应用
  • 纯css制作声波扩散动画、js+css3波纹催眠动画特效、【css3动画】圆波扩散效果、雷达光波效果完整代码
  • 【LeetCode】【算法】209. 课程表
  • 蓝桥杯备赛(持续更新)
  • 【C语言刷力扣】66.加一
  • 京东商品SKU信息的“窃听风云”:Python爬虫的幽默之旅
  • 搜维尔科技:【应用】Xsens在荷兰车辆管理局人体工程学评估中的应用
  • 『 Linux 』网络传输层 - TCP(三)
  • 基于百度飞桨paddle的paddlepaddle2.4.2等系列项目的运行
  • Tidb数据恢复
  • [CKS] Create/Read/Mount a Secret in K8S
  • 软考中级 软件设计师 上午考试内容笔记(个人向)Part.3