旅行推销员问题之C语言算法

旅行推销员问题:一位旅行推销员想要访问n个城市中每个城市恰好一次,并且返回到出发点。应如何走才能使总距离最短?其实就是求带权完全无向图中访问每个顶点恰好一次并且返回出发点的总权数最小的回路,即总权数最小的哈密顿回路。

思路:把n个城市做全排列,并求出每种排列对应的总距离,然后选择最短的一个。

推论:把上面求出的最短哈密顿回路看作一个圆,求出最短哈密顿回路以后,从回路上任一点开始出发而访问每个顶点一次,总距离是相等的。因为总距离与圆周相等。

代码(效率有待于提高,因为多判断了一半的回路,即一条路反方向和正方向各判断了一次):

/*
   程序功能:求解旅行推销员问题
   作者:BugEyes
   主页:http://bugeyes.blog.edu.cn
*/

int flag=0;/*用来标记回路是否检查完毕*/

int search(int distance[5][5],int start,int result[5]);
void next(int array[],int n);
void sort(int array[],int start,int end);
int dis_compute(int distance[5][5],int result[5]);
void chuli(int t_result[],int result[],int start,int n);


void main()
{
   int i,j,start;/*start表示出发点城市编号*/
   int distance[5][5]={{0,50,30,7,60},/*模拟城市之间的距离*/
                                {50,0,40,76,89},
                                {30,40,0,10,70},
                                {70,76,100,0,80},
                                {90,89,70,80,0}};
   int result[5]={0};/*存放最短哈密顿回路*/
   int dis_result=0;/*存放最短总距离*/
   printf("/nThe distance of cities are as follows:/n");
   for(i=0;i<5;i++)
   {
      for(j=0;j<=i;j++)
             printf("%4d",distance[i][j]);
      printf("/n/n");
   }
   printf("Please input the city number to start.Input -1 to exit./n");
   scanf("%d",&start);
   while(start<-1||start>4)
   {
      printf("Sorry,you are wrong./nPlease input the city number to start.Input -1 to exit./n");
      scanf("%d",&start);
   }
   if(start==-1)/*-1表示结束程序*/
   {
      printf("/nBye.Good luck!/n");
      return;
   }
   printf("/nOK.Let's go!!!/n");
   dis_result=search(distance,start,result);/*计算最短哈密顿回路,并求总距离*/
   printf("/nThe best route is:/n");
   for(i=0;i<5;i++)
      printf("%4d->",result[i]);
   printf("%4d/n",result[0]);
   printf("And the whole distance is:%d/n",dis_result);
}

int search(int distance[5][5],int start,int result[5])
{
   int i,j;
   int t_distance,r_distance;
   int t_result[5]={0,1,2,3,4};
   int r_result[5];
   printf("/nI am in search function...........");
   t_distance=r_distance=dis_compute(distance,t_result);
   while(flag==0)
   {
      next(t_result,5);
      chuli(t_result,r_result,start,5);
      if((t_distance=dis_compute(distance,r_result))<r_distance)
      {/*若当前回路总距离在检查过的回路中最小*/
             r_distance=t_distance;
             chuli(t_result,result,start,5);
      }
   }
   printf("/nI am leaving search function...............");
   return r_distance;
}

void next(int array[],int n)
{ /*计算下一个可能的回路,其实就是求排列*/
   int i,j,temp;
   printf("/nI am in next function..........................");
   flag=0;
   for(i=n-2;i>=0;i--)
      if(array[i]<array[i+1])
          break;
   for(j=n-1;j>i;j--)
      if(array[j]>array[i])
           break;
   if(j==i)
   {/*已是最后一条回路*/
      flag=1;
      return;
   }
   temp=array[i];
   array[i]=array[j];
   array[j]=temp;
   sort(array,i+1,n);
   printf("/nI am leaving next function..........................");
}

void sort(int array[],int start,int end)
{
   int i,j;
   printf("/nI am in sort function...............................");
   for(i=start;i<end-start;i++)
      for(j=start;j<end-1;j++)
  if(array[j]>array[j+1])
  {
     int t=array[j];
     array[j]=array[j+1];
     array[j+1]=t;
  }
   printf("/nI am leaving sort function...........................");
}

int dis_compute(int distance[5][5],int result[5])
{/*计算当前回路对应的总距离*/
    int r_distance=0;
    int i;
    printf("/nI am in dis_compute function.........................");
    for(i=0;i<5;i++)
        r_distance+=distance[result[i]][result[(i+1)%5]];
    printf("/nI am leaving dis_compute function...................");
    return r_distance;
}

void chuli(int t_result[],int result[],int start,int n)
{/*对选择的回路重新排列,按出发点---->路径---->出发点的顺序排列*/
   int i,pos;
   printf("/nI am in chuli function..............................");
   for(i=0;i<n;i++)
      if(t_result[i]==start)
             break;
   pos=i;
   for(i=0;i<n;i++)
      result[i]=t_result[(pos+i)%n];
   printf("/nI am leaving chuli function.......................");
}

 
  • 0
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值