

感慨:这几天都快被这个题目搞得没有耐心了,写了至少三天,但是刚刚灵光乍现,一下子明白了点,还是多读代码,多在脑子里面思考,培养耐心的能力不要浮躁
思路:想通了就很简单,由样例一可知,遍历时可以折返,那我们就想什么时候要折返才能使边权和最小呢,那就是当已经遍历过的节点的边权和小于接下来要顺时针遍历的边权和,以下是我的原始的错误的代码,因为我当时想当然的以为一定是遇见最大的边权后才会折返,但其实这么想就错了
#include<stdio.h>
#define ll long long
int main(){
ll T;
scanf("%lld",&T);
while(T--){
int n;
scanf("%d",&n);
ll str[n+3];
ll sum=0;
ll max=0;
for(int i=0;i<n;i++){
scanf("%lld",&str[i]);
sum+=str[i];
if(str[i]>max){
max=str[i];
}
}
//不返回走(一条路走下去)
ll min=0;
min=(str[0]>str[n-1])?(sum-str[0]):(sum-str[n-1]);
ll sum_i=0,t;
for(int i=0;i<n;i++){
sum_i+=str[i];
if(str[i]==max){
t=sum-max*2+sum_i;
if(t<min) min=t;
//printf("%d %d\n",min,sum_i);
}
}
sum_i=0;
for(int i=n-1;i>=0;i--){
sum_i+=str[i];
if(str[i]==max){
t=sum-max*2+sum_i;
if(t<min) min=t;
}
}
printf("%lld\n",min);
}
}
AC代码
#include<stdio.h>
#define ll long long
int main(){
ll T;
scanf("%lld",&T);
while(T--){
int n;
scanf("%d",&n);
ll str[n+3];
ll sum=0;
ll max=0;
for(int i=0;i<n;i++){
scanf("%lld",&str[i]);
sum+=str[i];
if(str[i]>max){
max=str[i];
}
}
//不返回走(一条路走下去)
ll min=0;
min=(str[0]>str[n-1])?(sum-str[0]):(sum-str[n-1]);
ll sum_i=str[0],t;
for(int i=1;i<n;i++){
if(str[i]>sum_i){
t=sum-str[i]+sum_i;
if(t<min) min=t;
// printf("%d %d\n",min,sum_i);
}
sum_i+=str[i];
}
sum_i=str[n-1];
for(int i=n-2;i>=0;i--){
if(str[i]>sum_i){
t=sum-str[i]+sum_i;
if(t<min) min=t;
// printf("%d %d\n",min,sum_i);
}
sum_i+=str[i];
}
printf("%lld\n",min);
}
}