8 Dimensionality reduction

Why do we need to do this?

Imagine each gene represents a dimension - or an axis on a plot. We could plot the expression of two genes with a simple scatterplot. But a genome has thousands of genes - how do you collate all the information from each of those genes in a way that allows you to visualise it in a 2 dimensional image. This is where dimensionality reduction comes in, we calculate meta-features that contains combinations of the variation of different genes. From thousands of genes, we end up with 10s of meta-features

8.1 Perform linear dimensional reduction

Next we perform PCA on the scaled data. By default, only the previously determined variable features are used as input, but can be defined using features argument if you wish to choose a different subset.

seurat_object <- RunPCA(seurat_object, features = VariableFeatures(object = seurat_object))
#> PC_ 1 
#> Positive:  CCR7, TRAT1, CREM, ALOX5AP, NKG7, TSC22D3, CST7, PASK, GPR171, CD8A 
#>     CD8B, ADTRP, SVIP, PRF1, MYC, NOP58, TTC39C, SESN3, CMTM8, C14orf1 
#>     GRAP2, GZMA, TARBP1, KLRD1, CD320, GCHFR, AMICA1, TUBA4A, DDIT4, NOB1 
#> Negative:  TYROBP, SOD2, TIMP1, TYMP, ANXA5, LGALS3, KYNU, FCN1, LYZ, APOBEC3A 
#>     CD68, NPC2, S100A11, CTSL, MAFB, HLA-DRA, SDCBP, S100A10, PLAUR, GSTO1 
#>     IL4I1, IDO1, PILRA, LILRB4, S100A9, MS4A7, FGL2, CXCL11, HLA-DRB1, C3AR1 
#> PC_ 2 
#> Positive:  CD14, S100A8, PID1, CD9, GPX1, THBS1, PLAUR, C19orf59, OSM, CTB-61M7.2 
#>     MGST1, S100A9, GAPDH, C5AR1, SLC7A11, ATP6V0B, PPIF, CXCL2, TGFBI, PFN1 
#>     LIMS1, OLR1, PLIN2, TIMP1, COTL1, CYP1B1, PDLIM7, SLC11A1, CYP27A1, CEBPB 
#> Negative:  IFIT3, MX1, IFIT2, TNFSF10, IFI6, RSAD2, OAS1, CXCL11, MT2A, IRF7 
#>     IFITM3, OASL, GBP1, IDO1, PLSCR1, DDX58, CMPK2, APOBEC3A, FAM26F, BST2 
#>     HES4, IFIH1, RABGAP1L, IL27, VAMP5, SERPING1, GMPR, SPATS2L, IRG1, IL4I1 
#> PC_ 3 
#> Positive:  ANXA1, NKG7, PRF1, GZMA, KLRD1, MT2A, S100A8, S100A9, OASL, CD300E 
#>     CST7, C3AR1, CD8A, TYROBP, CD14, S100A12, S100A6, FCGR3A, CTSL, FCN1 
#>     MAFB, GCHFR, KLRC1, S100A11, IFI6, C5AR1, AQP9, CD8B, C19orf59, FPR1 
#> Negative:  HLA-DQA1, CD83, HLA-DQB1, HLA-DRA, HLA-DRB1, HLA-DMA, HERPUD1, HSP90AB1, CCR7, ID3 
#>     PKIB, TCF4, FABP5, BANK1, HSPD1, CLIC2, CD79B, FSCN1, HSPH1, CMTM6 
#>     CD40, TNFRSF13B, SQLE, ALDH2, LY9, NME1, HSP90AA1, CKS2, HAPLN3, IGLL5 
#> PC_ 4 
#> Positive:  CCR7, ADTRP, TRAT1, MYC, CMTM8, PASK, TARBP1, CTSL, S100A9, SOCS3 
#>     S100A8, SGTB, EMP3, TSC22D3, FBLN7, SESN3, NEXN, GBP1, NPC2, MPRIP 
#>     IL27, S100A12, CCR1, HSP90AB1, PPA1, FCN1, SOD2, RSAD2, GPR171, APOBEC3A 
#> Negative:  NKG7, CST7, PRF1, KLRD1, GZMA, ID2, KLRC1, TNFRSF18, RAMP1, IGFBP7 
#>     ALOX5AP, CD8A, GCHFR, GNG2, GAPDH, FCGR3A, XCL2, PRR5L, ANXA1, OASL 
#>     RAB27A, EIF4EBP1, HAVCR2, PKIB, RHOC, GZMK, LINC00996, ADAM8, GSN, BST2 
#> PC_ 5 
#> Positive:  S100A9, SLC7A11, MGST1, S100A8, P2RY6, CCR5, LYZ, FPR3, FABP5, RSAD2 
#>     SDS, EMP1, CCR1, IFI6, DHRS9, PRF1, CSTB, LILRB4, MX1, SPHK1 
#>     IDO1, TGFBI, CXCL2, ANXA1, CCNA1, HSP90B1, NKG7, CMPK2, S100A12, LPXN 
#> Negative:  FCGR3A, MS4A4A, MS4A7, CXCL16, PPM1N, SMPDL3A, AIF1, SERPINA1, CDKN1C, ADA 
#>     CH25H, PLAC8, C3AR1, IL3RA, PILRA, CFD, CLEC12A, VMP1, FGL2, VNN2 
#>     FCGR3B, MTMR11, C1QA, MAPKAPK3, LILRB2, COTL1, FAM26F, FPR2, IFNGR2, IL15

