题目传送门

W神爷的题解

数论

小 M 的算式

【问题描述】

小 M 在做数学作业的时候遇到了一个有趣的问题:有一个长度为 n 的数字

串 S,小 M 需要在数字之间填入若干个“+”和恰好一个“=”,使其成为一个

合法的等式。如对于 S=“2349”,可以通过添加 2个“+”和 1 个“=”成为

“2+3+4=9”。

小 M 发现有些数字串是无法通过添加符号成为一个合法的等式的,她想知

道对于每一个给定的数字串 S,是否可以通过添加符号使之成为一个合法的等

式(允许前导 0)?

【输入】

第一行为数据组数 T,表示有 T组输入数据。

接下来 T行每行一个数字串 S。

【输出】

对于每组数据,若 S可以成为合法的等式,输出“Yes”,否则输出

“No”,以单行回车隔开。

【输入输出样例】

equation.in

4

2349

233233

122323

2344322322

equation.out

Yes

Yes

No

Yes

【输入输出样例解释】

2+3+4=9

233=233

2+34=4+3+2+2+3+22

【数据范围】

对于 50%的数据:1 ≤ T ≤ 3,1 ≤ n ≤ 4。

对于 100%的数据:1 ≤ T ≤ 5,1 ≤ n ≤ 10。

简单的搜索

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
inline long long read()
{
long long f=,ans=;char c;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){ans=ans*+c-'';c=getchar();}
return f*ans;
}
char str[];
long long t,len,n1,n2;
long long cnt1[][],cnt2[][],s1,s2;
bool ff;
void dfs2(long long pos,long long ans,long long dq,long long dqs)
{
if(dqs+dq>s1) return;
if(ff) return;
if(pos==n2+)
{
s2=dqs+dq;
if(s1==s2)
ff=;
return;
}
cnt2[ans][++cnt2[ans][]]=str[pos]-'';
dq=dq*+str[pos]-'';
dfs2(pos+,ans,dq,dqs);
dq=(dq-(str[pos]-''))/;
cnt2[ans][]--; long long t=dq;
cnt2[ans+][++cnt2[ans+][]]=str[pos]-'';
dq=str[pos]-'';
dfs2(pos+,ans+,dq,dqs+t);
cnt2[ans+][]--;
dq=;
return;
}
void dfs1(long long pos,long long ans,long long dq,long long dqs)
{
if(ff) return;
if(pos==n1+)
{
s1=dqs+dq;
dfs2(n1+,,,);
return;
}
cnt1[ans][++cnt1[ans][]]=str[pos]-'';
dq=dq*+str[pos]-'';
dfs1(pos+,ans,dq,dqs);
cnt1[ans][]--;
dq=(dq-(str[pos]-''))/; long long t=dq;
cnt1[ans+][++cnt1[ans+][]]=str[pos]-'';
dq=str[pos]-'';
dfs1(pos+,ans+,dq,dqs+t);
cnt1[ans+][]--; dq=t;
return;
}
int main()
{
t=read();
while(t--)
{
ff=false;
scanf("%s",str+);
len=strlen(str+);
for(long long i=;i<len;i++)
{
n1=i,n2=len;
dfs1(,,,);
if(ff){cout<<"Yes"<<endl;break;}
}
if(!ff)cout<<"No"<<endl;
}
}

小R的调度

【问题描述】

小 R 是一名 OIer,她刚刚学习了逆序对的相关知识:对于一个长度为 n的

序列 A,其逆序对数定义为满足 i < j 且 A[i] > A[j]的(i, j)对数。

现在小 R 有一个长度为 n的 01序列 A,小 R每次可以交换 当前 A[i]与 A[j]

的值,并需要付出 c[i] + c[j]的代价。在若干轮(可以为空)交换后,小 R的得

分为最终 A序列的逆序对数减去她在交换中付出的代价之和。

小 R 想让最后的得分尽量大,你能帮帮她吗?

【输入】

第一行为一个正整数 n。

第二行为一个长度为 n的字符串,表示 01序列 A。

第三行为 n个整数 c[i],表示交换的代价参数。

【输出】

输出最大得分。

【输入输出样例】

inverse.in

6

101010

1 1 1 1 1 1

inverse.out

7

【样例解释】

交换 A[2]和 A[5],付出 1 + 1 = 2 的代价。

最终序列为 111000,逆序对数为 9,得分为 9 - 2 = 7。

【数据范围】

对于 100%的数据:1 ≤ n ≤ 2501,0 ≤ c[i] ≤ 100。

