VST實例(5)節點(NODE) 二、節點的遍歷

来源:https://www.cnblogs.com/luohq001/archive/2023/05/29/17440597.html
-Advertisement-
Play Games

編碼如下:#include <stdio.h> void swap(int* x,int* y ){ int tmp; tmp=*x; *x=*y; *y=tmp ; }; int main(){ int a=4; int b=5; printf("befer\n"); printf("a=%d\n ...


二、節點的遍歷

每一個節點都有一個index值,用於描述其在相同level,相同父節點下的序號,但是,遍歷時並不能利用這個值,因為這個值是變化的,當進行排序,插入節點、刪除節點等等操作時,這個index會發生改變。

同樣的,vst.AbsoluteIndex(node)返回的是某一節點在VST中的絕對序號,也會在排序等操作時進行改變。

1、通過函數遍歷節點

下麵這些函數可以找到特定的第一個的節點:

  •          function GetFirst(ConsiderChildrenAbove: Boolean = False): PVirtualNode;
  •          function GetFirstChild(Node: PVirtualNode): PVirtualNode;
  •          function GetFirstCutCopy(ConsiderChildrenAbove: Boolean = False): PVirtualNode;
  •          function GetFirstInitialized(ConsiderChildrenAbove: Boolean = False): PVirtualNode;
  •          function GetFirstNoInit(ConsiderChildrenAbove: Boolean = False): PVirtualNode;
  •          function GetFirstSelected(ConsiderChildrenAbove: Boolean = False): PVirtualNode;
  •          function GetFirstVisible(Node: PVirtualNode = nil; ConsiderChildrenAbove: Boolean = True; IncludeFiltered: Boolean = False): PVirtualNode;
  •          function GetFirstVisibleNoInit(Node: PVirtualNode = nil; ConsiderChildrenAbove: Boolean = True; IncludeFiltered: Boolean = False): PVirtualNode;

找下一個節點的函數有:

  •          function GetNext(Node: PVirtualNode; ConsiderChildrenAbove: Boolean = False): PVirtualNode;
  •          function GetNextChecked(Node: PVirtualNode; ConsiderChildrenAbove: Boolean): PVirtualNode;
  •          function GetNextInitialized(Node: PVirtualNode; ConsiderChildrenAbove: Boolean = False): PVirtualNode;
  •          function GetNextLeaf(Node: PVirtualNode): PVirtualNode;
  •          function GetNextLevel(Node: PVirtualNode; NodeLevel: Cardinal): PVirtualNode;
  •          function GetNextSelected(Node: PVirtualNode; ConsiderChildrenAbove: Boolean = False): PVirtualNode;
  •          function GetNextSibling(Node: PVirtualNode): PVirtualNode;
  •          function GetNextVisible(Node: PVirtualNode; ConsiderChildrenAbove: Boolean = True): PVirtualNode;
  •          GetNextVisible...,其它的針對可見NODE的查找下一個。

如果沒有指定節點的下一個,則返回nil。

查找最後一個節點:

  •          function GetLast(Node: PVirtualNode = nil; ConsiderChildrenAbove: Boolean = False): PVirtualNode;
  •          function GetLastChild(Node: PVirtualNode): PVirtualNode;
  •          function GetLastVisible(Node: PVirtualNode = nil; ConsiderChildrenAbove: Boolean = True; IncludeFiltered: Boolean = False): PVirtualNode;
  •          ……

下麵的代碼可以實現遍歷:

var node:pvirtualnode;

……

begin

  node:=vst.getfirst;

  repeat

     ……

     node:=vst.getnext(node);

  until node=vst.getlast;

end;

 

當然,上面的代碼中未包含對最後一個節點的處理。

2、遍歷函數IterateSubtree

遍歷函數的原型是:

