`
ikeycn
  • 浏览: 146269 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
文章分类
社区版块
存档分类
最新评论

Apriori算法实现

阅读更多
Apriori算法的主题思想是:
1. 找出所有的频繁1项集
2. 递归地使用Apriori产生方法由频繁k-1项集生成k项集,直到产生的k项集为空
2.1 对每个k-1项集中的元素排序
2.2 找出k-1项集中每一对排序后的频繁集之间仅有最后一个位置不同的两个集合合并为k集合
                2.3 生成k集合的所有k-1项集,然后判断集合中每一个是否在频繁k-1项集中出现:如果未曾出现,则把当前生成的k集合剪掉;否则把当前的k集合加入到候选频繁k项集中
2.4 对候选频繁k项集中的每一个集合,遍历判断其在原始数据集中出现的频率是否满足给定的最小支持度,如果满足则保留,否则所该k项集删除

算法实现如下:
package com.ustc.apriori;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.TreeSet;

public class Apriori {

	private int minSup;
	private static List<String> data;
	private static List<Set<String>> dataSet;
	
	/**
	 * @param args
	 */
	public static void main(String[] args) {

		long startTime = System.currentTimeMillis();
		Apriori apriori = new Apriori();
		
		//使用书中的测试集
		/*apriori.setMinSup(2);
		data = apriori.buildData();*/
		
		//设置最小支持度
		apriori.setMinSup(1000);
		//构造数据集
		data = apriori.buildData("retail.dat");
		
		
		
		//构造频繁1项集
		List<Set<String>> f1Set = apriori.findF1Items(data);
		apriori.printSet(f1Set, 1);
		List<Set<String>> result = f1Set;
		
		int i = 2;
		do{
			
			result = apriori.arioriGen(result);
			apriori.printSet(result, i);
			i++;
		}while(result.size() != 0);
		long endTime = System.currentTimeMillis();
		System.out.println("共用时:" + (endTime - startTime) + "ms");
	}

	
	public void setMinSup(int minSup) {
		this.minSup = minSup;
	}


	/**
	 * 构造原始数据集,可以为之提供参数,也可以不提供
	 * 如果不提供参数,将按程序默认构造的数据集;
	 * 如果提供参数为文件名,则使用文件中的数据集
	 * 
	 * @return
	 */
	List<String> buildData(String...fileName) {
		List<String> data = new ArrayList<String>();
		if(fileName.length !=0){
			File file = new File(fileName[0]);
			try {
				BufferedReader reader = new BufferedReader(new FileReader(file));
				String line;
				while( (line = reader.readLine()) != null){
					data.add(line);
				}
				
			} catch (FileNotFoundException e) {
				
				e.printStackTrace();
			} catch (IOException e) {
				
				e.printStackTrace();
			}
		}else{
			
			data.add("I1 I2 I5");
			data.add("I2 I4");
			data.add("I2 I3");
			data.add("I1 I2 I4");
			data.add("I1 I3");
			data.add("I2 I3");
			data.add("I1 I3");
			data.add("I1 I2 I3 I5");
			data.add("I1 I2 I3");
		}
		
		dataSet = new ArrayList<Set<String>>();
		Set<String> dSet;
		for (String d : data) {
			dSet = new TreeSet<String>();
			String[] dArr = d.split(" ");
			for (String str : dArr) {
				dSet.add(str);
			}
			dataSet.add(dSet);
		}
		

		return data;
	}

	/**
	 * 找出候选1项集
	 * 
	 * @param data
	 * @return
	 */
	List<Set<String>> findF1Items(List<String> data) {

		List<Set<String>> result = new ArrayList<Set<String>>();
		Map<String, Integer> dc = new HashMap<String, Integer>();
		for (String d : data) {
			String[] items = d.split(" ");
			for (String item : items) {
				if (dc.containsKey(item)) {
					dc.put(item, dc.get(item) + 1);
				} else {
					dc.put(item, 1);
				}
			}
		}
		Set<String> itemKeys = dc.keySet();
		Set<String> tempKeys = new TreeSet<String>();
		for (String str : itemKeys) {
			tempKeys.add(str);
		}

		for (String item : tempKeys) {
			if (dc.get(item) >= minSup) {
				Set<String> f1Set = new TreeSet<String>();
				f1Set.add(item);
				result.add(f1Set);
			}
		}

		return result;
	}

	/**
	 * 利用arioriGen方法由k-1项集生成k项集
	 * 
	 * @param preSet
	 * @return
	 */
	List<Set<String>> arioriGen(List<Set<String>> preSet) {

		List<Set<String>> result = new ArrayList<Set<String>>();
		int preSetSize = preSet.size();
		for (int i = 0; i < preSetSize - 1; i++) {
			for (int j = i + 1; j < preSetSize; j++) {
				String[] strA1 = preSet.get(i).toArray(new String[0]);
				String[] strA2 = preSet.get(j).toArray(new String[0]);
				if (isCanLink(strA1, strA2)) { // 判断两个k-1项集是否符合连接成k项集的条件 
					Set<String> set = new TreeSet<String>();
					for (String str : strA1) {
						set.add(str);
					}
					set.add((String) strA2[strA2.length - 1]); // 连接成k项集
					// 判断k项集是否需要剪切掉,如果不需要被cut掉,则加入到k项集列表中
					if (!isNeedCut(preSet, set)) {

						result.add(set);
					}
				}

			}
		}
		return checkSupport(result);
	}

	/**
	 * 把set中的项集与数量集比较并进行计算,求出支持度大于要求的项集
	 * 
	 * @param set
	 * @return
	 */
	List<Set<String>> checkSupport(List<Set<String>> setList) {

		

		List<Set<String>> result = new ArrayList<Set<String>>();
		boolean flag = true;
		int[] counter = new int[setList.size()];
		for (int i = 0; i < setList.size(); i++) {
			for (Set<String> dSets : dataSet) {
				if (setList.get(i).size() > dSets.size()) {
					flag = true;

				} else {

					for (String str : setList.get(i)) {
						if (!dSets.contains(str)) {
							flag = false;
							break;
						}
					}
					if (flag) {
						counter[i] += 1;
					} else {
						flag = true;
					}
				}
			}
		}

		for (int i = 0; i < setList.size(); i++) {
			if (counter[i] >= minSup) {
				result.add(setList.get(i));
			}
		}

		return result;
	}

	/**
	 * 判断两个项集合能否执行连接操作
	 * 
	 * @param s1
	 * @param s2
	 * @return
	 */
	boolean isCanLink(String[] s1, String[] s2) {

		boolean flag = true;
		if (s1.length == s2.length) {
			for (int i = 0; i < s1.length - 1; i++) {
				if (!s1[i].equals(s2[i])) {
					flag = false;
					break;
				}
			}
			if (s1[s1.length - 1].equals(s2[s2.length - 1])) {
				flag = false;
			}
		} else {
			flag = false;
		}

		return flag;
	}

	/**
	 * 判断set是否需要被cut
	 * 
	 * @param setList
	 * @param set
	 * @return
	 */
	boolean isNeedCut(List<Set<String>> setList, Set<String> set) {
		boolean flag = false;
		List<Set<String>> subSets = getSubset(set); // 获得k项集的所有k-1项集
		for (Set<String> subSet : subSets) {
			// 判断当前的k-1项集set是否在频繁k-1项集中出现,如出现,则不需要cut
			// 若没有出现,则需要被cut
			if (!isContained(setList, subSet)) {
				flag = true;
				break;
			}
		}
		return flag;
	}

	/**
	 * 判断k项集的某k-1项集是否包含在频繁k-1项集列表中
	 * 
	 * @param setList
	 * @param set
	 * @return
	 */
	boolean isContained(List<Set<String>> setList, Set<String> set) {

		boolean flag = false;
		int position = 0;
		for (Set<String> s : setList) {

			String[] sArr = s.toArray(new String[0]);
			String[] setArr = set.toArray(new String[0]);
			for (int i = 0; i < sArr.length; i++) {
				if (sArr[i].equals(setArr[i])) { // 如果对应位置的元素相同,则position为当前位置的值
					position = i;
				} else {
					break;
				}
			}
			// 如果position等于了数组的长度,说明已找到某个setList中的集合与
			// set集合相同了,退出循环,返回包含
			// 否则,把position置为0进入下一个比较
			if (position == sArr.length - 1) {
				flag = true;
				break;
			} else {
				flag = false;
				position = 0;
			}

		}
		return flag;
	}

	/**
	 * 获得k项集的所有k-1项集
	 * 
	 * @param set
	 * @return
	 */
	List<Set<String>> getSubset(Set<String> set) {

		List<Set<String>> result = new ArrayList<Set<String>>();
		String[] setArr = set.toArray(new String[0]);
		for (int i = 0; i < setArr.length; i++) {
			Set<String> subSet = new TreeSet<String>();
			for (int j = 0; j < setArr.length; j++) {
				if (i != j) {
					subSet.add((String) setArr[j]);
				}
			}
			result.add(subSet);
		}
		return result;
	}

	void printSet(List<Set<String>> setList, int i) {
		System.out.print("频繁"+i+"项集: 共" +  setList.size()+"项:{");
		for (Set<String> set : setList) {
			System.out.print("[ ");
			for (String str : set) {
				System.out.print(str + " ");
			}
			System.out.print("], ");
		}
		System.out.println("}");
	}
}
分享到:
评论
3 楼 lvshuding 2013-01-12  
代码中每个方法的注释写的很详细,学习了。
另外,与1楼的同求。
2 楼 503718696 2011-11-28  
哥们,你那个数据retail.dat能打个包下载,用用吗?
1 楼 happystonehlj 2011-01-03  
能不能把这个算法后半部分,生成最大频繁项集以及挖掘的关联规则以及规则的支持度和置信度补充补充呢

相关推荐

    Apriori算法实现及实验报告

    - **算法实现**:详细解释如何使用Java实现Apriori,包括关键的数据结构(如项集、候选集的表示)、方法设计(如支持度计算、频繁项集生成)等。 - **实验设置**:最小支持度阈值的选择,以及其他可能的参数设定。 -...

    apriori算法实现

    在`www.pudn.com.txt`文件中,可能是提供了关于Apriori算法实现的更多细节或示例数据。而`Apriori源代码全部`很可能包含了一个完整的C++实现Apriori算法的代码库,包含了从读取数据到生成规则的完整流程。 总结,...

    关联规则挖掘算法apriori算法的实现

    在给定的文件名"apriori2"中,可能包含了Apriori算法的第二阶段实现,即候选集生成和剪枝过程的代码或者结果。这个阶段通常是算法中最耗时的部分,因为它涉及到大量的集合操作和计数。 总之,Apriori算法是关联规则...

    Apriori算法实现及改进

    ### Apriori算法实现及改进 #### 数据挖掘简介 数据挖掘(Data Mining)是指从大量数据中抽取有价值的、潜在有用的知识、模型或者规则的过程。它伴随着数据库技术和人工智能技术的发展而兴起,是计算机科学领域的...

    Apriori算法实现实验报告.docx

    总结来说,Apriori算法实现实验不仅验证了算法的正确性,还通过实验分析了算法在不同数据条件下的性能表现,为实际应用中的数据挖掘提供了有价值的信息。尽管Apriori算法在大数据集上可能效率较低,但在较小规模的...

    Apriori算法实现以及优化程序和报告

    在给定的标题“Apriori算法实现以及优化程序和报告”中,我们可以推断出该压缩包包含了关于Apriori算法的理论实现、程序代码优化以及实验结果的详细报告。 在描述中提到,“基于Apriori算法实现,并对该算法进行了...

    基于weka对银行客户信息的关联分析及apriori算法实现

    "基于weka对银行客户信息的关联分析及apriori算法实现"这一标题揭示了本文档的核心内容。首先,"weka"是新西兰怀卡托大学开发的一个开源数据挖掘工具,它提供了丰富的机器学习算法和数据预处理功能。在这里,weka被...

    基于.net、c#的Apriori算法实现

    **基于.NET、C#的Apriori算法实现** Apriori算法是一种经典的关联规则学习算法,主要用于挖掘数据集中的频繁项集和强关联规则。它最初由Rakesh Agrawal和Rameesh Srikant在1994年提出,广泛应用于市场篮子分析、...

    关联分析Apriori算法实现

    "Apriori"压缩包文件可能包含了实现Apriori算法的代码,可能包括数据加载模块、Apriori算法的核心逻辑、支持度和置信度计算函数,以及结果展示或存储的部分。学习和理解这些代码可以帮助我们更好地掌握Apriori算法的...

    python apriori算法实例

    **Python Apriori算法实现步骤:** 1. **生成初始候选集**:首先,从数据集中找出所有单个项目的集合,这些项目在数据集中至少出现一次,形成一个最小的支持度(支持度定义为项集在交易中出现的频率)。 2. **计算...

    apriori算法vc++实现

    《Apriori算法在VC++中的实现及其在数据挖掘中的应用》 Apriori算法是一种经典的数据挖掘算法,主要用于关联规则学习,它由R. Agrawal和R. Srikant于1994年提出。该算法的核心思想是通过频繁项集的生成和剪枝过程来...

    Apriori算法Python实现

    Apriori算法Python实现

    使用Apriori算法进行关联规则挖掘的实验报告与代码实现

    代码部分可能涉及Python编程语言,使用了像`pandas`库处理数据,`apriori`函数实现Apriori算法,以及`association_rules`函数生成和过滤关联规则。 实验报告可能会讨论实验过程中遇到的问题、优化策略以及对挖掘...

    数据挖掘Apriori算法实现.rar

    这个C语言实现的Apriori算法可以帮助学习者理解算法背后的逻辑,也可以作为基础进行进一步的优化或扩展,例如并行化处理、优化内存使用或调整为适应不同数据结构。对于想要深入理解数据挖掘和Apriori算法的人来说,...

    Apriori算法对购物篮进行关联分析-Apriori算法进行购物篮关联分析.rar

    Apriori算法对购物篮进行关联分析-Apriori算法进行购物篮关联分析.rar 大家好,出来乍到,看到好多高手分享自己的程序,我也想分享一下,做出自己的贡献。 虽然学MATLAB已经一年有余,但是一直忙着数学建模,对...

    Apriori.rar_Apriori_Apriori MATLAB_Apriori算法实现_关联_数据关联算法

    《Apriori算法在MATLAB中的实现及其应用》 Apriori算法是数据挖掘领域中一种经典的关联规则学习算法,由Raghu Ramakrishnan和Gehrke于1994年提出,主要用于发现数据库中项集之间的频繁模式。在商业智能、市场篮子...

Global site tag (gtag.js) - Google Analytics