这是w神爷的题解:

注意到每次交换只有当两位置的元素不同才有意义,进一步分析可知序列的每个位置最多被交换一次,而每个从 0 换成 1 的位置与每个从 1 换成 0 的位置是一一对应的。

我们可以设计一个动态规划算法,设 f(i,j,k) 表示序列的前 i 个位置已经确定下来,前 i 个位置中有 j 个位置从 0 交换成了 1,有 k个位置从 1 交换成了 0。

转移时枚举第 i + 1 个位置有没有发生交换,根据初始 A 1 到 A i有多少个 1,以及 j、k 的值可以计算出第 i+1 个位置确定为 0 产生的逆序对贡献。

简称就是一个dp,可以想到转移为从0->1 与 1->0

然后就瞎搞

但是为什么可以dp呢

因为它不需要管后面发生了什么

最后维护的是差值

有加有减才能出来最后答案

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
inline int read(){
int f=,ans=;char c;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){ans=ans*+c-'';c=getchar();}
return f*ans;
}
int n,dp[][],cur,co,A[],inf=<<-;
char str[];
int main(){
n=read();
scanf("%s",str+);
for(int i=;i<=n;i++) A[i]=read();
memset(dp,-,sizeof(dp));
dp[][n]=;
for(int i=;i<=n;i++){
cur^=;
for(int j=;j<=*n;j++){
int ans=dp[cur^][j];
if(ans==-inf) continue;
if(str[i]==''){
dp[cur][j]=max(dp[cur][j],ans);
dp[cur][j-]=max(dp[cur][j-],ans+co-A[i]+j-n);
}
if(str[i]==''){
dp[cur][j]=max(dp[cur][j],ans+co+j-n);
dp[cur][j+]=max(dp[cur][j+],ans-A[i]);
}
}
if(str[i]=='') co++;
}
cout<<dp[cur][n];
}

小 C  的机器人

【问题描述】

小 C 的机器人工厂里有 m个机器人,标号为 1 到 m号。给定一棵 n 个节点

的有根树,根结点为 1号点,第 i个机器人初始在第 p[i]号点。树上每个节点有

一盏灯,初始灯都是灭着的。树上每条边的距离均为 1,且机器人每次移动的

距离均为 1。

你是小 C 工厂的技术顾问,需要写一个程序模拟 q次操作,操作共 3种类

型:

操作 1:1 l r x,表示将标号在 l 和 r之间的所有机器人移动到 x号节点。

操作 2:2 l r x,表示将标号在 l 和 r之间的所有机器人向根结点移动 x步。

若移动步数超过机器人到根节点的距离,则机器人会停在根节点。

操作 3:3 x,表示将 x号机器人当前所在节点的灯的亮灭状态取反,并查

询当前所在节点到其子树内所有亮着灯的节点的距离之和。

【输入】

第一行为两个正整数 n,m。

第二行至第 n行每行两个正整数 u、v,表示一条从 u到 v 的树边。

第 n+1 行为 m 个正整数,第 i 个正整数表示 p[i]。

第 n+2 行为一个正整数 q。

接下来 q行表示 q 个操作,具体格式见样例。

【输出】

输出若干行,第 i行的整数表示第 i 个操作 3的答案。

【输入输出样例】

robot.in

5 4

1 2

1 3

3 4

3 5

1 1 1 1

8

1 2 2 4

1 3 3 5

3 3

3 2

2 2 2 1

3 2

3 3

3 2

robot.out

0

0

2

0

1

【数据范围】

对于 20%的数据满足:1 ≤ n、m、q ≤ 10。

另有 20%的数据满足:1 ≤ n ≤ 50。

另有 20%的数据满足:1 ≤ m ≤ 20。

对于 100%的数据满足:1 ≤ n、m、q ≤ 152501,1 ≤ u、v、p[i]、x ≤ n,1 ≤ l≤ r ≤ m

毒瘤题目,到最后也没有写完,还算错复杂度

将这个问题分解,不难想到其实就是两个子问题

机器人的移动与子树距离

第二个线段树用一下树链剖分的思想,十分简单

想一下dfs在树上的时间戳

若到达第i点的时间戳为k,并且它的子树大小为s,所以它在线段树中所

所以建两个线段树去维护,lca去加速往上跳

悲催的调试:第二个线段树写错,第一个线段树写错,lca写错

