# 1. Cài đặt và nạp gói
library(readxl)
## Warning: package 'readxl' was built under R version 4.4.3
library(urca)
## Warning: package 'urca' was built under R version 4.4.3
library(PerformanceAnalytics)
## Warning: package 'PerformanceAnalytics' was built under R version 4.4.3
## Loading required package: xts
## Warning: package 'xts' was built under R version 4.4.3
## Loading required package: zoo
## Warning: package 'zoo' was built under R version 4.4.3
##
## Attaching package: 'zoo'
## The following objects are masked from 'package:base':
##
## as.Date, as.Date.numeric
##
## Attaching package: 'PerformanceAnalytics'
## The following object is masked from 'package:graphics':
##
## legend
# 2. Đọc dữ liệu
library(readxl)
data <- read_excel("C:/Users/Hoang Quyen/Downloads/DL.xlsx", sheet = "Worksheet")
colnames(data) <- c("Date", "BID", "MBB", "ACB", "CTG", "SHB", "VNI") # chuẩn hóa tên cột
# 3. Tính log return
returns <- as.data.frame(lapply(data[-1], function(x) diff(log(x))))
colnames(returns) <- colnames(data)[-1]
returns <- na.omit(returns)
4. KẾT QUẢ NGHIÊN CỨU VÀ PHÂN TÍCH
4.1. Kiểm định tính dừng chuỗi lợi suất
Trước khi tiến hành ước lượng mô hình CAPM, nghiên cứu đã sử dụng
kiểm định Augmented Dickey-Fuller (ADF) để kiểm tra tính dừng của các
chuỗi lợi suất log của năm cổ phiếu: BID, MBB, ACB, CTG, SHB, cùng với
chỉ số thị trường VNI.
# 4. Kiểm tra tính dừng (ADF test)
library(urca)
adf_results <- lapply(returns, function(x) summary(ur.df(x, type = "drift", selectlags = "AIC")))
# Kiểm tra 1 ví dụ
print(adf_results$BID)
##
## ###############################################
## # Augmented Dickey-Fuller Test Unit Root Test #
## ###############################################
##
## Test regression drift
##
##
## Call:
## lm(formula = z.diff ~ z.lag.1 + 1 + z.diff.lag)
##
## Residuals:
## Min 1Q Median 3Q Max
## -0.073948 -0.007662 -0.000569 0.006908 0.067783
##
## Coefficients:
## Estimate Std. Error t value Pr(>|t|)
## (Intercept) 2.153e-05 7.866e-04 0.027 0.978
## z.lag.1 -9.419e-01 7.073e-02 -13.317 <2e-16 ***
## z.diff.lag 2.700e-02 5.215e-02 0.518 0.605
## ---
## Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
##
## Residual standard error: 0.01511 on 366 degrees of freedom
## Multiple R-squared: 0.459, Adjusted R-squared: 0.456
## F-statistic: 155.3 on 2 and 366 DF, p-value: < 2.2e-16
##
##
## Value of test-statistic is: -13.317 88.6714
##
## Critical values for test statistics:
## 1pct 5pct 10pct
## tau2 -3.44 -2.87 -2.57
## phi1 6.47 4.61 3.79
Kết quả kiểm định ADF cho cổ phiếu BID như sau:
- Giá trị thống kê kiểm định: –13.317
- Ngưỡng tới hạn tại mức ý nghĩa 5%: –2.87
Vì –13.317 < –2.87, chuỗi lợi suất log của cổ phiếu BID là
dừng tại mức ý nghĩa 5%. Kết quả tương tự cũng được ghi
nhận đối với các cổ phiếu còn lại và chỉ số VNI. Do đó, tất cả các chuỗi
được xác định là dừng và đủ điều kiện đưa vào hồi quy CAPM.
4.2. Ước lượng mô hình CAPM
Mô hình CAPM được ước lượng cho từng cổ phiếu theo phương trình:
\[
R_{it} - R_f = \alpha_i + \beta_i (R_{mt} - R_f) + \varepsilon_{it}
\]
Trong đó:
- \(R_{it}\): lợi suất của cổ phiếu
\(i\)
- \(R_f\): lãi suất phi rủi ro (giả
định 3%/năm, tương đương 0.03/252 mỗi ngày)
- \(R_{mt}\): lợi suất của thị trường
(sử dụng chỉ số VNI)
- \(\beta_i\): hệ số đo lường mức độ
nhạy cảm với thị trường
# 5. Ước lượng mô hình CAPM
rf <- 0.03 / 252 # lãi suất phi rủi ro theo ngày
stocks <- colnames(returns)[colnames(returns) != "VNI"]
CAPM_models <- lapply(stocks, function(stock) {
Ri_excess <- returns[[stock]] - rf
Rm_excess <- returns$VNI - rf
df <- na.omit(data.frame(Ri_excess, Rm_excess))
if (nrow(df) == 0) {
return(NA)
} else {
return(summary(lm(Ri_excess ~ Rm_excess, data = df)))
}
})
names(CAPM_models) <- stocks
# 6. Trích xuất kết quả từ CAPM_models hợp lệ
valid_CAPM <- CAPM_models[!sapply(CAPM_models, function(x) any(is.na(x)))]
# Tính các đại lượng
beta_vals <- sapply(valid_CAPM, function(x) coef(x)[2])
alpha_vals <- sapply(valid_CAPM, function(x) coef(x)[1])
rsq_vals <- sapply(valid_CAPM, function(x) x$r.squared)
# Lợi suất thực tế trung bình (%/năm)
mean_returns <- colMeans(returns[, names(valid_CAPM)], na.rm = TRUE) * 252 * 100
# Lợi suất kỳ vọng theo CAPM (%/năm)
market_return <- mean(returns$VNI, na.rm = TRUE) * 252
expected_returns <- (rf * 252 + beta_vals * (market_return - rf * 252)) * 100
# Tạo bảng kết quả
CAPM_results <- data.frame(
Stock = names(valid_CAPM),
Alpha = round(alpha_vals * 252 * 100, 4), # annualized %
Beta = round(beta_vals, 4),
R_Squared = round(rsq_vals, 4),
Return_Actual = round(mean_returns, 2),
Return_CAPM = round(expected_returns, 2)
)
# In kết quả
print(CAPM_results)
## Stock Alpha Beta R_Squared Return_Actual Return_CAPM
## BID BID -12.6728 1.0616 0.5384 1.54 14.21
## MBB MBB 17.9963 1.1071 0.5737 32.69 14.69
## ACB ACB -12.7868 0.9688 0.4109 0.44 13.23
## CTG CTG 13.9585 1.2269 0.5865 29.91 15.95
## SHB SHB 2.7264 0.8390 0.3099 14.58 11.86
| Stock |
Mã cổ phiếu |
| Alpha |
Sai số lợi suất (alpha), phản ánh hiệu suất vượt trội nếu >
0 |
| Beta |
Hệ số nhạy cảm với thị trường – càng cao, càng biến động mạnh |
| R_Squared |
Độ phù hợp của mô hình (giải thích bao nhiêu % biến thiên lợi
suất) |
| Return_Actual |
Lợi suất trung bình thực tế (%/năm) |
| Return_CAPM |
Lợi suất kỳ vọng theo mô hình CAPM (%/năm) |
# 6. Lọc mô hình hợp lệ
valid_CAPM <- CAPM_models[!sapply(CAPM_models, function(x) any(is.na(x)))]
betas <- sapply(valid_CAPM, function(x) coef(x)[2])
means <- colMeans(returns[, names(betas)], na.rm = TRUE) * 252
market_return <- mean(returns$VNI, na.rm = TRUE) * 252
expected_return_CAPM <- rf * 252 + betas * (market_return - rf * 252)
4.3. Phân tích đường thị trường chứng khoán
(SML)
Dựa trên kết quả mô hình CAPM, nghiên cứu tiến hành xây dựng đường
thị trường chứng khoán (Security Market Line – SML), thể hiện mối quan
hệ giữa beta và lợi suất kỳ vọng của các cổ phiếu.
Đường SML được xác định theo công thức:
\[
E(R_i) = R_f + \beta_i (E(R_m) - R_f)
\]
Các điểm dữ liệu đại diện cho các cổ phiếu được vẽ trên hệ tọa độ
\((\beta, E(R))\), với đường SML được
thể hiện bằng đường thẳng. So sánh giữa lợi suất trung bình thực tế và
lợi suất theo mô hình giúp xác định:
- Nếu một cổ phiếu nằm trên đường SML: cổ phiếu có
thể đang định giá thấp.
- Nếu nằm dưới đường SML: cổ phiếu có thể đang
định giá cao.
# 7. Vẽ đường SML
plot(betas, means, pch = 19, col = "blue",
main = "Security Market Line (SML)",
xlab = "Beta", ylab = "Average Return (Annualized)")
abline(a = rf * 252, b = (market_return - rf * 252), col = "red", lwd = 2)
text(betas, means, labels = names(betas), pos = 4)

