\(\color{white}{\mathbb{燕子來時青尚在,木蔭遙看杏花菲,名之以:杏紅}}\)
考場發現 \(t2\) 基本上是原題,\(t3\) 的套路見過,\(t4\) 像是並查集之類的算法,\(t1\) 是個擴歐,總之感覺今天的題非常可做
先寫 \(t3\) 打完二進制分組,過了大樣例,然后也沒測時間也沒對拍開始寫 \(t2\),很快也過了大樣例,然后推 \(t1\),像是個三分,打完稍調一下也過了大樣例。
發現時間還有兩個半小時,覺得良心出題人前三題都給了大樣例,估計不用對拍了吧,於是開始想 \(t4\)
想了一個小時復雜度大概對了,半個小時碼完,再寫個暴力掛上對拍
發現還有將近一個小時,開始喝水上廁所(一度以為有生之年可以AK了)
最后考完發現兩題都掛了,發現 \(t1\) 有個三分的細節寫錯了,\(t3\) 是位運算弄錯了,總共掛了 135 的分……
還是得認真檢查……
A. 數列
很明顯是擴歐,然后要求一組解使得 \(abs(x)+abs(y)\) 最小,這個是單峰函數,寫了個三分,但是左右 \(mid\) 寫錯了
更簡單的做法是令 \(a < b\),那么求通解的時候 \(x\) 每次跨度大,\(y\) 跨度小,所以當 \(abs(x)\) 小的時候一定答案最小
代碼實現
#include<bits/stdc++.h>
using namespace std;
#define int long long
const int maxn=1e5+5;
int n,a,b,g,x,y,p[maxn],ans,xx,yy,ax,ay;
int read(){
int x=0,f=1;
char ch=getchar();
while(!isdigit(ch)){
if(ch=='-')f=-1;
ch=getchar();
}
while(isdigit(ch)){
x=x*10+ch-48;
ch=getchar();
}
return x*f;
}
int gcd(int a,int b){
return b==0?a:gcd(b,a%b);
}
void exgcd(int a,int b){
if(!b){
x=1;
y=0;
return ;
}
exgcd(b,a%b);
int k=y;
y=x-a/b*y;
x=k;
return ;
}
int calc(int k){
return abs(xx+k*ax)+abs(yy-k*ay);
}
int solve(int val){
if(val%g){
puts("-1");
exit(0);
}
xx=x*val/g;
yy=y*val/g;
int l=-1e9,r=1e9,lastl=0,lastr=0,cnt=0;
while(l<r){
int lmid=(r-l+1)/3+l;
int rmid=l+r-lmid;
// cout<<l<<" "<<r<<" "<<lmid<<" "<<rmid<<endl;
if(calc(lmid)<calc(rmid))r=rmid;
else l=lmid;
if(l==lastl&&r==lastr)cnt++;
else lastl=l,lastr=r,cnt=0;
if(cnt>=2)break;
}
int mn=0x3f3f3f3f3f3f3f3f;
for(int i=l-1;i<=r+1;i++){
mn=min(mn,calc(i));
}
return mn;
}
signed main(){
// freopen("array.in","r",stdin);
// cout<<gcd(12,18);
n=read();
a=read();
b=read();
g=gcd(a,b);
// cout<<g<<endl;
exgcd(a,b);
ax=b/g;
ay=a/g;
// cout<<x<<" "<<y<<endl;
for(int i=1;i<=n;i++){
p[i]=read();
}
for(int i=1;i<=n;i++){
ans+=solve(abs(p[i]));
}
cout<<ans;
return 0;
}
B. 數對
類似與隊長快跑,只不過這道題無序,按 \(min(a_i,b_i)\) 排序即可
C. 最小距離
二進制拆分寫錯+卡常掛成零分
一定注意 \((1<< pos ) \& x\) 的值不是 \(1\)!!!!
正解是記錄每個點更新的源點或者次短路
代碼實現
#include<bits/stdc++.h>
using namespace std;
#define int long long
const int maxn=4e5+5;
const int maxm=1e6+5;
int n,m,x,y,w,hd[maxn],cnt,dis[maxn],a[maxn],k,ans[maxn],col[maxn];
bool vis[maxn];
int read(){
int x=0,f=1;
char ch=getchar();
while(!isdigit(ch)){
if(ch=='-')f=-1;
ch=getchar();
}
while(isdigit(ch)){
x=x*10+ch-48;
ch=getchar();
}
return x*f;
}
struct Edge{
int nxt,from,to,val;
}edge[maxm];
void add(int u,int v,int w){
edge[++cnt].nxt=hd[u];
edge[cnt].to=v;
edge[cnt].val=w;
edge[cnt].from=u;
hd[u]=cnt;
return ;
}
struct Node{
int dis,id;
Node(){}
Node(int x,int y):dis(x),id(y){}
bool operator < (const Node & x)const{
return dis>x.dis;
}
};
priority_queue<Node>q;
void dij(){
memset(dis,0x3f,sizeof dis);
memset(vis,0,sizeof vis);
for(int i=1;i<=k;i++){
col[a[i]]=i;
q.push(Node(0,a[i]));
dis[a[i]]=0;
}
while(!q.empty()){
int u=q.top().id;
q.pop();
if(vis[u])continue;
vis[u]=true;
for(int i=hd[u];i;i=edge[i].nxt){
int v=edge[i].to;
if(dis[v]>dis[u]+edge[i].val){
dis[v]=dis[u]+edge[i].val;
col[v]=col[u];
q.push(Node(dis[v],v));
}
}
}
return ;
}
signed main(){
// freopen("distance.in","r",stdin);
// freopen("my.out","w",stdout);
n=read();
m=read();
k=read();
memset(ans,0x3f,sizeof ans);
for(int i=1;i<=k;i++){
a[i]=read();
}
for(int i=1;i<=m;i++){
x=read();
y=read();
w=read();
add(x,y,w);
add(y,x,w);
}
dij();
for(int i=1;i<=cnt;i++){
if(col[edge[i].from]!=col[edge[i].to]){
ans[col[edge[i].from]]=min(ans[col[edge[i].from]],edge[i].val+dis[edge[i].from]+dis[edge[i].to]);
ans[col[edge[i].to]]=min(ans[col[edge[i].to]],edge[i].val+dis[edge[i].from]+dis[edge[i].to]);
}
}
for(int i=1;i<=k;i++){
printf("%lld ",ans[i]);
}
return 0;
}
D. 真相
這種題一看類似於圖論
按照真假拆點建圖后發現分成多個塊,每個塊的結尾都是 \(1\) 類節點
那么一種思路是枚舉每個結尾點,統計所有與其相同的節點的真話總數,判斷是否滿足即可,這個可以預處理優化成 \(O(n)\)
代碼實現
#include<bits/stdc++.h>
using namespace std;
const int maxn=1e6+5;
int t,n,a[maxn],val[maxn],pos[maxn],val1[maxn],tot,w[maxn],sumtot,sum1[maxn],sum2[maxn];
bool vis[maxn];
char c[5];
int read(){
int x=0,f=1;
char ch=getchar();
while(!isdigit(ch)){
if(ch=='-')f=-1;
ch=getchar();
}
while(isdigit(ch)){
x=x*10+ch-48;
ch=getchar();
}
return x*f;
}
void work(){
int op=0;
for(int i=1;i<=n;i++){
op^=a[i];
}
if(op==0)puts("consistent");
else puts("inconsistent");
return ;
}
void pre(){
for(int j=1;j<=tot;j++){
int p=pos[j];
int op=0;
while(!vis[p-1]){
p--;
if(!p)p=n;
if(vis[p])break;
op^=a[p];
val[j]+=(!op);
}
val[j]++;
p=pos[j],op=1;
while(!vis[p-1]){
p--;
if(!p)p=n;
if(vis[p])break;
op^=a[p];
val1[j]+=(!op);
// cout<<p<<" "<<a[p]<<" "<<op<<endl;
}
// cout<<val[j]<<" "<<val1[j]<<endl;
}
return ;
}
bool solve(){
for(int i=1;i<=n;i++){
sum1[w[i]]+=val[i];
sum2[w[i]]+=val1[i];
sumtot+=val1[i];
}
for(int i=1;i<=tot;i++){
int sum=sumtot-sum2[w[i]]+sum1[w[i]];
if(sum==w[i])return true;
}
int sum=0;
for(int i=1;i<=tot;i++){
sum+=val1[i];
}
for(int i=1;i<=tot;i++){
if(sum==w[i])return false;
}
return true;
}
void init(){
for(int i=1;i<=n;i++){
a[i]=vis[i]=val[i]=val1[i]=sum1[i]=w[i]=sum2[i]=0;
}
tot=sumtot=0;
return ;
}
int main(){
// freopen("shuju.in","r",stdin);
// freopen("my1.out","w",stdout);
t=read();
while(t--){
init();
n=read();
for(int i=1;i<=n;i++){
scanf("%s",c+1);
if(c[1]=='$')pos[++tot]=i,vis[i]=true,w[tot]=read();
else if(c[1]=='-')a[i]=1;
else a[i]=0;
}
if(!tot){
work();
continue;
}
pre();
if(solve())puts("consistent");
else puts("inconsistent");
}
return 0;
}
\(\color{white}{\mathbb{花褪殘紅青杏小,燕子來時,綠水人家繞。}}\)