function IterateSubtree(Node: PVirtualNode; Callback: TVTGetNodeProc; Data: Pointer; Filter: TVirtualNodeStates = []; DoInit: Boolean = False;

 

功能是對VST中的Node的每一個子(孫)節點進行一次回調函數(callback)的操作。如果回調函數返回了abord:=true,則終止遍歷,並返回終止遍歷的節點。

如果參數node=nil,則對所有的節點進行遍歷。

data是傳入回調函數的指針參數,而Filter是篩選器,如果為空,則對所有的節點進行遍歷,否則只對具有篩選器狀態的節點進行遍歷。例如如果filter包含“vsvisible”,則只對可見的節點進行遍歷。

回調函數的格式是:

TVTGetNodeProc = procedure (Sender: TBaseVirtualTree; Node: PVirtualNode; Data: Pointer; var Abort: Boolean) of object;

其中data指的就是IterateSubtree中指向的數據data。

以本程式為例,當編輯框中內容發生改變,則立即查找符合條件的節點,並設置為可見,不符合條件的節點則設置為不可見。

代碼如下:

procedure TForm2.lbledt1Change(Sender: TObject);

var data:pchar;

begin

   data:=PChar(lbledt1.Text);

   //beginupdate之後的語句只有當遇到endupdate之後才進行顯示刷新

   //可以避免程式的頻繁刷新導致的抖動,加快程式的運行。

   vst.BeginUpdate;

   //調用遍歷函數,傳入的參數是文本框的文本

   vst.IterateSubtree(nil,getnodebystr,data);

   //讓所有符合條件的節點展開

   vst.FullExpand();

   vst.EndUpdate;

end;

 

回調函數getnodebystr的定義和代碼如下:

procedure GETNODEBYSTR(Sender: TBaseVirtualTree; Node: PVirtualNode; Data: Pointer; var Abort: Boolean);

……

procedure TForm2.GETNODEBYSTR(Sender: TBaseVirtualTree; Node: PVirtualNode; Data: Pointer; var Abort: Boolean);

var str1,str2:string;

begin

  str1:=string(Data).Trim.ToUpper;

  with pcodes(Sender.GetNodeData(node))^ do

  begin

    str2:=icao+'|'+iata+'|'+names;

  end;

  str2:=str2.Trim.ToUpper;

  if str2.Contains(str1) or str1.IsEmpty then

  begin

    Sender.FullyVisible[NODE]:=true;

    //下麵部分是設置可見的另一種寫法,如果不包含子節點,則效率會更高

    //sender.IsVisible[node]:=True;

    //if Sender.GetNodeLevel(node)>0 then//有父節點的情況下

    //  Sender.IsVisible[Node.Parent]:=True;

  end

  else

  begin

    Sender.IsVisible[node]:=false;

  end;

end;

 

本程式共有630個節點,在執行中完全可以實時查詢無停頓。VST的效率非常高。

3、節點序列

前面我們提到過vst.SelectedNodes和vst.checkedNodes都是節點序列,節點序列的定義是這樣的:

TVTVirtualNodeEnumeration = record

  objectprivateFMode: TVZVirtualNodeEnumerationMode;

  FTree: TBaseVirtualTree;

  FConsiderChildrenAbove: Boolean;

  FNode: PVirtualNode;

  FNodeLevel: Cardinal;

  FState: TCheckState;

  FIncludeFiltered: Boolean;

end;

 

並提供了一個函數:GetEnumerator,用於從FTREE中獲取具有指定特征的節點,例如checkednodes就是從VST中取得具有某種TCheckState屬性的所有節點,而SelectedNodes則是獲得具有selected屬性的節點(node的states屬性),所以你也可以寫自己的節點序列程式。

函數GetEnumerator返回的是一個TVTVirtualNodeEnumerator類型的結構(record),其具體使用代碼例子如下:

var node:PVirtualNode;

begin

  with vst.CheckedNodes().GetEnumerator do

  begin

    MoveNext;

repeat

      node:=Current;

      ……

    until not MoveNext;

  end;

end;

 

需要註意的是,一定要針對GetEnumerator返回的結構進行操作,不能一次次的調用GetEnumerator.current 和GetEnumerator.movenext,因為每次調用一次GetEnumerator就會生成一次序列。

請註意,調用GetEnumerator後,current並沒有對應某個NODE,需要調用一次movenext後,current才指向第一個node,每調用一次movenext,current指向下一個node,如果已經是最後一個node,調用movenext則返回false.

程式中寫了一小段代碼,用於把選中的機場資料導出到一個CSV文件中。

procedure TForm2.N5Click(Sender: TObject);

var sl:TStringList;s:string;

begin

  if vst.CheckedCount=0 then Exit;

  sl:=TStringList.Create;

  with vst.CheckedNodes(csCheckedNormal).GetEnumerator do

  begin

    MoveNext;

    repeat

       //如果是根節點,即FIR,則跳到下一個迴圈

       if vst.GetNodeLevel(current)=0 then  Continue;

       s:='';

       if Assigned(current) then

       with pcodes(vst.GetNodeData(current))^ do

       begin

         s:=s.Join(',',[icao,iata,names,rwy_style,apt_type]);

       end;

       sl.Add(s);

    until not MoveNext;

  end;

  //直接把文檔存儲到執行文件所在目錄的文件中

  sl.SaveToFile('checkedapts.csv');

  FreeAndNil(sl);

end;

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • > 本文首發於公眾號:Hunter後端 > 原文鏈接:[Python連接es筆記二之查詢方式彙總](https://mp.weixin.qq.com/s/0Yn5c-U9pBWrSC5HrCgWog) 上一節除了介紹使用 Python 連接 es,還有最簡單的 query() 方法,這一節介紹一下幾 ...
  • Python3 支持int(整型數據)、float(浮點型數據)、bool(布爾類型) # 1.int(整型數據) 在Python 3里,**只有一種整數類型 int,表示為長整型**。像大多數語言一樣,數值類型的賦值和計算都是很直觀的。 ## 1.1數值運算 編寫程式如下所示 ![image](h ...
  • ## 聊一下MySQL 關於mysql關係型資料庫的一些分析: 1、從性能上:如果我們碰到需要執行耗時特別久,並且執行結果不是很頻繁變動的SQL語句,我們就沒有必要每次都去查詢資料庫,因為每次操作資料庫都很耗時。 2、從併發上:在大併發的情況下(比如618秒殺活動,你敢讓千萬級的請求直接打到資料庫上 ...
  • 歡迎來到系列第八篇,異常處理的深入探討。本文將分五部分展開。首先,我們將學習Python異常處理的基礎知識,理解`try/except`語句的用法。然後,我們將瞭解Python的常見異常類型並通過實例理解它們的作用。第三部分,我們將更深入地解析`try-except`塊,理解其工作原理及更加複雜的用... ...
  • 哈嘍大家好,我是鹹魚 相信小伙伴們在學習 python 數據分析的過程中或多或少都會聽說或者使用過 pandas pandas 是 python 的一個拓展庫,常用於數據分析 今天鹹魚將介紹幾個關於 pandas 導入數據的方法和技巧 ## 從 URL 獲取 csv 數據 關於 pandas 導入 ...
  • ## 教程簡介 Angular Highcharts是一個基於Angular的開源組件,可在Angular應用程式中提供優雅且功能豐富的高圖表可視化,並可與Angular組件無縫配合使用。 [Angular Highcharts入門教程](https://www.itbaoku.cn/tutoria ...
  • ## 教程簡介 Flume是Apache下麵的一個分散式組件,它提供高效,可靠的收集,整合,傳輸日誌數據的服務。Flume可以理解成一個管道,它連接數據的生產者和消費者,它從數據的生產者(Source)獲取數據,保存在自己的緩存(Channel)中,然後通過Sink發送到消費者。它不對數據做保存和復 ...
  • ## 測試環境: > MySQL版本:8.0 資料庫表:T (主鍵id,唯一索引c,普通欄位d) ![](http://img.javastack.cn/1685072039483867.png) 如果你的業務設計依賴於自增主鍵的連續性,這個設計假設自增主鍵是連續的。但實際上,這樣的假設是錯的,因為 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 在我們開發過程中基本上不可或缺的用到一些敏感機密數據,比如SQL伺服器的連接串或者是OAuth2的Secret等,這些敏感數據在代碼中是不太安全的,我們不應該在源代碼中存儲密碼和其他的敏感數據,一種推薦的方式是通過Asp.Net Core的機密管理器。 機密管理器 在 ASP.NET Core ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 順序棧的介面程式 目錄順序棧的介面程式頭文件創建順序棧入棧出棧利用棧將10進位轉16進位數驗證 頭文件 #include <stdio.h> #include <stdbool.h> #include <stdlib.h> 創建順序棧 // 指的是順序棧中的元素的數據類型,用戶可以根據需要進行修改 ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • C總結與剖析:關鍵字篇 -- <<C語言深度解剖>> 目錄C總結與剖析:關鍵字篇 -- <<C語言深度解剖>>程式的本質:二進位文件變數1.變數:記憶體上的某個位置開闢的空間2.變數的初始化3.為什麼要有變數4.局部變數與全局變數5.變數的大小由類型決定6.任何一個變數,記憶體賦值都是從低地址開始往高地 ...
  • 如果讓你來做一個有狀態流式應用的故障恢復,你會如何來做呢? 單機和多機會遇到什麼不同的問題? Flink Checkpoint 是做什麼用的?原理是什麼? ...
  • C++ 多級繼承 多級繼承是一種面向對象編程(OOP)特性,允許一個類從多個基類繼承屬性和方法。它使代碼更易於組織和維護,並促進代碼重用。 多級繼承的語法 在 C++ 中,使用 : 符號來指定繼承關係。多級繼承的語法如下: class DerivedClass : public BaseClass1 ...
  • 前言 什麼是SpringCloud? Spring Cloud 是一系列框架的有序集合,它利用 Spring Boot 的開發便利性簡化了分散式系統的開發,比如服務註冊、服務發現、網關、路由、鏈路追蹤等。Spring Cloud 並不是重覆造輪子,而是將市面上開發得比較好的模塊集成進去,進行封裝,從 ...
  • class_template 類模板和函數模板的定義和使用類似,我們已經進行了介紹。有時,有兩個或多個類,其功能是相同的,僅僅是數據類型不同。類模板用於實現類所需數據的類型參數化 template<class NameType, class AgeType> class Person { publi ...
  • 目錄system v IPC簡介共用記憶體需要用到的函數介面shmget函數--獲取對象IDshmat函數--獲得映射空間shmctl函數--釋放資源共用記憶體實現思路註意 system v IPC簡介 消息隊列、共用記憶體和信號量統稱為system v IPC(進程間通信機制),V是羅馬數字5,是UNI ...