2020牛客多校第八场K题
__int128(例题:2020牛客多校第八场K题)
题意:
有n道菜,第i道菜的利润为\(a_i\),且有\(b_i\)盘。你要按照下列要求给顾客上菜。
1.每位顾客至少有一道菜
2.给顾客上菜时,都必须从第一道菜开始,上连续的编号的菜,例如,你可能给一位顾客 上的菜为第一道,第二道,第三道,但是不能为只上第二道而不上第一道,或者第一道,第三道中间缺少第二道。求这些菜能够容纳的最大顾客数,并且求出在容纳最多顾客时的利润最大为多少。
数据范围
\(1\le n\le 10^{5}\),\(-10^{9}\le a_i\le10^{9}\),\(1\le b_i\le10^{5}\)
输入
2
3
2 -1 3
3 2 1
4
3 -2 3 -1
4 2 1 2
输出
Case #1: 3 8
Case #2: 4 13
说明
对于测试案例1,最大访问者数量为3,一种可能的解决方案是:
第一位访客获得食物1,利润是2。
第二位访客获得食物1,利润为2。
第三位访客获得食物1 +食物2 +食物3,利润为2 +(-1)+ 3。
题解
1.由于必须上第一道菜,因此第一道菜的数量即为最大招待顾客的数量。
2.计算最大利润时,采用贪心策略,在满足条件时,优先选取利润最大的即可,由于每次选取 1~i的连续的菜,可以先求出前i份菜的利润的前缀和。然后按利润前缀和大小排序,从大到小遍历,对于前n份菜利润为sum的贡献计算方法为,求出前n种菜剩余的菜的最小的数量 min,让答案累加\(min*sum\),然后将前n种菜的数量都减去min,这个操作可以用线段树来进行区间查询和修改。
3.由于答案最大值为 \(10^{19}\),稍微超过了long long int的范围,因此c++要采用 __int128,__int128为128位的整数,是gcc的扩展,大小范围在 [\(2^{127},2^{127}-1\)].且在windows环境下的gcc不支持编译,在linux环境下的gcc支持编译,不过没有对应的输入输出函数,要自定义输入输出函数。
__int128模板
`inline __int128 read(){//读入
__int128 x=0,f=1;
char ch=getchar();
while(ch<'0'||ch>'9'){
if(ch=='-')
f=-1;
ch=getchar();
}
while(ch>='0'&&ch<='9')
x=x*10+ch-'0';
ch=getchar();
}
return x*f;
}
inline void fprint(__int128 x){//输出。
if(x<0){
putchar('-');
x=-x;
}
if(x>9)
fprint(x/10);
putchar(x%10+'0');
}`
题目完整code
#include<bits/stdc++.h>
#define scd(x) scanf("%d",&x)
#define scdd(x,y) scanf("%d%d",&x,&y)
#define sclld(x) scanf("%lld",&x)
#define sclldd(x,y) scanf("%lld%lld",&x,&y)
#define print(x) cout<<x<<endl
#define CLR(a) memset(a,0,sizeof(a))
#define ls (rt<<1)
#define rs (rt<<1|1)
using namespace std;
typedef long long ll;
typedef double db;
const double eps = 1e-7;
const double pi = acos(-1.0);
const int inf = 0x3f3f3f3f;
const ll linf = 0x3f3f3f3f3f3f3f3f;
const ll mod = ll(1e9)+7;
const int maxn=int(2e5)+500;
template<typename tp>inline tp lowbit(tp x){return x&(-x);}
template<typename tp_>void debug(tp_ x) {cout<<"*********** "<<x<<" *************"<<endl;}
template<typename tp1_,typename tp2_>void debug(tp1_ x,tp2_ y){cout<<"*********** "<<x<<" "<<y<<" *************"<<endl;}
struct P{int x;__int128 y;P(){}P(int _x,__int128 _y):x(_x),y(_y){}};
bool cmp(const P elem1,const P elem2){
if(elem1.y==elem2.y)return elem1.x>elem2.x;
return elem1.y<elem2.y;
}
__int128 a[maxn];
__int128 b[maxn];
P sum[maxn];
int Case = 1;
struct node{
__int128 sum;
__int128 lazy;
}tree[maxn<<2];
inline __int128 read(){
__int128 x=0,f=1;
char ch=getchar();
while(ch<'0'||ch>'9'){
if(ch=='-')
f=-1;
ch=getchar();
}
while(ch>='0'&&ch<='9'){
x=x*10+ch-'0';
ch=getchar();
}
return x*f;
}
inline void fprint(__int128 x){
if(x<0){
putchar('-');
x=-x;
}
if(x>9)
fprint(x/10);
putchar(x%10+'0');
}
void build(int rt,int l,int r){
tree[rt].sum=tree[rt].lazy=0;
if(l==r){
tree[rt].sum = b[l];
return ;
}
int mid = (l+r)>>1;
build(ls,l,mid);
build(rs,mid+1,r);
tree[rt].sum = min(tree[ls].sum,tree[rs].sum);
}
void pushdown(int rt,int l,int r){
if(tree[rt].lazy){
__int128 val = tree[rt].lazy;
tree[ls].sum-=val;
tree[rs].sum-=val;
tree[ls].lazy += val;
tree[rs].lazy += val;
tree[rt].lazy = 0;
}
}
void update(int rt,int l,int r,int L,int R,__int128 val){
if(l>=L&&r<=R){
tree[rt].sum -=val;
tree[rt].lazy += val;
return ;
}
pushdown(rt,l,r);
int mid= (l+r)>>1;
if(L<=mid)update(ls,l,mid,L,R,val);
if(R>mid)update(rs,mid+1,r,L,R,val);
tree[rt].sum = min(tree[ls].sum,tree[rs].sum);
}
__int128 ask(int rt,int l,int r,int L,int R){
if(l>=L&&r<=R){
return tree[rt].sum;
}
pushdown(rt,l,r);
int mid = (l+r)>>1;
__int128 res = inf;
if(L<=mid)res=min(res,ask(ls,l,mid,L,R));
if(R>mid)res=min(res,ask(rs,mid+1,r,L,R));
return res;
}
void solve() {
int n;
scd(n);
sum[0].y = 0;
for(int i=1;i<=n;i++){
a[i]=read();
sum[i].x=i;
sum[i].y=a[i]+sum[i-1].y;
}
for(int i=1;i<=n;i++)b[i]=read();
sort(sum+1,sum+n+1,cmp);
__int128 visit = b[1];
build(1,1,n);
__int128 ans = 0;
for(int i=n;i>=1;i--){
__int128 cnt = ask(1,1,n,1,sum[i].x);
ans+= cnt*sum[i].y;
update(1,1,n,1,sum[i].x,cnt);
}
printf("Case #%d: ",Case);
fprint(visit);
printf(" ");
fprint(ans);
printf("\n");
}
void Main(){
int _;
scd(_);
while(_--)
solve(),++Case;
}
//#define _Debug
int main(){
#ifdef _Debug
freopen("data.in","r",stdin);
clock_t b = clock();
Main();
clock_t e = clock();
cout<<"the time of cost is "<<(e-b)<<endl;
#else
Main();
#endif
return 0;
}
2020牛客多校第八场K题的更多相关文章
- 2019牛客多校第八场 F题 Flowers 计算几何+线段树
2019牛客多校第八场 F题 Flowers 先枚举出三角形内部的点D. 下面所说的旋转没有指明逆时针还是顺时针则是指逆时针旋转. 固定内部点的答案的获取 anti(A)anti(A)anti(A)或 ...
- Distance(2019年牛客多校第八场D题+CDQ+树状数组)
题目链接 传送门 思路 这个题在\(BZOJ\)上有个二维平面的版本(\(BZOJ2716\)天使玩偶),不过是权限题因此就不附带链接了,我也只是在算法进阶指南上看到过,那个题的写法是\(CDQ\), ...
- Explorer(2019年牛客多校第八场E题+线段树+可撤销并查集)
题目链接 传送门 题意 给你一张无向图,每条边\(u_i,v_i\)的权值范围为\([L_i,R_i]\),要经过这条边的条件是你的容量要在\([L_i,R_i]\),现在问你你有多少种容量使得你可以 ...
- 2019牛客多校第四场 I题 后缀自动机_后缀数组_求两个串de公共子串的种类数
目录 求若干个串的公共子串个数相关变形题 对一个串建后缀自动机,另一个串在上面跑同时计数 广义后缀自动机 后缀数组 其他:POJ 3415 求两个串长度至少为k的公共子串数量 @(牛客多校第四场 I题 ...
- 暴力三维树状数组求曼哈顿距离求最值——牛客多校第八场D
涉及的知识点挺多,但是大多是套路 1.求曼哈顿距离的最值一般对所有情况进行讨论 2.三维树状数组用来求前缀最大值 /* 有一个三维坐标系(x,y,z),取值范围为[1,n],[1,m],[1,h],有 ...
- 线段树区间离散化维护按秩合并并查集(可撤销)——牛客多校第八场E
模板题..去网上学了可撤销的并查集.. /* 给定一个无向图,边的属性为(u,v,l,r),表示<u,v>可以通过的size为[l,r] 求出有多少不同的size可以从1->n 把每 ...
- 单调栈(最大子矩形强化版)——牛客多校第八场A
求01矩阵里有多少个不同的1矩阵 首先预处理出pre[i][j]表示i上面连续的1个数,对每行的高度进行单调栈处理 栈里的元素维护两个值:pre[i][j]和向前延伸最多能维护的位置pos 然后算贡献 ...
- 牛客多校第八场 G Gemstones 栈/贪心
题意: 对于一个序列,把可以把连着三个相同的字母拿走,问最多拿走多少组. 题解: 直接模拟栈,三个栈顶元素相同则答案+1,并弹出栈 #include<bits/stdc++.h> usin ...
- 牛客多校第八场 C CDMA 线性代数:沃尔什矩阵
题意: 构造出一个由1和-1组成的$2^k*2^k$的矩阵,使得矩阵任意两列内积为0 题解: 数学知识,沃尔什矩阵.沃尔什矩阵的特性被CDMA(码分多址)采用,使得编码成为无线信号的频段和振幅之外的第 ...
随机推荐
- DVWA学习记录 PartⅢ
CSRF 1. 题目 CSRF,全称Cross-site request forgery,翻译过来就是跨站请求伪造,是指利用受害者尚未失效的身份认证信息(cookie.会话等),诱骗其点击恶意链接或者 ...
- java 数据结构(五):数据结构简述
1.数据结构概述数据结构(Data Structure是一门和计算机硬件与软件都密切相关的学科,它的研究重点是在计算机的程序设计领域中探讨如何在计算机中组织和存储数据并进行高效率的运用,涉及的内容包含 ...
- 数据可视化之powerBI基础(十五)Power BI同步切片器,你知道怎么用吗?
https://zhuanlan.zhihu.com/p/67932754 在PowerBI报表中,切片器绝对是最常用的控件了,利用它可以进行各种维度的动态切换,同一个页面中的所有图表可以同步响应:利 ...
- SQLAlchemy01 /SQLAlchemy去连接数据库、ORM介绍、将ORM模型映射到数据库中
SQLAlchemy01 /SQLAlchemy去连接数据库.ORM介绍.将ORM模型映射到数据库中 目录 SQLAlchemy01 /SQLAlchemy去连接数据库.ORM介绍.将ORM模型映射到 ...
- unity-热更-InjectFix(一)
1 C#热更新预备知识 1.1 mono.cecil注入 使用Mono.Cecil实现IL代码注入 注入之后修改dll,新增mdb文件: 注意,待了解参数注释打开会报错: 1.2 InjectFix ...
- 网页排名算法PagaRank
网页排名算法PageRank PageRank,网页排名,又叫做网页级别.是一种利用网页之间的超链接数据进行计算的方法.它是由Google的两位创始人提出的. 对于用户而言,网页排名一般是比较主观的, ...
- java 正则提取字符串中的电话号码
public static void test2() { String str = "张三:13539558064,李四:15626829748,赵六:13718952204"; ...
- Ethical Hacking - Web Penetration Testing(5)
LOCAL FILE INCLUSION Allows an attacker to read ANY file on the same server. Access files outside ww ...
- 12C oracle 12.1.0.2版本打补丁
从support 1454618.1文档 中可以下载到各版本的补丁和PSU. 此次采用的是28349311 版本号的psu 和28440711 版本号的jvm.opatch版本号6880880_12 ...
- 关于ES6的let和const
变量 var存在的问题 可以重复声明 无法限制修改 没有块级作用域 (在全局范围内有效) 存在变量提升 const/let 不可以重复声明 let a = 1; let a = 2; var b = ...