2019牛客暑期多校训练营(第一场)

Wesley13
• 阅读 640

Solutions


A:Equivalent Prefixes

题意:

若$\text{RMQ(u,l,r)=RMQ(v,l,r)}$ for all $1{\leq}l{\leq}r{\leq}m$,则为$equivalent$

其中$\text{RMQ(w,l,r)}$的定义是${[l,r]}$区间内最小数的下标

给两个数组,每组元素各不相同,求最大的$p{\leq}n\ where\ \{a_1,a_2,{\ldots}a_p\} are\ equivalent$ 

单调栈思路:

处理出每一位左边第一个比它小的下标,然后相同的话可取,不同不可取。

假如当前处理$a_i$,(也就是$\{a_1,a_2,{\ldots}a_{i-1}\}$满足条件),假设比它第一个小的是$last[i]$,那么$last[i]$右边的$RMQ$一定是$a_i$,这样的话,$last[i]$左边的$RMQ$一定与新数无关,因为仅仅$last[i]$处就已经比$a_i$小了。

可以利用单调栈求每一位左边第一个比它小的下标

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 const int maxn=1e5+10;
 4 
 5 int n,a[maxn],b[maxn],ida[maxn],idb[maxn];
 6 
 7 stack<pair<int,int> > s;
 8 
 9 int main() {
10     while(~scanf("%d",&n)) {
11         s.push(make_pair(-1,-1));
12         for(int i=0;i<n;i++) {
13             scanf("%d",&a[i]);
14             while(!s.empty()&&s.top().first>a[i]) s.pop();
15             ida[i]=s.top().second;
16             s.push(make_pair(a[i],i));
17         }
18         while(!s.empty()) s.pop();
19         s.push(make_pair(-1,-1));
20         for(int i=0;i<n;i++) {
21             scanf("%d",&b[i]);
22             while(!s.empty()&&s.top().first>b[i]) s.pop();
23             idb[i]=s.top().second;
24             s.push(make_pair(b[i],i));
25         }
26         int cur=0;
27         while(cur<n) {
28             if(ida[cur]==idb[cur]) cur++;
29             else break;
30         }
31         printf("%d\n",cur);
32     }
33     return 0;
34 }

二分+分治+ST思路:

 通过$ST$表预处理出$RMQ$,$p$具有单调性,可以二分判断是否满足条件,然后询问$1{\sim}mid$最小值,再分治

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 const int maxn=100010;
 4 
 5 int dpa[maxn][20],dpb[maxn][20],a[maxn],b[maxn];
 6 int mm[maxn],n,ida[maxn],idb[maxn];
 7 
 8 void init() {
 9     mm[0]=-1;
10     for(int i=1;i<=n;i++) {
11         mm[i]=((i&(i-1))==0)?mm[i-1]+1:mm[i-1];
12         dpa[i][0]=a[i];
13         dpb[i][0]=b[i];
14     }
15     for(int j=1;j<=mm[n];j++) {
16         for(int i=1;i+(1<<j)-1<=n;i++) {
17             dpa[i][j]=min(dpa[i][j-1],dpa[i+(1<<(j-1))][j-1]);
18             dpb[i][j]=min(dpb[i][j-1],dpb[i+(1<<(j-1))][j-1]);
19         }
20     }
21 }
22 
23 int rmqa(int x,int y) {
24     int k=mm[y-x+1];
25     return ida[min(dpa[x][k],dpa[y-(1<<k)+1][k])];
26 }
27 
28 int rmqb(int x,int y) {
29     int k=mm[y-x+1];
30     return idb[min(dpb[x][k],dpb[y-(1<<k)+1][k])];
31 }
32 
33 bool dfs(int l,int r) {
34     if(l>=r) return true;
35     int x=rmqa(l,r);
36     int y=rmqb(l,r);
37     if(x!=y) return false;
38     else return dfs(1,x-1)&&dfs(y+1,r);
39 }
40 
41 bool check(int mid) {
42     return dfs(1,mid);
43 }
44 
45 int main() {
46     while(~scanf("%d",&n)) {
47         for(int i=1;i<=n;i++) {
48             scanf("%d",&a[i]);
49             ida[a[i]]=i;
50         }
51         for(int i=1;i<=n;i++) {
52             scanf("%d",&b[i]);
53             idb[b[i]]=i;
54         }
55         init();
56         int l=1,r=n,ans=1;
57         while(l<=r) {
58             int mid=(l+r)>>1;
59             if(check(mid)) {
60                 ans=mid;
61                 l=mid+1;
62             } else {
63                 r=mid-1;
64             }
65         }
66         printf("%d\n",ans);
67     }
68 }

