久久久久久久av_日韩在线中文_看一级毛片视频_日本精品二区_成人深夜福利视频_武道仙尊动漫在线观看

PHP實(shí)現(xiàn)圖的鄰接矩陣表示及幾種簡單遍歷算法分析

這篇文章主要介紹了PHP實(shí)現(xiàn)圖的鄰接矩陣表示及幾種簡單遍歷算法,結(jié)合實(shí)例形式分析了php基于鄰接矩陣實(shí)現(xiàn)圖的定義及相關(guān)遍歷操作技巧,需要的朋友可以參考下

本文實(shí)例講述了PHP實(shí)現(xiàn)圖的鄰接矩陣表示及幾種簡單遍歷算法。分享給大家供大家參考,具體如下:

在web開發(fā)中圖這種數(shù)據(jù)結(jié)構(gòu)的應(yīng)用比樹要少很多,但在一些業(yè)務(wù)中也常有出現(xiàn),下面介紹幾種圖的尋徑算法,并用PHP加以實(shí)現(xiàn).

佛洛依德算法,主要是在頂點(diǎn)集內(nèi),按點(diǎn)與點(diǎn)相鄰邊的權(quán)重做遍歷,如果兩點(diǎn)不相連則權(quán)重?zé)o窮大,這樣通過多次遍歷可以得到點(diǎn)到點(diǎn)的最短路徑,邏輯上最好理解,實(shí)現(xiàn)也較為簡單,時間復(fù)雜度為O(n^3);

迪杰斯特拉算法,OSPF中實(shí)現(xiàn)最短路由所用到的經(jīng)典算法,djisktra算法的本質(zhì)是貪心算法,不斷的遍歷擴(kuò)充頂點(diǎn)路徑集合S,一旦發(fā)現(xiàn)更短的點(diǎn)到點(diǎn)路徑就替換S中原有的最短路徑,完成所有遍歷后S便是所有頂點(diǎn)的最短路徑集合了.迪杰斯特拉算法的時間復(fù)雜度為O(n^2);

克魯斯卡爾算法,在圖內(nèi)構(gòu)造最小生成樹,達(dá)到圖中所有頂點(diǎn)聯(lián)通.從而得到最短路徑.時間復(fù)雜度為O(N*logN);

<?php
/**
 * PHP 實(shí)現(xiàn)圖鄰接矩陣
 */
