[BZOJ 2118]墨墨的等式
论如何把数论乱搞和图论乱搞出在一起……
这个题由于要求[tex]x\ge 0[/tex],所以不能gcd乱搞。我们可以先取[tex]\{a_n\}[/tex]的最小值[tex]p[/tex](忽略为0的情况,为啥取最小值待会再说),对方程两边模[tex]p[/tex]。然后对于任何能使某个同余方程成立的[tex]\{x_n\}[/tex],将其中所有[tex]x_i[/tex]同时加任意个[tex]p[/tex],同余方程都成立。
取模后,[tex]B\in Z_p[/tex],所以说只要对于[tex]Z_p[/tex]中的每个数找出最小的一组合法解即能推出其他解(所以说,剩余系越少效率越高,这也就要求取的[tex]a_i[/tex]要小)。不过这个最小的一组合法解怎么找?
我们先找出任意一个合法[tex]B[/tex](比如说0吧),然后尝试加上[tex]a_i[/tex],就可以推出其他[tex]B\in Z_p[/tex]的最小解。这个应用当然是需要最短路辣。
求出来的最短路,表示的是取最小解时的[tex]B[/tex]。这样的话就可以推出某个前缀区间中合法[tex]B[/tex]的个数(能加多少[tex]p[/tex],就有多少个,注意不要忽略加0个的情况),并且答案符合区间减法,最后做差分即可。
注意忽略[tex]a_i=0[/tex]的情况(相当于不存在)。
代码:
/**************************************************************
Problem: 2118
User: danihao123
Language: C++
Result: Accepted
Time:1952 ms
Memory:5508 kb
****************************************************************/
#include <cstdio>
#include <cstring>
#include <queue>
#include <algorithm>
#ifdef DEBUG
#include <cassert>
#endif
using namespace std;
typedef long long ll;
const int maxn=15;
const ll INF=0x7f7f7f7f7f7f7f7f;
ll A[maxn];
bool inQueue[500005];
ll d[500005];
int n;
ll minv;
inline void SPFA(){
register int i,u,to;
queue<int> Q;
memset(d,0x7f,sizeof(d));
d[0]=0;
Q.push(0);
inQueue[0]=true;
#ifdef DEBUG
assert(d[1]==INF);
#endif
while(!Q.empty()){
u=Q.front();
Q.pop();
inQueue[u]=false;
for(i=1;i<=n;i++){
to=(u+A[i])%minv;
if(d[u]<INF && d[u]+A[i]<d[to]){
d[to]=d[u]+A[i];
if(!inQueue[to]){
Q.push(to);
inQueue[to]=true;
}
}
}
}
}
inline ll Calc(ll x){
register ll ans=0;
register int i=0;
for(i=0;i<minv;i++)
if(d[i]<=x)
ans+=(x-d[i])/minv+1;
return ans;
}
int main(){
ll l,r;
register int i;
scanf("%d%lld%lld",&n,&l,&r);
minv=0x7fffffff;
for(i=1;i<=n;i++){
scanf("%d",&A[i]);
if(!A[i]){
i--;
n--;
continue;
}
minv=min(minv,A[i]);
}
SPFA();
printf("%lld\n",Calc(r)-Calc(l-1));
return 0;
}
[BZOJ 2662]冻结
又是一道分层图最短路水题……
我估计会卡SPFA(或许可能不卡?),所以再次写了pb_ds优化Dijkstra。
代码:
/**************************************************************
Problem: 2662
User: danihao123
Language: C++
Result: Accepted
Time:4 ms
Memory:868 kb
****************************************************************/
#include <cstdio>
#include <cctype>
#include <cstring>
#include <utility>
#include <ext/pb_ds/priority_queue.hpp>
#include <bitset>
#include <algorithm>
using namespace std;
const int maxn=51,maxk=51,maxm=2005;
int cnt=0;
int first[maxn];
int to[maxm],next[maxm];
int dist[maxm];
inline void Add_Edge(int x,int y,int z){
cnt++;
next[cnt]=first[x];
first[x]=cnt;
to[cnt]=y;
dist[cnt]=z;
}
int d[maxn][maxk];
bitset<maxn> vis[maxk];
struct Node{
int x,k,d;
bool operator <(const Node& itt) const{
return d<itt.d;
}
bool operator >(const Node& itt) const{
return d>itt.d;
}
};
typedef __gnu_pbds::priority_queue<Node,greater<Node> > Heap;
Heap::point_iterator ite[maxn][maxk];
Heap Q;
int n,k;
inline void relax(int x,int y,int p){
d[x][y]=p;
if(ite[x][y]!=0)
Q.modify(ite[x][y],(Node){x,y,p});
else
ite[x][y]=Q.push((Node){x,y,p});
}
int dij(){
register int i,u,v,ans=0x7f7f7f7f;
memset(d,0x7f,sizeof(d));
d[1][0]=0;
ite[1][0]=Q.push((Node){1,0,0});
while(!Q.empty()){
u=Q.top().x;
v=Q.top().k;
Q.pop();
if(vis[u][v])
continue;
vis[u][v]=true;
for(i=first[u];i;i=next[i]){
if(v<k){
if(d[to[i]][v+1]>(d[u][v]+dist[i]/2)){
relax(to[i],v+1,d[u][v]+dist[i]/2);
}
}
if(d[to[i]][v]>d[u][v]+dist[i]){
relax(to[i],v,d[u][v]+dist[i]);
}
}
}
for(i=0;i<=k;i++)
ans=min(ans,d[n][i]);
return ans;
}
// I/O优化
inline int readint(){
char c=getchar();
register int x=0;
while(!isdigit(c))
c=getchar();
while(isdigit(c)){
x=x*10+c-'0';
c=getchar();
}
return x;
}
int bf[10];
inline void writeint(int x){
register int p=0;
if(x==0){
bf[p++]=0;
}else{
while(x){
bf[p++]=x%10;
x/=10;
}
}
for(register int i=p-1;i>=0;i--)
putchar('0'+bf[i]);
}
int main(){
int m;
register int i,u,v,d;
n=readint();
m=readint();
k=readint();
for(i=1;i<=m;i++){
u=readint();
v=readint();
d=readint();
Add_Edge(u,v,d);
Add_Edge(v,u,d);
}
writeint(dij());
putchar('\n');
return 0;
}
[BZOJ 2763]飞行路线
分层图最短路处女作TAT
很明显要求你求一个分层图上的最短路。不过没必要把分层图构出来,手写转移即可。还有卡SPFA是怎么回事?出题人你粗来我保证不打死你……
然而沉迷pb_ds,不能自拔。
代码:
/**************************************************************
Problem: 2763
User: danihao123
Language: C++
Result: Accepted
Time:268 ms
Memory:3136 kb
****************************************************************/
#include <cstdio>
#include <queue>
#include <algorithm>
#include <utility>
#include <cstring>
#include <cctype>
#include <ext/pb_ds/priority_queue.hpp>
using namespace std;
const int maxn=10001,maxm=100001,maxk=11;
typedef pair<int,int> pii;
int first[maxn];
int next[maxm],to[maxm],dist[maxm];
int graph_cnt=0;
inline void Add_Edge(int u,int v,int d){
graph_cnt++;
next[graph_cnt]=first[u];
first[u]=graph_cnt;
to[graph_cnt]=v;
dist[graph_cnt]=d;
}
struct Node{
pii pnt;
int d;
bool operator <(const Node& x) const{
return d<x.d;
}
bool operator >(const Node& x) const{
return d>x.d;
}
};
typedef __gnu_pbds::priority_queue<Node,greater<Node> > Heap;
Heap::point_iterator ite[maxn][maxk];
Heap Q;
int n,k;
bool vis[maxn][maxk];
int d[maxn][maxk];
inline void relax(int u,int v,int di){
d[u][v]=di;
if(ite[u][v]!=0)
Q.modify(ite[u][v],(Node){make_pair(u,v),di});
else
ite[u][v]=Q.push((Node){make_pair(u,v),di});
}
int Dijkstra(int s,int t){
register int i,u,v;
pii temp;
memset(d,0x7f,sizeof(d));
d[s][0]=0;
ite[s][0]=Q.push((Node){make_pair(s,0),0});
while(!Q.empty()){
temp=Q.top().pnt;
Q.pop();
u=temp.first;
v=temp.second;
if(vis[u][v])
continue;
vis[u][v]=true;
for(i=first[u];i;i=next[i]){
if(v<k){
if(d[u][v]<d[to[i]][v+1]){
relax(to[i],v+1,d[u][v]);
}
}
if(d[u][v]+dist[i]<d[to[i]][v]){
relax(to[i],v,d[u][v]+dist[i]);
}
}
}
register int ans=0x7f7f7f7f;
for(i=0;i<=k;i++)
ans=min(ans,d[t][i]);
return ans;
}
int main(){
int m,s,t,u,v,d;
register int i;
scanf("%d%d%d%d%d",&n,&m,&k,&s,&t);
for(i=1;i<=m;i++){
scanf("%d%d%d",&u,&v,&d);
Add_Edge(u,v,d);
Add_Edge(v,u,d);
}
printf("%d\n",Dijkstra(s,t));
return 0;
}
[BZOJ 4152]The Captain
这题初看有些棘手。但是毕竟是最短路,[tex]min(abs(x_{1}-x_{2}),abs(y_{1}-y_{2}))[/tex]这样的边,大可以分成两条边来建,于是乎min就形同虚设了。并且我们可以看到这样建图有一个好处:跨越若干点的方案可以一定可以分成图上的若干边。问题大为简化,迎刃而解。
但这提有个丧病的地方:卡SPFA。加了一些优化照样挂。所以我一气之下写了可并堆优化的Dijkstra :)当然可并堆用的是pb_ds辣。
代码:
/**************************************************************
Problem: 4152
User: danihao123
Language: C++
Result: Accepted
Time:4888 ms
Memory:17412 kb
****************************************************************/
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <utility>
#include <ext/pb_ds/priority_queue.hpp>
#include <cctype>
#include <bitset>
#ifdef DEBUG
#include <cassert>
#endif
using namespace std;
const int maxn=200001;
int n;
inline int abs(int x){
return x<0?-x:x;
}
int first[maxn];
int next[maxn*4],to[maxn*4],dist[maxn*4];
int graph_cnt=0;
inline void Add_Edge(int x,int y,int d){
graph_cnt++;
next[graph_cnt]=first[x];
first[x]=graph_cnt;
to[graph_cnt]=y;
dist[graph_cnt]=d;
}
int d[maxn];
bitset<maxn> vis;
typedef pair<int,int> my_pair;
typedef __gnu_pbds::priority_queue<my_pair,greater<my_pair> > Heap;
Heap::point_iterator ite[maxn];
Heap Q;
int dij(){
register int i,u;
memset(d,0x7f,sizeof(d));
d[1]=0;
ite[1]=Q.push(make_pair(0,1));
while(!Q.empty()){
u=Q.top().second;
Q.pop();
if(vis[u])
continue;
vis[u]=true;
for(i=first[u];i;i=next[i]){
if(d[to[i]]>(dist[i]+d[u])){
d[to[i]]=dist[i]+d[u];
if(ite[to[i]]!=0)
Q.modify(ite[to[i]],make_pair(d[to[i]],to[i]));
else
ite[to[i]]=Q.push(make_pair(d[to[i]],to[i]));
}
}
}
return d[n];
}
int pr[maxn][2];
int order1[maxn],order2[maxn];
int cmp1(const int i,const int j){
return pr[i][0]<pr[j][0];
}
int cmp2(const int i,const int j){
return pr[i][1]<pr[j][1];
}
// I/O优化
inline int readint(){
char c=getchar();
register int x=0;
while(!isdigit(c))
c=getchar();
while(isdigit(c)){
x=x*10+c-'0';
c=getchar();
}
return x;
}
int main(){
register int i;
n=readint();
for(i=1;i<=n;i++){
pr[i][0]=readint();
pr[i][1]=readint();
order1[i]=i;
order2[i]=i;
}
sort(order1+1,order1+1+n,cmp1);
sort(order2+1,order2+1+n,cmp2);
for(i=1;i<=n;i++){
if(i!=1){
Add_Edge(order1[i],order1[i-1],pr[order1[i]][0]-pr[order1[i-1]][0]);
Add_Edge(order2[i],order2[i-1],pr[order2[i]][1]-pr[order2[i-1]][1]);
}
if(i!=n){
Add_Edge(order1[i],order1[i+1],pr[order1[i+1]][0]-pr[order1[i]][0]);
Add_Edge(order2[i],order2[i+1],pr[order2[i+1]][1]-pr[order2[i]][1]);
}
}
printf("%d\n",dij());
return 0;
}