B: Integration

题意:

给出$n$个数,$a_1,a_2,\ldots,a_n$,求$\frac1\pi\int_0^{\infty}\frac1{\prod_{i=0}^n(a_i^2+x^2)}dx$

因为分母是连乘积,所以可以列项拆分。推几项可以得出规律

具体参考这里

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 const int maxn=1010;
 4 typedef long long ll;
 5 const int mod=1e9+7;
 6 
 7 ll a[maxn];
 8 int n;
 9 
10 ll q_power(ll a,ll b) {
11     ll ans=1,tmp=a%mod;
12     while(b) {
13         if(b&1) ans=ans*tmp%mod;
14         tmp=tmp*tmp%mod;
15         b>>=1;
16     }
17     return ans;
18 }
19 
20 int main() {
21     while(~scanf("%d",&n)) {
22         for(int i=1;i<=n;i++) scanf("%lld",&a[i]);
23         ll ans=0;
24         for(int i=1;i<=n;i++) {
25             ll tmp=1;
26             for(int j=1;j<=n;j++) {
27                 if(i==j) continue;
28                 tmp=tmp*(((a[j]*a[j]%mod)-(a[i]*a[i]%mod)+mod)%mod)%mod;
29             }
30             tmp=tmp*2*a[i]%mod;
31             ans=(ans+q_power(tmp,mod-2))%mod;
32         }
33         printf("%lld\n",ans);
34     }
35 }

C:Euclidean Distance

题意:

给出$A$点的$n$维坐标$A=(a_1/m,a_2/m,\ldots,a_n/m)$,请找一点$P=(p_1,p_2,\ldots,p_n)$,满足一下条件

  • $p_1,p_2,\ldots,p_n\ \in{\mathbb{R}}$
  • $p_1,p_2,\ldots,p_n{\geq}0$
  • $p_1+p_2+\ldots+p_n=1$

使得$\sum_{i=1}^n(a_i/m-p_i)^2$最小。

思路:

参考这里

首先,我们可以将$p$的坐标放大m倍,那么答案可以转化为:

$\sum_{i=1}^n(a_i/m-p_i)^2\ =\ \sum_{i=1}^n[m(a_i/m-p_i)]^2\ \Rightarrow\ \frac1{m^2}\sum_{i=1}^n(a_i-mpi)^2$

考虑$a_i$为每个矩形的高,减去的$m$相当于削减矩形的高度,首先将$a_1,a_2,\ldots,a_n$降序排列。证:

削减高度较大的收益会比削减高度较小的收益更大,(比如负数,减去数反而变大了)。

如图中的绿线,把矩形一点一点往下推,然后下一步一起推,道理同上。

若不能进一步往下推,就把整体减一点点。可以预处理出前缀和。然后判断够不够把$a_i$前面的全部推平。

具体细节可以见上面大佬的博客。

2019牛客暑期多校训练营(第一场)

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 #define ll long long
 4 int n,m,a[10010],sum[10010];
 5 
 6 bool cmp(int a,int b) {
 7     return a>b;
 8 }
 9 
