FP树 高效发现频繁项集

1:a,b,c,d,e

2:a,b,c

3:e,f

4:b,c,d,e

现将其建立FP树

计算支持度

a=2,b=3,c=3,d=2,e=2,f=1

剔除支持度=1的,再排序

b3,c3,a2,d2,e2

1:b,c,a,d,e

2:b,c,a

3:e

4:b,c,d,e

第一个插入树中:

第二个插入树中:

第三个插入:

第四个:

这样找频繁项集的支持度只需要找路径就好了

from __future__ import print_function
print(__doc__)class treeNode:def __init__(self,nameValue,numOccur,parentNode):self.name=nameValueself.count=numOccurself.nodeLink=Noneself.parent=parentNodeself.children={}def inc(self,numOccur):self.count+=numOccurdef disp(self,ind=1):#将树以文本形式展示print('  '*ind,self.name,' ',self.count)for child in self.children.values():child.disp(ind+1)def loadSimpDat():simpDat = [['r', 'z', 'h', 'j', 'p'],['z', 'y', 'x', 'w', 'v', 'u', 't', 's'],['z'],['r', 'x', 'n', 'o', 's'],#    ['r', 'x', 'n', 'o', 's'],['y', 'r', 'x', 'z', 'q', 't', 'p'],['y', 'z', 'x', 'e', 'q', 's', 't', 'm']]return simpDatdef createInitSet(dataSet):retDict = {}for trans in dataSet:if not retDict.has_key(frozenset(trans)):retDict[frozenset(trans)] = 1else:retDict[frozenset(trans)] += 1return retDict# this version does not use recursion
def updateHeader(nodeToTest, targetNode):"""updateHeader(更新头指针,建立相同元素之间的关系,例如:  左边的r指向右边的r值,就是后出现的相同元素 指向 已经出现的元素)从头指针的nodeLink开始,一直沿着nodeLink直到到达链表末尾。这就是链表。性能: 如果链表很长可能会遇到迭代调用的次数限制。Args:nodeToTest  满足minSup {所有的元素+(value, treeNode)}targetNode  Tree对象的子节点"""# 建立相同元素之间的关系,例如:  左边的r指向右边的r值while(nodeToTest.nodeLink is not None):nodeToTest=nodeToTest.nodeLinknodeToTest.nodeLink=targetNodedef updateTree(items,inTree,headerTable,count):"""updateTree(更新FP-tree,第二次遍历)# 针对每一行的数据# 最大的key,  添加Args:items       满足minSup 排序后的元素key的数组(大到小的排序)inTree      空的Tree对象headerTable 满足minSup {所有的元素+(value, treeNode)}count       原数据集中每一组Kay出现的次数"""# 取出 元素 出现次数最高的# 如果该元素在 inTree.children 这个字典中,就进行累加# 如果该元素不存在 就 inTree.children 字典中新增key,value为初始化的 treeNode 对象if items[0] in inTree.children:# 更新 最大元素,对应的 treeNode 对象的count进行叠加inTree.children[items[0]].inc(count)else:# 如果不存在子节点,我们为该inTree添加子节点inTree.children[items[0]] = treeNode(items[0], count, inTree)# 如果满足minSup的dist字典的value值第二位为null, 我们就设置该元素为 本节点对应的tree节点# 如果元素第二位不为null,我们就更新header节点if headerTable[items[0]][1] is None:# headerTable只记录第一次节点出现的位置headerTable[items[0]][1] = inTree.children[items[0]]else:# 本质上是修改headerTable的key对应的Tree,的nodeLink值updateHeader(headerTable[items[0]][1], inTree.children[items[0]])if len(items) > 1:# 递归的调用,在items[0]的基础上,添加item0[1]做子节点, count只要循环的进行累计加和而已,统计出节点的最后的统计值。updateTree(items[1:], inTree.children[items[0]], headerTable, count)def createTree(dataSet, minSup=1):"""createTree(生成FP-tree)Args:dataSet  dist{行: 出现次数}的样本数据minSup   最小的支持度Returns:retTree  FP-treeheaderTable 满足minSup {所有的元素+(value, treeNode)}"""# 支持度>=minSup的dist{所有元素: 出现的次数}headerTable = {}# 循环 dist{行: 出现次数}的样本数据for trans in dataSet:# 对所有的行进行循环,得到行里面的所有元素# 统计每一行中,每个元素出现的总次数for item in trans:# 例如:  {'ababa': 3}  count(a)=3+3+3=9   count(b)=3+3=6headerTable[item] = headerTable.get(item, 0) + dataSet[trans]# 删除 headerTable中,元素次数<最小支持度的元素for k in headerTable.keys():if headerTable[k] < minSup:del(headerTable[k])# 满足minSup: set(各元素集合)freqItemSet = set(headerTable.keys())# 如果不存在,直接返回Noneif len(freqItemSet) == 0:return None, Nonefor k in headerTable:# 格式化:  dist{元素key: [元素次数, None]}headerTable[k] = [headerTable[k], None]# create treeretTree = treeNode('Null Set', 1, None)# 循环 dist{行: 出现次数}的样本数据for tranSet, count in dataSet.items():# print 'tranSet, count=', tranSet, count# localD = dist{元素key: 元素总出现次数}localD = {}for item in tranSet:# 判断是否在满足minSup的集合中if item in freqItemSet:# print 'headerTable[item][0]=', headerTable[item][0], headerTable[item]localD[item] = headerTable[item][0]# print 'localD=', localDif len(localD) > 0:# p=key,value; 所以是通过value值的大小,进行从大到小进行排序# orderedItems 表示取出元组的key值,也就是字母本身,但是字母本身是大到小的顺序orderedItems = [v[0] for v in sorted(localD.items(), key=lambda p: p[1], reverse=True)]# print 'orderedItems=', orderedItems, 'headerTable', headerTable, '\n\n\n'# 填充树,通过有序的orderedItems的第一位,进行顺序填充 第一层的子节点。updateTree(orderedItems, retTree, headerTable, count)return retTree, headerTabledef ascendTree(leafNode, prefixPath):"""ascendTree(如果存在父节点,就记录当前节点的name值)Args:leafNode   查询的节点对于的nodeTreeprefixPath 要查询的节点值"""if leafNode.parent is not None:prefixPath.append(leafNode.name)ascendTree(leafNode.parent, prefixPath)def findPrefixPath(basePat, treeNode):"""findPrefixPath 基础数据集Args:basePat  要查询的节点值treeNode 查询的节点所在的当前nodeTreeReturns:condPats 对非basePat的倒叙值作为key,赋值为count数"""condPats = {}# 对 treeNode的link进行循环while treeNode is not None:prefixPath = []# 寻找改节点的父节点,相当于找到了该节点的频繁项集ascendTree(treeNode, prefixPath)# 避免 单独`Z`一个元素,添加了空节点if len(prefixPath) > 1:# 对非basePat的倒叙值作为key,赋值为count数# prefixPath[1:] 变frozenset后,字母就变无序了# condPats[frozenset(prefixPath)] = treeNode.countcondPats[frozenset(prefixPath[1:])] = treeNode.count# 递归,寻找改节点的下一个 相同值的链接节点treeNode = treeNode.nodeLink# print treeNodereturn condPatsdef mineTree(inTree, headerTable, minSup, preFix, freqItemList):"""mineTree(创建条件FP树)Args:inTree       myFPtreeheaderTable  满足minSup {所有的元素+(value, treeNode)}minSup       最小支持项集preFix       preFix为newFreqSet上一次的存储记录,一旦没有myHead,就不会更新freqItemList 用来存储频繁子项的列表"""# 通过value进行从小到大的排序, 得到频繁项集的key# 最小支持项集的key的list集合bigL = [v[0] for v in sorted(headerTable.items(), key=lambda p: p[1])]print('-----', sorted(headerTable.items(), key=lambda p: p[1]))print('bigL=', bigL)# 循环遍历 最频繁项集的key,从小到大的递归寻找对应的频繁项集for basePat in bigL:# preFix为newFreqSet上一次的存储记录,一旦没有myHead,就不会更新newFreqSet = preFix.copy()newFreqSet.add(basePat)print('newFreqSet=', newFreqSet, preFix)freqItemList.append(newFreqSet)print('freqItemList=', freqItemList)condPattBases = findPrefixPath(basePat, headerTable[basePat][1])print('condPattBases=', basePat, condPattBases)# 构建FP-treemyCondTree, myHead = createTree(condPattBases, minSup)print('myHead=', myHead)# 挖掘条件 FP-tree, 如果myHead不为空,表示满足minSup {所有的元素+(value, treeNode)}if myHead is not None:myCondTree.disp(1)print('\n\n\n')# 递归 myHead 找出频繁项集mineTree(myCondTree, myHead, minSup, newFreqSet, freqItemList)print('\n\n\n')# import twitter
# from time import sleep
# import re# def getLotsOfTweets(searchStr):
#     """
#     获取 100个搜索结果页面
#     """
#     CONSUMER_KEY = ''
#     CONSUMER_SECRET = ''
#     ACCESS_TOKEN_KEY = ''
#     ACCESS_TOKEN_SECRET = ''
#     api = twitter.Api(consumer_key=CONSUMER_KEY, consumer_secret=CONSUMER_SECRET, access_token_key=ACCESS_TOKEN_KEY, access_token_secret=ACCESS_TOKEN_SECRET)#     # you can get 1500 results 15 pages * 100 per page
#     resultsPages = []
#     for i in range(1, 15):
#         print "fetching page %d" % i
#         searchResults = api.GetSearch(searchStr, per_page=100, page=i)
#         resultsPages.append(searchResults)
#         sleep(6)
#     return resultsPages# def textParse(bigString):
#     """
#     解析页面内容
#     """
#     urlsRemoved = re.sub('(http:[/][/]|www.)([a-z]|[A-Z]|[0-9]|[/.]|[~])*', '', bigString)    
#     listOfTokens = re.split(r'\W*', urlsRemoved)
#     return [tok.lower() for tok in listOfTokens if len(tok) > 2]# def mineTweets(tweetArr, minSup=5):
#     """
#     获取频繁项集
#     """
#     parsedList = []
#     for i in range(14):
#         for j in range(100):
#             parsedList.append(textParse(tweetArr[i][j].text))
#     initSet = createInitSet(parsedList)
#     myFPtree, myHeaderTab = createTree(initSet, minSup)
#     myFreqList = []
#     mineTree(myFPtree, myHeaderTab, minSup, set([]), myFreqList)
#     return myFreqList
# rootNode = treeNode('pyramid', 9, None)# rootNode.children['eye'] = treeNode('eye', 13, None)# rootNode.children['phoenix'] = treeNode('phoenix', 3, None)# # 将树以文本形式显示# # print rootNode.disp()# load样本数据simpDat = loadSimpDat()# print simpDat, '\n'# frozen set 格式化 并 重新装载 样本数据,对所有的行进行统计求和,格式: {行: 出现次数}initSet = createInitSet(simpDat)print(initSet)# 创建FP树# 输入: dist{行: 出现次数}的样本数据  和  最小的支持度# 输出: 最终的PF-tree,通过循环获取第一层的节点,然后每一层的节点进行递归的获取每一行的字节点,也就是分支。然后所谓的指针,就是后来的指向已存在的myFPtree, myHeaderTab = createTree(initSet, 3)myFPtree.disp()# 抽取条件模式基# 查询树节点的,频繁子项print('x --->', findPrefixPath('x', myHeaderTab['x'][1]))print('z --->', findPrefixPath('z', myHeaderTab['z'][1]))print('r --->', findPrefixPath('r', myHeaderTab['r'][1]))# 创建条件模式基freqItemList = []mineTree(myFPtree, myHeaderTab, 3, set([]), freqItemList)print(freqItemList)# # 项目实战# # 1.twitter项目案例# # 无法运行,因为没发链接twitter# lotsOtweets = getLotsOfTweets('RIMM')# listOfTerms = mineTweets(lotsOtweets, 20)# print len(listOfTerms)# for t in listOfTerms:#     print t# # 2.新闻网站点击流中挖掘,例如: 文章1阅读过的人,还阅读过什么?# parsedDat = [line.split() for line in open('data/12.FPGrowth/kosarak.dat').readlines()]# initSet = createInitSet(parsedDat)# myFPtree, myHeaderTab = createTree(initSet, 100000)# myFreList = []# mineTree(myFPtree, myHeaderTab, 100000, set([]), myFreList)# print myFreList

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/139381.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