4.4. Phân tích rủi ro danh mục đầu tư
Từ dữ liệu log-return, ma trận hiệp phương sai hàng năm giữa các cổ
phiếu được tính toán, từ đó ước lượng rủi ro tổng thể của danh mục đầu
tư gồm các cổ phiếu trên với tỷ trọng bằng nhau.
# 8. Tính ma trận hiệp phương sai & VaR danh mục
cov_matrix <- cov(returns[, names(betas)]) * 252
w <- rep(1 / length(betas), length(betas)) # tỷ trọng đều
var_p <- as.numeric(t(w) %*% cov_matrix %*% w)
sd_p <- sqrt(var_p)
VaR_95 <- qnorm(0.95) * sd_p
list(Var = var_p, SD = sd_p, VaR_95 = VaR_95)
## $Var
## [1] 0.04128751
##
## $SD
## [1] 0.2031933
##
## $VaR_95
## [1] 0.3342232
- Phương sai của danh mục (annualized): 0.0413
- Độ lệch chuẩn (standard deviation): 20.3%
- Giá trị rủi ro (Value at Risk - VaR) ở mức 95%:
33.42%
Ý nghĩa: Với mức độ tin cậy 95%, danh mục có thể lỗ tối đa
khoảng 33.42% giá trị trong một năm trong điều kiện thị trường
bình thường.
Phân tích cho thấy mô hình CAPM tương thích với dữ liệu cổ phiếu ngân
hàng giai đoạn nghiên cứu. Các hệ số beta cho thấy sự khác biệt trong độ
nhạy với thị trường giữa các mã cổ phiếu. Đồng thời, việc so sánh lợi
suất thực tế và kỳ vọng cùng với VaR giúp nhà đầu tư có cái nhìn đầy đủ
hơn về cả rủi ro và kỳ vọng lợi suất.
---
title: "CT5_DGTSTC2"
author: "Hoàng Quyên"
date: "`r format(Sys.time(), '%H:%M:%S, %d - %m - %Y')`"
output:
  html_document:
    code_folding: hide
    code_download: true
    number_sections: false  
    toc: true
    toc_depth: 5
    toc_float:
      collapsed: true
      smooth_scroll: true
  word_document:
    toc: true