Seurat provides several useful ways of visualizing both cells and features that define the PCA, including VizDimReduction(), DimPlot(), and DimHeatmap()

# Examine and visualize PCA results a few different ways
print(seurat_object$pca, dims = 1:5, nfeatures = 5)
#> PC_ 1 
#> Positive:  CCR7, TRAT1, CREM, ALOX5AP, NKG7 
#> Negative:  TYROBP, SOD2, TIMP1, TYMP, ANXA5 
#> PC_ 2 
#> Positive:  CD14, S100A8, PID1, CD9, GPX1 
#> Negative:  IFIT3, MX1, IFIT2, TNFSF10, IFI6 
#> PC_ 3 
#> Positive:  ANXA1, NKG7, PRF1, GZMA, KLRD1 
#> Negative:  HLA-DQA1, CD83, HLA-DQB1, HLA-DRA, HLA-DRB1 
#> PC_ 4 
#> Positive:  CCR7, ADTRP, TRAT1, MYC, CMTM8 
#> Negative:  NKG7, CST7, PRF1, KLRD1, GZMA 
#> PC_ 5 
#> Positive:  S100A9, SLC7A11, MGST1, S100A8, P2RY6 
#> Negative:  FCGR3A, MS4A4A, MS4A7, CXCL16, PPM1N
VizDimLoadings(seurat_object, dims = 1:2, reduction = 'pca')
DimPlot(seurat_object, reduction = 'pca')

In particular DimHeatmap() allows for easy exploration of the primary sources of heterogeneity in a dataset, and can be useful when trying to decide which PCs to include for further downstream analyses. Both cells and features are ordered according to their PCA scores. Setting cells to a number plots the ‘extreme’ cells on both ends of the spectrum, which dramatically speeds plotting for large datasets. Though clearly a supervised analysis, we find this to be a valuable tool for exploring correlated feature sets.

DimHeatmap(seurat_object, dims = 1, cells = 500, balanced = TRUE)
DimHeatmap(seurat_object, dims = 1:15, cells = 500, balanced = TRUE)

8.2 Determine the ‘dimensionality’ of the dataset

To overcome the extensive technical noise in any single feature for scRNA-seq data, Seurat clusters cells based on their PCA scores, with each PC essentially representing a ‘metafeature’ that combines information across a correlated feature set. The top principal components therefore represent a robust compression of the dataset. However, how many components should we choose to include? 10? 20? 100?