无人机“长坡”上,谁是滚出“厚雪球”的长期主义者?

“股神”巴菲特&#xff0c;曾提出过“长坡厚雪”的理论&#xff1a; 人生就像滚雪球&#xff0c;重要的是发现很湿的雪和很长的坡。 运用到企业经营上&#xff0c;“长坡”指的是企业所布局的领域发展潜力足、空间大&#xff1b;而“湿雪”&#xff0c;指的是企业竞争力强、…

大模型应用发展的方向|代理 Agent 的兴起及其未来(下)

“ 借助LLM作为代理大脑的优势&#xff0c;探讨了单一代理、多代理系统和人机协作等应用场景&#xff0c;探讨了代理的社会行为、心理活动以及在模拟社会环境中观察新兴社会现象和人类洞见的可能性。” 01 — 造福人类&#xff1a;代理实践 LLM型智能代理是一种新兴的方向&…

Linux——文件系统

✅<1>主页&#xff1a;&#xff1a;我的代码爱吃辣 &#x1f4c3;<2>知识讲解&#xff1a;Linux——文件系统 ☂️<3>开发环境&#xff1a;Centos7 &#x1f4ac;<4>前言&#xff1a;上期我们了解了文件在内存中得组织方式&#xff0c;那么文件在磁盘中…

