题目链接

loj#2721. 「NOI2018」屠龙勇士

题解

首先可以列出线性方程组

方程组转化为在模p意义下的同余方程

因为不保证pp 互素,考虑扩展中国剩余定理合并

方程组是带系数的,我们要做的是在%p意义下把系数除过去,(系数为atk[i])

(atk[i],p[i]) 不等于1时无逆元,此时仍可能有解

很显然无解的情况就是

瞎jb猜的,无解的话就是%p[i]意义下atk[i] != 0 ,a[i] = 0

考虑原方程式ai = atk{i] * x + p[i] * y

方程两边同除gcd(pi,atki)解不变 此时atki,pi此时保证了atki与pi互素

若ai不能被整除也是无解的

反正也没有-1的情况....

扩展crt

着实被winXP下输出lld坑了一把....顺带被自己抄的splay坑了一把

/*
苟活者在淡红的血色中,会依稀看到微茫的希望
*/ #include<bits/stdc++.h>
using namespace std;
inline long long read() {
long long x;
scanf("%lld",&x);
return x;
}
int n,m;
#define LL long long
const int maxn = 500007;
LL a[maxn],p[maxn]; // x atk[i] = a[i] ( % p[i])
LL atk[maxn],tatk[maxn]; struct Splay {
#define fa(x) T[x].fa
#define ls(x) T[x].ch[0]
#define rs(x) T[x].ch[1]
#define root T[0].ch[1]
struct node
{
LL val,rev,siz,fa,ch[2];
}T[maxn];
LL tot;
void clear() {
tot = 0; root = 0;
for(LL i = 1; i <= maxn; i++) T[i].val = T[i].rev = T[i].siz = T[i].fa = T[i].ch[0] = T[i].ch[1] = 0;
}
LL ident(LL x){return T[fa(x)].ch[0]==x?0:1;}
void connect(LL x,LL fa,LL how){T[fa].ch[how]=x;T[x].fa=fa;}
void update(LL x){T[x].siz=T[ls(x)].siz+T[rs(x)].siz+T[x].rev;}
void rotate(LL x)
{
LL Y=T[x].fa,R=T[Y].fa;
LL Yson=ident(x),Rson=ident(Y);
LL B=T[x].ch[Yson^1];
connect(B,Y,Yson);
connect(Y,x,Yson^1);
connect(x,R,Rson);
update(Y);update(x);
}
void splay(LL x,LL to)
{
to=T[to].fa;
while(T[x].fa!=to)
{
if(T[fa(x)].fa==to) rotate(x);
else if(ident(x)==ident(fa(x))) rotate(fa(x)),rotate(x);
else rotate(x),rotate(x);
}
}
LL newnode(LL fa,LL val) {
T[++tot].fa=fa;
T[tot].val=val;
T[tot].rev=T[tot].siz=1;
return tot;
}
LL find(LL val)
{
LL now=root;
while(1)
{
if(T[now].val==val) {splay(now,root);return now;}
LL nxt=T[now].val<val;
now=T[now].ch[nxt];
}
}
void insert(LL val)
{
if(root==0) {root=newnode(0,val);return ;}
LL now=root;
while(1)
{
T[now].siz++;
if(T[now].val==val) {T[now].rev++;splay(now,root);return ;}
LL nxt=val<T[now].val?0:1;
if(!T[now].ch[nxt]) {T[now].ch[nxt]=newnode(now,val);splay(now,root);return ;}
now=T[now].ch[nxt];
}
}
void erase(LL val)
{
LL now=find(val);
if(T[now].rev>1) {T[now].rev--;T[now].siz--;return ;}
else if(!ls(now)&&!rs(now)) {root=0;return ;}
else if(!ls(now)) {root=rs(now);T[rs(now)].fa=0;return ;}
LL left=ls(now);
while(rs(left)) left=rs(left);
splay(left,ls(now));
connect(rs(now),left,1);
connect(left,0,1);
//update(rs(now));
update(left);//
}
LL pre(LL val)
{
LL now=root,ans=-1e13;
while(now)
{
if(T[now].val<=val) ans=max(ans,T[now].val);
LL nxt=val<=T[now].val?0:1;
now=T[now].ch[nxt];
}
return ans == -1e13 ? -1 : ans;
}
LL nxt(LL val)
{
LL now=root,ans=1e13;
while(now)
{
if(T[now].val>val) ans=min(ans,T[now].val);
LL nxt=val<T[now].val?0:1;
now=T[now].ch[nxt];
}
return ans;
}
}Sp;
LL gcd(LL a,LL b) {return b == 0 ? a : gcd(b,a % b);}
LL exgcd(LL a,LL b,LL &x,LL &y) {
if(b == 0) {x = 1,y = 0;return a; }
LL ret = exgcd(b,a % b,x,y);
LL tmp = x;x = y;y = tmp - (a / b) * y;
return ret;
}
LL inv(LL a,LL b) {
LL x,y;
exgcd(a,b,x,y);
while(x < 0) x += b;
return x;
}
void work() {
LL ans = 0;
for(int i = 1;i <= n;++ i) {
ans = std::max(ans,a[i] % atk[i] == 0 ? a[i] / atk[i] : a[i] / atk[i] + 1);
}
printf("%lld\n",ans);
}
LL M[maxn],C[maxn];
inline LL add(LL x,LL y,LL mod) { return x + y >= mod ? x + y - mod : x + y; }
LL mul(LL x,LL k,LL mod) {
LL ret = 0 ;
x %= mod;
for(;k;k >>= 1,x = add(x,x,mod))
if(k & 1) ret = add(ret,x,mod);
return ret;
}
bool flag = false;
void init() {
Sp.clear();
n = read(),m = read();
flag = false;
//puts("haha");
for(int i = 1;i <= n;++ i) a[i] = read();
for(int i = 1;i <= n;++ i){ p[i] = read();if(p[i] != 1) flag = true; }
for(int j = 1;j <= n;++ j) tatk[j] = read();
//printf("%d %d %d\n",n,m,flag);
for(int k,i = 1;i <= m;++ i)
k = read(),Sp.insert(k);
for(int i = 1;i <= n;++ i) {
//puts("asdasd");
LL p = Sp.pre(a[i]);
if(p == -1) p = Sp.nxt(a[i]);
atk[i] = p;
Sp.erase(p);
Sp.insert(tatk[i]);
}
if(!flag) {work();/*puts("haha");*/return; }
int num = 0;
for(int i = 1;i <= n;++ i) {
a[i] %= p[i],atk[i] %= p[i];
if(!a[i] && !atk[i]) continue;
else if(!atk[i]){puts("-1");return;}
LL d = gcd(atk[i],p[i]);
if(a[i] % d != 0) {continue; }
a[i] /= d,atk[i] /= d,p[i] /= d;
C[++ num] = mul(a[i] , ((inv(atk[i],p[i]) % p[i] + p[i]) % p[i]),p[i]);
M[num] = p[i];
}
LL m = M[1],A = C[1],x,y,t;
for(int i = 2;i <= num;++ i) {
LL d = exgcd(m,M[i],x,y);
t = (M[i] / d);
//if((C[i]-A)%d == 0 || (a[i] - A) % d == -0 ) {
x = mul((x % t + t) % t,(((C[i] - A) / d) % t + t) % t,t);
LL MOD = (m / d) * (M[i]);
A = (mul(m , x, MOD) + A % MOD) % MOD;
m = MOD;
// else {puts("-1"); return;};
}
A = (A % m + m) % m;
printf("%lld\n",A);
} int main() {
freopen("dragon.in","r",stdin); freopen("dragon.out","w",stdout);
int t = read();
while(t --) {
init();
}
return 0;
}

