행렬 곱셈을 사용하여 Jaccard 계수를 계산할 수있는 방법이 있는지 알고 싶습니다.
이 코드를 사용했습니다
jaccard_sim <- function(x) {
# initialize similarity matrix
m <- matrix(NA, nrow=ncol(x),ncol=ncol(x),dimnames=list(colnames(x),colnames(x)))
jaccard <- as.data.frame(m)
for(i in 1:ncol(x)) {
for(j in i:ncol(x)) {
jaccard[i,j]= length(which(x[,i] & x[,j])) / length(which(x[,i] | x[,j]))
jaccard[j,i]=jaccard[i,j]
}
}
이것은 R에서 구현하는 것이 좋습니다. 주사위 유사성 하나를 수행했지만 Tanimoto / Jaccard에 붙어 있습니다. 누구든지 도울 수 있습니까?
답변
Jaccard (2 진 데이터의 두 열 사이에서 계산 됨)
)는
로저스-타니 모토는
, 어디
- a-두 열이 모두 1 인 행 수
- b-다른 열이 아닌 1 인 행 수
- c-이 열이 아닌 다른 열이 1 인 행 수
- d-두 열이 모두 0 인 행 수
의 행 수
그럼 우리는 :
의 제곱 대칭 행렬입니다
모든 열 사이.
의 제곱 대칭 행렬입니다
모든 열 사이에서 ( “X가 아님”은 X에서 1-> 0 및 0-> 1로 변환 됨)
그래서,
모든 열 사이의 Jaccard의 사각형 대칭 행렬입니다.
모든 열 사이의 Rogers-Tanimoto의 사각형 대칭 행렬입니다.
이 공식이 올바른 결과를 제공하는지 수치로 확인했습니다. 그들이하다.
Upd. 행렬을 얻을 수도 있습니다
과
:
여기서 “[1]”은 다음과 같이 크기가 지정된 1의 행렬을 나타냅니다.
.
제곱 비대칭 행렬입니다
모든 열 사이; 요소 ij 는 행의 수입니다.
컬럼에서 0 I 열에서 1 J .
따라서,
.
매트릭스
물론 이런 식으로 계산할 수도 있습니다.
.
행렬 알기
, 이진 데이터에 대해 발명 된 쌍 유사 (비 유사) 계수의 행렬을 계산할 수 있습니다.
답변
X가 드문 경우 위의 해결책은 그리 좋지 않습니다. ! X를 사용하면 밀도가 높은 행렬이 만들어 지므로 엄청난 양의 메모리와 계산이 필요합니다.
더 좋은 해결책은 공식 Jaccard [i, j] = #common / (#i + #j-#common)을 사용하는 것 입니다. 희소 행렬을 사용하면 다음과 같이 수행 할 수 있습니다 (코드는 희소 행렬에도 적용됨).
library(Matrix)
jaccard <- function(m) {
## common values:
A = tcrossprod(m)
## indexes for non-zero common values
im = which(A > 0, arr.ind=TRUE)
## counts for each row
b = rowSums(m)
## only non-zero values of common
Aim = A[im]
## Jacard formula: #common / (#i + #j - #common)
J = sparseMatrix(
i = im[,1],
j = im[,2],
x = Aim / (b[im[,1]] + b[im[,2]] - Aim),
dims = dim(A)
)
return( J )
}
답변
이것은 귀하의 필요에 따라 귀하에게 유용 할 수도 있고 그렇지 않을 수도 있습니다. 클러스터링 할당 간의 유사성에 관심이 있다고 가정합니다.
Jaccard 유사성 계수 또는 Jaccard 색인 을 사용하여 두 군집 지정의 유사성을 계산할 수 있습니다.
주어진 labelings L1
하고 L2
, 벤허, Elisseeff 및 귀용 (2002) 인 Jaccard 인덱스 중간 행렬의 내적을 이용하여 계산 될 수 있음을 보여 주었다. 아래 코드는이를 활용 하여 중간 행렬을 메모리에 저장하지 않고도 Jaccard Index 를 빠르게 계산합니다.
코드는 C ++로 작성되었지만 sourceCpp
명령을 사용하여 R에로드 할 수 있습니다 .
/**
* The Jaccard Similarity Coefficient or Jaccard Index is used to compare the
* similarity/diversity of sample sets. It is defined as the size of the
* intersection of the sets divided by the size of the union of the sets. Here,
* it is used to determine how similar to clustering assignments are.
*
* INPUTS:
* L1: A list. Each element of the list is a number indicating the cluster
* assignment of that number.
* L2: The same as L1. Must be the same length as L1.
*
* RETURNS:
* The Jaccard Similarity Index
*
* SIDE-EFFECTS:
* None
*
* COMPLEXITY:
* Time: O(K^2+n), where K = number of clusters
* Space: O(K^2)
*
* SOURCES:
* Asa Ben-Hur, Andre Elisseeff, and Isabelle Guyon (2001) A stability based
* method for discovering structure in clustered data. Biocomputing 2002: pp.
* 6-17.
*/
// [[Rcpp::export]]
NumericVector JaccardIndex(const NumericVector L1, const NumericVector L2){
int n = L1.size();
int K = max(L1);
int overlaps[K][K];
int cluster_sizes1[K], cluster_sizes2[K];
for(int i = 0; i < K; i++){ // We can use NumericMatrix (default 0)
cluster_sizes1[i] = 0;
cluster_sizes2[i] = 0;
for(int j = 0; j < K; j++)
overlaps[i][j] = 0;
}
//O(n) time. O(K^2) space. Determine the size of each cluster as well as the
//size of the overlaps between the clusters.
for(int i = 0; i < n; i++){
cluster_sizes1[(int)L1[i] - 1]++; // -1's account for zero-based indexing
cluster_sizes2[(int)L2[i] - 1]++;
overlaps[(int)L1[i] - 1][(int)L2[i] - 1]++;
}
// O(K^2) time. O(1) space. Square the overlap values.
int C1dotC2 = 0;
for(int j = 0; j < K; j++){
for(int k = 0; k < K; k++){
C1dotC2 += pow(overlaps[j][k], 2);
}
}
// O(K) time. O(1) space. Square the cluster sizes
int C1dotC1 = 0, C2dotC2 = 0;
for(int i = 0; i < K; i++){
C1dotC1 += pow(cluster_sizes1[i], 2);
C2dotC2 += pow(cluster_sizes2[i], 2);
}
return NumericVector::create((double)C1dotC2/(double)(C1dotC1+C2dotC2-C1dotC2));
}