diff --git a/Clustering_Knowhow.qmd b/Clustering_Knowhow.qmd
new file mode 100644
index 00000000..277045d1
--- /dev/null
+++ b/Clustering_Knowhow.qmd
@@ -0,0 +1,264 @@
+---
+title: "Clustering_knowhow"
+author: "Niladri Dasgupta"
+date: "2024-08-12"
+output: html_document
+---
+
+```{r setup, include=FALSE}
+knitr::opts_chunk$set(echo = TRUE)
+```
+
+## **What is clustering?**
+
+Clustering is a method of segregating unlabeled data or data points into different groups/clusters such that similar data points fall in the same cluster than those which differ from the others. The similarity measures are calculated using distance based metrics like Euclidean distance, Cosine similarity, Manhattan distance, etc.
+
+For Example, In the graph given below, we can clearly see that the data points can be grouped into 3 clusters
+
+![](images/Clustering/clustering_ex.PNG)
+
+
+## **Type of Clustering Algorithm**
+
+Some of the popular clustering algorithms are:
+
+1. Centroid-based Clustering (Partitioning methods)
+2. Density-based Clustering (Model-based methods)
+3. Connectivity-based Clustering (Hierarchical clustering)
+4. Distribution-based Clustering
+
+### 1.Centroid-based Clustering (Partitioning methods)
+
+Partitioning methods group data points on the basis of their closeness. The similarity measure chosen for these algorithms are Euclidean distance, Manhattan Distance or Minkowski Distance.
+
+The primary drawback for these algorithms is we need to pre define the number of clusters before allocating the data points to a group.
+
+One of the popular centroid based clustering technique is K means Clustering.
+
+
+#### **K Means Clustering**
+
+K means is an iterative clustering algorithm that works in these 5 steps:
+
+1. Specify the desired number of clusters K: Let us choose k=2 for these 5 data points in 2-D space.
+
+ ![](images/Clustering/kmeans_1.png)
+
+2. Randomly assign each data point to a cluster: Let’s assign three points in cluster 1, shown using orange color, and two points in cluster 2, shown using grey color.
+
+ ![](images/Clustering/kmeans_2.png)
+
+3. Compute cluster centroids: Centroids correspond to the arithmetic mean of data points assigned to the cluster. The centroid of data points in the orange cluster is shown using the orange cross, and those in the grey cluster using a grey cross.
+
+ ![](images/Clustering/kmeans_3.png)
+
+4. Assigns each observation to their closest centroid, based on the Euclidean distance between the object and the centroid
+
+ ![](images/Clustering/kmeans_4.png)
+
+5. Re-computing the centroids for both clusters.
+
+ ![](images/Clustering/kmeans_5.png)
+
+
+We will repeat the 4th and 5th steps until no further switching of data points between two clusters for two successive repeats.
+
+
+
+#### K-Means Clustering in R
+
+
+**Step 1: Load packages**
+
+First, we’ll load below packages that contain several useful functions regarding k-means clustering in R.
+
+
+```{r, message=FALSE}
+library(cluster) #Contain cluster function
+library(dplyr) #Data manipulation
+library(ggplot2) #Plotting function
+library(readr) #Read and write excel/csv files
+library(factoextra) #Extract and Visualize the Results of Multivariate Data Analyses
+```
+
+**Step 2: Load Data**
+
+We have used the “Mall_Customer” dataset in R for this case study.
+
+```{r, message=FALSE}
+#Loading the data
+df <- read_csv("data/Mall_Customers.csv")
+
+#Structure of the data
+str(df)
+```
+
+dataset consists of 200 customers data with their age, annual income and Spending score.
+
+
+```{r}
+
+#Rename the columns
+df <- df %>%
+ rename("Annual_Income"= `Annual Income (k$)`, "Spending_score"= `Spending Score (1-100)`)
+
+#remove rows with missing values
+df <- na.omit(df)
+
+#scale each variable to have a mean of 0 and sd of 1
+df1 <- df %>%
+ mutate(across(where(is.numeric), scale))
+
+#view first six rows of dataset
+head(df1)
+
+```
+
+
+We have separated the CustomerID and Genre from the dataset. The reason for removing these variables from the cluster dataset as Kmeans can handle only numerical variables.
+To create cluster with categorical or ordinal variable we can use k-Medoid clustering.
+
+
+```{r}
+df1 <- df1[,4:5]
+
+```
+
+**Step 3: Find the Optimal Number of Clusters**
+
+To perform k-means clustering in R we can use the built-in kmeans() function, which uses the following syntax:
+
+
+ kmeans(data, centers, iter.max, nstart)
+ where:
+ - data: Name of the dataset.
+ - centers: The number of clusters, denoted k.
+ - iter.max (optional): The maximum number of iterations allowed. Default value is 10.
+ - nstart (optional): The number of initial configurations. Default value is 1.
+
+
+
+- Centers is the k of K Means. centers = 5 would results in 5 clusters being created. We need to **predefine the k** before the cluster process starts.
+- iter.max is the number of times the algorithm will repeat the cluster assignment and update the centers / centroids. Iteration stops after this many iterations even if the convergence criterion is not satisfied
+- nstart is the number of times the initial starting points are re-sampled.
+It means at the initialization of Clusters you need to specify how many clusters you want and the algorithm will randomly find same number of centroids to initialize. nstart gives you an edge to initialize the centroids through re sampling.
+For example if total number of cluster is 3 and nstart=25 then it extracts 3 sets of data, 25 times, and for each of these times, the algorithm is run (up to iter.max # of iterations) and the cost function (total sum of the squares) is evaluated and finally 3 centroids with lowest cost function are chosen to start the clustering process.
+
+
+To find the best number of clusters/centroids there are two popular methods as shown below.
+
+[**A. Elbow Method:**]{.underline}
+
+It has two parts as explained below-
+
+- WSS: The Within Sum of Squares (WSS) is the sum of distance between the centroids and every other data points within a cluster. Small WSS indicates that every data point is close to its nearest centroids.
+
+- Elbow rule/method: Here we plot out the WSS score against the number of K. Because with the number of K increasing, the WSS will always decrease; however, the magnitude of decrease between each k will be diminishing, and the plot will be a curve which looks like an arm that curled up. In this way, we can find out which point falls on the elbow.
+
+```{r}
+set.seed(1)
+wss<- NULL
+
+#Feeding different centroid/cluster and record WSS
+
+for (i in 1:10){
+ fit = kmeans(df1,centers = i,nstart=25)
+ wss = c(wss, fit$tot.withinss)
+}
+
+#Visualize the plot
+plot(1:10, wss, type = "o", xlab='Number of clusters(k)')
+
+```
+
+Based on the above plot at k=5 we can see an “elbow” where the sum of squares begins to “bend” or level off so the ideal number of clusters should be 5.
+
+
+The above process to compute the “Elbow method” has been wrapped up in a single function (fviz_nbclust):
+
+```{r}
+fviz_nbclust(df1, kmeans, method = "wss",nstart=25)
+```
+
+
+[**B. Silhouette Method:**]{.underline}
+
+The silhouette coefficient or silhouette score is a measure of how similar a data point is within-cluster (intra-cluster) compared to other clusters (inter-cluster).
+The Silhouette Coefficient is calculated using the mean *intra-cluster distance (a)* and the *mean nearest-cluster distance (b)* for each sample. The Silhouette Coefficient for a sample is *(b - a) / max(a, b)*
+
+Here we will plot the silhouette width/coefficient for different number of clusters and will choose the point where the silhouette width is highest.
+
+**Points to Remember While Calculating Silhouette Coefficient:**
+
+The value of the silhouette coefficient is between [-1, 1].
+A score of 1 denotes the best, meaning that the data points are very compact within the cluster to which it belongs and far away from the other clusters.
+The worst value is -1. Values near 0 denote overlapping clusters.
+
+In this demonstration, we are going to see how silhouette method is used.
+
+```{r}
+
+silhouette_score <- function(k){
+ km <- kmeans(df1, centers = k,nstart = 25)
+ ss <- silhouette(km$cluster, dist(df1))
+ mean(ss[, 3])
+}
+k <- 2:10
+
+avg_sil <- sapply(k, silhouette_score)
+plot(k, type='b', avg_sil, xlab='Number of clusters', ylab='Average Silhouette Scores', frame=FALSE)
+```
+
+From the above method we can see the silhouette width is highest at cluster 5 so the optimal number of cluster should be 5.
+
+Similar to the elbow method, this process to compute the “average silhoutte method” has been wrapped up in a single function (fviz_nbclust):
+
+```{r}
+fviz_nbclust(df1, kmeans, method='silhouette',nstart=25)
+```
+
+The optimal number of clusters is 5.
+
+
+**Step 4: Perform K-Means Clustering with Optimal K**
+
+Lastly, we can perform k-means clustering on the dataset using the optimal value for k of 5:
+
+```{r}
+
+#make this example reproducible
+set.seed(1)
+
+#perform k-means clustering with k = 5 clusters
+fit <- kmeans(df1, 5, nstart=25)
+#view results
+fit
+```
+
+We can visualize the clusters on a scatterplot that displays the first two principal components on the axes using the fivz_cluster() function:
+
+```{r}
+#plot results of final k-means model
+
+fviz_cluster(fit, data = df1)
+```
+
+
+**Step 5: Exporting the data by adding generated clusters**
+
+```{r}
+#Adding the clusters in the main data
+
+df_cluster <- df %>%
+ mutate(cluster=fit$cluster)
+
+#Creating Summary of created clusters based on existing variables
+
+df_summary <- df_cluster %>%
+ group_by(cluster) %>%
+ summarise(records=n(),avg_age=mean(Age),avg_annual_income=mean(Annual_Income),avg_spending_score=mean(Spending_score))
+
+print(df_summary)
+```
+
+We can create a group of potential customers to target based on their age, average annual income and average spending score.
diff --git a/data/Mall_Customers.csv b/data/Mall_Customers.csv
new file mode 100644
index 00000000..9d16c774
--- /dev/null
+++ b/data/Mall_Customers.csv
@@ -0,0 +1,201 @@
+CustomerID,Genre,Age,Annual Income (k$),Spending Score (1-100)
+0001,Male,19,15,39
+0002,Male,21,15,81
+0003,Female,20,16,6
+0004,Female,23,16,77
+0005,Female,31,17,40
+0006,Female,22,17,76
+0007,Female,35,18,6
+0008,Female,23,18,94
+0009,Male,64,19,3
+0010,Female,30,19,72
+0011,Male,67,19,14
+0012,Female,35,19,99
+0013,Female,58,20,15
+0014,Female,24,20,77
+0015,Male,37,20,13
+0016,Male,22,20,79
+0017,Female,35,21,35
+0018,Male,20,21,66
+0019,Male,52,23,29
+0020,Female,35,23,98
+0021,Male,35,24,35
+0022,Male,25,24,73
+0023,Female,46,25,5
+0024,Male,31,25,73
+0025,Female,54,28,14
+0026,Male,29,28,82
+0027,Female,45,28,32
+0028,Male,35,28,61
+0029,Female,40,29,31
+0030,Female,23,29,87
+0031,Male,60,30,4
+0032,Female,21,30,73
+0033,Male,53,33,4
+0034,Male,18,33,92
+0035,Female,49,33,14
+0036,Female,21,33,81
+0037,Female,42,34,17
+0038,Female,30,34,73
+0039,Female,36,37,26
+0040,Female,20,37,75
+0041,Female,65,38,35
+0042,Male,24,38,92
+0043,Male,48,39,36
+0044,Female,31,39,61
+0045,Female,49,39,28
+0046,Female,24,39,65
+0047,Female,50,40,55
+0048,Female,27,40,47
+0049,Female,29,40,42
+0050,Female,31,40,42
+0051,Female,49,42,52
+0052,Male,33,42,60
+0053,Female,31,43,54
+0054,Male,59,43,60
+0055,Female,50,43,45
+0056,Male,47,43,41
+0057,Female,51,44,50
+0058,Male,69,44,46
+0059,Female,27,46,51
+0060,Male,53,46,46
+0061,Male,70,46,56
+0062,Male,19,46,55
+0063,Female,67,47,52
+0064,Female,54,47,59
+0065,Male,63,48,51
+0066,Male,18,48,59
+0067,Female,43,48,50
+0068,Female,68,48,48
+0069,Male,19,48,59
+0070,Female,32,48,47
+0071,Male,70,49,55
+0072,Female,47,49,42
+0073,Female,60,50,49
+0074,Female,60,50,56
+0075,Male,59,54,47
+0076,Male,26,54,54
+0077,Female,45,54,53
+0078,Male,40,54,48
+0079,Female,23,54,52
+0080,Female,49,54,42
+0081,Male,57,54,51
+0082,Male,38,54,55
+0083,Male,67,54,41
+0084,Female,46,54,44
+0085,Female,21,54,57
+0086,Male,48,54,46
+0087,Female,55,57,58
+0088,Female,22,57,55
+0089,Female,34,58,60
+0090,Female,50,58,46
+0091,Female,68,59,55
+0092,Male,18,59,41
+0093,Male,48,60,49
+0094,Female,40,60,40
+0095,Female,32,60,42
+0096,Male,24,60,52
+0097,Female,47,60,47
+0098,Female,27,60,50
+0099,Male,48,61,42
+0100,Male,20,61,49
+0101,Female,23,62,41
+0102,Female,49,62,48
+0103,Male,67,62,59
+0104,Male,26,62,55
+0105,Male,49,62,56
+0106,Female,21,62,42
+0107,Female,66,63,50
+0108,Male,54,63,46
+0109,Male,68,63,43
+0110,Male,66,63,48
+0111,Male,65,63,52
+0112,Female,19,63,54
+0113,Female,38,64,42
+0114,Male,19,64,46
+0115,Female,18,65,48
+0116,Female,19,65,50
+0117,Female,63,65,43
+0118,Female,49,65,59
+0119,Female,51,67,43
+0120,Female,50,67,57
+0121,Male,27,67,56
+0122,Female,38,67,40
+0123,Female,40,69,58
+0124,Male,39,69,91
+0125,Female,23,70,29
+0126,Female,31,70,77
+0127,Male,43,71,35
+0128,Male,40,71,95
+0129,Male,59,71,11
+0130,Male,38,71,75
+0131,Male,47,71,9
+0132,Male,39,71,75
+0133,Female,25,72,34
+0134,Female,31,72,71
+0135,Male,20,73,5
+0136,Female,29,73,88
+0137,Female,44,73,7
+0138,Male,32,73,73
+0139,Male,19,74,10
+0140,Female,35,74,72
+0141,Female,57,75,5
+0142,Male,32,75,93
+0143,Female,28,76,40
+0144,Female,32,76,87
+0145,Male,25,77,12
+0146,Male,28,77,97
+0147,Male,48,77,36
+0148,Female,32,77,74
+0149,Female,34,78,22
+0150,Male,34,78,90
+0151,Male,43,78,17
+0152,Male,39,78,88
+0153,Female,44,78,20
+0154,Female,38,78,76
+0155,Female,47,78,16
+0156,Female,27,78,89
+0157,Male,37,78,1
+0158,Female,30,78,78
+0159,Male,34,78,1
+0160,Female,30,78,73
+0161,Female,56,79,35
+0162,Female,29,79,83
+0163,Male,19,81,5
+0164,Female,31,81,93
+0165,Male,50,85,26
+0166,Female,36,85,75
+0167,Male,42,86,20
+0168,Female,33,86,95
+0169,Female,36,87,27
+0170,Male,32,87,63
+0171,Male,40,87,13
+0172,Male,28,87,75
+0173,Male,36,87,10
+0174,Male,36,87,92
+0175,Female,52,88,13
+0176,Female,30,88,86
+0177,Male,58,88,15
+0178,Male,27,88,69
+0179,Male,59,93,14
+0180,Male,35,93,90
+0181,Female,37,97,32
+0182,Female,32,97,86
+0183,Male,46,98,15
+0184,Female,29,98,88
+0185,Female,41,99,39
+0186,Male,30,99,97
+0187,Female,54,101,24
+0188,Male,28,101,68
+0189,Female,41,103,17
+0190,Female,36,103,85
+0191,Female,34,103,23
+0192,Female,32,103,69
+0193,Male,33,113,8
+0194,Female,38,113,91
+0195,Female,47,120,16
+0196,Female,35,120,79
+0197,Female,45,126,28
+0198,Male,32,126,74
+0199,Male,32,137,18
+0200,Male,30,137,83
\ No newline at end of file
diff --git a/images/Clustering/clustering_ex.PNG b/images/Clustering/clustering_ex.PNG
new file mode 100644
index 00000000..218698a5
Binary files /dev/null and b/images/Clustering/clustering_ex.PNG differ
diff --git a/images/Clustering/kmeans_1.png b/images/Clustering/kmeans_1.png
new file mode 100644
index 00000000..cdea98cd
Binary files /dev/null and b/images/Clustering/kmeans_1.png differ
diff --git a/images/Clustering/kmeans_2.png b/images/Clustering/kmeans_2.png
new file mode 100644
index 00000000..f74b8c40
Binary files /dev/null and b/images/Clustering/kmeans_2.png differ
diff --git a/images/Clustering/kmeans_3.png b/images/Clustering/kmeans_3.png
new file mode 100644
index 00000000..91b0e355
Binary files /dev/null and b/images/Clustering/kmeans_3.png differ
diff --git a/images/Clustering/kmeans_4.png b/images/Clustering/kmeans_4.png
new file mode 100644
index 00000000..49e5c83a
Binary files /dev/null and b/images/Clustering/kmeans_4.png differ
diff --git a/images/Clustering/kmeans_5.png b/images/Clustering/kmeans_5.png
new file mode 100644
index 00000000..4745e4cb
Binary files /dev/null and b/images/Clustering/kmeans_5.png differ
diff --git a/renv/activate.R b/renv/activate.R
index cb5401f9..d13f9932 100644
--- a/renv/activate.R
+++ b/renv/activate.R
@@ -2,11 +2,13 @@
local({
# the requested version of renv
- version <- "1.0.3"
+ version <- "1.0.7"
attr(version, "sha") <- NULL
# the project directory
- project <- getwd()
+ project <- Sys.getenv("RENV_PROJECT")
+ if (!nzchar(project))
+ project <- getwd()
# use start-up diagnostics if enabled
diagnostics <- Sys.getenv("RENV_STARTUP_DIAGNOSTICS", unset = "FALSE")
@@ -31,6 +33,14 @@ local({
if (!is.null(override))
return(override)
+ # if we're being run in a context where R_LIBS is already set,
+ # don't load -- presumably we're being run as a sub-process and
+ # the parent process has already set up library paths for us
+ rcmd <- Sys.getenv("R_CMD", unset = NA)
+ rlibs <- Sys.getenv("R_LIBS", unset = NA)
+ if (!is.na(rlibs) && !is.na(rcmd))
+ return(FALSE)
+
# next, check environment variables
# TODO: prefer using the configuration one in the future
envvars <- c(
@@ -50,9 +60,22 @@ local({
})
- if (!enabled)
+ # bail if we're not enabled
+ if (!enabled) {
+
+ # if we're not enabled, we might still need to manually load
+ # the user profile here
+ profile <- Sys.getenv("R_PROFILE_USER", unset = "~/.Rprofile")
+ if (file.exists(profile)) {
+ cfg <- Sys.getenv("RENV_CONFIG_USER_PROFILE", unset = "TRUE")
+ if (tolower(cfg) %in% c("true", "t", "1"))
+ sys.source(profile, envir = globalenv())
+ }
+
return(FALSE)
+ }
+
# avoid recursion
if (identical(getOption("renv.autoloader.running"), TRUE)) {
warning("ignoring recursive attempt to run renv autoloader")
@@ -108,6 +131,21 @@ local({
}
+ heredoc <- function(text, leave = 0) {
+
+ # remove leading, trailing whitespace
+ trimmed <- gsub("^\\s*\\n|\\n\\s*$", "", text)
+
+ # split into lines
+ lines <- strsplit(trimmed, "\n", fixed = TRUE)[[1L]]
+
+ # compute common indent
+ indent <- regexpr("[^[:space:]]", lines)
+ common <- min(setdiff(indent, -1L)) - leave
+ paste(substring(lines, common), collapse = "\n")
+
+ }
+
startswith <- function(string, prefix) {
substring(string, 1, nchar(prefix)) == prefix
}
@@ -610,6 +648,9 @@ local({
# if the user has requested an automatic prefix, generate it
auto <- Sys.getenv("RENV_PATHS_PREFIX_AUTO", unset = NA)
+ if (is.na(auto) && getRversion() >= "4.4.0")
+ auto <- "TRUE"
+
if (auto %in% c("TRUE", "True", "true", "1"))
return(renv_bootstrap_platform_prefix_auto())
@@ -801,24 +842,23 @@ local({
# the loaded version of renv doesn't match the requested version;
# give the user instructions on how to proceed
- remote <- if (!is.null(description[["RemoteSha"]])) {
+ dev <- identical(description[["RemoteType"]], "github")
+ remote <- if (dev)
paste("rstudio/renv", description[["RemoteSha"]], sep = "@")
- } else {
+ else
paste("renv", description[["Version"]], sep = "@")
- }
# display both loaded version + sha if available
friendly <- renv_bootstrap_version_friendly(
version = description[["Version"]],
- sha = description[["RemoteSha"]]
+ sha = if (dev) description[["RemoteSha"]]
)
- fmt <- paste(
- "renv %1$s was loaded from project library, but this project is configured to use renv %2$s.",
- "- Use `renv::record(\"%3$s\")` to record renv %1$s in the lockfile.",
- "- Use `renv::restore(packages = \"renv\")` to install renv %2$s into the project library.",
- sep = "\n"
- )
+ fmt <- heredoc("
+ renv %1$s was loaded from project library, but this project is configured to use renv %2$s.
+ - Use `renv::record(\"%3$s\")` to record renv %1$s in the lockfile.
+ - Use `renv::restore(packages = \"renv\")` to install renv %2$s into the project library.
+ ")
catf(fmt, friendly, renv_bootstrap_version_friendly(version), remote)
FALSE
@@ -1041,7 +1081,7 @@ local({
# if jsonlite is loaded, use that instead
if ("jsonlite" %in% loadedNamespaces()) {
- json <- catch(renv_json_read_jsonlite(file, text))
+ json <- tryCatch(renv_json_read_jsonlite(file, text), error = identity)
if (!inherits(json, "error"))
return(json)
@@ -1050,7 +1090,7 @@ local({
}
# otherwise, fall back to the default JSON reader
- json <- catch(renv_json_read_default(file, text))
+ json <- tryCatch(renv_json_read_default(file, text), error = identity)
if (!inherits(json, "error"))
return(json)
@@ -1063,14 +1103,14 @@ local({
}
renv_json_read_jsonlite <- function(file = NULL, text = NULL) {
- text <- paste(text %||% read(file), collapse = "\n")
+ text <- paste(text %||% readLines(file, warn = FALSE), collapse = "\n")
jsonlite::fromJSON(txt = text, simplifyVector = FALSE)
}
renv_json_read_default <- function(file = NULL, text = NULL) {
# find strings in the JSON
- text <- paste(text %||% read(file), collapse = "\n")
+ text <- paste(text %||% readLines(file, warn = FALSE), collapse = "\n")
pattern <- '["](?:(?:\\\\.)|(?:[^"\\\\]))*?["]'
locs <- gregexpr(pattern, text, perl = TRUE)[[1]]
@@ -1118,14 +1158,14 @@ local({
map <- as.list(map)
# remap strings in object
- remapped <- renv_json_remap(json, map)
+ remapped <- renv_json_read_remap(json, map)
# evaluate
eval(remapped, envir = baseenv())
}
- renv_json_remap <- function(json, map) {
+ renv_json_read_remap <- function(json, map) {
# fix names
if (!is.null(names(json))) {
@@ -1152,7 +1192,7 @@ local({
# recurse
if (is.recursive(json)) {
for (i in seq_along(json)) {
- json[i] <- list(renv_json_remap(json[[i]], map))
+ json[i] <- list(renv_json_read_remap(json[[i]], map))
}
}