【C语言】错题本(4)

一. 题目及选项: 答案解析: 知识点: 字符型在内存中的数据存储 char类型数据在内存中的图示: unsigned char类型数据在内存中的图示: 二. 题目及选项: 答案解析: A: B: C: D: 三. 题目及选项: 答案解析: 数据在计算机中是先转换成补码,再进行运算的!

C语言自定义类型详解(1)结构体知识汇总

本篇概要 本篇主要讲述C语言结构体的相关知识&#xff0c;包括结构体的基本声明&#xff0c;结构体的匿名结构&#xff0c;结构体的自引用&#xff0c;结构体变量的定义和初始化以及结构体的内存对齐等相关知识。 文章目录 本篇概要1.结构体1.1结构体的基本声明1.2结构体的特殊…

精华回顾:Web3 前沿创新者在 DESTINATION MOON 共话未来

9 月 17 日&#xff0c;由 TinTinLand 主办的「DESTINATION MOON: Web3 Dev Summit Shanghai 2023」线下活动在上海黄浦如约而至。 本次 DESTINATION MOON 活动作为 2023 上海区块链国际周的 Side Event&#xff0c;设立了 4 场主题演讲与 3 个圆桌讨论&#xff0c;聚集了诸多…

高压放大器电源有什么作用和用途

