好蛋疼啊,由于double 不能用memset,害的我调了一个多小时才发现。由于用二分搜索,时间有点大
#include
#include
#include
#include
#define maxn 1005
using namespace std;
const int INF = 0x3f3f3f;
struct node{
int x,y,z;
}Node[maxn];
double d[maxn];
double G[maxn][maxn];
double cost[maxn][maxn];
double benefit[maxn][maxn];
int n;
double mid,ans;
void calculate(int i,int j){
cost[i][j] = cost[j][i] = abs(Node[i].z - Node[j].z);
int x1=Node[i].x, y1=Node[i].y;
int x2=Node[j].x, y2=Node[j].y;
benefit[i][j] = benefit[j][i] = sqrt(1.0*(x1-x2)*(x1-x2) + 1.0*(y1-y2)*(y1-y2));
}
void makemap(){
for(int i=1;i<=n;i++)
for(int j=i+1;j<=n;j++){
G[i][j] = G[j][i] = cost[i][j] - mid * benefit[i][j];
}
return;
}
void prim(){
bool vis[maxn];
double mindist;
double lowdist[maxn];
memset(vis,0,sizeof(vis));
for(int i=1;i<=n;i++) lowdist[i] = INF;
int point;
ans = 0.0;
int s=1;
int num=1;
while(true){
if(num == n) break;
vis[s] = true;
mindist = INF;
for(int i=1;i<=n;i++){
if(!vis[i] && lowdist[i] > G[s][i]){
lowdist[i] = G[s][i];
}
if(!vis[i] && lowdist[i] < mindist){
mindist = lowdist[i];
point = i;
}
}
s = point;
ans += mindist;
num++;
}
return;
}
int main()
{
//if(freopen("input.txt","r",stdin)== NULL) {printf("Error\n"); exit(0);}
while(cin>>n && n){
for(int i=1;i<=n;i++){
cin>>Node[i].x>>Node[i].y>>Node[i].z;
}
for(int i=1;i<=n;i++)
for(int j=i+1;j<=n;j++){
calculate(i,j);
}
double R=10000000,L=0.0;
while( (R-L)>1e-6 ){
mid = (R+L)/2;
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++){
G[i][j] = INF;
}
makemap();
prim();
if(ans >= 0) L = mid;
else R = mid;
}
printf("%.3lf\n",L);
}
return 0;
}
还是迭代好上面的用时2500多MS,而这个只要297MS,空间复杂度是降不下来了。
#include
#include
#include
#include
#define maxn 1005
using namespace std;
const int INF = 0x3f3f3f;
struct node{
int x,y,z;
}Node[maxn];
double d[maxn];
double G[maxn][maxn];
double cost[maxn][maxn];
double benefit[maxn][maxn];
int n;
double mid,ans;
void calculate(int i,int j){
cost[i][j] = cost[j][i] = abs(Node[i].z - Node[j].z);
int x1=Node[i].x, y1=Node[i].y;
int x2=Node[j].x, y2=Node[j].y;
benefit[i][j] = benefit[j][i] = sqrt(1.0*(x1-x2)*(x1-x2) + 1.0*(y1-y2)*(y1-y2));
}
void makemap(){
for(int i=1;i<=n;i++)
for(int j=i+1;j<=n;j++){
G[i][j] = G[j][i] = cost[i][j] - mid * benefit[i][j];
}
return;
}
void prim(){
bool vis[maxn];
double mindist;
struct low{
double lowdist;
int s;
}L[maxn];
memset(vis,0,sizeof(vis));
for(int i=1;i<=n;i++) L[i].lowdist = INF;
int point;
ans = 0.0;
int s=1;
int num=1;
double suma = 0.0,sumb = 0.0;
while(true){
if(num == n) break;
vis[s] = true;
mindist = INF;
for(int i=1;i<=n;i++){
if(!vis[i] && L[i].lowdist > G[s][i]){
L[i].lowdist = G[s][i];
L[i].s = s;
}
if(!vis[i] && L[i].lowdist < mindist){
mindist = L[i].lowdist;
point = i;
}
}
int a = L[point].s;
suma += cost[a][point];
sumb += benefit[a][point];
s = point;
num++;
}
ans = suma/sumb;
return;
}
int main()
{
//if(freopen("input.txt","r",stdin)== NULL) {printf("Error\n"); exit(0);}
while(cin>>n && n){
for(int i=1;i<=n;i++){
cin>>Node[i].x>>Node[i].y>>Node[i].z;
}
for(int i=1;i<=n;i++)
for(int j=i+1;j<=n;j++){
calculate(i,j);
}
mid = 0.5;
while( true){
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++){
G[i][j] = INF;
}
makemap();
prim();
if(abs(ans -mid) < 1e-6) break;
else mid = ans;
}
printf("%.3lf\n",ans);
}
return 0;
}