本题乍看像是线段树之类的区间操作,只是由于仅仅是须要查找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. Log4Qt 使用(二)

    本文基于上一篇<Log4Qt 使用(一)>来继续完善一下关于Log4Qt的使用. 在讲解之前,我们首先来看一个例子: int main(int argc, char *argv[]) { ...

  2. js操作符

    JS操作符 x + y(数字) 将x和y相加; x + y(字符串) 将x和y拼接在一起; x - y 从x中减去y; x * y 将x和y相乘; x / y 将x除以y; x % y x和y的模(既 ...

  3. 使用Teleport Pro离线下载网页所有内容

    在学习生活中,碰到网页中内容太多,如何讲其保存到本地,已方便随时查看呢? 使用Teleport Pro就可以解决问题:     首先下载Teleport Pro V1.54 汉化绿色版的,解压完之后 ...

  4. Ajax客户登陆验证

        服务器端操作方便之处我就不吹了,地球人都知道,它最烦莫过于页面刷新,头都被刷晕了,而且他在刷新的时候,还触发服务器端的事件,现在Ajax的出现,他们的结合是发展的必然!    一.介绍一下Aj ...

  5. php模块memcache和memcached区别分析

    zm总结:尽量使用memcached就好了 1.目前大多数php环境里使用的都是不带d的memcache版本,这个版本出的比较早,是一个原生版本,完全在php框架内开发的.与之对应的带d的memcac ...

  6. AngularJs练习Demo13Provider_factory

    @{ Layout = null; } <!DOCTYPE html> <html> <head> <meta name="viewport&quo ...

  7. js中substring和substr的用法 (转)

    1.substring 方法 定义和用法 substring 方法用于提取字符串中介于两个指定下标之间的字符. 语法 stringObject.substring(start,stop) 参数     ...

  8. Swift中可能失败的构造器的传播(调用)和重写

    import Foundation /* 可能失败构造器的传播(调用) 1.可能失败的构造器可以调用同一个类中的普通构造器 2.普通构造器不能调用同一个类中的可能失败构造器 3.结构体中, 普通构造器 ...

  9. css中的颜色值

    下面是比较适合在测试页面中用来设置背景颜色的淡颜色,最好记住一些,dark.blue.red.green.gray.olive颜色较深.

  10. 会话控制session,cookie(0521)

    简单介绍: 一.什么是session? 1. 定义: Session,在计算机中,尤其是在网络应用中,称为“会话”.在计算机专业术语中,Session是指一个终端用户与交互系统进行通信的时间间隔,通常 ...