10 int main() {
11     while(~scanf("%d%d",&n,&m)) {
12         for(int i=1;i<=n;i++) scanf("%d",&a[i]);
13         sort(a+1,a+n+1,cmp);
14         sum[0]=-m;
15         for(int i=1;i<=n;i++) {
16             sum[i]=sum[i-1]+a[i];
17         }
18         ll frac1,frac2;
19         int now=n;
20         for(int i=1;i<n;i++) {
21             if(sum[i]>a[i+1]*i) {
22                 now=i;
23                 break;
24             }
25         }
26         frac1=1ll*sum[now]*sum[now]*now;
27         frac2=now*now;
28         for(int i=now+1;i<=n;i++) {
29             frac1+=1ll*a[i]*a[i]*frac2;
30         }
31         frac2=frac2*1ll*m*m;
32         ll tmp=__gcd(frac1,frac2);
33         if(frac2/tmp==1) printf("%lld\n",frac1/tmp);
34         else if(!(frac1/tmp)) printf("0\n");
35         else printf("%lld/%lld\n",frac1/tmp,frac2/tmp);
36     }
37 }

E:ABBA

题意:

给出$2(n+m)$个$A,B$构成序列,求可以划分为$n$个$AB$,$m$个$BA$的序列的个数。

思路:

首先贪心的想,前$n$个$A$是$AB$的$A$,前$m$个$B$是$BA$的$B$。

然后$dp[i][j]$表示放了$i$个$A$,$j$个$B$的方案数。

若$i{\leq}n$,$A$可以直接放

若$j{\leq}m$,$B$可以直接放

若$i-n{\leq}j$,即现在用于组合$BA$的$A$的数量小于$B$的数量,$A$可以放

若$j-m{\leq}i$,即现在用于组合$AB$的$B$的数量小于$A$的数量,$B$可以放

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 const int mod=1e9+7;
 4 
 5 int dp[2010][2010];
 6 int n,m;
 7 
 8 int main() {
 9     while(~scanf("%d%d",&n,&m)) {
10         for(int i=0;i<=n+m;i++) {
11             for(int j=0;j<=n+m;j++)
12                 dp[i][j]=0;
13         }
14         dp[0][0]=1;
15         for(int i=0;i<=n+m;i++) {
16             for(int j=0;j<=n+m;j++) {
17                 if(i-n<j) dp[i+1][j]=(dp[i+1][j]+dp[i][j])%mod;
18                 if(j-m<i) dp[i][j+1]=(dp[i][j+1]+dp[i][j])%mod;
19             }
20         }
21         printf("%d\n",dp[n+m][n+m]);
22     }
23 }

F:Random Point in Triangle

题意:

给出三角形三点坐标$A,B,C$,求

$$E=max\{S_{PAB},S_{PBC},S_{PCA}\}$$

输出$36{\times}E$

思路:

暂时只知道$ans=22{\times}S_{ABC}$,证明不会,记一下叉积求面积。$S_{ABC}=\frac12\vec{AB}{\times}\vec{AC}$

 1 #include<iostream>
 2 #include<cstdio>
 3 #define lowbit(x) (x&(-x))
 4 #include<cstring>
 5 using namespace std;
 6 typedef long long ll;
 7 
 8 int main() {
 9     ll x1,y1,x2,y2,x3,y3;
10     while(~scanf("%lld%lld%lld%lld%lld%lld",&x1,&y1,&x2,&y2,&x3,&y3)) {
11         ll ans=abs(x2*y3-x2*y1-x1*y3-y2*x3+x1*y2+y1*x3);
12         printf("%lld\n",22/2*ans);
13     }
14 }

I:Points Division

题意:

给出$n$个点,每个点有两个权值$a_i,b_i$,然后把这些点划分为两个集合,若点$i$在$A$中,则贡献为$a_i$,若若点$i$在$B$中,则贡献为$b_i$,

要求集合$A$在集合$B$的左上方。求划分后的最大权值和。

思路:

参考这里

显然可以用一条递增的折线划分为两个区域。然后就开始玄学了。

$dp[i]$表示第$i$个点在折线上时的最大权值和。

当加入$i$时,如果$y_i$大于之前的$y_j$,则$i$的贡献为$a_i$,如果$y_i$小于之前的$y_j$,则$i$的贡献为$b_i$

那么$dp[j]=
\begin{cases}
dp[j]+b_i & j<i,y_j>y_i\\
dp[j]+a_i & j<i,y_j<y_i
\end{cases}$

$dp[i]=b_i+max_{1{\leq}<i,y_j<y_i}dp[j]$