最后就完了

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
inline int read()
{
int f=,ans=;char c;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){ans=ans*+c-'';c=getchar();}
return ans*f;
} int p[],head[],tt[];
int cnt,deep[],fa[][],pd;
int n,m,pig,size[],seg[];
int rev[*],ans[*],q; struct node{
int u,v,nex;
}x[]; void dfs(int f,int fath)
{
size[f]=;
tt[f]=++pig;
seg[pig]=f;
deep[f]=deep[fath]+;
fa[f][]=fath;
for(int i=;(<<i)<=deep[f];i++) fa[f][i]=fa[fa[f][i-]][i-];
for(int i=head[f];i!=-;i=x[i].nex)
{
if(x[i].v==fath) continue;
dfs(x[i].v,f);
size[f]+=size[x[i].v];
}
return;
} int jump(int u,int h)
{
for(int i=;i>=;i--)
if(h-(<<i)>=) u=fa[u][i],h-=(<<i);
if(u==) return ;
return u;
} void add(int u,int v)
{
x[cnt].u=u,x[cnt].v=v,x[cnt].nex=head[u],head[u]=cnt++;
} void build_seg(int k,int l,int r)
{
if(l==r){
rev[k]=p[l];return;
}
int mid=l+r>>;
build_seg(k<<,l,mid);
build_seg(k<<|,mid+,r);
return;
}
void pushdown_seg(int k,int l,int r)
{
if(rev[k]!=)
{
ans[k<<]=ans[k<<|]=;
rev[k<<]=rev[k<<|]=rev[k];
rev[k]=;
}
if(ans[k]!=)
{
ans[k<<]+=ans[k];
ans[k<<|]+=ans[k];
ans[k]=;
}
return;
}
void change_seg(int k,int l,int r,int x,int y,int s)
{ if(x<=l&&r<=y){
if(pd==){
rev[k]=s;
ans[k]=;
}
else if(pd==){
ans[k]+=s;
}
return;
}
pushdown_seg(k,l,r);
int mid=l+r>>;
if(x<=mid) change_seg(k<<,l,mid,x,y,s);
if(mid<y) change_seg(k<<|,mid+,r,x,y,s);
return;
} int query_seg(int k,int l,int r,int x,int y)
{
if(x==l&&r==y){return jump(rev[k],ans[k]);}
pushdown_seg(k,l,r);
int mid=l+r>>;
if(x<=mid) return query_seg(k<<,l,mid,x,y);
if(mid<y) return query_seg(k<<|,mid+,r,x,y);
}
int sum[*];
int flag[*];
void change(int k,int l,int r,int x,int y)
{
if(l==x&&r==y){
if(flag[k]==) {
flag[k]=;
sum[k]=;
}else if(flag[k]==){
flag[k]=;
sum[k]=deep[seg[l]]-;
} return;
}
int mid=l+r>>;
if(x<=mid) change(k<<,l,mid,x,y);
if(mid<y) change(k<<|,mid+,r,x,y);
sum[k]=sum[k<<|]+sum[k<<];
flag[k]=flag[k<<]+flag[k<<|];
return;
}
int ff; int query(int k,int l,int r,int x,int y)
{ if(x<=l&&r<=y){
ff+=flag[k];
return sum[k];
}
int mid=l+r>>,st=;
if(x<=mid) st+=query(k<<,l,mid,x,y);
if(mid<y) st+=query(k<<|,mid+,r,x,y);
return st;
} int main()
{
memset(head,-,sizeof(head));
n=read(),m=read();
for(int i=;i<n;i++)
{
int u=read(),v=read();
add(u,v),add(v,u);
}
dfs(,);
for(int i=;i<=m;i++) p[i]=read();
build_seg(,,m);
q=read();
while(q--)
{
pd=read();
if(pd==||pd==){
int l=read(),r=read(),xx=read();
change_seg(,,m,l,r,xx);
}
if(pd==)
{
int st=read();
int pos=query_seg(,,m,st,st);
ff=;
int kk=query(,,n,tt[pos]+,tt[pos]+size[pos]-);
change(,,n,tt[pos],tt[pos]);
printf("%d\n",kk-ff*(deep[pos]-));
}
}
}