class MGraph{
  private $vexs; //頂點(diǎn)數(shù)組
  private $arc; //邊鄰接矩陣,即二維數(shù)組
  private $arcData; //邊的數(shù)組信息
  private $direct; //圖的類型(無向或有向)
  private $hasList; //嘗試遍歷時存儲遍歷過的結(jié)點(diǎn)
  private $queue; //廣度優(yōu)先遍歷時存儲孩子結(jié)點(diǎn)的隊(duì)列,用數(shù)組模仿
  private $infinity = 65535;//代表無窮,即兩點(diǎn)無連接,建帶權(quán)值的圖時用,本示例不帶權(quán)值
  private $primVexs; //prim算法時保存頂點(diǎn)
  private $primArc; //prim算法時保存邊
  private $krus;//kruscal算法時保存邊的信息
  public function MGraph($vexs, $arc, $direct = 0){
    $this->vexs = $vexs;
    $this->arcData = $arc;
    $this->direct = $direct;
    $this->initalizeArc();
    $this->createArc();
  }
  private function initalizeArc(){
    foreach($this->vexs as $value){
      foreach($this->vexs as $cValue){
        $this->arc[$value][$cValue] = ($value == $cValue ? 0 : $this->infinity);
      }
    }
  }
  //創(chuàng)建圖 $direct:0表示無向圖,1表示有向圖
  private function createArc(){
    foreach($this->arcData as $key=>$value){
      $strArr = str_split($key);
      $first = $strArr[0];
      $last = $strArr[1];
      $this->arc[$first][$last] = $value;
      if(!$this->direct){
        $this->arc[$last][$first] = $value;
      }
    }
  }
  //floyd算法
  public function floyd(){
    $path = array();//路徑數(shù)組
    $distance = array();//距離數(shù)組
    foreach($this->arc as $key=>$value){
      foreach($value as $k=>$v){
        $path[$key][$k] = $k;
        $distance[$key][$k] = $v;
      }
    }
    for($j = 0; $j < count($this->vexs); $j ++){
      for($i = 0; $i < count($this->vexs); $i ++){
        for($k = 0; $k < count($this->vexs); $k ++){
          if($distance[$this->vexs[$i]][$this->vexs[$k]] > $distance[$this->vexs[$i]][$this->vexs[$j]] + $distance[$this->vexs[$j]][$this->vexs[$k]]){
            $path[$this->vexs[$i]][$this->vexs[$k]] = $path[$this->vexs[$i]][$this->vexs[$j]];
            $distance[$this->vexs[$i]][$this->vexs[$k]] = $distance[$this->vexs[$i]][$this->vexs[$j]] + $distance[$this->vexs[$j]][$this->vexs[$k]];
          }
        }
      }
    }
    return array($path, $distance);
  }
  //djikstra算法
  public function dijkstra(){
    $final = array();
    $pre = array();//要查找的結(jié)點(diǎn)的前一個結(jié)點(diǎn)數(shù)組
    $weight = array();//權(quán)值和數(shù)組
    foreach($this->arc[$this->vexs[0]] as $k=>$v){
      $final[$k] = 0;
      $pre[$k] = $this->vexs[0];
      $weight[$k] = $v;
    }
    $final[$this->vexs[0]] = 1;
    for($i = 0; $i < count($this->vexs); $i ++){
      $key = 0;
      $min = $this->infinity;
      for($j = 1; $j < count($this->vexs); $j ++){
        $temp = $this->vexs[$j];
        if($final[$temp] != 1 && $weight[$temp] < $min){
          $key = $temp;
          $min = $weight[$temp];
        }
      }
      $final[$key] = 1;
      for($j = 0; $j < count($this->vexs); $j ++){
        $temp = $this->vexs[$j];
        if($final[$temp] != 1 && ($min + $this->arc[$key][$temp]) < $weight[$temp]){
          $pre[$temp] = $key;
          $weight[$temp] = $min + $this->arc[$key][$temp];
        }
      }
    }
    return $pre;
  }
  //kruscal算法
  private function kruscal(){
    $this->krus = array();
    foreach($this->vexs as $value){
      $krus[$value] = 0;
    }
    foreach($this->arc as $key=>$value){
      $begin = $this->findRoot($key);
      foreach($value as $k=>$v){
        $end = $this->findRoot($k);
        if($begin != $end){
          $this->krus[$begin] = $end;
        }
      }
    }
  }
  //查找子樹的尾結(jié)點(diǎn)
  private function findRoot($node){
    while($this->krus[$node] > 0){
      $node = $this->krus[$node];
    }
    return $node;
  }
  //prim算法,生成最小生成樹
  public function prim(){
    $this->primVexs = array();
    $this->primArc = array($this->vexs[0]=>0);
    for($i = 1; $i < count($this->vexs); $i ++){
      $this->primArc[$this->vexs[$i]] = $this->arc[$this->vexs[0]][$this->vexs[$i]];
      $this->primVexs[$this->vexs[$i]] = $this->vexs[0];
    }
    for($i = 0; $i < count($this->vexs); $i ++){
      $min = $this->infinity;
      $key;
      foreach($this->vexs as $k=>$v){
        if($this->primArc[$v] != 0 && $this->primArc[$v] < $min){
          $key = $v;
          $min = $this->primArc[$v];
        }
      }
      $this->primArc[$key] = 0;
      foreach($this->arc[$key] as $k=>$v){
        if($this->primArc[$k] != 0 && $v < $this->primArc[$k]){
          $this->primArc[$k] = $v;
          $this->primVexs[$k] = $key;
        }
      }
    }
    return $this->primVexs;
  }
  //一般算法,生成最小生成樹
  public function bst(){
    $this->primVexs = array($this->vexs[0]);
    $this->primArc = array();
    next($this->arc[key($this->arc)]);
    $key = NULL;
    $current = NULL;
    while(count($this->primVexs) < count($this->vexs)){
      foreach($this->primVexs as $value){
        foreach($this->arc[$value] as $k=>$v){
          if(!in_array($k, $this->primVexs) && $v != 0 && $v != $this->infinity){
            if($key == NULL || $v < current($current)){
              $key = $k;
              $current = array($value . $k=>$v);
            }
          }
        }
      }
      $this->primVexs[] = $key;
      $this->primArc[key($current)] = current($current);
      $key = NULL;
      $current = NULL;
    }
    return array('vexs'=>$this->primVexs, 'arc'=>$this->primArc);
  }
  //一般遍歷
  public function reserve(){
    $this->hasList = array();
    foreach($this->arc as $key=>$value){
      if(!in_array($key, $this->hasList)){
        $this->hasList[] = $key;
      }
      foreach($value as $k=>$v){
        if($v == 1 && !in_array($k, $this->hasList)){
          $this->hasList[] = $k;
        }
      }
    }
    foreach($this->vexs as $v){
      if(!in_array($v, $this->hasList))
        $this->hasList[] = $v;
    }
    return implode($this->hasList);
  }
  //廣度優(yōu)先遍歷
  public function bfs(){
    $this->hasList = array();
    $this->queue = array();
    foreach($this->arc as $key=>$value){
      if(!in_array($key, $this->hasList)){
        $this->hasList[] = $key;
        $this->queue[] = $value;
        while(!empty($this->queue)){
          $child = array_shift($this->queue);
          foreach($child as $k=>$v){
            if($v == 1 && !in_array($k, $this->hasList)){
              $this->hasList[] = $k;
              $this->queue[] = $this->arc[$k];
            }
          }
        }
      }
    }
    return implode($this->hasList);
  }
  //執(zhí)行深度優(yōu)先遍歷
  public function excuteDfs($key){
    $this->hasList[] = $key;
    foreach($this->arc[$key] as $k=>$v){
      if($v == 1 && !in_array($k, $this->hasList))
        $this->excuteDfs($k);
    }
  }
  //深度優(yōu)先遍歷
  public function dfs(){
    $this->hasList = array();
    foreach($this->vexs as $key){
      if(!in_array($key, $this->hasList))
        $this->excuteDfs($key);
    }
    return implode($this->hasList);
  }
  //返回圖的二維數(shù)組表示
  public function getArc(){
    return $this->arc;
  }
  //返回結(jié)點(diǎn)個數(shù)
  public function getVexCount(){
    return count($this->vexs);
  }
}
$a = array('a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i');
$b = array('ab'=>'10', 'af'=>'11', 'bg'=>'16', 'fg'=>'17', 'bc'=>'18', 'bi'=>'12', 'ci'=>'8', 'cd'=>'22', 'di'=>'21', 'dg'=>'24', 'gh'=>'19', 'dh'=>'16', 'de'=>'20', 'eh'=>'7','fe'=>'26');//鍵為邊,值權(quán)值
$test = new MGraph($a, $b);
print_r($test->bst());