式子可以用线段树维护区间最值(但是我比较迷,看不太懂)

还需要多加一个点。因为dp初始化为0。

(仅复习下线段树区间修改时的最大值,注意可能L>R,所以判一下)

  1 #include<bits/stdc++.h>
  2 using namespace std;
  3 const int maxn=1e5+10;
  4 typedef long long ll;
  5 const int inf=0x3f3f3f3f;
  6 
  7 struct node{
  8     ll x,y,a,b;
  9 }p[maxn];
 10 ll t[maxn];
 11 int tot,cnt,n;
 12 
 13 bool cmp(node a,node b) {
 14     if(a.x==b.x) return a.y>b.y;
 15     else return a.x<b.x;
 16 }
 17 
 18 ll maxx[maxn<<2],lazy[maxn<<2];
 19 
 20 void pushup(int rt) {
 21     maxx[rt]=max(maxx[rt<<1],maxx[rt<<1|1]);
 22 }
 23 
 24 void build(int l,int r,int rt) {
 25     lazy[rt]=maxx[rt]=0;
 26     if(l==r) return ;
 27     int mid=(l+r)>>1;
 28     build(l,mid,rt<<1);
 29     build(mid+1,r,rt<<1|1);
 30     pushup(rt);
 31 }
 32 
 33 void pushdown(int rt) {
 34     if(lazy[rt]) {
 35         lazy[rt<<1]+=lazy[rt];
 36         lazy[rt<<1|1]+=lazy[rt];
 37         maxx[rt<<1]+=lazy[rt];
 38         maxx[rt<<1|1]+=lazy[rt];
 39         lazy[rt]=0;
 40     }
 41 }
 42 
 43 void updata1(int pos,ll val,int l,int r,int rt) {
 44     if(l==r) {
 45         maxx[rt]=max(maxx[rt],val);
 46         return ;
 47     }
 48     int mid=(l+r)>>1;
 49     pushdown(rt);
 50     if(pos<=mid) updata1(pos,val,l,mid,rt<<1);
 51     else updata1(pos,val,mid+1,r,rt<<1|1);
 52     pushup(rt);
 53 }
 54 
 55 void updata2(int L,int R,int val,int l,int r,int rt) {
 56     if(L<=l&&R>=r) {
 57         maxx[rt]+=val;
 58         lazy[rt]+=val;
 59         return ;
 60     }
 61     int mid=(l+r)>>1;
 62     pushdown(rt);
 63     if(L<=mid) updata2(L,R,val,l,mid,rt<<1);
 64     if(R>mid) updata2(L,R,val,mid+1,r,rt<<1|1);
 65     pushup(rt);
 66 }
 67 
 68 ll query(int L,int R,int l,int r,int rt) {
 69     if(L<=l&&R>=r) {
 70         return maxx[rt];
 71     }
 72     ll ans=0;
 73     int mid=(l+r)>>1;
 74     pushdown(rt);
 75     if(L<=mid) ans=max(ans,query(L,R,l,mid,rt<<1));
 76     if(R>mid) ans=max(ans,query(L,R,mid+1,r,rt<<1|1));
 77     return ans;
 78 }
 79 
 80 int main() {
 81     while(~scanf("%d",&n)) {
 82         cnt=0;
 83         for(int i=1;i<=n;i++) {
 84             scanf("%lld%lld%lld%lld",&p[i].x,&p[i].y,&p[i].a,&p[i].b);
 85             t[++cnt]=p[i].y;
 86         }
 87         sort(p+1,p+n+1,cmp);
 88         sort(t+1,t+cnt+1);
 89         tot=unique(t+1,t+cnt+1)-t-1;
 90         for(int i=1;i<=n;i++) {
 91             p[i].y=lower_bound(t+1,t+tot+1,p[i].y)-t+1;
 92         }
 93         tot++;
 94         build(1,tot,1);
 95         for(int i=1;i<=n;i++) {
 96             ll ans=query(1,p[i].y,1,tot,1);
 97             updata1(p[i].y,ans+p[i].b,1,tot,1);
 98             updata2(1,p[i].y-1,p[i].a,1,tot,1);
 99             if(p[i].y+1<=tot) updata2(p[i].y+1,tot,p[i].b,1,tot,1);
100         }
101         printf("%lld\n",maxx[1]);
102     }
103 }