高压放大器是一种专门用于放大高压信号的电子设备。它可以将低幅度的输入信号放大成高幅度的输出信号&#xff0c;用于驱动高压负载或处理高压信号。然而&#xff0c;高压放大器需要特定的电能来运行&#xff0c;而这就是电源的作用。 高压放大器电源的主要作用是为高压放大器提…

Linux常用命令—find命令大全

文章目录 一、find命令常用功能1、find命令的基本信息如下。2、按照文件名搜索3、按照文件大小搜索4、按照修改时间搜索5、按照权限搜索举例&#xff1a;6、按照所有者和所属组搜索7、按照文件类型搜索8、逻辑运算符 一、find命令常用功能 1、find命令的基本信息如下。 命令名…

伪原创文章生成器软件的崛起-哪个伪原创文章生成器软件好?

在当今数字化的时代&#xff0c;内容创作已经成为了无处不在的需求。不论您是个人博主、企业家还是网站管理员&#xff0c;都会面临一个共同的挑战&#xff1a;如何在互联网上脱颖而出&#xff0c;吸引更多的读者和访客。 gpt批量图文改写润色软件-147SEO gpt批量图文改写润色…

栈的简单应用(利用Stack进行四则混合运算)(JAVA)

目录 中缀表达式转后缀表达式 图解 代码实现过程&#xff1a; 完整代码&#xff1a; 利用后缀表达式求值&#xff1a; 完整代码&#xff1a; 首先我们得先了解逆波兰表达式。 中缀表达式转后缀表达式 所谓的中缀表达式其实就是我们平时写的例如&#xff1a;&#xff1…

ReadPaper论文阅读工具