---

```{r setup, include=FALSE}
knitr::opts_chunk$set(echo = TRUE)
```

```{css,echo = FALSE}
h1 {
  font-family: "Times New Roman", Times, serif;
  font-size: 32px;
  font-weight: bold
  }

h2 {
  font-family: "Times New Roman", Times, serif;
  font-size: 28px;
  font-weight: bold;
 
}

h3 {
  font-family: "Times New Roman", Times, serif;
  font-size: 24px;
  font-weight: bold;
  font-style: italic;
}

h4 {font-family: "Times New Roman", Times, serif;
  font-size: 20px;
  font-style: italic}

body {
  font-family: "Times New Roman", Times, serif;
  font-size: 18px;
  
}
p:not(h1):not(h2):not(h3):not(h4):not(h5) {
  text-indent: 2em;}
p {
  text-align: justify;
  }
.tocify-header {
  font-weight: bold;
}

``` 


```{r}
# 1. Cài đặt và nạp gói
library(readxl)
library(urca)
library(PerformanceAnalytics)
```


```{r}
# 2. Đọc dữ liệu
library(readxl)
data <- read_excel("C:/Users/Hoang Quyen/Downloads/DL.xlsx", sheet = "Worksheet")
colnames(data) <- c("Date", "BID", "MBB", "ACB", "CTG", "SHB", "VNI")  # chuẩn hóa tên cột
```