20181015 考试记录&数论的更多相关文章

  1. 【考试记录】4.8 Table ( 数论数学 --组合数 & 杨辉三角)

    陆陆续续的开始考很多的试,也会更新这些题目记录下来,免得做完了之后毫无印象,就这么水过去了(以前的考试都是如此,哎……) Table (T1) : 样例: 出于对数学题本能的恐惧考场上放弃了此题专攻T ...

  2. [总结] NOIP 前的考试记录

    sb博主又犯sb错误了! 他觉得以往模拟赛因为犯sb错误扔的分足足有1k分了! 于是他想记录一下自己犯的sb错误看看自己到底有多sb! 嗯就从今天开始吧 2018.9.28 1. 二分边界写错.骚什么 ...

  3. 20181022 考试记录&高级数据结构

    题目 W神爷的题解 高级数据结构 T1: 其实是一道easy题,$O(n^3log n)$ 也是能卡过去的,本着要的70分的心态,最后尽然A了. 如果是正解则是$O(n^3)$,当确定你要选择的列时, ...

  4. 【考试记录】Educational Codeforces Round 59 (Rated for Div. 2)

    本来准备划水,结果被垃圾题艹翻了…… T2题意: 定义一个数$x$的数字根$S(x)$为:将其各位数字相加得到一个新数,再将新数的数字和相加直到得到一个个位数,就是该数的数字根. 例如:$S(38)= ...

  5. 【考试记录】4.8 Path (网络流 —— 劲题)

    手抄代码 + 学习指针 + 冥思苦想一晚上终于——在一瞬间开窍了.果然题目都是这样:突破了一个点,一切都是柳暗花明. 题面描述: 样例: 这道题目,首先注意到给定的边的性质:这些边在平面上构成了一棵树 ...

  6. 【考试记录】2018 山东省队集训第一轮D4(雾)

    T1题意: 给你一个$n\times m$的矩阵$B$,求它能由最少多少个形如两个向量之积$(n\times 1)\times(1\times m)$的矩阵相加得到. 题解: 考虑上界,最多需要$mi ...

  7. CCF考试

    第八次CCF考试记录 代码还不知道对不对,过两天出成绩. 成绩出来了,310分. 100+100+100+10+0: 考试13:27开始,17:30结束,提交第4题后不再答题,只是检查前四题的代码 第 ...

  8. PHPEMS在线模拟考试系统 v4.2

    官网地址 :http://www.phpems.net/ 下载地址 : http://www.phpems.net/index.php?content-app-content&contenti ...

  9. 基于java开发jsp+ssm+mysql实现的在线考试系统 源码下载

    实现的关于在线考试的功能有:用户前台:用户注册登录.查看考试信息.进行考试.查看考试成绩.查看历史考试记录.回顾已考试卷.修改密码.修改个人信息等,后台管理功能(脚手架功能不在这里列出),科目专业管理 ...

随机推荐

  1. mysql bin log配置及查看

    mysql执行sql可以通过设置mysql bin 日志进行记录查看   mysql bin日志配置如下:   log_bin:on log_bin_basename:bin文件路径及名前缀(/var ...

  2. JS变量定义时连续赋值的坑!

    在定义变量时,可以将值相同的变量采用连续赋值的方式,如下代码: var a = b = c = ''; 其实这里面有一个很大很大的坑,以代码说明问题: <script language=&quo ...

  3. 从零开始的Python学习Episode 3——字符串格式化与for循环

    一.字符串格式化 利用一段注释记录想要输出的字符串格式,并用 %s . %d 或 %f 依次代替要输出的数据(%s代表字符串,%d代表数字,%f代表浮点数),然后在这段注释之后依次加上要输出的数据. ...

  4. 2018(容斥定理 HDU6286)

    2018 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  5. 吴恩达j机器学习之过拟合

    五.编程作业: 见:https://www.cnblogs.com/tommyngx/p/9933803.html

  6. LeetCode 240——搜索二维矩阵 II

    1. 题目 2. 解答 2.1. 方法一 从矩阵的左下角开始比较 目标值等于当前元素,返回 true: 目标值大于当前元素,j 增 1,向右查找,排除掉此列上边的数据(都比当前元素更小): 目标值小于 ...

  7. springMVC使用拦截器检查用户登录

    参考文章 编写拦截器类 package cultivate_web.interceptor; import javax.servlet.http.HttpServletRequest; import ...

  8. Map Reduce Application(Top 10 IDs base on their value)

    Top 10 IDs base on their value First , we need to set the reduce to 1. For each map task, it is not ...

  9. [Data Structures and Algorithms - 1] Introduction & Mathematics

    References: 1. Stanford University CS97SI by Jaehyun Park 2. Introduction to Algorithms 3. Kuangbin' ...

  10. Python3 小工具-TCP半连接扫描

    from scapy.all import * import optparse import threading def scan(ip,port): pkt=IP(dst=ip)/TCP(dport ...