之前看文献一直用的EndNote嘛&#xff0c;但是突然发现了它的一个弊端&#xff0c;就是说每次没看完退出去之后&#xff0c;下次再接着看的时候它不能保留我上一次的位置信息&#xff0c;又要重头开始翻阅&#xff0c;这让我感到很烦躁哈哈。&#xff08;当然也不知道是不是我哪…

自研多模态追踪算法 PICO 为「手柄小型化」找到新思路

作者&#xff1a;张韬、林泽一 、闻超 、赵洋 研发背景 作为头戴的追踪配件&#xff0c;VR手柄可以通过HMD&#xff08;头戴显示设备&#xff09;的inside-out光学追踪定位原理&#xff0c;计算出手柄的空间运动轨迹&#xff0c;同时结合6轴传感器实现6DoF空间定位。与此同时&a…

基于若依ruoyi-nbcio增加flowable流程待办消息的提醒,并提供右上角的红字数字提醒(六)

更多ruoyi-nbcio功能请看演示系统 gitee源代码地址 前后端代码&#xff1a; https://gitee.com/nbacheng/ruoyi-nbcio 演示地址&#xff1a;RuoYi-Nbcio后台管理系统 这个部分主要是前端方面的。 1、在Navbar.vue显示右上角的图标栏里增加一项显示消息提醒的组件 <el-…

【PyTorch攻略(2/7)】 加载数据集

一、说明 PyTorch提供了两个数据原语&#xff1a;torch.utils.data.DataLoader和torch.utils.data.Dataset&#xff0c;允许您使用预加载的数据集以及您自己的数据。数据集存储样本及其相应的标签&#xff0c;DataLoader 围绕数据集包装一个可迭代对象&#xff0c;以便轻松访问…

c++STL案列一评委打分

案例描述 有5名选手:选手ABCDE&#xff0c;10个评委分别对每一名选手打分&#xff0c;去除最高分&#xff0c;去除评委中最低分&#xff0c;取平均分 实现步骤 1.创建五名选手&#xff0c;放到vector中 2.遍历vector容器&#xff0c;取出来每一个选手&#xff0c;执行for循环…

autosar 诊断入门

AUTOSAR (汽车开放系统架构) 是一个国际汽车行业的开放和标准化的软件架构。它的主要目标是为了创建一种独立于硬件的软件架构&#xff0c;以提高汽车电子系统的模块化和可重用性。 AUTOSAR架构主要分为两个部分&#xff1a;AUTOSAR Runtime Environment (RTE) 和 AUTOSAR Soft…

第一百五十一回 自定义组件综合实例:游戏摇杆二

文章目录 内容回顾实现方法位置细节示例代码我们在上一章回中介绍了如何实现 游戏摇杆相关的内容,本章回中将继续介绍这方面的知识.闲话休提,让我们一起Talk Flutter吧。 内容回顾 我们在上一章回中介绍了游戏摇杆的概念以及实现方法,并且通过示例代码演示了实现游戏摇杆的…

用flask框架flask-sock和websocket创建一个自己的聊天界面

WebSocket 协议在10年前就已经标准化了(在2011年&#xff0c;你能相信吗?)所以我相信你不需要介绍。但是如果你不熟悉它&#xff0c;WebSocket 是 HTTP 协议的一个扩展&#xff0c;它在客户端和服务器之间提供了一个永久的、双向的通信通道&#xff0c;在这里双方可以实时地发…

Spring Security :一【权限管理概述、Spring Security 认证与授权】

文章目录 Spring Security一、权限管理概述1.1.什么是认证1.2 什么是授权1.3 授权的数据模型RBAC1.3.1 基于角色的访问控制1.3.2 基于资源的访问控制 1.4 权限管理框架1.4.1 Apache Shiro1.4.2 Spring Security1.4.3 Shiro 和 Spring Security 比较 二、Spring Security 认证与…

Redis学习 - 了解Redis(三)

1. 什么是缓存击穿、缓存穿透、缓存雪崩&#xff1f; 1.1 缓存穿透问题 先来看一个常见的缓存使用方式&#xff1a;读请求来了&#xff0c;先查下缓存&#xff0c;缓存有值命中&#xff0c;就直接返回&#xff1b;缓存没命中&#xff0c;就去查数据库&#xff0c;然后把数据库…