HDU 4760 Good FireWall 完好Trie题解
本题乍看像是线段树之类的区间操作,只是由于仅仅是须要查找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题解的更多相关文章
- HDU 4760 Good Firewall ( Trie树 )
一开始看的时候就想歪了,比赛的时候一直在YY线段树区间覆盖,然后纠结节点数太多开不下怎么办啊啊啊啊…… 然后昨天吃饭的时候也在纠结这到底是个啥题,后来发现公共前缀->前缀??!!!!->这 ...
- HDU 1671 Phone List(Trie的应用与内存释放)
Phone List Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- HDU 1251 统计难题(Trie模版题)
统计难题 Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 131070/65535 K (Java/Others) Total Subm ...
- HDU 2846:Repository(Trie)
http://acm.hdu.edu.cn/showproblem.php?pid=2846 题意:给出N个模式串,再给出M个文本串,问每一个文本串在多少个模式串中出现. 思路:平时都是找前缀的,这里 ...
- hdu 1251 统计难题(trie 树的简单应用)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1251 题意:给你多个字符串,求以某个字符串为前缀的字符串数量. 思路:简单的trie数应用,在trie ...
- 【hdu 3177 Crixalis's Equipment】 题解
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3177 \(describe\): 有一个山洞,山洞的容积最大为\(v\).现在你有\(n\)个物品,这 ...
- HDU 5687 Problem C(Trie+坑)
Problem C Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others) Tota ...
- HDU 4099 Revenge of Fibonacci Trie+高精度
Revenge of Fibonacci Problem Description The well-known Fibonacci sequence is defined as following: ...
- HDU 4758 Walk Through Squares ( Trie图 && 状压DP && 数量限制类型 )
题意 : 给出一个 n 行.m 列的方格图,现从图左上角(0, 0) 到右下角的 (n, m)走出一个字符串(规定只能往下或者往右走),向右走代表' R ' 向下走则是代表 ' D ' 最后从左上角到 ...
随机推荐
- 2d-x中Lua类型强转问题
在Lua中,使用CCDictionary进行保存CCSprite对象,但是,在CCDictionary取出来的时候,此时是一个CCObject对象,无法调用子类精灵的一些方法.那只能进行强转的. 那么 ...
- Append加载动态轮播
前几天遇到了些小麻烦,不过很快就解决了.之所以要记下来是因为作为一名前端的程序员,要理解页面的加载顺序是最重要的.要不然自己写程序意外的出现bug~~ 刚开始写利用Append的时候,利用火狐的fir ...
- rpm包安装
RPM全称是“RedHatPackageManager”是由RedHat公司发发展起来的,本质是将软件源码包经过编译并且打包成rpm的格式,rpm文件包含的有二进制文件,配置文件,库文件等,同时RPM ...
- css让div水平垂直居中
示例1: .div1{ width:200px; height:300px; border:1px solid #000; position: relative; } .div2{ width: 40 ...
- Fragment实现底部Tab,切换可保存状态
activity_main.xml <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android& ...
- SqlServer 不同服务器之间数据库连接、数据库登录、数据传递
需求:我是本地数据库想纯SQL访问其它服务器上的数据库,而不使用数据库客户端的连接.这里面就想到了数据库link,通过下面的代码进行创建以后,就可以在本地对链接的服务器数据库进行操作了--添加SQLS ...
- UILable / UITextField / UIButton
// 获取屏幕大小的view UIView *contentView = [[UIView alloc] initWithFrame:[UIScreen mainScreen].bounds]; // ...
- ionic开发ios app
注意必须是mac系统 1. 首先要安装node环境,Ionic的安装和后续的许多前端工具的安装都依赖于node的包管理器npm. nodeJs环境的安装很简单,去官网下载最新版的NodeJs直接安装即 ...
- 读书笔记_Effective_C++_条款二十一:当必须返回对象时,别妄想返回其reference
在栈空间的临时成员变量在函数生命期结束后无法传出 friend A& operator*(const A& a, const A& b) { A temp; temp.data ...
- C++ Primer 5th 第3章 字符串、向量和数组
*****代码在Debian g++ 5.40 / clang++ 3.8(C++11)下编写调试***** 本章主要是关于字符串.数组的内容,以及一些简单的容器知识. 1.using的声明 usin ...