Note: The Seurat developers suggest using a JackStraw resampling test to determine this. See Macosko et al, and the original seurat_object3 vignette. We’re going to use an Elbow Plot instead here, because its much quicker.


An alternative heuristic method generates an ‘Elbow plot’: a ranking of principle components based on the percentage of variance explained by each one (ElbowPlot() function). In this example, we can observe an ‘elbow’ around PC9-10, suggesting that the majority of true signal is captured in the first 10 PCs.

ElbowPlot(seurat_object)

Identifying the true dimensionality of a dataset – can be challenging/uncertain for the user. We therefore suggest these three approaches to consider. The first is more supervised, exploring PCs to determine relevant sources of heterogeneity, and could be used in conjunction with GSEA for example. The second implements a statistical test based on a random null model, but is time-consuming for large datasets, and may not return a clear PC cutoff. The third is a heuristic that is commonly used, and can be calculated instantly. In this example, all three approaches yielded similar results, but we might have been justified in choosing anything between PC 7-12 as a cutoff.

We chose 10 here, but encourage users to consider the following:

  • In the original version of this vignette with the PBMC3k dataset, genes strongly associated with PCs 12 and 13 defined rare immune subsets (i.e. MZB1 is a marker for plasmacytoid DCs). However, these groups are so rare, they are difficult to distinguish from background noise for a dataset of this size without prior knowledge.
  • We encourage users to repeat downstream analyses with a different number of PCs (10, 15, or even 50!). As you will observe, the results often do not differ dramatically.
  • We advise users to err on the higher side when choosing this parameter. For example, performing downstream analyses with only 5 PCs does significantly and adversely affect results.

8.3 Run non-linear dimensional reduction (UMAP/tSNE)

Seurat offers several non-linear dimensional reduction techniques, such as tSNE and UMAP, to visualize and explore these datasets. The goal of these algorithms is to learn the underlying manifold of the data in order to place similar cells together in low-dimensional space. Cells within the graph-based clusters determined above should co-localize on these dimension reduction plots. As input to the UMAP and tSNE, we suggest using the same PCs as input to the clustering analysis.

seurat_object <- RunUMAP(seurat_object, dims = 1:10)
#> Warning: The default method for RunUMAP has changed from calling Python UMAP via reticulate to the R-native UWOT using the cosine metric
#> To use Python UMAP via reticulate, set umap.method to 'umap-learn' and metric to 'correlation'
#> This message will be shown once per session
#> 13:12:11 UMAP embedding parameters a = 0.9922 b = 1.112
#> 13:12:11 Read 4877 rows and found 10 numeric columns
#> 13:12:11 Using Annoy for neighbor search, n_neighbors = 30
#> 13:12:11 Building Annoy index with metric = cosine, n_trees = 50
#> 0%   10   20   30   40   50   60   70   80   90   100%
#> [----|----|----|----|----|----|----|----|----|----|
#> **************************************************|
#> 13:12:12 Writing NN index file to temp file /var/folders/ww/bxqtlszx6cz42v_7kxxc2r7w0000gn/T//RtmpIbpXxA/filed8a1d897432
#> 13:12:12 Searching Annoy index using 1 thread, search_k = 3000
#> 13:12:13 Annoy recall = 100%
#> 13:12:13 Commencing smooth kNN distance calibration using 1 thread with target n_neighbors = 30
#> 13:12:13 Initializing from normalized Laplacian + noise (using RSpectra)
#> 13:12:13 Commencing optimization for 500 epochs, with 201290 positive edges
#> 13:12:17 Optimization finished
DimPlot(seurat_object, reduction = 'umap')

Challenge: PC genes

You can plot gene expression on the UMAP with the FeaturePlot() function.

Try out some genes that were highly weighted in the principal component analysis. How do they look?

8.4 Save

You can save the object at this point so that it can easily be loaded back in with readRDS() without having to rerun the computationally intensive steps performed above, or easily shared with collaborators.

saveRDS(seurat_object, file = "seurat_object_tutorial_saved.rds") 

Tip: For faster saving and loading, try the “qs” package.