Paso 1. Instalar paquetes y llamar librerías

# install.packages("e1071")
library(e1071)
# install.packages("caret")
library(caret)

Paso 2. Crear la base de datos

# file.choose()
df <- read.csv("C:\\Users\\USUARIO\\Downloads\\heart.csv")

Paso 3. Análisis exploratorio

summary(df)
##       age             sex               cp            trestbps    
##  Min.   :29.00   Min.   :0.0000   Min.   :0.0000   Min.   : 94.0  
##  1st Qu.:48.00   1st Qu.:0.0000   1st Qu.:0.0000   1st Qu.:120.0  
##  Median :56.00   Median :1.0000   Median :1.0000   Median :130.0  
##  Mean   :54.43   Mean   :0.6956   Mean   :0.9424   Mean   :131.6  
##  3rd Qu.:61.00   3rd Qu.:1.0000   3rd Qu.:2.0000   3rd Qu.:140.0  
##  Max.   :77.00   Max.   :1.0000   Max.   :3.0000   Max.   :200.0  
##       chol          fbs            restecg          thalach     
##  Min.   :126   Min.   :0.0000   Min.   :0.0000   Min.   : 71.0  
##  1st Qu.:211   1st Qu.:0.0000   1st Qu.:0.0000   1st Qu.:132.0  
##  Median :240   Median :0.0000   Median :1.0000   Median :152.0  
##  Mean   :246   Mean   :0.1493   Mean   :0.5298   Mean   :149.1  
##  3rd Qu.:275   3rd Qu.:0.0000   3rd Qu.:1.0000   3rd Qu.:166.0  
##  Max.   :564   Max.   :1.0000   Max.   :2.0000   Max.   :202.0  
##      exang           oldpeak          slope             ca        
##  Min.   :0.0000   Min.   :0.000   Min.   :0.000   Min.   :0.0000  
##  1st Qu.:0.0000   1st Qu.:0.000   1st Qu.:1.000   1st Qu.:0.0000  
##  Median :0.0000   Median :0.800   Median :1.000   Median :0.0000  
##  Mean   :0.3366   Mean   :1.072   Mean   :1.385   Mean   :0.7541  
##  3rd Qu.:1.0000   3rd Qu.:1.800   3rd Qu.:2.000   3rd Qu.:1.0000  
##  Max.   :1.0000   Max.   :6.200   Max.   :2.000   Max.   :4.0000  
##       thal           target      
##  Min.   :0.000   Min.   :0.0000  
##  1st Qu.:2.000   1st Qu.:0.0000  
##  Median :2.000   Median :1.0000  
##  Mean   :2.324   Mean   :0.5132  
##  3rd Qu.:3.000   3rd Qu.:1.0000  
##  Max.   :3.000   Max.   :1.0000
str(df)
## 'data.frame':    1025 obs. of  14 variables:
##  $ age     : int  52 53 70 61 62 58 58 55 46 54 ...
##  $ sex     : int  1 1 1 1 0 0 1 1 1 1 ...
##  $ cp      : int  0 0 0 0 0 0 0 0 0 0 ...
##  $ trestbps: int  125 140 145 148 138 100 114 160 120 122 ...
##  $ chol    : int  212 203 174 203 294 248 318 289 249 286 ...
##  $ fbs     : int  0 1 0 0 1 0 0 0 0 0 ...
##  $ restecg : int  1 0 1 1 1 0 2 0 0 0 ...
##  $ thalach : int  168 155 125 161 106 122 140 145 144 116 ...
##  $ exang   : int  0 1 1 0 0 0 0 1 0 1 ...
##  $ oldpeak : num  1 3.1 2.6 0 1.9 1 4.4 0.8 0.8 3.2 ...
##  $ slope   : int  2 0 0 2 1 1 0 1 2 1 ...
##  $ ca      : int  2 0 0 1 3 0 3 1 0 2 ...
##  $ thal    : int  3 3 3 3 2 2 1 3 3 2 ...
##  $ target  : int  0 0 0 0 0 1 0 0 0 0 ...
df$target <- as.factor(df$target)
summary(df)
##       age             sex               cp            trestbps    
##  Min.   :29.00   Min.   :0.0000   Min.   :0.0000   Min.   : 94.0  
##  1st Qu.:48.00   1st Qu.:0.0000   1st Qu.:0.0000   1st Qu.:120.0  
##  Median :56.00   Median :1.0000   Median :1.0000   Median :130.0  
##  Mean   :54.43   Mean   :0.6956   Mean   :0.9424   Mean   :131.6  
##  3rd Qu.:61.00   3rd Qu.:1.0000   3rd Qu.:2.0000   3rd Qu.:140.0  
##  Max.   :77.00   Max.   :1.0000   Max.   :3.0000   Max.   :200.0  
##       chol          fbs            restecg          thalach     
##  Min.   :126   Min.   :0.0000   Min.   :0.0000   Min.   : 71.0  
##  1st Qu.:211   1st Qu.:0.0000   1st Qu.:0.0000   1st Qu.:132.0  
##  Median :240   Median :0.0000   Median :1.0000   Median :152.0  
##  Mean   :246   Mean   :0.1493   Mean   :0.5298   Mean   :149.1  
##  3rd Qu.:275   3rd Qu.:0.0000   3rd Qu.:1.0000   3rd Qu.:166.0  
##  Max.   :564   Max.   :1.0000   Max.   :2.0000   Max.   :202.0  
##      exang           oldpeak          slope             ca        
##  Min.   :0.0000   Min.   :0.000   Min.   :0.000   Min.   :0.0000  
##  1st Qu.:0.0000   1st Qu.:0.000   1st Qu.:1.000   1st Qu.:0.0000  
##  Median :0.0000   Median :0.800   Median :1.000   Median :0.0000  
##  Mean   :0.3366   Mean   :1.072   Mean   :1.385   Mean   :0.7541  
##  3rd Qu.:1.0000   3rd Qu.:1.800   3rd Qu.:2.000   3rd Qu.:1.0000  
##  Max.   :1.0000   Max.   :6.200   Max.   :2.000   Max.   :4.0000  
##       thal       target 
##  Min.   :0.000   0:499  
##  1st Qu.:2.000   1:526  
##  Median :2.000          
##  Mean   :2.324          
##  3rd Qu.:3.000          
##  Max.   :3.000
str(df)
## 'data.frame':    1025 obs. of  14 variables:
##  $ age     : int  52 53 70 61 62 58 58 55 46 54 ...
##  $ sex     : int  1 1 1 1 0 0 1 1 1 1 ...
##  $ cp      : int  0 0 0 0 0 0 0 0 0 0 ...
##  $ trestbps: int  125 140 145 148 138 100 114 160 120 122 ...
##  $ chol    : int  212 203 174 203 294 248 318 289 249 286 ...
##  $ fbs     : int  0 1 0 0 1 0 0 0 0 0 ...
##  $ restecg : int  1 0 1 1 1 0 2 0 0 0 ...
##  $ thalach : int  168 155 125 161 106 122 140 145 144 116 ...
##  $ exang   : int  0 1 1 0 0 0 0 1 0 1 ...
##  $ oldpeak : num  1 3.1 2.6 0 1.9 1 4.4 0.8 0.8 3.2 ...
##  $ slope   : int  2 0 0 2 1 1 0 1 2 1 ...
##  $ ca      : int  2 0 0 1 3 0 3 1 0 2 ...
##  $ thal    : int  3 3 3 3 2 2 1 3 3 2 ...
##  $ target  : Factor w/ 2 levels "0","1": 1 1 1 1 1 2 1 1 1 1 ...

