淺談 trie樹 及其實現

定義又稱字典樹,單詞查找樹或者前綴樹,是一種用于快速檢索的多叉樹結構,

如英文字母的字典樹是一個26叉樹,數字的字典樹是一個10叉樹。

核心思想:是空間換時間.利用字符串的公共前綴來降低查詢時間的開銷以達到提高效率的目的。

三個基本性質:

1.?根結點不包含字符,除根結點外每一個結點都只包含一個字符。

2.?從根結點到某一結點,路徑上經過的字符連接起來,為該結點對應的字符串。

3.?每個結點的所有子結點包含的字符都不相同。

優點:利用字符串的公共前綴來節約存儲空間,最大限度地減少無謂的字符串比較,查詢效率比哈希表高。

缺點:如果存在大量字符串且這些字符串基本沒有公共前綴,則相應的trie樹將非常消耗內存。

典型應用:統計和排序大量的字符串(但不僅限于字符串),所以經常被搜索引擎系統用于文本詞頻統計

至于Trie樹的實現,可以用數組,靜態分配空間,也可以用指針動態分配

Trie樹的操作

在Trie樹中主要有3個操作,插入、查找和刪。一般情況下Trie樹中很少存在刪除單獨某個結點的情況,因此只考慮刪除整棵樹。

假設存在字符串str(都為小寫字母),Trie樹的根結點為root。i=0,p=root。

typedef struct stu
{int n,flag;         //n記錄前綴及單詞的個數,flag標記單詞是否存在struct stu *next[26];          //子節點
}node;
View Code
node* creat_node()
{node *p=(node *)malloc(sizeof(node));p->n=p->flag=0;memset(p->next,0,sizeof(p->next));return p;
}
建立新節點并初始化

1、插入

??1)取str[i],判斷p->next[str[i]-'a']是否為空,若為空,則建立結點temp,并將p->next[str[i]-'a']指向temp,然后p指向temp;

???若不為空,則p=p->next[str[i]-'a'];

??2)i++,繼續取str[i],循環1)中的操作,直到遇到結束符'\0',此時將當前結點p中的?flag?置為true。

void trie_insert(node *p,char *s)
{int i;while(*s!='\0'){i=*s-'a';if(p->next[i]==0)p->next[i]=creat_node();p=p->next[i];s++;p->n++;}p->flag=1;
}
插入字符串

2、查找

??1)取str[i],判斷判斷p->next[str[i]-'a']是否為空,若為空,則返回false;若不為空,則p=p->next[str[i]-'a'],繼續取字符。

??2)重復1)中的操作直到遇到結束符'\0',若當前結點p不為空并且 flag?為true,則返回true,否則返回false。

int trie_search(node *p,char *s)
{int i;while(*s!='\0'){i=*s-'a';p=p->next[i];if(p==0)return 0;s++;}return p->n;
}
查找字符串,并返回其個數

3、刪除

??刪除可以以遞歸的形式進行刪除。

void trie_del(node *root)
{int i;for(i=0;i<M;i++)                     //M為子節點的個數if(root->next[i]!=NULL)trie_del(root->next[i]);free(root);
}
遞歸刪除整棵樹

?

?

轉載于:https://www.cnblogs.com/happy-lcj/p/3890417.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/273640.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/273640.shtml
英文地址,請注明出處:http://en.pswp.cn/news/273640.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Docker-compose 安裝與基本使用(四)

安裝 Docker-Compose Compose有多種安裝方式,例如通過 shell, pip以及將 Compose作為容器安裝等。本次安裝以Shell 為主。 通過以下命令自動下載并安裝適應系統版本的 Compose: curl -L "https://github.com/docker/compose/releases/download/1.10.0/docker-compose-$(un…

如何開始DDD(完)

連續寫了兩篇文章&#xff0c;這一篇我想是序的完結篇了。結合用戶注冊的例子再將他簡單豐富一下。在這里只添加一個簡單需求&#xff0c;就是用戶注冊成功后給用戶發一封郵件。補充一下之前的代碼 public class DomainService {public void Register(User user){if (_userRepo…

git pull 報錯:Untracked Fles Preventing Merge

場景 使用 git pull 命令更新報錯解決 找到對應的文件刪除后重新打開項目。

關于string,我今天科普的

今天下午朋友討論組上討論一個關于string的問題&#xff0c;問題是這樣的&#xff0c;string a"aaa";string ba;a"bbb",為什么測試b的值不改變&#xff1f;之前我看過一個文章&#xff0c;知道肯定不相等&#xff0c;因為引用地址的一系列問題&#xff0c;…

git pull 報錯:The following untracked working tree files would be overwritten by merge

場景 使用 git pull 命令更新報錯 Updating d652d1c..fa05549 error: The following untracked working tree files would be overwritten by merge:.idea/encodings.xmlPlease move or remove them before you can merge. Aborting 解決 使用 git clean -d -fx 命令即可。

SpringBoot 配置多數據源

項目Git地址&#xff1a;SpringBoot 配置多數據源&#xff1a;Jacob-multi-data-source 準備工作 準備兩個數據庫(此模塊中兩個數據庫一個為本地 一個為遠程&#xff0c;本地為主&#xff0c;遠程為從)。然后建表。 #本地庫 CREATE TABLE username (id bigint(11) NOT NULL AUT…

