前缀树java实现

1.定义:每个样本都从头节点开始,根据前缀字符或前缀数字建立出一棵树来。没有路就新建,有就复用。

2.准备

 public static int  MAXN = 150001;
    public static int cnt = 1;
    public static int[][] tree = new int[MAXN][26];
    public static int[] pass = new int[MAXN];
    public static int[] end = new int[MAXN];

用静态数组实现前缀树,tree数组表示数的具体实现,pass数组存储路径被是使用次数,end数组表示一共有多少字符串。

功能实现:

①插入②搜索某字符串是否被插入过返回次数③返回以某一字符串为前缀的字符串的数量④删除字符串⑤清空树

public static int  MAXN = 150001;
    public static int cnt = 1;
    public static int[][] tree = new int[MAXN][26];
    public static int[] pass = new int[MAXN];
    public static int[] end = new int[MAXN];
    
    public static void build(){
        cnt = 1;
    }

    public static void insert(String word){
        int cur = 1;
        pass[cur]++;
        for(int i=0,path;i<word.length();i++){
            path = word.charAt(i) - 'a';
            if(tree[cur][path]==0){
                tree[cur][path] = ++cnt;
            }
            cur = tree[cur][path];
            pass[cur]++;
        }
        end[cur]++;
    }

    public static int search(String word){
        int cur =1;
        for(int i=0,path;i<word.length();i++){
            path = word.charAt(i) - 'a';
            if(tree[cur][path]==0){
                return 0;
            }
            cur = tree[cur][path];
        }
        return end[cur];
    }

    public static int preFixNumber(String word){
        int cur = 1;
        for(int i=0,path;i<word.length();i++){
            path  = word.charAt(i) - 'a';
            if(tree[cur][path]==0){
                return 0;
            }
            cur = tree[cur][path];
        }
        return pass[cur];
    }

    public static void delete(String word){
        if(search(word)>0){
            int cur = 1;
            for(int i=0,path;i<word.length();i++){
                path = word.charAt(i) - 'a';
                if(--pass[tree[cur][path]]==0){
                    tree[cur][path] = 0;
                    return ;
                }
                cur = tree[cur][path];
            }
            end[cur]--;
        }
    }
    public static void clear(){
        for(int i=0;i<=cnt;i++){
            for(int j=0;j<tree[i].length;j++){
                tree[i][j] = 0;
            }
            pass[i] = 0;
            end[i] = 0;
        }
    }

前缀树,也称为字典树或Trie树,是一种树形数据结构,用于有效地处理字符串查找问题。每个节点表示一个字符串前缀,节点可以有零个或多个子节点。这里提供一份用Java实现前缀树的代码示例: ```java class TrieNode { // 存储子节点 private TrieNode[] children; // 标识是否是一个单词的结束节点 private boolean isEndOfWord; public TrieNode() { children = new TrieNode[26]; isEndOfWord = false; } public void insert(String word) { // 从根节点开始遍历 TrieNode curNode = this; for(char c : word.toCharArray()) { int index = c - 'a'; if(curNode.children[index] == null) { // 如果子节点不存在,则新建一个 curNode.children[index] = new TrieNode(); } // 继续遍历下一个节点 curNode = curNode.children[index]; } // 最后一个节点标识为单词结束节点 curNode.isEndOfWord = true; } public boolean search(String word) { TrieNode curNode = this; for(char c : word.toCharArray()) { int index = c - 'a'; if(curNode.children[index] == null) { // 如果不存在该节点 return false; } curNode = curNode.children[index]; } // 如果最后一个节点标识为单词结束节点,则表示查找到该单词 return curNode.isEndOfWord; } public boolean startsWith(String prefix) { TrieNode curNode = this; for(char c : prefix.toCharArray()) { int index = c - 'a'; if(curNode.children[index] == null) { // 如果不存在该节点 return false; } curNode = curNode.children[index]; } // 如果遍历完前缀,最后一个节点不为null,则表示该前缀存在 return true; } } public class Trie { private TrieNode root; public Trie() { root = new TrieNode(); } public void insert(String word) { root.insert(word); } public boolean search(String word) { return root.search(word); } public boolean startsWith(String prefix) { return root.startsWith(prefix); } } ``` 使用方法: ```java public static void main(String[] args) { Trie trie = new Trie(); trie.insert("apple"); boolean result1 = trie.search("apple"); // 返回 true boolean result2 = trie.search("app"); // 返回 false boolean result3 = trie.startsWith("app"); // 返回 true trie.insert("app"); boolean result4 = trie.search("app"); // 返回 true System.out.println(result1 + " " + result2 + " " + result3 + " " + result4); } ``` 输出结果: ``` true false true true ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值