Paso 4. Partir la base de datos

# Partir la base de datos
set.seed(123)
renglones_entrenamiento <- createDataPartition(df$target, p=0.8, list=FALSE)
entrenamiento <- df[renglones_entrenamiento, ]
prueba <- df[-renglones_entrenamiento, ]

Paso 5. Construir modelo SVM Linear

modelo <- svm(target ~ ., data = entrenamiento, kernel = "linear")
resultado_entrenamiento <- predict(modelo,entrenamiento)
resultado_prueba <- predict(modelo,prueba)

Paso 6. Matriz de Confusión

mcre <- confusionMatrix(resultado_entrenamiento, entrenamiento$target) # matriz de confusión del resultado del entrenamiento mcre
mcrp <- confusionMatrix(resultado_prueba,prueba$target) # matriz de confusión del resultado de la prueba mcrp
resultados <- data.frame("svm lineal" = c(mcre$overall["Accuracy"],
mcrp$overall["Accuracy"]))
rownames(resultados) <- c("Precision de entrenamiento", "Precision de prueba")
resultados
##                            svm.lineal
## Precision de entrenamiento  0.8343484
## Precision de prueba         0.8480392

Paso 7. Obtener predicción

paciente <- data.frame(
age = 58,
sex = 0,
cp = 0,
trestbps = 100,
chol = 248,
fbs = 0,
restecg = 0,
thalach = 122,
exang = 0,
oldpeak = 1,
slope = 1,
ca = 0,
thal = 2
)

