本题乍看像是线段树之类的区间操作,只是由于仅仅是须要查找ip的前缀,故此事实上是使用Trie来做。

挺高难度的Trie应用,做完这道题之后说明Trie功力有一定火候了。

这里的Trie使用到了Delete函数。这是个Trie函数中最难的函数了,当然要使用数组记录的方法水掉。也是能够的。这里不水。给出delete函数。

考点难点:

1 Trie的操作函数的灵活运用。主要难点是delete函数的灵活运用

2  在叶子节点全部的group id, 删除的时候要注意,不能一气删除了,有多个group id会挂在同一颗树中

3  源ip和目的ip或许在多个叶子节点中,要使用两个vector记录全部group id,然后使用查找两集合是否有公共值的算法解之

4 ip值转换为一个整数值记录就能够了,这里使用了unsigned,只是应该是int也能够的,由于仅仅须要操作其二进制值,和整数值无关。

5 使用mask值,mask值后面的0能够无论,加速程序

全部问题考虑全。高级数据结构中包含各种小问题处理,大算法中使用到多种小算法,综合难度超过5星级。做完后非常有成就感O(∩_∩)O哈哈~。

#include <stdio.h>
#include <bitset>
#include <vector>
#include <algorithm>
using namespace std; const int MAX_ID = 1024;
const int MAX_LINE = 32769;
const int MAX_N = 15;
const int ARR_SIZE = 2;
const int MAX_DIGIT = 32;
const int MAX_NODE = MAX_ID*MAX_DIGIT*MAX_N;
unsigned gEnableIdIpMask[MAX_ID][MAX_N<<1];
int gLen[MAX_ID], id; struct Node
{
int n;
vector<int> id;
Node *arr[ARR_SIZE];
}; void clearNode(Node *rt)
{
rt->n = 0;
rt->id.clear();
for (int i = 0; i < ARR_SIZE; i++)
{
rt->arr[i] = NULL;
}
} Node pool[MAX_NODE];
int poolID; void insertTrie(Node *trie, unsigned dig, unsigned mark, int id)
{
bitset<MAX_DIGIT> bi = dig;
int m = MAX_DIGIT - mark;
for (int i = MAX_DIGIT-1; i >= m; i--)
{
Node *&p = trie->arr[bi[i]];//注意*&p,操作同名地址变量
if (!p)//推断成if(p)。程序崩溃。
{
p = &pool[poolID++];
clearNode(p);
}
trie = p;
}
trie->n++;
trie->id.push_back(id);
} bool searchNode(Node *trie, unsigned dig, vector<int> &id)
{
bitset<MAX_DIGIT> bi = dig;
bool flag = false;
for (int i = MAX_DIGIT-1; i >= 0; i--)
{
trie = trie->arr[bi[i]];
if (!trie) return flag;
if (trie->n)
{
flag = true;
id.insert(id.end(), trie->id.begin(), trie->id.end());
}
}
return flag;
} inline bool isFreeNode(Node *rt)
{
for (int i = 0; i < ARR_SIZE; i++)
{
if (rt->arr[i]) return false;
}
return true;
} bool deleteNodeHelper(Node *trie, bitset<MAX_DIGIT> &bi, int mask, int id,
int lv = MAX_DIGIT-1)//注意lv含义。准确赋值
{
if (trie)
{
if (mask-1 == lv)//注意:下标计算。对齐
{
if (trie->n)
{
if (trie->n > 1)
{
int j = 0;
for (; j < trie->n; j++) if (trie->id[j] == id) break;
trie->id.erase(trie->id.begin()+j);
trie->n--;
}
else
{
trie->n = 0;
trie->id.clear();
return isFreeNode(trie);
}
}
}
else
{
if (deleteNodeHelper(trie->arr[bi[lv]], bi, mask, id, lv-1))
{
trie->arr[bi[lv]] = NULL;
return !trie->n && isFreeNode(trie);
}
}
}
return false;
} inline void deleteNode(Node *trie, unsigned dig, unsigned mask, int id)
{
bitset<MAX_DIGIT> bi = dig;
int m = MAX_DIGIT - mask; //注意长度计算
deleteNodeHelper(trie, bi, m, id);
} unsigned getIP()
{
unsigned ip = 0;
unsigned part;
for (int j = 3; j >= 0; j--)
{
scanf("%u", &part);
getchar();
ip |= part<<(j<<3);
}
return ip;
} int main()
{
char cmd;
unsigned ip, mask;
Node *trie = &pool[0];
clearNode(trie);
poolID = 1;
while (scanf("%c", &cmd) != EOF)
{
if (cmd == 'E')
{
scanf("%d", &id);
scanf("%d", gLen+id);
for (int i = 0; i < gLen[id]; i++)
{
ip = getIP();
scanf("%u", &mask);
gEnableIdIpMask[id][i<<1] = ip;
gEnableIdIpMask[id][i<<1|1] = mask;
insertTrie(trie, ip, mask, id);
}
getchar();
}
else if (cmd == 'D')
{
scanf("%d", &id);
getchar();
for (int i = 0; i < gLen[id]; i++)
{
deleteNode(trie, gEnableIdIpMask[id][i<<1],
gEnableIdIpMask[id][i<<1|1], id);
}
gLen[id] = 0;
}
else// if (cmd == 'F')
{
ip = getIP();
vector<int> ipSrc, ipDes;
bool src = searchNode(trie, ip, ipSrc);
ip = getIP();
if (src) src = searchNode(trie, ip, ipDes);
if (src)
{
sort(ipSrc.begin(), ipSrc.end());
sort(ipDes.begin(), ipDes.end());
bool hasSame = false;
for (int i = 0, j = 0;
i < (int)ipSrc.size() && j < (int)ipDes.size(); )
{
if (ipSrc[i] == ipDes[j])
{
hasSame = true;
break;
}
else if (ipSrc[i] < ipDes[j]) i++;
else j++;
}
if (hasSame) puts("F");
else puts("D");
}
else puts("D");
}
}
return 0;
}

