contesthunter暑假NOIP模拟赛第一场题解
contesthunter暑假NOIP模拟赛#1题解:
第一题:杯具大派送
水题。枚举A,B的公约数即可。
#include <algorithm>
#include <cmath>
#include <iostream>
using namespace std;
#define MAXN 100010
struct node
{
int a,b,c;
}ans[MAXN];
int main() {
int R, G;
scanf("%d%d",&R,&G);
int x=min(R,G);
,t=MAXN-;
; i*i<=x; ++i )
{
)
{
&&R%i==)
{ans[s].a=i;
ans[s].b=R/i;
ans[s].c=G/i;
s++;
}
&&R%(x/i)==)
{
ans[t].a=x/i;
ans[t].b=R/(x/i);
ans[t].c=G/(x/i);
t--;
}
}
}
;i>t;i--)
printf("%d %d %d\n",ans[i].a,ans[i].b,ans[i].c);
printf("%d %d %d\n",ans[i].a,ans[i].b,ans[i].c);
;
}
第二题:另类区间和
动态规划、记忆化搜索
官方题解:
Let f(n, prefix) consider all numbers in the interval [prefix·10n, (prefix+1)·10n>. The function calculates
the total contribution towards the result of n digits yet to be placed to the right of the given prefix.
To calculate f, we add a group of any digit other than the last one in prefix. For example, suppose n=4
and prefix=112. f(4, 112) considers the numbers 1120000 through 1129999. If we decide to append the
group 55 to the number, the contribution of this recursive branch is 52 + f(2, 11255).
It may seem that it takes too many recursive calls to calculate the result. However, many of these yield
the same result for different prefixes, more precisely when the entire interval considered is contained
inside [A, B]. For these cases we can memoize the result (it depends only on n). There are only
O(log B) recursive calls in which we branch without memoization.
#include <algorithm>
#include <cstdio>
#include <cstring>
using namespace std;
typedef long long llint;
llint A, B;
llint p10[16];
llint memo[16][11];
llint intersect( int n, llint prefix ) {
if( n < 0 ) return 0;
llint lo = max( prefix * p10[n], A );
llint hi = min( (prefix+1) * p10[n] - 1, B );
if( lo > hi ) return 0;
return hi-lo+1;
}
llint rec( int n, int prev, llint prefix ) {
llint mini = prefix * p10[n];
llint maxi = (prefix+1) * p10[n] - 1;
if( mini > B || maxi < A ) return 0;
if( mini >= A && maxi <= B && memo[n][prev] != -1 ) return memo[n][prev];
llint ret = 0;
for( int digit = 0; digit <= 9; ++digit ) {
if( digit == prev ) continue;
llint t = prefix;
for( int k = 1; k <= n; ++k ) {
t = t*10+digit;
ret += digit * k * k * (intersect( n-k, t )-intersect( n-k-1, t*10+digit)) + rec( n-k, digit, t );
}
}
if( mini >= A && maxi <= B ) memo[n][prev] = ret;
return ret;
}
int main( void ) {
scanf( "%lld%lld", &A, &B );
p10[0] = 1;
for( int i = 1; i <= 15; ++i ) p10[i] = p10[i-1] * 10;
memset( memo, -1, sizeof memo );
printf( "%lld\n", rec( 15, 10, 0 ) );
return 0;
}
第三题:01序列
线段树
交替出现的序列才是合法的序列,题目要求最长序列的长度。我们可以用线段树来解决。
当前区间最长序列长度应该等于下面三者的最大值:
左儿子区间的最长序列、右儿子区间的最长序列、以及
左儿子的后缀与右儿子的前缀组成的序列。
所以每个节点需要保存5个值:区间最长的合法序列的长度,前缀长度,后缀长度,前缀的第一个字符,后缀的最后一个字符。
其他的我们还需要知道前缀的最后一个字符,但我们可以通过前缀长度和前缀的第一个字符推导出来,后缀的第一个字符我们也可以通过后缀的最后字符和后缀长度推导出来。
标程:(01序列)
#include <cstdio>
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
const int mxn = 1<<19;
struct node{
int mx;
int left, right;
char start, end;
int len;
} data[2*mxn];
node operator +( const node &a, const node &b ){
node ret;
ret.start = a.start;
ret.end = b.end;
ret.len = a.len + b.len;
ret.left = a.left;
if( a.len == a.left && a.end != b.start ) ret.left += b.left;
ret.right = b.right;
if( b.len == b.right && a.end != b.start ) ret.right += a.right;
ret.mx = max( max( a.mx, b.mx ), max( ret.left, ret.right ) );
if( a.end != b.start ) ret.mx = max( ret.mx, a.right+b.left );
return ret;
}
int n;
void construct( int i, int lo, int hi ){
if( hi-lo == 1 ){
if( lo >= n ) data[i].mx = data[i].left = data[i].right = 0;
else data[i].mx = data[i].left = data[i].right = 1;
data[i].start = data[i].end = 'L';
data[i].len = 1;
return;
}
construct( 2*i, lo, (lo+hi)/2 );
construct( 2*i+1, (lo+hi)/2, hi );
data[i] = data[2*i] + data[2*i+1];
}
void change( int i ){
i += mxn;
if( data[i].start == 'L' ) data[i].start = 'R';
else data[i].start = 'L';
data[i].end = data[i].start;
for( i /= 2; i > 0; i /= 2 )
data[i] = data[2*i] + data[2*i+1];
}
int main(){
int q;
scanf( "%d %d", &n, &q );
construct( 1, 0, mxn );
for( ; q > 0; q-- ){
int i;
scanf( "%d", &i ); i--;
change(i);
printf( "%d\n", data[1].mx );
}
return 0;
}
contesthunter暑假NOIP模拟赛第一场题解的更多相关文章
- NOI.AC NOIP模拟赛 第一场 补记
NOI.AC NOIP模拟赛 第一场 补记 candy 题目大意: 有两个超市,每个超市有\(n(n\le10^5)\)个糖,每个糖\(W\)元.每颗糖有一个愉悦度,其中,第一家商店中的第\(i\)颗 ...
- NOI.AC NOIP模拟赛 第二场 补记
NOI.AC NOIP模拟赛 第二场 补记 palindrome 题目大意: 同[CEOI2017]Palindromic Partitions string 同[TC11326]Impossible ...
- nowcoder(牛客网)提高组模拟赛第一场 解题报告
T1 中位数(二分) 这个题是一个二分(听说是上周atcoder beginner contest的D题???) 我们可以开一个数组b存a,sort然后二分b进行check(从后往前直接遍历check ...
- WC2019 全国模拟赛第一场 T1 题解
由于只会T1,没法写游记,只好来写题解了... 题目链接 题目大意 给你一个数列,每次可以任取两个不相交的区间,取一次的贡献是这两个区间里所有数的最小值,求所有取法的贡献和,对 \(10^9+7\) ...
- nowcoder(牛客网)普及组模拟赛第一场 解题报告
蒟蒻我可能考了一场假试 T1 绩点 这题没什么好说的,应该是只要会语言的就会做. T2 巨大的棋盘 一个模拟题吧qwq,但是要注意取模的时候先加上n或者m再取模,要不然会错的. #include< ...
- NOI.AC省选模拟赛第一场 T1 (树上高斯消元)
link 很容易对于每个点列出式子 \(f_{x,y}=(f_{x,y-1}+f_{x,y}+f_{x,y+1}+f_{x+1,y})/4\)(边角转移类似,略) 这个转移是相互依赖的就gg了 不过你 ...
- CH Round #52 - Thinking Bear #1 (NOIP模拟赛)
A.拆地毯 题目:http://www.contesthunter.org/contest/CH%20Round%20%2352%20-%20Thinking%20Bear%20%231%20(NOI ...
- CH Round #49 - Streaming #4 (NOIP模拟赛Day2)
A.二叉树的的根 题目:http://www.contesthunter.org/contest/CH%20Round%20%2349%20-%20Streaming%20%234%20(NOIP 模 ...
- CH Round #48 - Streaming #3 (NOIP模拟赛Day1)
A.数三角形 题目:http://www.contesthunter.org/contest/CH%20Round%20%2348%20-%20Streaming%20%233%20(NOIP模拟赛D ...
随机推荐
- python学习笔记 - assert用法
[转自]http://blog.sina.com.cn/s/blog_76e94d210100vz37.html 1.assert语句用来声明某个条件是真的. 2.如果你非常确信某个你使用的列表中 ...
- 268. Missing Number
Given an array containing n distinct numbers taken from 0, 1, 2, ..., n, find the one that is missin ...
- leetcode 120 Triangle ----- java
Given a triangle, find the minimum path sum from top to bottom. Each step you may move to adjacent n ...
- VC调用系统的调色板
调用系统的调色板可以用到ChooseColor这个函数,这个函数传入一个参数为CHOOSECOLOR结构体的指针,其函数原型为 BOOL ChooseColor(LPCHOOSECOLOR lpCC) ...
- dir:一行代码,提取出所有视频文件名称及路径
某次,部门接到一个任务,要求对公司现有的视频文件资料做一个统计整理分类的工作. 领导召集开会,问:两周时间够用吗? 统计整理分类工作的第一步骤是把视频文件名称来源类别信息录入到 excel 表格中,才 ...
- Linux 用户态和内核态
1.特权级特权级用来管理和控制程序执行.如Intel x86架构的CPU,有0~3四个特权级,0级最高,3级最低.硬件在执行每条指令时都会检查指令具有的特权级.硬件提供了特权级使用机制,对操作系统来说 ...
- 怎么提高OCR文字识别软件的识别正确率
在OCR文字识别软件当中,ABBYY FineReader是比较好用的程序之一,但再好的识别软件也不能保证100%的识别正确率,用户都喜欢软件的正确率高一些,以减轻识别后修正的负担,很多用户也都提过这 ...
- C#中的预处理指令
C#中的预处理指令 作为预处理中的一对:#region name ,#endregion可能是大家使用得最多的,我也常用它来进行代码分块,在一个比较长的cs文件中,这么做确实是一件可以让你使代码更清晰 ...
- shell脚本实例-菜单样例
1.9.1 实例需求 用户在进行Linux系统管理的过程中,经常需要用到查看进程的信息.用户的信息等常用的功能.本例针对这一需求,使用shell编程实现基本的系统管理 功能.通过本程序,可以按照要求实 ...
- shell之条件表达式
conditional expressions are used by the [[ compound command and the test and [ builtin commands. ari ...