聚类算法之BIRCH(Java实现)转载
http://www.cnblogs.com/zhangchaoyang/articles/2200800.html
http://blog.csdn.net/qll125596718/article/details/6895291
BIRCH(Balanced Iterative Reducing and Clustering using Hierarchies)天生就是为处理超大规模(至少要让你的内存容不下)的数据集而设计的,它可以在任何给定的内存下运行。关于BIRCH的更多特点先不介绍,我先讲一下算法的完整实现细节,对算法的实现过程搞清楚后再去看别人对该算法的评价才会感受深刻。
你不需要具备B树的相关知识,我接下来会讲得很清楚。
BIRCH算法的过程就是要把待分类的数据插入一棵树中,并且原始数据都在叶子节点上。这棵树看起来是这个样子:
在这棵树中有3种类型的节点:Nonleaf、Leaf、MinCluster,Root可能是一种Nonleaf,也可能是一种Leaf。所有的Leaf放入一个双向链表中。每一个节点都包含一个CF值,CF是一个三元组,其中data point instance的个数,
和
是与数据点同维度的向量,
是线性和,
是平方和。比如有一个MinCluster里包含3个数据点(1,2,3),(4,5,6),(7,8,9),则
N=3,
=(1+4+7,2+5+8,3+6+9)=(12,15,18),
=(1+16+49,4+25+64,9+36+81)。
就拿这个MinCluster为例,我们可以计算它的
簇中心
簇半径
簇直径
我们还可以计算两个簇之间的距离,当然你也可以使用D0,D1,D3等等,不过在这里我们使用D2。
有意思的是簇中心、簇半径、簇直径以及两簇之间的距离D0到D3都可以由CF来计算,比如
簇直径
簇间距离,这里的N,LS和SS是指两簇合并后大簇的N,LS和SS。所谓两簇合并只需要两个对应的CF相加那可
CF1 + CF2 = (N1 + N2 , LS1 + LS2, SS1 + SS2)
每个节点的CF值就是其所有孩子节点CF值之和,以每个节点为根节点的子树都可以看成 是一个簇。
Nonleaf、Leaf、MinCluster都是有大小限制的,Nonleaf的孩子节点不能超过B个,Leaf最多只能有L个MinCluster,而一个MinCluster的直径不能超过T。
算法起初,我们扫描数据库,拿到第一个data point instance--(1,2,3),我们创建一个空的Leaf和MinCluster,把点(1,2,3)的id值放入Mincluster,更新MinCluster的CF值为(1,(1,2,3),(1,4,9)),把MinCluster作为Leaf的一个孩子,更新Leaf的CF值为(1,(1,2,3),(1,4,9))。实际上只要往树中放入一个CF(这里我们用CF作为Nonleaf、Leaf、MinCluster的统称),就要更新从Root到该叶子节点的路径上所有节点的CF值。
当又有一个数据点要插入树中时,把这个点封装为一个MinCluster(这样它就有了一个CF值),把新到的数据点记为CF_new,我们拿到树的根节点的各个孩子节点的CF值,根据D2来找到CF_new与哪个节点最近,就把CF_new加入那个子树上面去。这是一个递归的过程。递归的终止点是要把CF_new加入到一个MinCluster中,如果加入之后MinCluster的直径没有超过T,则直接加入,否则譔CF_new要单独作为一个簇,成为MinCluster的兄弟结点。插入之后注意更新该节点及其所有祖先节点的CF值。
插入新节点后,可能有些节点的孩子数大于了B(或L),此时该节点要分裂。对于Leaf,它现在有L+1个MinCluster,我们要新创建一个Leaf,使它作为原Leaf的兄弟结点,同时注意每新创建一个Leaf都要把它插入到双向链表中。L+1个MinCluster要分到这两个Leaf中,怎么分呢?找出这L+1个MinCluster中距离最远的两个Cluster(根据D2),剩下的Cluster看离哪个近就跟谁站在一起。分好后更新两个Leaf的CF值,其祖先节点的CF值没有变化,不需要更新。这可能导致祖先节点的递归分裂,因为Leaf分裂后恰好其父节点的孩子数超过了B。Nonleaf的分裂方法与Leaf的相似,只不过产生新的Nonleaf后不需要把它放入一个双向链表中。如果是树的根节点要分裂,则树的高度加1。
CF.java
package birch; public class CF { private int N; private double [] LS; private double [] SS; public CF() { LS= new double [BIRCH.dimen]; SS= new double [BIRCH.dimen]; } // 根据一个data point instance创建一个Clustering Feature public CF( double [] data) { int len = data.length; this .N = 1 ; this .LS = data; this .SS= new double [len]; for ( int i = 0 ; i < len; i++) this .SS[i] = Math.pow(data[i], 2 ); } //复制构造函数(深复制) public CF(CF cf){ this .N=cf.getN(); int len=cf.getLS().length; this .LS= new double [len]; this .SS= new double [len]; for ( int i= 0 ;i<len;i++){ this .LS[i]=cf.getLS()[i]; this .SS[i]=cf.getSS()[i]; } } // 采用D2计算两个CF Entry之间的距离 public double getDistanceTo(CF entry) { double dis = 0.0 ; int len = this .LS.length; // 采用D2 for ( int i = 0 ; i < len; i++) { dis += this .SS[i] / this .N + entry.getSS()[i] / entry.getN() - 2 * this .LS[i] * entry.getLS()[i] / ( this .N * entry.getN()); } return Math.sqrt(dis); } //采用D0计算两个簇心之间的欧氏距离 // public double getDistanceTo(CF entry) { // int len=entry.getLS().length; // double[] a=new double[len]; // double[] b=new double[len]; // for(int i=0;i<len;i++){ // a[i]=this.getLS()[i]/this.N; // b[i]=this.getSS()[i]/this.N; // } // return calEuraDist(a,b,len); // } // 加上或减去一个CF的值 public void addCF(CF entry, boolean add) { int opt = 1 ; // 默认为相加 if (!add) // 如果add为false则为相减 opt = - 1 ; this .N = this .N + entry.getN() * opt; int len = this .LS.length; for ( int i = 0 ; i < len; i++) { this .LS[i] = this .LS[i] + entry.getLS()[i] * opt; this .SS[i] = this .SS[i] + entry.getSS()[i] * opt; } } //计算两个向量的欧氏距离 public static double calEuraDist( double [] arr1, double [] arr2, int len){ double result= 0.0 ; for ( int i= 0 ;i<len;i++){ result+=Math.pow(arr1[i]-arr2[i], 2.0 ); } return Math.sqrt(result); } public int getN() { return N; } public void setN( int n) { N = n; } public double [] getLS() { return LS; } public void setLS( double [] lS) { LS = lS; } public double [] getSS() { return SS; } public void setSS( double [] sS) { SS = sS; } } |
MinCluster.java
package birch; import java.util.ArrayList; //最小簇 public class MinCluster { private CF cf; private ArrayList<String> inst_marks; public MinCluster(){ cf= new CF(); inst_marks= new ArrayList<String>(); } public CF getCf() { return cf; } public void setCf(CF cf) { this .cf = cf; } public ArrayList<String> getInst_marks() { return inst_marks; } public void setInst_marks(ArrayList<String> inst_marks) { this .inst_marks = inst_marks; } //计算簇的直径 public static double getDiameter(CF cf){ double diameter= 0.0 ; int n=cf.getN(); for ( int i= 0 ;i<cf.getLS().length;i++){ double ls=cf.getLS()[i]; double ss=cf.getSS()[i]; diameter=diameter+( 2 *n*ss- 2 *ls*ls); } diameter=diameter/(n*n-n); return Math.sqrt(diameter); } //计算和另外一个簇合并后的直径 public static double getDiameter(MinCluster cluster1,MinCluster cluster2){ CF cf= new CF(cluster1.getCf()); cf.addCF(cluster2.getCf(), true ); return getDiameter(cf); } public void mergeCluster(MinCluster cluster){ this .getCf().addCF(cluster.getCf(), true ); for ( int i= 0 ;i<cluster.getInst_marks().size();i++){ this .getInst_marks().add(cluster.getInst_marks().get(i)); } } } |
TreeNode.java
package birch; public abstract class TreeNode extends CF { private TreeNode parent; public TreeNode() { } public TreeNode( double [] data) { super (data); } public TreeNode getParent() { return parent; } public void setParent(TreeNode parent) { this .parent = parent; } public void addCFUpToRoot(CF cf){ TreeNode node= this ; while (node!= null ){ node.addCF(cf, true ); node=node.getParent(); } } abstract void split(); abstract void absorbSubCluster(MinCluster cluster); } |
NonleafNode.java
package birch; import java.util.ArrayList; public class NonleafNode extends TreeNode { private int B= 5 ; private ArrayList<TreeNode> children; public NonleafNode() { children= new ArrayList<TreeNode>(); } public NonleafNode( double [] data) { super (data); } // 节点分裂 public void split() { // 找到距离最远的两个孩子节点 int c1 = 0 ; int c2 = 0 ; double maxDist = 0 ; int len = this .getChildren().size(); for ( int i = 0 ; i < len - 1 ; i++) { for ( int j = i + 1 ; j < len; j++) { double dist = this .getChildren().get(i) .getDistanceTo( this .getChildren().get(j)); if (dist > maxDist) { maxDist = dist; c1 = i; c2 = j; } } } // 以距离最远的孩子节点为中心,把B+1个孩子分为两个大簇。其中一个簇仍留作本节点的孩子,另外一簇需要新创建一个节点来领养它们 NonleafNode newNode = new NonleafNode(); newNode.addChild( this .getChildren().get(c2)); //如果本节点已经是Root节点,则需要创建一个新的Root节点 if ( this .getParent()== null ){ NonleafNode root= new NonleafNode(); root.setN( this .getN()); root.setLS( this .getLS()); root.setSS( this .getSS()); root.addChild( this ); this .setParent(root); } newNode.setParent( this .getParent()); ((NonleafNode) this .getParent()).addChild(newNode); for ( int i = 0 ; i < len; i++) { if (i != c1 && i != c2) { if ( this .getChildren().get(i) .getDistanceTo( this .getChildren().get(c2)) < this .getChildren().get(i) .getDistanceTo( this .getChildren().get(c1))) { newNode.addChild( this .getChildren().get(i)); } } } for (TreeNode entry : newNode.getChildren()) { newNode.addCF(entry, true ); this .deleteChild(entry); this .addCF(entry, false ); } //如果本节点分裂导致父节点的孩子数超过了分枝因子,引发父节点分裂 NonleafNode pn=(NonleafNode) this .getParent(); if (pn.getChildren().size()>B){ this .getParent().split(); } } public void absorbSubCluster(MinCluster cluster){ //从本节点的孩子中寻找与cluster最近的子节点 CF cf=cluster.getCf(); int nearIndex= 0 ; double minDist=Double.MAX_VALUE; for ( int i= 0 ;i< this .getChildren().size();i++){ double dist=cf.getDistanceTo( this .getChildren().get(i)); if (dist<minDist){ nearIndex=i; } } //让那个最近的子节点absorb掉这个新到的cluster this .getChildren().get(nearIndex).absorbSubCluster(cluster); } public ArrayList<TreeNode> getChildren() { return children; } public void setChildren(ArrayList<TreeNode> children) { this .children = children; } public void addChild(TreeNode child) { this .children.add(child); } public void deleteChild(TreeNode child) { this .children.remove(children.indexOf(child)); } public int getB() { return B; } public void setB( int b) { B = b; } } |
LeafNode.java
package birch; import java.util.ArrayList; public class LeafNode extends TreeNode { private int L= 10 ; private double T= 2.8 ; private ArrayList<MinCluster> children; private LeafNode pre; private LeafNode next; public LeafNode() { children= new ArrayList<MinCluster>(); } public LeafNode( double [] data) { super (data); } // 节点分裂 public void split() { // 找到距离最远的两个孩子节点 int c1 = 0 ; int c2 = 0 ; double maxDist = 0 ; int len = this .getChildren().size(); for ( int i = 0 ; i < len - 1 ; i++) { for ( int j = i + 1 ; j < len; j++) { double dist = this .getChildren().get(i).getCf() .getDistanceTo( this .getChildren().get(j).getCf()); if (dist > maxDist) { maxDist = dist; c1 = i; c2 = j; } } } // 以距离最远的孩子节点为中心,把B+1个孩子分为两个大簇。其中一个簇仍留作本节点的孩子,另外一簇需要新创建一个节点来领养它们 LeafNode newNode = new LeafNode(); newNode.addChild( this .getChildren().get(c2)); // 如果本节点已经是Root节点,则需要创建一个新的Root节点 if ( this .getParent() == null ) { NonleafNode root = new NonleafNode(); root.setN( this .getN()); root.setLS( this .getLS()); root.setSS( this .getSS()); this .setParent(root); root.addChild( this ); } //建立新节点和本节点的父节点的父子关系 newNode.setParent( this .getParent()); ((NonleafNode) this .getParent()).addChild(newNode); //把离newNode近的孩子节点归到newNode这个簇里面 for ( int i = 0 ; i < len; i++) { if (i != c1 && i != c2) { if ( this .getChildren().get(i).getCf() .getDistanceTo( this .getChildren().get(c2).getCf()) < this .getChildren().get(i).getCf() .getDistanceTo( this .getChildren().get(c1).getCf())) { newNode.addChild( this .getChildren().get(i)); } } } //把离newNode近的孩子节点从本节点中删除 for (MinCluster cluster : newNode.getChildren()) { newNode.addCF(cluster.getCf(), true ); this .deleteChild(cluster); this .addCF(cluster.getCf(), false ); } // 把新增加的LeafNode添加到LeafNode双向链表中 if ( this .getNext() != null ) { newNode.setNext( this .getNext()); this .getNext().setPre(newNode); } this .setNext(newNode); newNode.setPre( this ); // 如果本节点分裂导致父节点的孩子数超过了分枝因子,引发父节点分裂 NonleafNode pn = (NonleafNode) this .getParent(); if (pn.getChildren().size() > pn.getB()) { this .getParent().split(); } } @Override public void absorbSubCluster(MinCluster cluster) { // 先试图找到叶子节点的孩子(一些subcluster)中与cluster最近的簇 CF cf = cluster.getCf(); int nearIndex = 0 ; double minDist = Double.MAX_VALUE; int len = this .getChildren().size(); if (len > 0 ) { for ( int i = 0 ; i < len; i++) { double dist = cf.getDistanceTo( this .getChildren().get(i) .getCf()); if (dist < minDist) { nearIndex = i; } } // 计算两个簇合并后的直径 double mergeDiameter = MinCluster.getDiameter(cluster, this .getChildren().get(nearIndex)); // 如果合并后发现簇的直径超过了阈值,则把cluster作为一个单独的孩子插入本叶子节点下 if (mergeDiameter > T) { this .addChild(cluster); if ( this .getChildren().size() > L) { this .split(); } } // 如果不超过阈值,则直接合并两个簇 else { this .getChildren().get(nearIndex).mergeCluster(cluster); } } // 创建B树之初,叶子节点还没有children else { this .addChild(cluster); } this .addCFUpToRoot(cluster.getCf()); } public ArrayList<MinCluster> getChildren() { return children; } public void setChildren(ArrayList<MinCluster> children) { this .children = children; } public void addChild(MinCluster child) { this .children.add(child); } public void deleteChild(MinCluster child) { this .children.remove(children.indexOf(child)); } public LeafNode getPre() { return pre; } public void setPre(LeafNode pre) { this .pre = pre; } public LeafNode getNext() { return next; } public void setNext(LeafNode next) { this .next = next; } public int getL() { return L; } public void setL( int l) { L = l; } public double getT() { return T; } public void setT( double t) { T = t; } } |
BIRCH.java
package birch; import java.io.BufferedReader; import java.io.File; import java.io.FileReader; import java.io.IOException; public class BIRCH { public static final int dimen= 4 ; LeafNode leafNodeHead= new LeafNode(); int point_num= 0 ; //point instance计数 //逐条扫描数据库,建立B-树 public TreeNode buildBTree(String filename){ //先建立一个叶子节点 LeafNode leaf= new LeafNode(); TreeNode root=leaf; //把叶子节点加入存储叶子节点的双向链表 leafNodeHead.setNext(leaf); leaf.setPre(leafNodeHead); //打开文件,从文件中读取原始数据 File file = new File(filename); if (!file.exists()){ System.out.println( "Data File Not Exists." ); System.exit( 2 ); } try { FileReader fr = new FileReader(file); BufferedReader br= new BufferedReader(fr); String line= null ; while ((line=br.readLine())!= null && line.trim()!= "" ){ point_num++; String[] cont=line.split( "[,|\\s+]" ); //读入point instance double [] data= new double [dimen]; for ( int i= 0 ;i<data.length;i++){ data[i]=Double.parseDouble(cont[i]); } String mark=String.valueOf(point_num)+cont[data.length]; //根据一个point instance创建一个MinCluster CF cf= new CF(data); MinCluster subCluster= new MinCluster(); subCluster.setCf(cf); subCluster.getInst_marks().add(mark); //把新到的point instance插入树中 root.absorbSubCluster(subCluster); //要始终保证root是树的根节点 while (root.getParent()!= null ){ root=root.getParent(); } } br.close(); } catch (IOException e) { e.printStackTrace(); } return root; } //打印B-树的所有叶子节点 public void printLeaf(LeafNode header){ //point_num清0 point_num= 0 ; while (header.getNext()!= null ){ System.out.println( "\n一个叶子节点:" ); header=header.getNext(); for (MinCluster cluster:header.getChildren()){ System.out.println( "\n一个最小簇:" ); for (String mark:cluster.getInst_marks()){ point_num++; System.out.print(mark+ "\t" ); } } } } //打印指定根节点的子树 public void printTree(TreeNode root){ if (!root.getClass().getName().equals( "birch.LeafNode" )){ NonleafNode nonleaf=(NonleafNode)root; for (TreeNode child:nonleaf.getChildren()){ printTree(child); } } else { System.out.println( "\n一个叶子节点:" ); LeafNode leaf=(LeafNode)root; for (MinCluster cluster:leaf.getChildren()){ System.out.println( "\n一个最小簇:" ); for (String mark:cluster.getInst_marks()){ System.out.print(mark+ "\t" ); point_num++; } } } } public static void main(String[] args) { BIRCH birch= new BIRCH(); TreeNode root=birch.buildBTree( "/home/orisun/test/iris.shuffled" ); birch.point_num= 0 ; birch.printTree(root); System.out.println(); //birch.printLeaf(birch.leafNodeHead); //确认被分类的point instance和扫描数据库时录入的point instance的个数是一致的 System.out.println(birch.point_num); } } |
最后我们来总结一BIRCH的优势和劣势。
优点:
- 节省内在。叶子节点放在磁盘分区上,非叶子节点仅仅是存储了一个CF值,外加指向父节点和孩子节点的指针。
- 快。合并两个两簇只需要两个CF算术相加即可;计算两个簇的距离只需要用到(N,LS,SS)这三个值足矣。
- 一遍扫描数据库即可建立B树。
- 可识别噪声点。建立好B树后把那些包含数据点少的MinCluster当作outlier。
- 由于B树是高度平衡的,所以在树上进行插入或查找操作很快。
缺点:
- 结果依赖于数据点的插入顺序。本属于同一个簇的点可能由于插入顺序相差很远而分到不同的簇中,即使同一个点在不同的时刻被插入,也会被分到不同的簇中。
- 对非球状的簇聚类效果不好。这取决于簇直径和簇间距离的计算方法。
- 对高维数据聚类效果不好。
- 由于每个节点只能包含一定数目的子节点,最后得出来的簇可能和自然簇相差很大。
- BIRCH适合于处理需要数十上百小时聚类的数据,但在整个过程中算法一旦中断,一切必须从头再来。
- 局部性也导致了BIRCH的聚类效果欠佳。当一个新点要插入B树时,它只跟很少一部分簇进行了相似性(通过计算簇间距离)比较,高的efficient导致低的effective。
1.BIRCH算法概念
BIRCH(Balanced Iterative Reducing and Clustering Using Hierarchies)全称是:利用层次方法的平衡迭代规约和聚类。BIRCH算法是1996年由Tian Zhang提出来的,参考文献1。首先,BIRCH是一种聚类算法,它最大的特点是能利用有限的内存资源完成对大数据集的高质量的聚类,同时通过单遍扫描数据集能最小化I/O代价。
首先解释一下什么是聚类,从统计学的观点来看,聚类就是给定一个包含N个数据点的数据集和一个距离度量函数F(例如计算簇内每两个数据点之间的平均距离的函数),要求将这个数据集划分为K个簇(或者不给出数量K,由算法自动发现最佳的簇数量),最后的结果是找到一种对于数据集的最佳划分,使得距离度量函数F的值最小。从机器学习的角度来看,聚类是一种非监督的学习算法,通过将数据集聚成n个簇,使得簇内点之间距离最小化,簇之间的距离最大化。
BIRCH算法特点:
(1)BIRCH试图利用可用的资源来生成最好的聚类结果,给定有限的主存,一个重要的考虑是最小化I/O时间。
(2)BIRCH采用了一种多阶段聚类技术:数据集的单边扫描产生了一个基本的聚类,一或多遍的额外扫描可以进一步改进聚类质量。
(3)BIRCH是一种增量的聚类方法,因为它对每一个数据点的聚类的决策都是基于当前已经处理过的数据点,而不是基于全局的数据点。
(4)如果簇不是球形的,BIRCH不能很好的工作,因为它用了半径或直径的概念来控制聚类的边界。
BIRCH算法中引入了两个概念:聚类特征和聚类特征树,以下分别介绍。
1.1 聚类特征(CF)
CF是BIRCH增量聚类算法的核心,CF树中得节点都是由CF组成,一个CF是一个三元组,这个三元组就代表了簇的所有信息。给定N个d维的数据点{x1,x2,....,xn},CF定义如下:
CF=(N,LS,SS)
其中,N是子类中节点的数目,LS是N个节点的线性和,SS是N个节点的平方和。
CF有个特性,即可以求和,具体说明如下:CF1=(n1,LS1,SS1),CF2=(n2,LS2,SS2),则CF1+CF2=(n1+n2, LS1+LS2, SS1+SS2)。
例如:
假设簇C1中有三个数据点:(2,3),(4,5),(5,6),则CF1={3,(2+4+5,3+5+6),(2^2+4^2+5^2,3^2+5^2+6^2)}={3,(11,14),(45,70)},同样的,簇C2的CF2={4,(40,42),(100,101)},那么,由簇C1和簇C2合并而来的簇C3的聚类特征CF3计算如下:
CF3={3+4,(11+40,14+42),(45+100,70+101)}={7,(51,56),(145,171)}
另外在介绍两个概念:簇的质心和簇的半径。假如一个簇中包含n个数据点:{Xi},i=1,2,3...n.,则质心C和半径R计算公式如下:
C=(X1+X2+...+Xn)/n,(这里X1+X2+...+Xn是向量加)
R=(|X1-C|^2+|X2-C|^2+...+|Xn-C|^2)/n
其中,簇半径表示簇中所有点到簇质心的平均距离。CF中存储的是簇中所有数据点的特性的统计和,所以当我们把一个数据点加入某个簇的时候,那么这个数据点的详细特征,例如属性值,就丢失了,由于这个特征,BIRCH聚类可以在很大程度上对数据集进行压缩。
1.2 聚类特征树(CF tree)
CF tree的结构类似于一棵B-树,它有两个参数:内部节点平衡因子B,叶节点平衡因子L,簇半径阈值T。树中每个节点最多包含B个孩子节点,记为(CFi,CHILDi),1<=i<=B,CFi是这个节点中的第i个聚类特征,CHILDi指向节点的第i个孩子节点,对应于这个节点的第i个聚类特征。例如,一棵高度为3,B为6,L为5的一棵CF树的例子如图所示:
一棵CF树是一个数据集的压缩表示,叶子节点的每一个输入都代表一个簇C,簇C中包含若干个数据点,并且原始数据集中越密集的区域,簇C中包含的数据点越多,越稀疏的区域,簇C中包含的数据点越少,簇C的半径小于等于T。随着数据点的加入,CF树被动态的构建,插入过程有点类似于B-树。加入算法表示如下:
- (1)从根节点开始,自上而下选择最近的孩子节点
- (2)到达叶子节点后,检查最近的元组CFi能否吸收此数据点
- 是,更新CF值
- 否,是否可以添加一个新的元组
- 是,添加一个新的元组
- 否则,分裂最远的一对元组,作为种子,按最近距离重新分配其它元组
- (3)更新每个非叶节点的CF信息,如果分裂节点,在父节点中插入新的元组,检查分裂,直到root
计算节点之间的距离函数有多种选择,常见的有欧几里得距离函数和曼哈顿距离函数,具体公式如下:
构建CF树的过程中,一个重要的参数是簇半径阈值T,因为它决定了CF tree的规模,从而让CF tree适应当前内存的大小。如果T太小,那么簇的数量将会非常的大,从而导致树节点数量也会增大,这样可能会导致所有数据点还没有扫描完之前内存就不够用了。
2.算法流程
BIRCH算法流程如下图所示:
整个算法的实现分为四个阶段:
(1)扫描所有数据,建立初始化的CF树,把稠密数据分成簇,稀疏数据作为孤立点对待
(2)这个阶段是可选的,阶段3的全局或半全局聚类算法有着输入范围的要求,以达到速度与质量的要求,所以此阶段在阶段1的基础上,建立一个更小的CF树
(3)补救由于输入顺序和页面大小带来的分裂,使用全局/半全局算法对全部叶节点进行聚类
(4)这个阶段也是可选的,把阶段3的中心点作为种子,将数据点重新分配到最近的种子上,保证重复数据分到同一个簇中,同时添加簇标签
详细流程请参考文献1。
3.算法实现
BIRCH算法的发明者于1996年完成了BIRCH算法的实现,是用c++语言实现的,已在solaris下编译通过。
另外算法的实现也可参考:http://blog.sina.com.cn/s/blog_6e85bf420100om1i.html
参考文献:
1.BIRCH:An Efficient Data Clustering Method for Very Large Databases
聚类算法之BIRCH(Java实现)转载的更多相关文章
- 关于k-means聚类算法的matlab实现
在数据挖掘中聚类和分类的原理被广泛的应用. 聚类即无监督的学习. 分类即有监督的学习. 通俗一点的讲就是:聚类之前是未知样本的分类.而是根据样本本身的相似性进行划分为相似的类簇.而分类 是已知样本分类 ...
- 挑子学习笔记:两步聚类算法(TwoStep Cluster Algorithm)——改进的BIRCH算法
转载请标明出处:http://www.cnblogs.com/tiaozistudy/p/twostep_cluster_algorithm.html 两步聚类算法是在SPSS Modeler中使用的 ...
- BIRCH聚类算法原理
在K-Means聚类算法原理中,我们讲到了K-Means和Mini Batch K-Means的聚类原理.这里我们再来看看另外一种常见的聚类算法BIRCH.BIRCH算法比较适合于数据量大,类别数K也 ...
- 转载: scikit-learn学习之K-means聚类算法与 Mini Batch K-Means算法
版权声明:<—— 本文为作者呕心沥血打造,若要转载,请注明出处@http://blog.csdn.net/gamer_gyt <—— 目录(?)[+] ================== ...
- 多种聚类算法概述(BIRCH, DBSCAN, K-means, MEAN-SHIFT)
BIRCH:是一种使用树分类的算法,适用的范围是样本数大,特征数小的算法,因为特征数大的话,那么树模型结构就会要复杂很多 DBSCAN:基于概率密度的聚类方法:速度相对较慢,不适用于大型的数据,输入参 ...
- Java实现聚类算法k-means
2016-07 java简单实现聚类算法 但是有一个小问题,,,,我其实每次迭代之后(就是达不到我的收敛标准之前,聚类中心的误差达不到指定小的时候),虽然重新算了聚类中心,但是其实我的那些点并没有变, ...
- 数据挖掘聚类算法(DBSCAN、Kmeans)Java实现
学习聚类算法时,参考算法说明随手写的java实现,代码很简单,不多做说明啦,有需要的童鞋可以看看,自己也做个备录. http://files.cnblogs.com/files/yuananyun/% ...
- 转载:K-means聚类算法
转载地址:http://www.cnblogs.com/jerrylead/archive/2011/04/06/2006910.html K-means也是聚类算法中最简单的一种了,但是里面包含的思 ...
- 转载 | Python AI 教学│k-means聚类算法及应用
关注我们的公众号哦!获取更多精彩哦! 1.问题导入 假如有这样一种情况,在一天你想去某个城市旅游,这个城市里你想去的有70个地方,现在你只有每一个地方的地址,这个地址列表很长,有70个位置.事先肯定要 ...
随机推荐
- Java_一致性哈希算法与Java实现
摘自:http://blog.csdn.net/wuhuan_wp/article/details/7010071 一致性哈希算法是分布式系统中常用的算法.比如,一个分布式的存储系统,要将数据存储到具 ...
- IntelliSense: namespace "osgDB" 没有成员 "BEGIN_BRACKET"
IntelliSense: namespace "osgDB" 没有成员 "BEGIN_BRACKET" 转自:http://bbs.osgchina.org/ ...
- C语言-结构体
#include<stdio.h> struct stu //定义结构体类型 { int num; char *name; char *sex; float score; } boy[]= ...
- ASP.NET生成静态方法
CMS系统如果新闻多了,全部生成静态的话.不现实,而且占用空间比较大.那么只生成网站首页是必须的了,下面列出JCMS首页生成静态的方法.换一种思路其实更简单. 当点击生成首页静态的时候.去获取动态首页 ...
- How parse REST service JSON response
1. get JSON responses and go to : http://json2csharp.com/ 2. write data contracts using C# All class ...
- [LintCode] Add Binary 二进制数相加
Given two binary strings, return their sum (also a binary string). Have you met this question in a r ...
- oracle中复制表和数据 && 多表插入语句
创建测试表和测试数据 create table test (id number,name varchar(10)); insert into test values(1,'liufang'); ...
- Caused by: java.io.NotSerializableException: com.omhy.common.model.entity.XXX解决方法
启动tomact时引起的Caused by: java.io.NotSerializableException异常 种种情况就是没有序列化.序列化可以将内存中的类写入文件或数据库中 Serializa ...
- 读《深入php面向对象、模式与实践》有感(二)
书中关于设计模式的介绍很详细. 一.单例模式 作者建了一个preferences类来解释什么是单例模式.这个preferences类并非我第一次见到,在android中也有这个类,不过以前都是只管用即 ...
- FK JavaScript之:ArcGIS JavaScript添加Graphic,地图界面却不显示
使用ArcGIS JavaScript,往地图中添加几个Graphic,基本是与官网示例代码一致.绘制的图形一闪而过之后,就没了 核心代码如下: iniToolBar: function () { t ...