【網(wǎng)站聲明】本站除付費(fèi)源碼經(jīng)過測試外,其他素材未做測試,不保證完整性,網(wǎng)站上部分源碼僅限學(xué)習(xí)交流,請勿用于商業(yè)用途。如損害你的權(quán)益請聯(lián)系客服QQ:2655101040 給予處理,謝謝支持。

相關(guān)文檔推薦

這篇文章主要介紹了PHP定義字符串的四種方式,非常不錯,具有參考借鑒價(jià)值,需要的朋友可以參考下
下面小編就為大家分享一篇php 替換文章中的圖片路徑,下載圖片到本地服務(wù)器的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
下面小編就為大家分享一篇PHP給源代碼加密的幾種方法匯總(推薦),具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
下面小編就為大家分享一篇php打開本地exe程序,js打開本地exe應(yīng)用程序,并傳遞相關(guān)參數(shù)方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
這篇文章主要介紹了PHP類的反射來實(shí)現(xiàn)依賴注入過程以及相關(guān)知識點(diǎn)分享,對此有興趣的朋友跟著小編學(xué)習(xí)下吧。
php遍歷一個文件夾內(nèi)的所有文件和文件夾,并刪除所有文件夾和子文件夾下的所有文件的代碼,通過遞歸方式實(shí)現(xiàn)達(dá)到清空一個目錄的效果。本文給大家分享實(shí)例代碼,需要的朋友參考
主站蜘蛛池模板: 亚洲国产中文在线 | 91免费观看| 国产免费视频 | 中文字幕一区在线观看视频 | 夜夜艹天天干 | 天天看天天操 | 成年人免费网站 | 国产成人艳妇aa视频在线 | 国产一区二区三区精品久久久 | 亚洲成人一级 | 激情av在线 | 国产欧美一区二区三区久久手机版 | 久久亚洲综合 | 最新日韩欧美 | 天天插天天舔 | 91成人在线视频 | 日本一区二区不卡 | 精品成人佐山爱一区二区 | 免费高清av | 国产精品视频网 | 日韩av网址在线观看 | 天天操操操操操 | 影音先锋成人资源 | 欧美黄色网络 | 欧美一级免费 | 亚洲福利一区二区 | 欧美一区二区三区日韩 | 亚洲精品欧美精品 | 日韩日b视频 | 日本不卡一区二区 | 在线观看视频你懂得 | 在线永久看片免费的视频 | 精品国产91亚洲一区二区三区www | 午夜99| 在线播放国产一区二区三区 | 在线免费av电影 | 日韩色图视频 | 国产精品资源在线 | 日韩福利 | 日本久久视频 | 99久久亚洲 |