hihocoder[Offer收割]编程练习赛3及参考

2024-05-25 01:18

本文主要是介绍hihocoder[Offer收割]编程练习赛3及参考,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目1 : hiho密码

时间限制:10000ms
单点时限:1000ms
内存限制:256MB
描述
小Ho根据最近在密码学课上学习到的知识,开发出了一款hiho密码,这款密码的秘钥是这样生成的:对于一种有N个字母的语言,选择一个长度为M的单词;将组成这个单词的所有字母按照顺序不重复的写出(即遇到相同字母时跳过);然后将字母表剩下的没有使用过的字母按照顺序在其后进行排列。

如对于有5个字母的hiho语,选择单词1, 2, 2, 4, 3(此处数字表示字母在字母表中的顺序),则秘钥为1,2,4,3,5。

但是有一天小Ho在计算出了秘钥之后,却发现他弄丢了一开始选择的单词,于是他找到了你,希望你能够帮他找到能够生成这个秘钥的最短的单词。

输入
每个输入文件包含单组测试数据。

每组测试数据的第一行为一个正整数N,意义如前文所述。

每组测试数据的第二行为N个正整数,用来描述一个秘钥,其中第i个正整数Ai表示秘钥的第i个字符在字母表中的顺序。

对于100%的数据,满足N<=1000,1<=Ai<=N。

对于100%的数据,满足对于任意1<=i, j<=N,若i≠j,则Ai≠Aj。

输出
对于每组测试数据,输出能够生成输入给出的秘钥的最短的单词(空串不认为是单词)。由于字母表没有给出,所以对于每个字母,输出其在字母表中的顺序即可(用空格隔开)。

样例输入
5
1 2 4 3 5
样例输出
1 2 4

#include <iostream>
#include <cstdio>
using namespace std;int a[2000];
int main()
{int n;scanf("%d",&n);for(int i = 1; i <= n; i++)scanf("%d", &a[i]);int p = n-1;while(p >= 1){if(a[p] < a[p+1]) p--;else break;}if(p == 0){printf("%d\n", a[1]);}else{for(int i = 1; i <= p; i++)printf("%d%c", a[i], (i==p)?'\n':' ');}}

题目2 : 机会渺茫

时间限制:5000ms
单点时限:1000ms
内存限制:256MB
描述
小Hi最近在追求一名学数学的女生小Z。小Z其实是想拒绝他的,但是找不到好的说辞,于是提出了这样的要求:对于给定的两个正整数N和M,小Hi随机选取一个N的约数N’,小Z随机选取一个M的约数M’,如果N’和M’相等,她就答应小Hi。

小Z让小Hi去编写这个随机程序,到时候她review过没有问题了就可以抽签了。但是小Hi写着写着,却越来越觉得机会渺茫。那么问题来了,小Hi能够追到小Z的几率是多少呢?

输入
每个输入文件仅包含单组测试数据。

每组测试数据的第一行为两个正整数N和M,意义如前文所述。

对于40%的数据,满足1<=N,M<=106

对于100%的数据,满足1<=N,M<=1012

输出
对于每组测试数据,输出两个互质的正整数A和B(以A分之B表示小Hi能够追到小Z的几率)。

样例输入
3 2
样例输出
4 1

#include<cstdio>
#include<iostream>
using namespace std;
typedef long long LL;
LL gcd(LL n,LL m){if(m==0)return n;return gcd(m,n%m);
}
LL fnum(LL t){LL a=0;for(int i=1;i<=t/i;i++){if(t%i==0)a++;else continue;if(t/i!=i)a++;}return a;
}
int main(){LL n,m;while(scanf("%lld%lld\n",&n,&m)!=EOF){LL t=gcd(n,m);LL a=fnum(t),b=fnum(n),c=fnum(m);b*=c;c=gcd(a,b);printf("%lld %lld\n",b/c,a/c);}return 0;
}

题目3 : 智力竞赛

时间限制:5000ms
单点时限:1000ms
内存限制:256MB
描述
小Hi、小Ho还有被小Hi强拉来的小Z,准备组队参加一个智力竞赛。竞赛采用过关制,共计N个关卡。在第i个关卡中,小Hi他们需要获得Ai点分数才能够进入下一关。每一关的分数都是独立计算的,即使在一关当中获得超过需要的分数,也不会对后面的关卡产生影响。

小Hi他们可以通过答题获得分数。答对一道题获得S点分数,答错一道题获得T点分数。在所有的N个关卡中,小Hi他们一共有M次答题机会。在每个关卡中,都可以在累计答题次数不超过M的情况下使用任意次的答题机会。