```{r}
# 3. Tính log return
returns <- as.data.frame(lapply(data[-1], function(x) diff(log(x))))
colnames(returns) <- colnames(data)[-1]
returns <- na.omit(returns)
```

## **4. KẾT QUẢ NGHIÊN CỨU VÀ PHÂN TÍCH**

### **4.1. Kiểm định tính dừng chuỗi lợi suất**

Trước khi tiến hành ước lượng mô hình CAPM, nghiên cứu đã sử dụng kiểm định Augmented Dickey-Fuller (ADF) để kiểm tra tính dừng của các chuỗi lợi suất log của năm cổ phiếu: BID, MBB, ACB, CTG, SHB, cùng với chỉ số thị trường VNI.


```{r}
# 4. Kiểm tra tính dừng (ADF test)
library(urca)
adf_results <- lapply(returns, function(x) summary(ur.df(x, type = "drift", selectlags = "AIC")))
# Kiểm tra 1 ví dụ
print(adf_results$BID)
```

Kết quả kiểm định ADF cho cổ phiếu BID như sau:

* Giá trị thống kê kiểm định: **–13.317**
* Ngưỡng tới hạn tại mức ý nghĩa 5%: **–2.87**

Vì –13.317 < –2.87, chuỗi lợi suất log của cổ phiếu BID là **dừng** tại mức ý nghĩa 5%. Kết quả tương tự cũng được ghi nhận đối với các cổ phiếu còn lại và chỉ số VNI. Do đó, tất cả các chuỗi được xác định là dừng và đủ điều kiện đưa vào hồi quy CAPM.

### **4.2. Ước lượng mô hình CAPM**

Mô hình CAPM được ước lượng cho từng cổ phiếu theo phương trình:

$$
R_{it} - R_f = \alpha_i + \beta_i (R_{mt} - R_f) + \varepsilon_{it}
$$

Trong đó:

* $R_{it}$: lợi suất của cổ phiếu $i$
* $R_f$: lãi suất phi rủi ro (giả định 3%/năm, tương đương 0.03/252 mỗi ngày)
* $R_{mt}$: lợi suất của thị trường (sử dụng chỉ số VNI)
* $\beta_i$: hệ số đo lường mức độ nhạy cảm với thị trường

```{r}
# 5. Ước lượng mô hình CAPM
rf <- 0.03 / 252  # lãi suất phi rủi ro theo ngày
stocks <- colnames(returns)[colnames(returns) != "VNI"]

CAPM_models <- lapply(stocks, function(stock) {
  Ri_excess <- returns[[stock]] - rf
  Rm_excess <- returns$VNI - rf
  df <- na.omit(data.frame(Ri_excess, Rm_excess))
  if (nrow(df) == 0) {
    return(NA)
  } else {
    return(summary(lm(Ri_excess ~ Rm_excess, data = df)))
  }
})
names(CAPM_models) <- stocks
```


```{r}
# 6. Trích xuất kết quả từ CAPM_models hợp lệ
valid_CAPM <- CAPM_models[!sapply(CAPM_models, function(x) any(is.na(x)))]

# Tính các đại lượng
beta_vals <- sapply(valid_CAPM, function(x) coef(x)[2])
alpha_vals <- sapply(valid_CAPM, function(x) coef(x)[1])
rsq_vals <- sapply(valid_CAPM, function(x) x$r.squared)

# Lợi suất thực tế trung bình (%/năm)
mean_returns <- colMeans(returns[, names(valid_CAPM)], na.rm = TRUE) * 252 * 100

# Lợi suất kỳ vọng theo CAPM (%/năm)
market_return <- mean(returns$VNI, na.rm = TRUE) * 252
expected_returns <- (rf * 252 + beta_vals * (market_return - rf * 252)) * 100

# Tạo bảng kết quả
CAPM_results <- data.frame(
  Stock = names(valid_CAPM),
  Alpha = round(alpha_vals * 252 * 100, 4),             # annualized %
  Beta = round(beta_vals, 4),
  R_Squared = round(rsq_vals, 4),
  Return_Actual = round(mean_returns, 2),
  Return_CAPM = round(expected_returns, 2)
)

# In kết quả
print(CAPM_results)

```