Paso 8. Hacer la predicción

prediccion <- predict(modelo, paciente)
if(prediccion == 1) {
print("Tiene enfermedad cardíaca")
} else {
print("No tiene enfermedad cardíaca")
}
## [1] "Tiene enfermedad cardíaca"

Conclusión

El modelo de Máquina de Vectores de Soporte (SVM) demostró ser una herramienta eficiente para la predicción de enfermedades cardíacas a partir de datos clínicos.

Los resultados de la matriz de confusión indican que el modelo tiene una buena capacidad para clasificar correctamente a los pacientes, minimizando errores en el diagnóstico. Al utilizar un kernel lineal, se obtuvo un balance entre precisión y generalización, permitiendo que el modelo sea interpretable y aplicable en escenarios médicos.

LS0tDQp0aXRsZTogIk3DoXF1aW5hIGRlIFZlY3RvcmVzIGRlIFNvcG9ydGUiDQphdXRob3I6ICJGYWJpYW5hIE1lZGluYWNlbGxpIC0gQTAwODM1ODk2Ig0KZGF0ZTogIjIwMjUtMDItMjciDQpvdXRwdXQ6DQogIGh0bWxfZG9jdW1lbnQ6DQogICAgdG9jOiBUUlVFDQogICAgdG9jX2Zsb2F0OiBUUlVFDQogICAgY29kZV9kb3dubG9hZDogVFJVRQ0KICAgIHRoZW1lOiBjb3Ntbw0KICAgIGhpZ2hsaWdodDogdGFuZ28NCi0tLQ0KDQohW10oQzpcXFVzZXJzXFxVU1VBUklPXFxEZXNrdG9wXFxCbG9xdWUgLSBNb2R1bG8gMlxcbGF0aWRvc2NvLmdpZikgIA0KDQojIDxzcGFuIHN0eWxlPSJjb2xvcjogYmx1ZTsiPlBhc28gMS4gSW5zdGFsYXIgcGFxdWV0ZXMgeSBsbGFtYXIgbGlicmVyw61hczwvc3Bhbj4NCg0KYGBge3IgbWVzc2FnZT1GQUxTRSwgd2FybmluZz1GQUxTRX0NCiMgaW5zdGFsbC5wYWNrYWdlcygiZTEwNzEiKQ0KbGlicmFyeShlMTA3MSkNCiMgaW5zdGFsbC5wYWNrYWdlcygiY2FyZXQiKQ0KbGlicmFyeShjYXJldCkNCmBgYA0KDQojIDxzcGFuIHN0eWxlPSJjb2xvcjogYmx1ZTsiPlBhc28gMi4gQ3JlYXIgbGEgYmFzZSBkZSBkYXRvczwvc3Bhbj4NCg0KYGBge3J9DQojIGZpbGUuY2hvb3NlKCkNCmRmIDwtIHJlYWQuY3N2KCJDOlxcVXNlcnNcXFVTVUFSSU9cXERvd25sb2Fkc1xcaGVhcnQuY3N2IikNCmBgYA0KIyA8c3BhbiBzdHlsZT0iY29sb3I6IGJsdWU7Ij5QYXNvIDMuIEFuw6FsaXNpcyBleHBsb3JhdG9yaW88L3NwYW4+DQpgYGB7cn0NCnN1bW1hcnkoZGYpDQpzdHIoZGYpDQpkZiR0YXJnZXQgPC0gYXMuZmFjdG9yKGRmJHRhcmdldCkNCnN1bW1hcnkoZGYpDQpzdHIoZGYpDQpgYGANCiMgPHNwYW4gc3R5bGU9ImNvbG9yOiBibHVlOyI+UGFzbyA0LiBQYXJ0aXIgbGEgYmFzZSBkZSBkYXRvczwvc3Bhbj4NCmBgYHtyfQ0KIyBQYXJ0aXIgbGEgYmFzZSBkZSBkYXRvcw0Kc2V0LnNlZWQoMTIzKQ0KcmVuZ2xvbmVzX2VudHJlbmFtaWVudG8gPC0gY3JlYXRlRGF0YVBhcnRpdGlvbihkZiR0YXJnZXQsIHA9MC44LCBsaXN0PUZBTFNFKQ0KZW50cmVuYW1pZW50byA8LSBkZltyZW5nbG9uZXNfZW50cmVuYW1pZW50bywgXQ0KcHJ1ZWJhIDwtIGRmWy1yZW5nbG9uZXNfZW50cmVuYW1pZW50bywgXQ0KYGBgDQojIDxzcGFuIHN0eWxlPSJjb2xvcjogYmx1ZTsiPlBhc28gNS4gQ29uc3RydWlyIG1vZGVsbyBTVk0gTGluZWFyPC9zcGFuPg0KYGBge3J9DQptb2RlbG8gPC0gc3ZtKHRhcmdldCB+IC4sIGRhdGEgPSBlbnRyZW5hbWllbnRvLCBrZXJuZWwgPSAibGluZWFyIikNCnJlc3VsdGFkb19lbnRyZW5hbWllbnRvIDwtIHByZWRpY3QobW9kZWxvLGVudHJlbmFtaWVudG8pDQpyZXN1bHRhZG9fcHJ1ZWJhIDwtIHByZWRpY3QobW9kZWxvLHBydWViYSkNCmBgYA0KIyA8c3BhbiBzdHlsZT0iY29sb3I6IGJsdWU7Ij5QYXNvIDYuIE1hdHJpeiBkZSBDb25mdXNpw7NuPC9zcGFuPg0KYGBge3J9DQptY3JlIDwtIGNvbmZ1c2lvbk1hdHJpeChyZXN1bHRhZG9fZW50cmVuYW1pZW50bywgZW50cmVuYW1pZW50byR0YXJnZXQpICMgbWF0cml6IGRlIGNvbmZ1c2nDs24gZGVsIHJlc3VsdGFkbyBkZWwgZW50cmVuYW1pZW50byBtY3JlDQptY3JwIDwtIGNvbmZ1c2lvbk1hdHJpeChyZXN1bHRhZG9fcHJ1ZWJhLHBydWViYSR0YXJnZXQpICMgbWF0cml6IGRlIGNvbmZ1c2nDs24gZGVsIHJlc3VsdGFkbyBkZSBsYSBwcnVlYmEgbWNycA0KYGBgDQoNCmBgYHtyfQ0KcmVzdWx0YWRvcyA8LSBkYXRhLmZyYW1lKCJzdm0gbGluZWFsIiA9IGMobWNyZSRvdmVyYWxsWyJBY2N1cmFjeSJdLA0KbWNycCRvdmVyYWxsWyJBY2N1cmFjeSJdKSkNCnJvd25hbWVzKHJlc3VsdGFkb3MpIDwtIGMoIlByZWNpc2lvbiBkZSBlbnRyZW5hbWllbnRvIiwgIlByZWNpc2lvbiBkZSBwcnVlYmEiKQ0KcmVzdWx0YWRvcw0KYGBgDQojIDxzcGFuIHN0eWxlPSJjb2xvcjogYmx1ZTsiPlBhc28gNy4gT2J0ZW5lciBwcmVkaWNjacOzbjwvc3Bhbj4NCmBgYHtyfQ0KcGFjaWVudGUgPC0gZGF0YS5mcmFtZSgNCmFnZSA9IDU4LA0Kc2V4ID0gMCwNCmNwID0gMCwNCnRyZXN0YnBzID0gMTAwLA0KY2hvbCA9IDI0OCwNCmZicyA9IDAsDQpyZXN0ZWNnID0gMCwNCnRoYWxhY2ggPSAxMjIsDQpleGFuZyA9IDAsDQpvbGRwZWFrID0gMSwNCnNsb3BlID0gMSwNCmNhID0gMCwNCnRoYWwgPSAyDQopDQpgYGANCiMgPHNwYW4gc3R5bGU9ImNvbG9yOiBibHVlOyI+UGFzbyA4LiBIYWNlciBsYSBwcmVkaWNjacOzbjwvc3Bhbj4NCmBgYHtyfQ0KcHJlZGljY2lvbiA8LSBwcmVkaWN0KG1vZGVsbywgcGFjaWVudGUpDQppZihwcmVkaWNjaW9uID09IDEpIHsNCnByaW50KCJUaWVuZSBlbmZlcm1lZGFkIGNhcmTDrWFjYSIpDQp9IGVsc2Ugew0KcHJpbnQoIk5vIHRpZW5lIGVuZmVybWVkYWQgY2FyZMOtYWNhIikNCn0NCmBgYA0KIyA8c3BhbiBzdHlsZT0iY29sb3I6IGJsdWU7Ij5Db25jbHVzacOzbjwvc3Bhbj4gIA0KDQpFbCBtb2RlbG8gZGUgKipNw6FxdWluYSBkZSBWZWN0b3JlcyBkZSBTb3BvcnRlIChTVk0pKiogZGVtb3N0csOzIHNlciB1bmEgaGVycmFtaWVudGEgZWZpY2llbnRlIHBhcmEgbGEgcHJlZGljY2nDs24gZGUgZW5mZXJtZWRhZGVzIGNhcmTDrWFjYXMgYSBwYXJ0aXIgZGUgZGF0b3MgY2zDrW5pY29zLiAgDQoNCkxvcyByZXN1bHRhZG9zIGRlIGxhIG1hdHJpeiBkZSBjb25mdXNpw7NuIGluZGljYW4gcXVlIGVsIG1vZGVsbyB0aWVuZSB1bmEgYnVlbmEgY2FwYWNpZGFkIHBhcmEgY2xhc2lmaWNhciBjb3JyZWN0YW1lbnRlIGEgbG9zIHBhY2llbnRlcywgbWluaW1pemFuZG8gZXJyb3JlcyBlbiBlbCBkaWFnbsOzc3RpY28uIEFsIHV0aWxpemFyIHVuICoqa2VybmVsIGxpbmVhbCoqLCBzZSBvYnR1dm8gdW4gYmFsYW5jZSBlbnRyZSBwcmVjaXNpw7NuIHkgZ2VuZXJhbGl6YWNpw7NuLCBwZXJtaXRpZW5kbyBxdWUgZWwgbW9kZWxvIHNlYSBpbnRlcnByZXRhYmxlIHkgYXBsaWNhYmxlIGVuIGVzY2VuYXJpb3MgbcOpZGljb3MuICA=