Expand a matrix of rankings (1 ~ 4) to a bigger binary matrix
Asked Answered
S

3

6

I have a matrix which I want to convert to one with binary output (0 vs 1). The matrix to be converted contains four rows of rankings (1 to 4):

mat1.data <- c(4,   3,  3,  3,  3,  2,  2,  1,  1,  1,
               3,   4,  2,  4,  2,  3,  1,  3,  3,  2,
               2,   2,  4,  1,  1,  1,  4,  4,  2,  4,
               1,   1,  1,  2,  4,  4,  3,  2,  4,  3)
mat1 <- matrix(mat1.data,nrow=4,ncol=10,byrow=TRUE)
mat1
     [,1] [,2] [,3] [,4] [,5] [,6] [,7] [,8] [,9] [,10]
[1,]    4    3    3    3    3    2    2    1    1     1
[2,]    3    4    2    4    2    3    1    3    3     2
[3,]    2    2    4    1    1    1    4    4    2     4
[4,]    1    1    1    2    4    4    3    2    4     3

For each row in the input matrix, I want to create four binary rows - one row for each value of the ranks (1-4). In the binary matrix, each row-wise entry is 1 on positions where the focal rank occurs in the input matrix, and 0 otherwise. Each row from the original matrix should produce 10*4=40 entries in the output matrix.

For example, for the first row of in the input matrix...

4   3   3   3   3   2   2   1   1   1

...the output should be:

0   0   0   0   0   0   0   1   1   1 # Rank 1 in input
0   0   0   0   0   1   1   0   0   0 # Rank 2 in input
0   1   1   1   1   0   0   0   0   0 # Rank 3 in input
1   0   0   0   0   0   0   0   0   0 # Rank 4 in input

Continue with this process, the expected output for all four rows of rankings should look like this:

0   0   0   0   0   0   0   1   1   1 #first row of rankings starts
0   0   0   0   0   1   1   0   0   0
0   1   1   1   1   0   0   0   0   0
1   0   0   0   0   0   0   0   0   0 #first row of rankings ends
0   0   0   0   0   0   1   0   0   0 #second row of rankings starts
0   0   1   0   1   0   0   0   0   1
1   0   0   0   0   1   0   1   1   0
0   1   0   1   0   0   0   0   0   0 #second row of rankings ends
0   0   0   1   1   1   0   0   0   0 #third row of rankings starts
1   1   0   0   0   0   0   0   1   0
0   0   0   0   0   0   0   0   0   0
0   0   1   0   0   0   1   1   0   1 #third row of rankings ends
1   1   1   0   0   0   0   0   0   0 #fourth row of rankings starts
0   0   0   1   0   0   0   1   0   0
0   0   0   0   0   0   1   0   0   1
0   0   0   0   1   1   0   0   1   0 #fourth row of rankings ends

How do I do achieve this? I have a larger dataset and so a more efficient method is preferred but any help will be greatly appreciated!

Syck answered 16/7, 2022 at 14:3 Comment(5)
Thanks to all the wonderful answers Henrik, Zheyuan Li, and ThomasIsCoding. I do wonder if it is possible to reverse the process. That is, given the binary matrix can we return to the original matrix?Syck
rowsum(1:4 * m, rep(1:4, each=4)) , where m is the binary matrix output (and change 4 to generalise the number of ranks)Tuggle
@Tuggle Impressively concise. I was thinking mat <- matrix(which(m == 1) %% 4, 4); mat[mat == 0] <- 4.Moses
That's clever @ZheyuanLi . (I think you can do the wee mod trick so simplify matrix( (which(m == 1) - 1) %% 4 + 1 , 4))Tuggle
@Tuggle You rock!! I couldn't think of this -1 then +1 shift. Your solutions are great. Perhaps OP can post this as a second question (well, it is) and you answer it!Moses
M
5
matrix(sapply(mat1, \(i) replace(numeric(4), i, 1)), ncol = ncol(mat1))
#      [,1] [,2] [,3] [,4] [,5] [,6] [,7] [,8] [,9] [,10]
# [1,]    0    0    0    0    0    0    0    1    1     1
# [2,]    0    0    0    0    0    1    1    0    0     0
# [3,]    0    1    1    1    1    0    0    0    0     0
# [4,]    1    0    0    0    0    0    0    0    0     0
# [5,]    0    0    0    0    0    0    1    0    0     0
# [6,]    0    0    1    0    1    0    0    0    0     1
# [7,]    1    0    0    0    0    1    0    1    1     0
# [8,]    0    1    0    1    0    0    0    0    0     0
# [9,]    0    0    0    1    1    1    0    0    0     0
#[10,]    1    1    0    0    0    0    0    0    1     0
#[11,]    0    0    0    0    0    0    0    0    0     0
#[12,]    0    0    1    0    0    0    1    1    0     1
#[13,]    1    1    1    0    0    0    0    0    0     0
#[14,]    0    0    0    1    0    0    0    1    0     0
#[15,]    0    0    0    0    0    0    1    0    0     1
#[16,]    0    0    0    0    1    1    0    0    1     0

It takes 2 steps, and piping syntax may look clearer:

sapply(mat1, \(i) replace(numeric(4), i, 1)) |>  ## each value to binary vector
  matrix(ncol = ncol(mat1))  ## reshape