HDU 4760 Good FireWall 完好Trie题解的更多相关文章

  1. HDU 4760 Good Firewall ( Trie树 )

    一开始看的时候就想歪了,比赛的时候一直在YY线段树区间覆盖,然后纠结节点数太多开不下怎么办啊啊啊啊…… 然后昨天吃饭的时候也在纠结这到底是个啥题,后来发现公共前缀->前缀??!!!!->这 ...

  2. HDU 1671 Phone List(Trie的应用与内存释放)

    Phone List Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total ...

  3. HDU 1251 统计难题(Trie模版题)

    统计难题 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 131070/65535 K (Java/Others) Total Subm ...

  4. HDU 2846:Repository(Trie)

    http://acm.hdu.edu.cn/showproblem.php?pid=2846 题意:给出N个模式串,再给出M个文本串,问每一个文本串在多少个模式串中出现. 思路:平时都是找前缀的,这里 ...

  5. hdu 1251 统计难题(trie 树的简单应用)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1251 题意:给你多个字符串,求以某个字符串为前缀的字符串数量. 思路:简单的trie数应用,在trie ...

  6. 【hdu 3177 Crixalis's Equipment】 题解

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3177 \(describe\): 有一个山洞,山洞的容积最大为\(v\).现在你有\(n\)个物品,这 ...

  7. HDU 5687 Problem C(Trie+坑)

    Problem C Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) Tota ...

  8. HDU 4099 Revenge of Fibonacci Trie+高精度

    Revenge of Fibonacci Problem Description The well-known Fibonacci sequence is defined as following: ...

  9. HDU 4758 Walk Through Squares ( Trie图 && 状压DP && 数量限制类型 )

    题意 : 给出一个 n 行.m 列的方格图,现从图左上角(0, 0) 到右下角的 (n, m)走出一个字符串(规定只能往下或者往右走),向右走代表' R ' 向下走则是代表 ' D ' 最后从左上角到 ...

随机推荐

  1. web_api vs2015 新加标题无法打开

    HomeController 去掉特性[Authorize]

  2. 反射那些事儿——Java动态装载和反射技术

    一直以来反射都是只闻其声,却无法将之使用,近日尽心下来学习下,发现了很多精妙之处. Java动态装载和反射技术 一.类的动态装载 1.Java代码编译和执行的整个过程包含了以下三个重要的机制: ● J ...

  3. Linq/List/Array/IEnumerable等集合操作

    来源:http://www.cnblogs.com/liushanshan/archive/2011/01/05/1926263.html 目录 1    LINQ查询结果集    1 2    Sy ...

  4. GitHub 相关内容

    1. Git是分布式版本控制系统 集中式版本控制系统:版本库是集中存放在中央服务器的,而干活的时候,用的都是自己的电脑,所以要先从中央服务器取得最新的版本,然后开始干活,干完活了,再把自己的活推送给中 ...

  5. hadoop文本转换为序列文件

    在以前使用hadoop的时候因为mahout里面很多都要求输入文件时序列文件,所以涉及到把文本文件转换为序列文件或者序列文件转为文本文件(因为当时要分析mahout的源码,所以就要看到它的输入文件是什 ...

  6. C++ Primer 5th 第4章 表达式

    表达式是运算对象及运算符组成的式子,表达式求值将得到一个结果,单独的变量或者字面值也算表达式,结果是其本身. 运算符分为:一元运算符.二元运算符.三元运算符.一元即一个作用对象,二元两个作用对象,以此 ...

  7. github 推送时can't be established.

    http://www.xuebuyuan.com/2095099.html 飞凡@FANZ /e/learngit (master)$ git push origin masterThe authen ...

  8. js深入理解"闭包"

    一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量 ...

  9. 更换ios 开发者账号与下载别人的代码 真机调试时注意切换

    Buid Setting search sign

  10. Ajax实现三级联动(0520)

    查询数据库中的chinastates表,通过父级代号查询相应省市区. 实现界面: 在js页面实现三级联动 在JQuery中调用Ajax方法(引用JQuery文件一定放在最上面) 用插件的形式,创建三个 ...