HDU 2912

直線關于球的多次反射&#xff0c;求最后一次反射點 #include <iostream> #include <cstdio> #include <cstring> #include <algorithm> #include <cmath>using namespace std; const double inf1e10; const double eps1e-8; struct point {doub…

EMVTag系列3《持卡人基本信息數據》

9F61 持卡人證件號 L&#xff1a;2–26 R&#xff08;需求&#xff09;&#xff1a;數據應存在&#xff0c;在讀應用數據過程中&#xff0c;終端不檢查&#xff1b; (PBOC2.0第五部分中規定)芯片中持卡人姓名 5F20與持卡人姓名擴展9F0B只能使用一個&#xff0c;另一個必須不…

BindingException: Parameter 'XXX' not found. Available parameters are [collection, list]

應業務需求&#xff0c;需要使用到MQ進行數據上傳和下發。傳遞格式為JSON,服務那邊下發JSON數組&#xff0c;接收端將JSON數組轉換成List集合&#xff0c;調用Mybatis-plus批量添加saveBatch()。提示字段未找到... org.apache.ibatis.exceptions.PersistenceException: ### Er…

JDK 8 新特性 之 default關鍵字

前言 Jdk1.8之前的接口中只聲明方法&#xff0c;方法具體實現應在子類中進行。Jdk1.8打破了這樣的用法&#xff1a;接口中可以實現具體的方法體&#xff0c;只需要加上關鍵字static或者default修飾即可。 default關鍵字 public interface UserService {//自定義方法void getUse…

headroom.js插件使用方法

1.什么是headroom.js&#xff1f; headroom是用純Javascript寫的插件&#xff0c;用來隱藏和展示頁面元素&#xff0c;從而為頁面留下更多空間。比如使用headroom能使導航欄當頁面下滾時消失&#xff0c;當頁面上滾時候又出現。&#xff08;查看效果&#xff09; 2.工作原理 通…

JDK 8 新特性 之 方法引用

概述 方法引用&#xff1a;當要傳遞給Lambda體的操作&#xff0c;已經有實現的方法了&#xff0c;就可以使用方法引用方法引用&#xff1a;在Lambda的基礎上進一步的簡化。換句話說&#xff0c;方法引用就是Lambda表達式&#xff0c;也就是函數式接口的一個實例&#xff0c;通過…

項目記錄:springmvc forward redirect 問題

RequestMapping("/redirect")public String redirect(RedirectAttributes redirectAttributes){redirectAttributes.addFlashAttribute("test", "testdata"); //專供此種情況下使用。return "redirect:read";} 注意&#xff1a;此種情…

JDK 8 新特性 之 Lambda表達式

前言 Lambda 表達式&#xff0c;也可稱為閉包&#xff0c;它是推動 Java 8 發布的最重要新特性。Lambda 允許把函數作為參數傳遞進方法中。使用 Lambda 表達式可以使代碼變的更加簡潔緊湊。lambda表達式的重要特征: 可選類型聲明&#xff1a;不需要聲明參數類型&#xff0c;編譯…

開源組件DocX導出Word

1、使用Docx替換Word模板里書簽里內容的一個方法 using Novacode;public class ExportWord{/// <summary>/// 導出word/// </summary>/// <param name"lBookMarks">書簽數據源</param>/// <param name"sTemplatePath">導出W…

JDK 8 新特性 之 Strams簡單使用

概述 Java 8 API添加了一個新的抽象稱為流Stream&#xff0c;可以讓你以一種聲明的方式處理數據。 Stream 使用一種類似用 SQL 語句從數據庫查詢數據的直觀方式來提供一種對 Java 集合運算和表達的高階抽象。 Stream API可以極大提供Java程序員的生產力&#xff0c;讓程序員寫出…

Cannot open include file: jni.h: No such file or directory解決方法

在此運行Visual Studio 2012 項目時出現 #include <stdio.h> #include <jni.h> int main() { printf("Hello World"); } But when I try to build, I get the following error - 1>c:testtest.cpp(2) : fatal error C1083: Cannot open include file:…

JDK 8 新特性 之 函數接口

函數接口 定義:接口中只有唯一的一個抽象方法&#xff0c;該接口就稱之為函數接口。 //函數接口 public interface FunctionInterface1 {//1、只有一個方法的接口&#xff0c;默認稱之為函數接口void get(); }//非函數接口 public interface FunctionInterface2 {void get1();v…

微服務之基礎知識

什么是微服務架構 微服務是系統架構上的一種設計風格&#xff0c; 它的主旨是將一個原本獨立的系統拆分成多個小型服務&#xff0c;這些小型服務都在各自獨立的進程中運行&#xff0c;服務之間通過基于HTTP的RESTful API進行通信協作。 被拆分成的每一個小型服務都圍繞著系統中…

LightOj 1078 Basic Math

思路&#xff1a; 設輸入的兩個數分別為n和a,每一次所得到的數為update&#xff1a; 開始updatea,依次update分別為update*10a,這樣數據會超出范圍&#xff0c;則update每次為update(update*10a)%n即可&#xff0c; 如果update0,跳出循環&#xff1b; 只需證明&#xff1a;(upd…