Actually, I don't need that anonymous function \(i). I can pass replace, and its arguments, to sapply directly.

matrix(sapply(mat1, replace, x = numeric(4), values = 1), ncol = ncol(mat1))

sapply(mat1, replace, x = numeric(4), values = 1) |> matrix(ncol = ncol(mat1))

Misc

user20650 and I discussed a little bit in comments, and here is a "vectorized" approach using outer:

matrix(+outer(1:4, c(mat1), "=="), ncol = ncol(mat1))

Henrik's answer is a more memory-efficient "vectorized" approach, but it over-complicates the index computation. Here is something simpler:

out <- matrix(0, nrow(mat1) * 4, ncol(mat1))
pos1 <- seq(0, length(mat1) - 1) * 4 + c(mat1)
out[pos1] <- 1

All methods so far create a dense output matrix. This is OK because the percentage of nonzero elements is 25%, which is not typically sparse. However, in case we want a sparse one, it is also straightforward:

## in fact, this is what Henrik aims to compute
ij <- arrayInd(pos1, c(4 * nrow(mat1), ncol(mat1)))
## sparse matrix
Matrix::sparseMatrix(i = ij[, 1], j = ij[, 2], x = rep(1, length(mat1)))
#16 x 10 sparse Matrix of class "dgCMatrix"
#                         
# [1,] . . . . . . . 1 1 1
# [2,] . . . . . 1 1 . . .
# [3,] . 1 1 1 1 . . . . .
# [4,] 1 . . . . . . . . .
# [5,] . . . . . . 1 . . .
# [6,] . . 1 . 1 . . . . 1
# [7,] 1 . . . . 1 . 1 1 .
# [8,] . 1 . 1 . . . . . .
# [9,] . . . 1 1 1 . . . .
#[10,] 1 1 . . . . . . 1 .
#[11,] . . . . . . . . . .
#[12,] . . 1 . . . 1 1 . 1
#[13,] 1 1 1 . . . . . . .
#[14,] . . . 1 . . . 1 . .
#[15,] . . . . . . 1 . . 1
#[16,] . . . . 1 1 . . 1 .
Moses answered 16/7, 2022 at 14:25 Comment(0)
O
4

Using row, col, and matrix indexing:

m = matrix(0, nr = 4 * nrow(mat1), nc = ncol(mat1))
m[cbind(c(row(mat1) + seq(0, by = (4 - 1), len = nrow(mat1)) + (mat1 - 1)), 
        c(col(mat1)))] = 1

      [,1] [,2] [,3] [,4] [,5] [,6] [,7] [,8] [,9] [,10]
 [1,]    0    0    0    0    0    0    0    1    1     1
 [2,]    0    0    0    0    0    1    1    0    0     0
 [3,]    0    1    1    1    1    0    0    0    0     0
 [4,]    1    0    0    0    0    0    0    0    0     0
 [5,]    0    0    0    0    0    0    1    0    0     0
 [6,]    0    0    1    0    1    0    0    0    0     1
 [7,]    1    0    0    0    0    1    0    1    1     0
 [8,]    0    1    0    1    0    0    0    0    0     0
 [9,]    0    0    0    1    1    1    0    0    0     0
[10,]    1    1    0    0    0    0    0    0    1     0
[11,]    0    0    0    0    0    0    0    0    0     0
[12,]    0    0    1    0    0    0    1    1    0     1
[13,]    1    1    1    0    0    0    0    0    0     0
[14,]    0    0    0    1    0    0    0    1    0     0
[15,]    0    0    0    0    0    0    1    0    0     1
[16,]    0    0    0    0    1    1    0    0    1     0
Obannon answered 16/7, 2022 at 14:32 Comment(0)
C
2

Probably we can benefit from using kronecker + rep like below

> +(kronecker(mat1, matrix(rep(1, 4))) == rep(1:4, nrow(mat1)))
      [,1] [,2] [,3] [,4] [,5] [,6] [,7] [,8] [,9] [,10]
 [1,]    0    0    0    0    0    0    0    1    1     1
 [2,]    0    0    0    0    0    1    1    0    0     0
 [3,]    0    1    1    1    1    0    0    0    0     0
 [4,]    1    0    0    0    0    0    0    0    0     0
 [5,]    0    0    0    0    0    0    1    0    0     0
 [6,]    0    0    1    0    1    0    0    0    0     1
 [7,]    1    0    0    0    0    1    0    1    1     0
 [8,]    0    1    0    1    0    0    0    0    0     0
 [9,]    0    0    0    1    1    1    0    0    0     0
[10,]    1    1    0    0    0    0    0    0    1     0
[11,]    0    0    0    0    0    0    0    0    0     0
[12,]    0    0    1    0    0    0    1    1    0     1
[13,]    1    1    1    0    0    0    0    0    0     0
[14,]    0    0    0    1    0    0    0    1    0     0
[15,]    0    0    0    0    0    0    1    0    0     1
[16,]    0    0    0    0    1    1    0    0    1     0
Carnarvon answered 16/7, 2022 at 23:18 Comment(1)
@ZheyuanLi kronecker helps to expand the dimmension of one matrix with another given matrix, which is the same as outer only if both matrices are reduced to column/row vectors.Carnarvon

© 2022 - 2025 — McMap. All rights reserved.