那么现在问题来了,对于给定的N、M、S、T和A,小Hi他们至少需要答对多少道题目才能够完成所有的关卡呢?

输入
每个输入文件包含多组测试数据,在每个输入文件的第一行为一个整数Q,表示测试数据的组数。
每组测试数据的第一行为四个正整数N、M、S和T,意义如前文所述。
第二行为N个正整数,分别表示A1~AN。
对于40%的数据,满足1<=N,M<=100
对于100%的数据,满足1<=N,M<=1000,1<=T< S<=10,1<=Ai<=50
对于100%的数据,满足1<=Q<=100
输出
对于每组测试数据,如果小Hi他们能够顺利完成关卡,则输出一个整数Ans,表示小Hi他们至少需要答对的题目数量,否则输出No。

样例输入
1
2 10 9 1
12 35
样例输出
5

#include <iostream>
#include <string.h>
#include <stdio.h>
#include <map>
#define maxn 1010
#define ll long long
using namespace std;
int dp[maxn][maxn];
int a[maxn];
int main()
{int ncase;scanf("%d",&ncase);while(ncase--){int n,m,s,t,limit=0;scanf("%d%d%d%d",&n,&m,&s,&t);for(int i=1;i<=n;i++){scanf("%d",&a[i]);limit+=(a[i]+s-1)/s;}if(limit>m){printf("No\n");continue;}memset(dp,1,sizeof(dp));dp[0][0]=0;int num=0;for(int i=0;i<n;i++){num+=(a[i]+s-1)/s;for(int j=0;j<=num;j++){if(dp[i][j]<=m){int ed=(a[i+1]+s-1)/s;for(int k=0;k<=ed;k++){int left=a[i+1]-k*s,fal=0;if(left>0)fal=(left+t-1)/t;dp[i+1][j+k]=min(dp[i+1][j+k],dp[i][j]+fal+k);}}}}int ans=0;for(int i=0;i<=limit;i++){if(dp[n][i]<=m){ans=i;break;}}printf("%d\n",ans);}return 0;
}

题目4 : 子矩阵求和

时间限制:20000ms
单点时限:2000ms
内存限制:256MB
描述
给定一个无限矩阵A,如下图所示,其中Aij=min(i,j)。

1 1 1 1 1 1 1 1 1
1 2 2 2 2 2 2 2 2
1 2 3 3 3 3 3 3 3
1 2 3 4 4 4 4 4 4
1 2 3 4 5 5 5 5 5 … y
1 2 3 4 5 6 6 6 6
1 2 3 4 5 6 7 7 7
1 2 3 4 5 6 7 8 8
1 2 3 4 5 6 7 8 9
⋮ ⋱
x
小Hi希望你从中找到一个N*M的子矩阵,使得子矩阵中元素的和是K的整数倍。

如果不存在这样的子矩阵,输出-1;否则输出子矩阵左上角元素的下标x和y。如果存在多个满足条件的子矩阵,输出x+y最小的一个;如果仍有多个,输出其中x最小的一个。

输入
输入的第一行是一个整数Q (1 <= Q <= 100)表示输入数据的组数。

对于每组数据,输入一行三个整数N, M, K。1 <= N, M <= 105, 1 <= K <= 106。

输出
对于每组数据输出一行,-1或者子矩阵左上角元素的下标x和y。

样例输入
2
2 2 10
3 3 31
样例输出
2 3
6 7