| Cột                | Ý nghĩa                                                             |
| ------------------ | ------------------------------------------------------------------- |
| **Stock**          | Mã cổ phiếu                                                         |
| **Alpha**          | Sai số lợi suất (alpha), phản ánh hiệu suất vượt trội nếu > 0       |
| **Beta**           | Hệ số nhạy cảm với thị trường – càng cao, càng biến động mạnh       |
| **R\_Squared**     | Độ phù hợp của mô hình (giải thích bao nhiêu % biến thiên lợi suất) |
| **Return\_Actual** | Lợi suất trung bình thực tế (%/năm)                                 |
| **Return\_CAPM**   | Lợi suất kỳ vọng theo mô hình CAPM (%/năm)                          |


```{r}
# 6. Lọc mô hình hợp lệ
valid_CAPM <- CAPM_models[!sapply(CAPM_models, function(x) any(is.na(x)))]
betas <- sapply(valid_CAPM, function(x) coef(x)[2])
means <- colMeans(returns[, names(betas)], na.rm = TRUE) * 252
market_return <- mean(returns$VNI, na.rm = TRUE) * 252
expected_return_CAPM <- rf * 252 + betas * (market_return - rf * 252)
```


### **4.3. Phân tích đường thị trường chứng khoán (SML)**

Dựa trên kết quả mô hình CAPM, nghiên cứu tiến hành xây dựng đường thị trường chứng khoán (Security Market Line – SML), thể hiện mối quan hệ giữa beta và lợi suất kỳ vọng của các cổ phiếu.

Đường SML được xác định theo công thức:

$$
E(R_i) = R_f + \beta_i (E(R_m) - R_f)
$$

Các điểm dữ liệu đại diện cho các cổ phiếu được vẽ trên hệ tọa độ $(\beta, E(R))$, với đường SML được thể hiện bằng đường thẳng. So sánh giữa lợi suất trung bình thực tế và lợi suất theo mô hình giúp xác định:

* Nếu một cổ phiếu nằm **trên đường SML**: cổ phiếu có thể đang **định giá thấp**.
* Nếu nằm **dưới đường SML**: cổ phiếu có thể đang **định giá cao**.

```{r}
# 7. Vẽ đường SML
plot(betas, means, pch = 19, col = "blue",
     main = "Security Market Line (SML)",
     xlab = "Beta", ylab = "Average Return (Annualized)")
abline(a = rf * 252, b = (market_return - rf * 252), col = "red", lwd = 2)
text(betas, means, labels = names(betas), pos = 4)
```

### **4.4. Phân tích rủi ro danh mục đầu tư**

Từ dữ liệu log-return, ma trận hiệp phương sai hàng năm giữa các cổ phiếu được tính toán, từ đó ước lượng rủi ro tổng thể của danh mục đầu tư gồm các cổ phiếu trên với tỷ trọng bằng nhau.


```{r}
# 8. Tính ma trận hiệp phương sai & VaR danh mục
cov_matrix <- cov(returns[, names(betas)]) * 252
w <- rep(1 / length(betas), length(betas))  # tỷ trọng đều
var_p <- as.numeric(t(w) %*% cov_matrix %*% w)
sd_p <- sqrt(var_p)
VaR_95 <- qnorm(0.95) * sd_p

list(Var = var_p, SD = sd_p, VaR_95 = VaR_95)

```

* **Phương sai của danh mục (annualized)**: 0.0413
* **Độ lệch chuẩn (standard deviation)**: 20.3%
* **Giá trị rủi ro (Value at Risk - VaR) ở mức 95%**: **33.42%**

Ý nghĩa: Với mức độ tin cậy 95%, **danh mục có thể lỗ tối đa khoảng 33.42% giá trị trong một năm** trong điều kiện thị trường bình thường.

Phân tích cho thấy mô hình CAPM tương thích với dữ liệu cổ phiếu ngân hàng giai đoạn nghiên cứu. Các hệ số beta cho thấy sự khác biệt trong độ nhạy với thị trường giữa các mã cổ phiếu. Đồng thời, việc so sánh lợi suất thực tế và kỳ vọng cùng với VaR giúp nhà đầu tư có cái nhìn đầy đủ hơn về cả rủi ro và kỳ vọng lợi suất.