loj#2721. 「NOI2018」屠龙勇士的更多相关文章

  1. LOJ #2721. 「NOI2018」屠龙勇士(set + exgcd)

    题意 LOJ #2721. 「NOI2018」屠龙勇士 题解 首先假设每条龙都可以打死,每次拿到的剑攻击力为 \(ATK\) . 这个需要支持每次插入一个数,查找比一个 \(\le\) 数最大的数(或 ...

  2. LOJ 2721 「NOI2018」屠龙勇士——扩展中国剩余定理

    题目:https://loj.ac/problem/2721 1.注意别一输入 p[ i ] 就 a[ i ] %= p[ i ] ,因为在 multiset 里找的时候还需要真实值. 2.注意用 m ...

  3. 「NOI2018」屠龙勇士

    「NOI2018」屠龙勇士 题目描述 小\(D\)最近在网上发现了一款小游戏.游戏的规则如下: 游戏的目标是按照编号\(1-n\)顺序杀掉\(n\) 条巨龙,每条巨龙拥有一个初始的生命 值ai .同时 ...

  4. 「NOI2018」屠龙勇士(EXCRT)

    「NOI2018」屠龙勇士(EXCRT) 终于把传说中 \(NOI2018D2\) 的签到题写掉了... 开始我还没读懂题目...而且这题细节巨麻烦...(可能对我而言) 首先我们要转换一下,每次的 ...

  5. POJ1061 青蛙的约会 和 LOJ2721 「NOI2018」屠龙勇士

    青蛙的约会 Language:Default 青蛙的约会 Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 133470 Accep ...

  6. 「NOI2018」屠龙勇士 解题报告

    「NOI2018」屠龙勇士 首先对于每个龙用哪个剑砍,我们可以用set随便模拟一下得到. 然后求出拿这个剑砍这条龙的答案 \[ atk_ix-p_iy=a_i \] 其中\(atk_i\)是砍第\(i ...

  7. 「NOI2018」屠龙勇士(CRT)

    /* 首先杀每条龙用到的刀是能够确定的, 然后我们便得到了许多形如 ai - x * atki | pi的方程 而且限制了x的最小值 那么exgcd解出来就好了 之后就是扩展crt合并了 因为全T调了 ...

  8. loj#2718. 「NOI2018」归程

    题目链接 loj#2718. 「NOI2018」归程 题解 按照高度做克鲁斯卡尔重构树 那么对于询问倍增找到当前点能到达的高度最小可行点,该点的子树就是能到达的联通快,维护子树中到1节点的最短距离 s ...

  9. Loj #2719. 「NOI2018」冒泡排序

    Loj #2719. 「NOI2018」冒泡排序 题目描述 最近,小 S 对冒泡排序产生了浓厚的兴趣.为了问题简单,小 S 只研究对 *\(1\) 到 \(n\) 的排列*的冒泡排序. 下面是对冒泡排 ...

随机推荐

  1. MySQL练习-主外键多表查询

    练习: 1.建立表关系: 请创建如下表,并创建相关约束 USE db1; CREATE TABLE class( cid INT AUTO_INCREMENT PRIMARY KEY, caption ...

  2. JavaScript 优雅的实现方式包含你可能不知道的知识点

    有些东西很好用,但是你未必知道:有些东西你可能用过,但是你未必知道原理. 实现一个目的有多种途径,俗话说,条条大路通罗马.很多内容来自平时的一些收集以及过往博客文章底下的精彩评论,收集整理拓展一波,发 ...

  3. mybatis的配置文件中<selectKey>标签问题

    1.mybatis的配置文件中,使用sequence生成主键 未执行add方法之前,主键未生成(null):刚执行add之后,主键即生成(212) 这里的重点是,一旦执行add方法,配置文件中的sel ...

  4. jQuery和Prototype的兼容性和冲突的五种解决方法

    第一种情况:先加载Prototype,再加载jQuery方法一:jQuery 库和它的所有插件都是在jQuery名字空间内的,包括全局变量也是保存在jQuery 名字空间内的. 使用jQuery.no ...

  5. iframe测试

    iframe类似于frameset,iframe 和 frameset 都用于html页面的框架布局.(都可用于后台管理界面,iframe放于body标签内,frameset不能放在body标签) 1 ...

  6. margin-bottom无效问题以及div里内容动态居中样式!

    最近调前端样式时候,遇到一个需求,在中间文字不对等的情况下想让下面的操作文字距离底部对齐,如图: , 刚开始觉得使用margin-bottom就可以,后来发现只有margin-top是管用的,查了资料 ...

  7. Angular CLI 命令行工具

    工欲善其事必先利其器.好的工具让开发更加简单便捷. 1.全局安装angular cli npm install -g @angular/cli 2.安装完成后就可以使用angular-cli命令行工具 ...

  8. 缓存数据库-redis数据类型和操作(hash)

    一:Redis 哈希(Hash) Redis hash 是一个string类型的field和value的映射表,hash特别适合用于存储对象. Redis 中每个 hash 可以存储 232 - 1 ...

  9. Codeforces 801C Voltage Keepsake(二分枚举+浮点(模板))

    题目链接:http://codeforces.com/contest/801/problem/C 题目大意:给你一些电器以及他们的功率,还有一个功率一定的充电器可以给这些电器中的任意一个充电,并且不计 ...

  10. sass问题

     用sass的minix定义一些代码片段,且可传参数 /** * @module 功能 * @description 生成全屏方法 * @method fullscreen * @version 1. ...