#include<algorithm>
#include<cstring>
#include<cstdio>
#include<string>
#include<iostream>
#include<fstream>
#include<map>
#include<vector>
#include <stdlib.h>
#include <time.h>
#include<cmath>
#include<sstream>
using namespace std;
long long a[300005];
long long b[300005];
int num[1000005];
int flag[1000005];
int main()
{int Q;cin>>Q;while(Q--){long long n,m,k;cin>>n>>m>>k;int p=max(n,m)+2;int anp=1e8,anq=1e8;memset(flag,0,sizeof(flag));memset(num,-1,sizeof(num));long long s=n*m%k;long long h=0;for(int i=0;i<=k;i++){if(flag[h]==1)break;flag[h]=1;num[h]=i;h=(h+s)%k;//       cout<<h<<'g'<<endl;}//   for(int i=0;i<=10;i++)//      cout<<num[i]<<endl;for(long long i=1;i<=2*p;i++){if(i<=n)a[i]=((i*(i-1))/2+i*(n-i+1))%k;elsea[i]=(n*(n+1))/2%k;//      cout<<i<<' '<<a[i]<<endl;}b[0]=0;for(int i=1;i<=2*p;i++){b[i]=(b[i-1]+a[i])%k;}for(int i=1;i<=p;i++){int g=(b[i+m-1]-b[i-1]+k)%k;//     cout<<i<<' '<<g<<endl;int o;if(g==0)o=0;elseo=k-g;if(num[o]==-1)continue;int np=num[o]+1;int nq=num[o]+i;if(np+nq<anp+anq||((np+nq)==(anp+anq)&&np<anp)){anp=np;anq=nq;}}for(long long i=1;i<=2*p;i++){if(i<=m)a[i]=((i*(i-1))/2+i*(m-i+1))%k;elsea[i]=(m*(m+1))/2%k;//     cout<<i<<' '<<a[i]<<endl;}b[0]=0;for(int i=1;i<=2*p;i++){b[i]=(b[i-1]+a[i])%k;}for(int i=1;i<=p;i++){int g=(b[i+n-1]-b[i-1]+k)%k;int o;if(g==0)o=0;elseo=k-g;if(num[o]==-1)continue;int np=num[o]+i;int nq=num[o]+1;if(np+nq<anp+anq||((np+nq)==(anp+anq)&&np<anp)){anp=np;anq=nq;}}if(anp==1e8)cout<<-1<<endl;elsecout<<anp<<' '<<anq<<endl;}return 0;
}代码取自排名靠前的优秀选手,简洁值得学习

这篇关于hihocoder[Offer收割]编程练习赛3及参考的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1000071

相关文章

C#反射编程之GetConstructor()方法解读

《C#反射编程之GetConstructor()方法解读》C#中Type类的GetConstructor()方法用于获取指定类型的构造函数,该方法有多个重载版本,可以根据不同的参数获取不同特性的构造函... 目录C# GetConstructor()方法有4个重载以GetConstructor(Type[]

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

【编程底层思考】垃圾收集机制,GC算法,垃圾收集器类型概述

Java的垃圾收集(Garbage Collection,GC)机制是Java语言的一大特色,它负责自动管理内存的回收,释放不再使用的对象所占用的内存。以下是对Java垃圾收集机制的详细介绍: 一、垃圾收集机制概述: 对象存活判断:垃圾收集器定期检查堆内存中的对象,判断哪些对象是“垃圾”,即不再被任何引用链直接或间接引用的对象。内存回收:将判断为垃圾的对象占用的内存进行回收,以便重新使用。

Go Playground 在线编程环境

For all examples in this and the next chapter, we will use Go Playground. Go Playground represents a web service that can run programs written in Go. It can be opened in a web browser using the follow

深入理解RxJava:响应式编程的现代方式

在当今的软件开发世界中,异步编程和事件驱动的架构变得越来越重要。RxJava,作为响应式编程(Reactive Programming)的一个流行库,为Java和Android开发者提供了一种强大的方式来处理异步任务和事件流。本文将深入探讨RxJava的核心概念、优势以及如何在实际项目中应用它。 文章目录 💯 什么是RxJava?💯 响应式编程的优势💯 RxJava的核心概念

函数式编程思想

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想. 如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c… 面向过程的指令式编程 面向过程,简单理解就是y=a(x)+b(x)+c(x)

Java并发编程之——BlockingQueue(队列)

一、什么是BlockingQueue BlockingQueue即阻塞队列,从阻塞这个词可以看出,在某些情况下对阻塞队列的访问可能会造成阻塞。被阻塞的情况主要有如下两种: 1. 当队列满了的时候进行入队列操作2. 当队列空了的时候进行出队列操作123 因此,当一个线程试图对一个已经满了的队列进行入队列操作时,它将会被阻塞,除非有另一个线程做了出队列操作;同样,当一个线程试图对一个空

生信代码入门:从零开始掌握生物信息学编程技能

少走弯路,高效分析;了解生信云,访问 【生信圆桌x生信专用云服务器】 : www.tebteb.cc 介绍 生物信息学是一个高度跨学科的领域,结合了生物学、计算机科学和统计学。随着高通量测序技术的发展,海量的生物数据需要通过编程来进行处理和分析。因此,掌握生信编程技能,成为每一个生物信息学研究者的必备能力。 生信代码入门,旨在帮助初学者从零开始学习生物信息学中的编程基础。通过学习常用

基于SpringBoot的宠物服务系统+uniapp小程序+LW参考示例

系列文章目录 1.基于SSM的洗衣房管理系统+原生微信小程序+LW参考示例 2.基于SpringBoot的宠物摄影网站管理系统+LW参考示例 3.基于SpringBoot+Vue的企业人事管理系统+LW参考示例 4.基于SSM的高校实验室管理系统+LW参考示例 5.基于SpringBoot的二手数码回收系统+原生微信小程序+LW参考示例 6.基于SSM的民宿预订管理系统+LW参考示例 7.基于