J:Fraction Comparision

题意:

判断$\frac{x}a$与$\frac{y}b$的大小关系

  • $0{\leq}x,y{\leq}10^{18}$
  • $1{\leq}a,b{\leq}10^9$

思路:

  • 不能用浮点数判断(精度不够?)

  • 把$\frac{x}a$写成${\lfloor}{\frac{x}a}{\rfloor}+{x\%a}$ 先比较整数部分,再交叉相乘比较分数

    1 #include<bits/stdc++.h> 2 using namespace std; 3 typedef long long ll; 4 5 void out(ll x,ll y) { 6 if(x==y) puts("="); 7 else if(x>y) puts(">"); 8 else puts("<"); 9 } 10 11 12 int main() { 13 ll x,a,y,b; 14 while(~scanf("%lld%lld%lld%lld",&x,&a,&y,&b)) { 15 ll A=x/a; 16 ll B=y/b; 17 x%=a; 18 y%=b; 19 if(A!=B) out(A,B); 20 else out(xb,ya); 21 } 22 return 0; 23 }

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Easter79 Easter79
3年前
swap空间的增减方法
(1)增大swap空间去激活swap交换区:swapoff v /dev/vg00/lvswap扩展交换lv:lvextend L 10G /dev/vg00/lvswap重新生成swap交换区:mkswap /dev/vg00/lvswap激活新生成的交换区:swapon v /dev/vg00/lvswap
Wesley13 Wesley13
3年前
PS打包实现AI图像论文,英伟达在实时视频PS之路上越走越远
  编辑:Panda  !(https://nimg.ws.126.net/?urlhttp%3A%2F%2Fdingyue.ws.126.net%2F2020%2F1028%2F88b5e6ffj00qiwi6t000id000l000kup.jpg&thumbnail650x2147483647&quality80&typejpg)
九路 九路
4年前
一个int数组,奇数排前,偶数排后,算法实现
//奇数排前,偶数排后publicstaticvoidsortOdd(intdata){intl0;intrdata.length1;while(l<r){while(l<r&&datar%20)
Wesley13 Wesley13
3年前
2019字节跳动ACM程序设计冬令营题解
喜欢这张图~!(https://oscimg.oschina.net/oscnet/15565e2766de7fef626499419e6f29708cc.jpg)day1【E√,F√,G√,L√】  水题/老套路题。【B,H,I】  待补。【A√】  题意:有$k(\\leq1000)$个选手和$N(
Stella981 Stella981
3年前
Extjs标悬浮在grid单元格上时,显示单元格中内容的值
{xtype:'gridcolumn',text:'名称',width:120,dataIndex:'name',editor:{xtype:'textfield',allowBlank:false,maxLength:20},renderer:function(v,m,r,i){
Stella981 Stella981
3年前
Android蓝牙连接汽车OBD设备
//设备连接public class BluetoothConnect implements Runnable {    private static final UUID CONNECT_UUID  UUID.fromString("0000110100001000800000805F9B34FB");
Stella981 Stella981
3年前
Google自己的下拉刷新组件SwipeRefreshLayout(黄柳淞
Google自己的下拉刷新组件SwipeRefreshLayout(黄柳淞转载请注明出处)!图片(http://r.photo.store.qq.com/psb?/V13qOEX02lnBLg/R5jBW34TiSoJEl1GXYQlL2.bL2PkdZeQ6ps1WmiYc5A!/o/dGIAAAAAAAAA&ek1&kp1&pt0&bo
暗箭伤人 暗箭伤人
1年前
【www.ithunter.club】 20230922下午
不容易的2023年,我们一起努力【www.ithunter.club】(2023092208:00:00.8872062023092216:00:00.887206)1.人事招聘专员数名(可选远程或入职)2.招聘向坐标东京Yahoo、Shift、L
Python进阶者 Python进阶者
11个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这