[比赛|考试]nowcoder NOIPpj组第二场
nowcoder NOIPpj组第二场
370pts/400pts(100,100,100,70) rank3
给自己的反思:前3题都A了,T4O(N^2)不会就是不会(没准是我懒得推了),DP了70pts,没什么好反思的,反正就是保持状态吧(qtmd我就是个喜欢做水题的菜鸡)反正以后水题保证一遍AC难题保证暴力就行
以下是题解
T1.你好诶加币
牛牛刚学习了输入输出,他遇到了一道这样的题目。
输入2个整数a和b
保证输入的a和b在long long范围之内,即满足
-9223372036854775808 <= a, b <= 9223372036854775807
计算a+b的值,即这两个数字的和。
如果a+b在long long范围之内,即满足
-9223372036854775808 <= a + b <= 9223372036854775807
那么输出一行一个整数表示a+b的结果。
如果a+b不在long long范围之内,即越界了,那么输出"hello, %lld\n",包含引号。
用python来cheatforscore是一个好方法,现在luogu,nowcoder都资磁python3了
既然是python那么就简单了直接模拟就行,如下
def main():
s = input().split()
res = int(s[0]) + int(s[1])
if res <= 9223372036854775807 and res >= -9223372036854775808:
print(res)
else:
print('''"hello, %lld\\n"''')
main()
唠唠正解吧,可以用__int128(至少在nowcoder可以)(zwfymqz大佬残忍地没有开类型转换还是炸了)。打个高精就能过。懒得写代码。
T2. 最后一次
牛牛最近学习了质数的概念。 质数指在大于1的自然数中,除了1和它本身以外不再有其他因数。 输入一个n,输出小于等于n最大的质数。
可以从n往下枚举并判断这个数是否为素数。
暴力判素数可做,记得全程开long long,注意是全程
暴力判断素数代码(17ms)
#include <bits/stdc++.h>
using namespace std;
long long n;
bool isprime(long long x)
{
for (long long i = 2; i * i <= x; i++)
if (x % i == 0)
return false;
return true;
}
int main()
{
cin >> n;
for (long long i = n; i >= 1; i--)
{
if (isprime(i))
{
cout << i << endl;
return 0;
}
}
return 0;
}
然而我的Miller-Rabin也过了(3ms),注意乘法不能直接乘会炸long long,要写龟速乘法(程序中的turtle())
#include <bits/stdc++.h>
using namespace std;
long long turtle(long long x, long long y, long long p)
{
long long res = 0;
while (y > 0)
{
if (y & 1)
res = (res + x) % p;
(x <<= 1) %= p;
y >>= 1;
}
return res;
}
long long qpow(long long x, long long y, long long p)
{
long long res = 1;
while (y > 0)
{
if (y & 1)
res = turtle(res, x, p);
x = turtle(x, x, p);
y >>= 1;
}
return res;
}
bool Miller_Rabin(long long N, long long a)
{
if (N == a)
return true;
if (N == 46856248255981LL || N < 2 || N % a == 0)
return false;
long long d = N - 1;
while (!(d & 1))
d >>= 1;
long long t = qpow(a, d, N);
while (d != N - 1 && t != 1 && t != N - 1)
{
t = turtle(t, t, N);
d <<= 1;
}
return (t == N - 1) || ((d & 1) == 1);
}
inline bool prime(long long x)
{
return (Miller_Rabin(x, 2) && Miller_Rabin(x, 3) && Miller_Rabin(x, 7) && Miller_Rabin(x, 61) && Miller_Rabin(x, 24251));
}
int main()
{
long long n;
cin >> n;
for (long long i = n; i >= 1; i--)
{
if (prime(i))
{
cout << i << endl;
return 0;
}
}
return 0;
}
T3.选择颜色
n个人排成一个环形,每个人要从c种颜色中选择一个。 牛牛希望相邻的人选择的颜色是不同的,问有多少种方案。输出方案数对10007取模的结果。 人是有顺序的,环旋转同构算不同的方案。
结论题。有通项公式,我不会推。这里给出递推公式的推的过程。
考虑一个有n个节点的环,按照逆时针顺序依次给环上节点染色。
第一个节点有c种颜色,第2~n-1个节点由于不能与他上一个节点的颜色重复,每个节点都有(c-1)种颜色。(其实这句话没用)
考虑第n个节点,设情况数量为f(n)。如果他两边的节点(第1个和第n-1个)颜色相同,那么他有c-1种颜色。否则有c-2种颜色。
对于两边珠子颜色相同的情况,把第(n-1)个节点和第n个节点删除,剩余n-2个节点环--第(n-2)个节点和第1个节点连接在一起。由于第(n-1)个节点的颜色和第1个节点颜色相同,所以一定是合法的。这个环有f(n-2)个节点,也就是有f(n-2)种情况,新加的珠子有c-1种,即一共有(c-1)f(n-2)种。
对于两边珠子颜色不同的情况,把该节点(第n个节点)删除,剩余n-1个节点的环--第(n-1)个节点和第1个节点连接在一起,由于第(n-1)个节点的颜色和第1个节点颜色不相同,所以这个环也一定是合法的。这个环有f(n-1)个节点,新加的珠子有(c-2)种颜色,所以一共有(c-2)f(n-1)种颜色。
根据加法原理和乘法原理,f(n)=(c-2)f(n-1)+(c-1)f(n-2)。(感谢@_Lancy的指正)这是一个2阶线性递推式,可以O(n)推出。如果使用矩阵快速幂可以加速到O(logn)。(原谅我用四个变量写矩阵。。。。。)
#include <bits/stdc++.h>
#define p 10007
using namespace std;
struct matrix
{
int a, b, c, d;
};
matrix operator*(const matrix &x, const matrix &y)
{
return (matrix){(x.a * y.a + x.b * y.c) % p, (x.a * y.b + x.b * y.d) % p, (x.c * y.a + x.d * y.c) % p, (x.c * y.b + x.d * y.d) % p};
}
matrix qpow(matrix x, long long y)
{
matrix res;
res.a = res.d = 1;
res.b = res.c = 0;
while (y > 0)
{
if (y & 1)
res = res * x;
x = x * x;
y >>= 1;
}
return res;
}
int main()
{
long long n, c;
cin >> n >> c;
matrix start3;
start3.a = c * (c - 1) * (c - 2) % p;
start3.c = c * (c - 1) % p;
start3.b = 0;
start3.d = 0;
matrix d;
d.a = c - 2;
d.b = c - 1;
d.c = 1;
d.d = 0;
cout << (qpow(d, n - 3) * start3).a << endl;
return 0;
}
T4.合法括号序列
键盘上有左括号(,右括号),和退格键-,共三个键。 牛牛希望按键n次,使得输入的字符串恰好一个合法的括号序列。 每按一次左括号(,字符串末尾追加一个左括号( 每按一次右括号),字符串末尾追加一个右括号) 每按一次退格键-,会删掉字符串的最后一个字符, 特别的,如果字符串为空,牛牛也可以按退格,但是什么都不会发生。 输出方案数对p取模,注意p可能不是质数。 只要按键方法不同,就是不同的方案,即使得到的序列一样。
因为我很弱,所以这里只给出70pts的solution
区间上的问题考虑dp。但是这题尝试设状态的时候发现会有后效性---你在按backspace后不知道你删除了什么。所以这题考虑倒着做:按backspace键相当于取消掉后面最近的一个非backspace字符的效果,在下面我们称之为forwardspace。(backspace的效果可以累积,自己脑洞打开想一下这是什么个操作)这样就可以设状态了。令f[i][j][k]代表后i个字符,括号深度为j,此时有k个累积的backspace的效果。我们不难很难得到状态转移如下:(以下均为填表法)
| 状态 | 加一个左括号 | 加一个右括号 | 加一个forwardspace |
|---|---|---|---|
f[i][0][0] |
f[i-1][0][1] |
f[i-1][1][0],f[i-1][0][1] |
|
f[i][j][0] |
f[i-1][j-1][0],f[i-1][j][1] |
f[i-1][j+1][0],f[i-1][j][1] |
|
f[i][j][k] |
f[i-1][j][k+1] |
f[i-1][j][k+1] |
f[i-1][j][k-1] |
f[i][0][0] = f[i-1][0][1] + f[i-1][1][0] + f[i-1][0][1]
f[i][j][0] = f[i-1][j-1][0] + f[i-1][j][1] + f[i-1][j+1][0] + f[i-1][j][1]
f[i][j][k] = f[i-1][j][k+1] + f[i-1][j][k+1] + f[i-1][j][k-1]
注意到当前字符串没有文本按backspace没有效果,转化一下就是最后剩余多余的forwardspace没有效果。答案是所有f[n][0][k]的和。
然后我们开三维的1000数组,这样开空间会炸,注意到i只与i-1有关,开滚动数组即可,把第1维滚掉。
时间复杂度:O(n3),空间复杂度O(n2),期望得分70pts
我的代码如下
#include <bits/stdc++.h>
using namespace std;
int f[2][1010][1010];
int n, p, ans;
int main()
{
scanf("%d%d", &n, &p);
f[0][0][0] = 1;
for (int i = 1, t = 1; t <= n; i ^= 1, t++)
{
f[i][0][0] = (f[i ^ 1][1][0] + 2 * f[i ^ 1][0][1]) % p;
for (int j = 1; j <= n; j++)
f[i][j][0] = (f[i ^ 1][j - 1][0] + f[i ^ 1][j + 1][0] + 2 * f[i ^ 1][j][1]) % p;
for (int k = 1; k <= n; k++)
for (int j = 0; j <= n; j++)
f[i][j][k] = (f[i ^ 1][j][k + 1] * 2 + f[i ^ 1][j][k - 1]) % p;
}
for (int k = 0; k <= n; k++)
(ans += f[1 & n][0][k]) %= p;
printf("%d\n", ans);
return 0;
}
[比赛|考试]nowcoder NOIPpj组第二场的更多相关文章
- 2018.9.9 nowcoder 普及组第一场
2018.9.9 nowcoder 普及组第一场 C-括号 题目大意:一个只包含左右括号的字符串\(S\),希望删掉S中若干个字符,使得剩下的字符串是一个合法的括号串,有多少不同的方案. Soluti ...
- [比赛|考试]nowcoder NOIP提高组组第二场
160/300pts,rank16(100,30,30) 在自身找毛病,首先做题感觉还不不够认真,比如T3那个我一开始审出来了,然后tmd忘了...gg...T1AC没啥好赞美的,T2T3暴力没拿全啊 ...
- [比赛|考试]nowcoder 小白月赛7
牛客小白月赛7 比赛地址.本次比赛我切了8道(ACM赛制),rank(20). 反思:刚入手ACM赛,光追求刺激了,没有总结ACM赛制的经验.是应该多提交>..还是少提交...小白赛还有两道不会 ...
- 牛客网提高组第二场---solution
T1 方差 根据题目要求将式子先写出来注意下面式子中的 $n$ 全部都是 $n-1$$$\begin{aligned}ans&=n^2\times \frac{1}{n}\times \sum ...
- 牛客网NOIP赛前集训营-普及组(第二场)和 牛客网NOIP赛前集训营-提高组(第二场)解题报告
目录 牛客网NOIP赛前集训营-普及组(第二场) A 你好诶加币 B 最后一次 C 选择颜色 D 合法括号序列 牛客网NOIP赛前集训营-提高组(第二场) A 方差 B 分糖果 C 集合划分 牛客网N ...
- Nowcoder 提高组练习赛-R7
Nowcoder 提高组练习赛-R7 https://www.nowcoder.com/acm/contest/179#question 中间空了两场,因为实在是太难了... 第五场的第二题好像还比较 ...
- 2014百度之星预赛(第二场)——Best Financing
2014百度之星预赛(第二场)--Best Financing Problem Description 小A想通过合理投资银行理財产品达到收益最大化.已知小A在未来一段时间中的收入情况,描写叙述为两个 ...
- 2018牛客暑期ACM多校训练营第二场(有坑未填)
第二场终于等来学弟 开始(被队友带飞)的开心(被虐)多校之旅 A run A题是一个递推(dp?)+前缀和 因为看数据量比较大 就直接上前缀和了 一个比较简单的递推 没有太多难点 签到题 需要注意 ...
- 2018 计算之道初赛第二场 阿里巴巴的手机代理商(困难)(反向可持久化Trie)
阿里巴巴的手机代理商(困难) 阿里巴巴的手机代理商正在研究 infra 输入法的新功能.他们需要分析单词频率以改进用户输入法的体验.于是需要你在系统内核里面写一个 API. API 有如下功能: 添加 ...
随机推荐
- c# linq查询语句详细使用介绍
本文介绍Linq的使用方法 linq介绍 LINQ只不过是实现IEnumerable和IQueryable接口的类的扩展方法的集合. LINQ可以查询IEnumerable集合或者IQueryable ...
- 10-23C#基础--特殊集合(stack、queue、hashtable)
特殊集合一:stack集合--堆集合 1.定义:堆集合是集合中一种特殊的类,在Stack中也有许多方法和属性,下面一一列举: 命名格式:Stack ss=new Stack(); 2.如何添加数据:p ...
- 10-08C#基础--进制转换
(一).数制 计算机中采用的是二进制,因为二进制具有运算简单,易实现且可靠,为逻辑设计提供了有利的途径.节省设备等优点,为了便于描述,又常用八.十六进制作为二进制的缩写.一般计数都采用进位计数,其特点 ...
- [Python Study Notes]pynput实现对键盘控制与监控
''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''' ...
- for xml path 按分类合并行数据
) as itemnum FROM ( SELECT Sonum, (SELECT ItemNum+',' FROM testtb WHERE Sonum=A.Sonum FOR XML ...
- solr-用mmseg4j配置同义词索引和检索(IKanlyzer需要修改源码适应solr接口才能使用同义词功能)
概念说明:同义词大体的意思是指,当用户输入一个词时,solr会把相关有相同意思的近义词的或同义词的term的语段内容从索引中取出,展示给用户,提高交互的友好性(当然这些同义词的定义是要在配置文件中事先 ...
- Codeforces 719E (线段树教做人系列) 线段树维护矩阵
题面简洁明了,一看就懂 做了这个题之后,才知道怎么用线段树维护递推式.递推式的递推过程可以看作两个矩阵相乘,假设矩阵A是初始值矩阵,矩阵B是变换矩阵,求第n项相当于把矩阵B乘了n - 1次. 那么我们 ...
- 高性能MySQL笔记-第5章Indexing for High Performance-003索引的作用
一. 1. 1). Indexes reduce the amount of data the server has to examine.2). Indexes help the server av ...
- Linux kgdb命令
一.简介 kgdb是一种源码级的Linux内核调试器.使用kgdb调试内核时,需要结合gdb一起使用,使用他们可以对内核进行单步调试,设置断点,观察变量.寄存器的值等与应用调试相关的功能.然而也有其限 ...
- poj2287 Tian Ji -- The Horse Racing
传送门 分析 这个题和传统的田忌赛马不一样的地方就是多了平局情况,所有我们不难想到要用dp.我们先将两人的马均降序排列,用dpij表示考虑前i匹马,田忌有几匹马是按从大到小的顺序从头取的(剩下的是从尾 ...