2023.01.06 模拟赛小结
2023.01.06 模拟赛小结
更好的阅读体验戳此进入
今天的题是 sssmzy 组的,他的题居然有一道签到,真的,我哭死。
这次模拟赛没有太寄。
赛时思路
T1
要求你构造一个序列,满足对应条件。具体条件略。
比较简单,构造方案很多且均不难想到,总之简单想一下就知道了,妥妥的签到题。
Code
#define _USE_MATH_DEFINES
#include <bits/stdc++.h>
#define PI M_PI
#define E M_E
#define npt nullptr
#define SON i->to
#define OPNEW void* operator new(size_t)
#define ROPNEW(arr) void* Edge::operator new(size_t){static Edge* P = arr; return P++;}
using namespace std;
mt19937 rnd(random_device{}());
int rndd(int l, int r){return rnd() % (r - l + 1) + l;}
bool rnddd(int x){return rndd(1, 100) <= x;}
typedef unsigned int uint;
typedef unsigned long long unll;
typedef long long ll;
typedef long double ld;
template < typename T = int >
inline T read(void);
int N, K;
basic_string < int > ans;
deque < int > values;
int main(){
freopen("structure.in", "r", stdin);
freopen("structure.out", "w", stdout);
N = read(), K = read();
if(!K){
for(int i = 1; i <= N * 2; ++i)printf("%d%c", i, i == N * 2 ? '\n' : ' ');
exit(0);
}
ans += {K + 1, 1};
for(int i = 1; i <= N * 2; ++i)if(i != K + 1 && i != 1)values.emplace_back(i);
while(!values.empty())ans += {values.front(), values.back()}, values.pop_front(), values.pop_back();
for(auto it = ans.begin(); it != ans.end(); ++it)printf("%d%c", *it, it == prev(ans.end()) ? '\n' : ' ');
fprintf(stderr, "Time: %.6lf\n", (double)clock() / CLOCKS_PER_SEC);
return 0;
}
template < typename T >
inline T read(void){
T ret(0);
int flag(1);
char c = getchar();
while(c != '-' && !isdigit(c))c = getchar();
if(c == '-')flag = -1, c = getchar();
while(isdigit(c)){
ret *= 10;
ret += int(c - '0');
c = getchar();
}
ret *= flag;
return ret;
}
checker
#define _USE_MATH_DEFINES
#include <bits/stdc++.h>
#define PI M_PI
#define E M_E
#define npt nullptr
#define SON i->to
#define OPNEW void* operator new(size_t)
#define ROPNEW(arr) void* Edge::operator new(size_t){static Edge* P = arr; return P++;}
using namespace std;
mt19937 rnd(random_device{}());
int rndd(int l, int r){return rnd() % (r - l + 1) + l;}
bool rnddd(int x){return rndd(1, 100) <= x;}
typedef unsigned int uint;
typedef unsigned long long unll;
typedef long long ll;
typedef long double ld;
template < typename T = int >
inline T read(void);
int a[111000];
int main(){
while(true){
FILE* input = fopen("in.txt", "w");
int N = rndd(1, 50000), K = rndd(0, N >> 1);
fprintf(input, "%d %d\n", N, K);
fcloseall();
system("./std < in.txt > my.out");
FILE* output = fopen("my.out", "r");
for(int i = 1; i <= N * 2; ++i)fscanf(output, "%d", &a[i]);
fcloseall();
ll sum1(0), sum2(0);
for(int i = 1; i <= N * 2; i += 2)sum1 += abs(a[i + 1] - a[i]), sum2 += a[i + 1] - a[i];
sum2 = abs(sum2);
if(sum1 - sum2 == K << 1)printf("Accept!\n");
else printf("Wrong!\n"), exit(0);
}
fprintf(stderr, "Time: %.6lf\n", (double)clock() / CLOCKS_PER_SEC);
return 0;
}
template < typename T >
inline T read(void){
T ret(0);
int flag(1);
char c = getchar();
while(c != '-' && !isdigit(c))c = getchar();
if(c == '-')flag = -1, c = getchar();
while(isdigit(c)){
ret *= 10;
ret += int(c - '0');
c = getchar();
}
ret *= flag;
return ret;
}
T2
给定
读完题之后想到的思路是把所有
Code
#define _USE_MATH_DEFINES
#include <bits/stdc++.h>
#define PI M_PI
#define E M_E
#define npt nullptr
#define SON i->to
#define OPNEW void* operator new(size_t)
#define ROPNEW(arr) void* Edge::operator new(size_t){static Edge* P = arr; return P++;}
using namespace std;
mt19937 rnd(random_device{}());
int rndd(int l, int r){return rnd() % (r - l + 1) + l;}
bool rnddd(int x){return rndd(1, 100) <= x;}
typedef unsigned int uint;
typedef unsigned long long unll;
typedef long long ll;
typedef long double ld;
template < typename T = int >
inline T read(void);
ll K, MOD;
ll pow10[51000000];
ll sum(1);
ll ans(LONG_LONG_MAX);
const ll mxCnt = (ll)(1e8 + 114514);
int main(){
freopen("date_struct.in", "r", stdin);
freopen("date_struct.out", "w", stdout);
K = read < ll >(), MOD = read < ll >();
ll lim = mxCnt / (MOD - 1);
pow10[0] = 1;
for(int i = 1; i <= MOD - 2; ++i)pow10[i] = pow10[i - 1] * 10 % MOD, (sum += pow10[i]) %= MOD;
ll cur(0);
for(int i = 0; i <= MOD - 2; ++i){
(cur += pow10[i]) %= MOD;
if(cur == K)printf("%d\n", i + 1), exit(0);
}cur = 0;
for(int i = 0; i <= MOD - 2; ++i){
(cur += pow10[i]) %= MOD;
for(int j = 1; j <= lim; ++j)
if((cur + sum * j % MOD) % MOD == K)ans = min(ans, (ll)i + 1 + (MOD - 1) * j);
}
printf("%lld\n", ans);
fprintf(stderr, "Time: %.6lf\n", (double)clock() / CLOCKS_PER_SEC);
return 0;
}
template < typename T >
inline T read(void){
T ret(0);
int flag(1);
char c = getchar();
while(c != '-' && !isdigit(c))c = getchar();
if(c == '-')flag = -1, c = getchar();
while(isdigit(c)){
ret *= 10;
ret += int(c - '0');
c = getchar();
}
ret *= flag;
return ret;
}
T3
有一个长为
给你
求操作完成后数列的最大值的期望。
一眼没什么思路,糊了个暴力
Code
#define _USE_MATH_DEFINES
#include <bits/stdc++.h>
#define PI M_PI
#define E M_E
#define npt nullptr
#define SON i->to
#define OPNEW void* operator new(size_t)
#define ROPNEW(arr) void* Edge::operator new(size_t){static Edge* P = arr; return P++;}
using namespace std;
mt19937 rnd(random_device{}());
int rndd(int l, int r){return rnd() % (r - l + 1) + l;}
bool rnddd(int x){return rndd(1, 100) <= x;}
typedef unsigned int uint;
typedef unsigned long long unll;
typedef long long ll;
typedef long double ld;
template < typename T = int >
inline T read(void);
int N, Q;
int a[110000];
int l[5100], r[5100];
ld p[5100];
ld expect(0.0);
class SegTree{
private:
int tr[110000 << 2];
int lz[110000 << 2];
#define LS (p << 1)
#define RS (LS | 1)
#define MID ((gl + gr) >> 1)
public:
void Pushup(int p){
tr[p] = max(tr[LS], tr[RS]);
}
void Pushdown(int p){
if(!lz[p])return;
lz[LS] += lz[p], lz[RS] += lz[p];
tr[LS] += lz[p], tr[RS] += lz[p];
lz[p] = 0;
}
void Build(int p = 1, int gl = 1, int gr = N){
if(gl == gr)return tr[p] = a[gl = gr], void();
Build(LS, gl, MID), Build(RS, MID + 1, gr);
Pushup(p);
}
void Modify(int l, int r, int v = 1, int p = 1, int gl = 1, int gr = N){
if(l <= gl && gr <= r)return tr[p] += v, lz[p] += v, void();
Pushdown(p);
if(l <= MID)Modify(l, r, v, LS, gl, MID);
if(MID + 1 <= r)Modify(l, r, v, RS, MID + 1, gr);
Pushup(p);
}
int Query(void){
return tr[1];
}
}st;
void dfs(int dep = 1, ld cur = 1.0){
if(dep > Q){
expect += cur * (ld)st.Query();
return;
}
st.Modify(l[dep], r[dep]);
dfs(dep + 1, cur * p[dep]);
st.Modify(l[dep], r[dep], -1);
dfs(dep + 1, cur * (1.0 - p[dep]));
}
int main(){
freopen("expectation.in", "r", stdin);
freopen("expectation.out", "w", stdout);
N = read(), Q = read();
for(int i = 1; i <= N; ++i)a[i] = read();
st.Build();
for(int i = 1; i <= Q; ++i)l[i] = read(), r[i] = read(), scanf("%Lf", &p[i]);
dfs();
printf("%.9Lf\n", expect);
fprintf(stderr, "Time: %.6lf\n", (double)clock() / CLOCKS_PER_SEC);
return 0;
}
template < typename T >
inline T read(void){
T ret(0);
int flag(1);
char c = getchar();
while(c != '-' && !isdigit(c))c = getchar();
if(c == '-')flag = -1, c = getchar();
while(isdigit(c)){
ret *= 10;
ret += int(c - '0');
c = getchar();
}
ret *= flag;
return ret;
}
/*
5 2
1 7 2 4 3
1 3 0.500
2 2 0.500
5 2
281 280 279 278 282
1 4 1.000
1 4 0.000
3 5
1 2 3
1 3 0.500
2 2 0.250
1 2 0.800
1 1 0.120
2 2 0.900
*/
T4
Alice和Bob在一棵
他们轮流执行以下操作:
从当前树中选择一个白色节点
树上博弈论,较为阴间,赛时感觉暴力过于麻烦就没写,实际上这题的暴力精细实现的话也没有太麻烦。
正解
T2
尝试进行转化,对
然后套一下 BSGS 模板即可。
依然可以不使用光速幂,最终复杂度
注意过程中部分需要使用 __int128_t
。
Code
#define _USE_MATH_DEFINES
#include <bits/stdc++.h>
#define PI M_PI
#define E M_E
#define npt nullptr
#define SON i->to
#define OPNEW void* operator new(size_t)
#define ROPNEW(arr) void* Edge::operator new(size_t){static Edge* P = arr; return P++;}
using namespace std;
mt19937 rnd(random_device{}());
int rndd(int l, int r){return rnd() % (r - l + 1) + l;}
bool rnddd(int x){return rndd(1, 100) <= x;}
typedef unsigned int uint;
typedef unsigned long long unll;
typedef long long ll;
typedef long double ld;
template < typename T = int >
inline T read(void);
ll K, M;
unordered_map < ll, ll > mp;
ll qpow(ll a, ll b){
ll ret(1), mul(a);
while(b){
if(b & 1)ret = (__int128_t)ret * mul % M;
b >>= 1;
mul = (__int128_t)mul * mul % M;
}return ret;
}
int main(){
K = read < ll >(), M = read < ll >();
ll lim = (ll)ceil(sqrt(M));
for(int i = 1; i <= lim; ++i)mp.insert({(__int128_t)(9 * K + 1) % M * qpow(10, i) % M, i});
for(int i = 1; i <= lim; ++i){
ll val = qpow(10, (ll)i * lim);
if(mp.find(val) != mp.end())printf("%lld\n", (ll)i * lim - mp[val]), exit(0);
}
fprintf(stderr, "Time: %.6lf\n", (double)clock() / CLOCKS_PER_SEC);
return 0;
}
template < typename T >
inline T read(void){
T ret(0);
int flag(1);
char c = getchar();
while(c != '-' && !isdigit(c))c = getchar();
if(c == '-')flag = -1, c = getchar();
while(isdigit(c)){
ret *= 10;
ret += int(c - '0');
c = getchar();
}
ret *= flag;
return ret;
}
T3
有一个很高妙的转化,即因为区间无交错,所以考虑将所有操作区间抽象成森林的结构,可以加一个操作
转化之后就是经典的 树形DP 了,首先一个较为显然的状态就是
对于没有子节点的点自然为
最终答案即为枚举所有
同时需要注意对于
Code
#define _USE_MATH_DEFINES
#include <bits/stdc++.h>
#define PI M_PI
#define E M_E
#define npt nullptr
#define SON i->to
#define OPNEW void* operator new(size_t)
#define ROPNEW(arr) void* Edge::operator new(size_t){static Edge* P = arr; return P++;}
using namespace std;
mt19937 rnd(random_device{}());
int rndd(int l, int r){return rnd() % (r - l + 1) + l;}
bool rnddd(int x){return rndd(1, 100) <= x;}
typedef unsigned int uint;
typedef unsigned long long unll;
typedef long long ll;
typedef long double ld;
template < typename T = int >
inline T read(void);
struct Edge{
Edge* nxt;
int to;
OPNEW;
}ed[11000];
ROPNEW(ed);
Edge* head[5100];
int N, Q;
int A[110000];
struct Mdf{
int L, R;
double P;
int mx;
}mdf[5100];
double dp[5100][5100];
double ans(0.0);
class SegTree{
private:
int tr[110000 << 2];
#define LS (p << 1)
#define RS (LS | 1)
#define MID ((gl + gr) >> 1)
public:
void Pushup(int p){
tr[p] = max(tr[LS], tr[RS]);
}
void Build(int p = 1, int gl = 1, int gr = N){
if(gl == gr)return tr[p] = A[gl = gr], void();
Build(LS, gl, MID), Build(RS, MID + 1, gr);
Pushup(p);
}
int Query(int l, int r, int p = 1, int gl = 1, int gr = N){
if(l <= gl && gr <= r)return tr[p];
if(gr < l || r < gl)return -1;
return max(Query(l, r, LS, gl, MID), Query(l, r, RS, MID + 1, gr));
}
}st;
void TreeDP(int p = 1, int fa = 0){
for(auto i = head[p]; i; i = i->nxt)if(SON != fa)TreeDP(SON, p);
for(int j = 0; j <= Q; ++j){
double D1 = mdf[p].P, D2 = (1.0 - mdf[p].P);
for(auto i = head[p]; i; i = i->nxt){
if(SON == fa)continue;
if(j + mdf[p].mx - mdf[SON].mx - 1 >= 0)D1 *= dp[SON][min(Q, j + mdf[p].mx - mdf[SON].mx - 1)];
if(j + mdf[p].mx - mdf[SON].mx >= 0)D2 *= dp[SON][min(Q, j + mdf[p].mx - mdf[SON].mx)];
}dp[p][j] = j ? D1 + D2 : D2;
}
}
int main(){
N = read(), Q = read();
for(int i = 1; i <= N; ++i)A[i] = read();
st.Build();
for(int i = 1; i <= Q; ++i)mdf[i].L = read(), mdf[i].R = read(), scanf("%lf", &mdf[i].P), mdf[i].mx = st.Query(mdf[i].L, mdf[i].R);
mdf[++Q] = {1, N, 0.0, st.Query(1, N)};
sort(mdf + 1, mdf + Q + 1, [](const Mdf &a, const Mdf &b)->bool{return a.L == b.L ? a.R > b.R : a.L < b.L;});
for(int i = 1; i <= Q; ++i)
for(int j = i - 1; j; --j)
if(mdf[j].L <= mdf[i].L && mdf[i].R <= mdf[j].R){
head[i] = new Edge{head[i], j}, head[j] = new Edge{head[j], i};
break;
}
TreeDP();
for(int i = 0; i <= Q; ++i)ans += (dp[1][i] - (i ? dp[1][i - 1] : 0.0)) * (double)(mdf[1].mx + i);
printf("%.6lf\n", ans);
fprintf(stderr, "Time: %.6lf\n", (double)clock() / CLOCKS_PER_SEC);
return 0;
}
template < typename T >
inline T read(void){
T ret(0);
int flag(1);
char c = getchar();
while(c != '-' && !isdigit(c))c = getchar();
if(c == '-')flag = -1, c = getchar();
while(isdigit(c)){
ret *= 10;
ret += int(c - '0');
c = getchar();
}
ret *= flag;
return ret;
}
T4
Code
UPD
update-2022__ 初稿
本文作者:tsawke
本文链接:https://www.cnblogs.com/tsawke/p/17032818.html
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步