{"cells":[{"cell_type":"markdown","metadata":{"id":"pVk1SP3EaUvn"},"source":["# Classification with Convolutional Neural Network Using PyTorch\n","\n","ในบทเรียนนี้เราจะทดลองสร้าง Convolutional Neural Network ซึ่งเป็น Neural Network ที่ใช้ฟีเจอร์เชิงพื้นที่ (spatial features) ของภาพเพื่อนำมาช่วยใช้ในการเทรนและทำนายประเภทของภาพ โดยในตัวอย่างนี้เราจะใช้ภาพที่เราได้ทดลองใช้ตอนเรียนเรื่อง Principal Component Analysis (PCA) ซึ่งเป็นภาพที่ผู้ป่วยทดลองวาดนาฬิกา มาทดลองสร้างโมเดลเพื่อแบ่งประเภทกัน\n","\n","ชุดข้อมูล: https://github.com/cccnlab/CDT-API-Network"]},{"cell_type":"markdown","metadata":{},"source":["โดยในขั้น ตอนของการสร้าง Convolutional Neural Network (CNN) นั้น ประกอบไปด้วย\n","1. เตรียม Dataset\n","2. กําหนด input layer โดยมีจํานวนมิติเท่ากับข้อมูลที่เตรียมมา\n","3. กําหนด convolutional layer และ pooling layer โดยอาจทําซํ้าได้หลายรอบสําหรับ\n","feature extraction และการลดขนาดมิติของข้อมูลในแต่ละ layer\n","4. กําหนด fully-connected (dense) layer\n","5. กําหนด logics layer หรือ out put layer ซึ่งในกรณีนี้คือการทํา classification ของ\n","ภาพนาฬิกาที่ถูกวาดโดยผู้ป่วยว่าปกติ (normal) หรือ ผิดปกติ (abnormal)"]},{"cell_type":"markdown","metadata":{},"source":["![Basic-architecture-of-CNN.ppm.png]()\n","
\n","[ที่มา](https://www.researchgate.net/figure/Basic-architecture-of-CNN_fig3_335086346)"]},{"cell_type":"markdown","metadata":{},"source":["Convolution คือการดําเนินการทางคณิตศาสตร์ $f * g$ เพื่อแสดงให้เห็นผลของฟังก์ชัน $f$ ที่ เปลี่ยนแปลงไปเมื่อมีฟังก์ชัน $g$ โดยในรูปด้านล่าง สีม่วงคือ $f = [6,2,5,2,1,0,4,3,6,4]$ สีเขียว คือ $g = [6,5,4]$ และสีแดงคือ $f * g = [66,45,44,17,22,32,63,64]$"]},{"cell_type":"markdown","metadata":{},"source":["![Screenshot_2566-07-13_at_20.56.03.png]()"]},{"cell_type":"markdown","metadata":{},"source":["เพื่อเปรียบเทียบให้เห็นภาพชัด Convolution ถูกนํามาใช้ใน classic machine learning อย่างมาก สําหรับ การทํา feature extraction ในรูปแบบ 2D โดยมี $f$ คือ image และ $g$ คือ\n","filter (หรืออาจเรียกว่า kernel) โดยใน convolutional layer สามารถเลือกปรับ ขนาดของ input (padding) หรือ ขนาดการเลื่อนของการแสกน (stride) ได้ มีผลลัพธ์ท่ีได้จากการ คํานวณเรียกว่า Feature Map\n"]},{"cell_type":"markdown","metadata":{},"source":["![1Fw-ehcNBR9byHtho-Rxbtw.png]()"]},{"cell_type":"markdown","metadata":{},"source":["ในกรณี Back propagation จะมีกระบวนการเรียนรู้เพื่อปรับ filter (kernel) ของ convolutionallayerเพื่อให้มีผลลัพธ์ที่สูงที่สุด\n","โดย convolutional layer จะถูกส่งต่อไปยัง pooling layer เพื่อปรับขนาดภาพ และลดมิติของfeaturemapลง เพื่อให้โมเดลไม่จําเป็นต้องเรียนรู้ค่า weight ที่มีจํานวนมากเกินไป โดยมีเทคนิคที่นิยมคือ max pooling และ average pooling\n","โดย pooling layer สุดท้ายจะถูกรวม และเปลี่ยนรูปไปเป็น fully-connected layer เพื่อคํานวนเป็นoutputlayerใช้สําหรับ classification ภาพนาฬิกาที่ได้ยกนํามาเป็นตัว อย่างในบท นี้"]},{"cell_type":"markdown","metadata":{},"source":["## Download the Data"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"elapsed":1155,"status":"ok","timestamp":1688465945839,"user":{"displayName":"Korrawiz Chotayapa","userId":"12619977060284176060"},"user_tz":-420},"id":"PMI86_jqaUvq","outputId":"646ab883-3dcd-4e8e-e857-1c90d3c07305"},"outputs":[{"name":"stdout","output_type":"stream","text":["--2023-07-04 10:19:02-- https://github.com/ichatnun/brainCodeCamp/raw/main/Fundamentals/DimensionalityReduction/data/CDT/clock_images.pickle\n","Resolving github.com (github.com)... 140.82.112.3\n","Connecting to github.com (github.com)|140.82.112.3|:443... connected.\n","HTTP request sent, awaiting response... 302 Found\n","Location: https://raw.githubusercontent.com/ichatnun/brainCodeCamp/main/Fundamentals/DimensionalityReduction/data/CDT/clock_images.pickle [following]\n","--2023-07-04 10:19:03-- https://raw.githubusercontent.com/ichatnun/brainCodeCamp/main/Fundamentals/DimensionalityReduction/data/CDT/clock_images.pickle\n","Resolving raw.githubusercontent.com (raw.githubusercontent.com)... 185.199.111.133, 185.199.109.133, 185.199.110.133, ...\n","Connecting to raw.githubusercontent.com (raw.githubusercontent.com)|185.199.111.133|:443... connected.\n","HTTP request sent, awaiting response... 200 OK\n","Length: 1106083 (1.1M) [application/octet-stream]\n","Saving to: ‘images.pickle’\n","\n","images.pickle 100%[===================>] 1.05M --.-KB/s in 0.05s \n","\n","2023-07-04 10:19:03 (20.2 MB/s) - ‘images.pickle’ saved [1106083/1106083]\n","\n","--2023-07-04 10:19:03-- https://github.com/ichatnun/brainCodeCamp/raw/main/Fundamentals/DimensionalityReduction/data/CDT/labels.pickle\n","Resolving github.com (github.com)... 140.82.113.4\n","Connecting to github.com (github.com)|140.82.113.4|:443... connected.\n","HTTP request sent, awaiting response... 302 Found\n","Location: https://raw.githubusercontent.com/ichatnun/brainCodeCamp/main/Fundamentals/DimensionalityReduction/data/CDT/labels.pickle [following]\n","--2023-07-04 10:19:03-- https://raw.githubusercontent.com/ichatnun/brainCodeCamp/main/Fundamentals/DimensionalityReduction/data/CDT/labels.pickle\n","Resolving raw.githubusercontent.com (raw.githubusercontent.com)... 185.199.108.133, 185.199.109.133, 185.199.110.133, ...\n","Connecting to raw.githubusercontent.com (raw.githubusercontent.com)|185.199.108.133|:443... connected.\n","HTTP request sent, awaiting response... 200 OK\n","Length: 630 [application/octet-stream]\n","Saving to: ‘labels.pickle’\n","\n","labels.pickle 100%[===================>] 630 --.-KB/s in 0s \n","\n","2023-07-04 10:19:04 (55.1 MB/s) - ‘labels.pickle’ saved [630/630]\n","\n"]}],"source":["# Download the dataset which contrains images and labels of the Clock Drawing Test (CDT)\n","\n","!wget -O images.pickle https://github.com/ichatnun/brainCodeCamp2023/raw/main/Fundamentals/DimensionalityReduction/data/CDT/clock_images.pickle\n","!wget -O labels.pickle https://github.com/ichatnun/brainCodeCamp2023/raw/main/Fundamentals/DimensionalityReduction/data/CDT/labels.pickle"]},{"cell_type":"code","execution_count":1,"metadata":{"id":"k-ILPP8ZaUvt"},"outputs":[],"source":["# import libraries\n","import torch\n","import torch.nn as nn\n","import torch.nn.functional as F\n","import torch.optim as optim # module ที่เก็บ optimizer ต่างๆ ที่เราจะเอามาใช้ในการ train model\n","from torch.utils.data import Dataset, DataLoader # โหลด dataset มาใช้งาน\n","import torchvision # module สำหรับ computer vision ของ PyTorch\n","import torchvision.transforms as transforms # module สำหรับ transform รูปภาพ เช่น การปรับขนาดรูปภาพ การทำ data augmentation\n","from sklearn.model_selection import train_test_split # ใช้สำหรับแบ่งข้อมูลเป็น train/validation/test set\n","from glob import glob # ใช้สำหรับหา path ของไฟล์ในโฟลเดอร์\n","from PIL import Image # ใช้สำหรับโหลดรูปภาพ และแปลงรูปภาพให้เป็น array\n","from matplotlib import pyplot as plt # ใช้สำหรับ plot รูปภาพ\n","import pickle # ใช้สำหรับ save และ load ข้อมูล\n","import pandas as pd # ใช้สำหรับจัดการข้อมูลในรูปแบบของตาราง\n","import numpy as np # ใช้สำหรับจัดการข้อมูลในรูปแบบของ array\n","import os # ใช้สำหรับจัดการไฟล์และโฟลเดอร์"]},{"cell_type":"markdown","metadata":{"id":"mVUmiMvDaUvt"},"source":["## Prepare Data\n","\n","การเตรียมข้อมูลทำได้โดยการอ่านข้อมูลภาพจาก `images.pickle` และประเภทของภาพจาก `labels.pickle` เพื่อใช้ในการเตรียมข้อมูล จากนั้นเปลี่ยนเป็น dataframe"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"vDNsj1yjaUvu"},"outputs":[],"source":["# โหลดข้อมูลไฟล์ .pickle ที่เราได้ทำการ save ไว้ก่อนหน้านี้ และโหลดมาผ่าน wget จาก github\n","with open(\"images.pickle\", \"rb\") as handle:\n"," images = pickle.load(handle)\n","with open(\"labels.pickle\", \"rb\") as handle:\n"," labels = pickle.load(handle)"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"C0CyAxm6aUvu"},"outputs":[],"source":["# แปลงรูปภาพให้เป็น RGB array และเก็บไว้ใน list\n","list_images = []\n","for image in images:\n"," image = Image.fromarray(image) # Convert array to PIL image\n"," image = image.convert(\"RGB\") # Convert to RGB mode\n"," list_images.append(image)"]},{"cell_type":"markdown","metadata":{"id":"65F202JBaUvv"},"source":["สร้าง dataframe ที่เก็บข้อมูลที่จำเป็นต้องใช้ โดยมี column ดังนี้\n","- `Images` : ภาพที่ใช้ train\n","- `encoded_labels` : ค่า label โดย 0 คือ ปกติ และ 1 คือ ผิดปกติ"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"elapsed":656,"status":"ok","timestamp":1688465954893,"user":{"displayName":"Korrawiz Chotayapa","userId":"12619977060284176060"},"user_tz":-420},"id":"FTOeHGI-aUvv","outputId":"8fc4a0b1-f088-44fa-8ba3-69862da901a5"},"outputs":[{"name":"stderr","output_type":"stream","text":[":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n",":4: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)\n"]}],"source":["df_data = pd.DataFrame(columns=[\"Images\", \"encoded_labels\"]) # สร้าง dataframe เพื่อเก็บข้อมูลรูปภาพและ label ที่เป็นตัวเลข\n","\n","# นำข้อมูลรูปภาพและ label มาเก็บไว้ใน dataframe\n","for img, label in zip(list_images, labels):\n"," df_data = df_data.append({\"Images\": img, \"encoded_labels\": label}, ignore_index=True)"]},{"cell_type":"markdown","metadata":{"id":"gPyZ2YYJaUvv"},"source":["## Split Data and Prepare DataLoaders\n","\n","แบ่งข้อมูลเป็น train set (90%) และ test set (10%) โดยใช้ `train_test_split` จาก `sklearn.model_selection`"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"elapsed":20,"status":"ok","timestamp":1688465954894,"user":{"displayName":"Korrawiz Chotayapa","userId":"12619977060284176060"},"user_tz":-420},"id":"NGTw-NzUaUvw","outputId":"fde46b8f-87c0-42d1-a11d-4f991115a838"},"outputs":[{"data":{"text/plain":["(54, 6)"]},"execution_count":6,"metadata":{},"output_type":"execute_result"}],"source":["# แบ่งข้อมูลเป็น train/validation/test set โดยใช้ train/validation set อยู่ในสัดส่วน 90:10\n","train_data, test_data = train_test_split(df_data, test_size=0.1, random_state=42)\n","len(train_data), len(test_data)"]},{"cell_type":"markdown","metadata":{"id":"roaNKSnnaUvw"},"source":["สร้างชุดข้อมูล dataset จาก training และ testing โดยใช้ `torch.utils.data.Dataset` and `torch.utils.data.DataLoader` โดย\n","\n","- `Dataset` เป็น class ที่ใช้ในการกำหนดชุดข้อมูลของเรา โดยปกติจะประกอบด้วย 3 ฟังก์ชันหลักคือ `__init__` ใช้เพื่อประกาศค่าต่างๆ, `__len__` เป็นฟังก์ชันที่ให้ขนาดของชุดข้อมูล, และ `__getitem__` ที่มี index เป็น input และให้ `img` และ `label` ของ index นั้นๆออกมา\n","- `DataLoader` ส่วนมากจะใช้ร่วมกับ `Dataset` ที่เราสร้างขึ้น โดยเมื่อเราประกาศ `Dataset` แล้วจะสามารถนำ `DataLoader` มาครอบ `Dataset` เพื่อใช้ในการดึงข้อมูลออกมาเป็น batch ได้"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"hvHga8XtaUvw"},"outputs":[],"source":["# สร้างขั้นตอนสำหรับการแปลงข้อมูลรูปภาพรวมเป็นตัวแปร transform โดย\n","# 1. ทำการแปลงรูปภาพให้เป็น tensor\n","# 2. ทำการ normalize รูปภาพด้วย mean และ standard deviation ของชุดข้อมูล\n","transform = transforms.Compose(\n"," [transforms.ToTensor(), transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))]\n",")\n","\n","base_path = \"CDT_dataset\" # กำหนด path ของ dataset ที่เราจะเก็บไว้\n","\n","# สร้าง class สำหรับการโหลด dataset ที่เราสร้างขึ้นมา\n","class CDT_dataset(Dataset): \n"," def __init__(self, img_data, base_path, transform):\n"," self.img_data = img_data\n"," self.base_path = base_path\n"," self.transform = transform\n","\n"," def __len__(self): # สร้าง function สำหรับ return จำนวนข้อมูลใน dataset ด้วยการใช้ len(...)\n"," return len(self.img_data)\n","\n"," def __getitem__(self, index): # สร้าง function สำหรับ return ข้อมูลและ label ของข้อมูลใน dataset ด้วยการใช้ index\n"," # โหลดรูปภาพจาก path ที่เก็บไว้\n"," img = self.img_data.iloc[index][\"Images\"]\n"," # ลดขนาดภาพให้มีขนาด 32x32 เพื่อใช้ในการทำนาย\n"," img = img.resize((32, 32)) \n"," # ถ้ามีการสร้างขั้นตอนการแปลงข้อมูลรูปภาพ ให้ทำการแปลงข้อมูลรูปภาพด้วย\n"," if self.transform is not None:\n"," img = self.transform(img)\n"," # สร้าง label ของข้อมูล\n"," label = self.img_data.iloc[index][\"encoded_labels\"]\n"," # ส่งข้อมูลรูปภาพและ label กลับไป\n"," return img, label\n","\n","\n","train_dataset = CDT_dataset(train_data, base_path, transform) # สร้าง train dataset\n","test_dataset = CDT_dataset(test_data, base_path, transform) # สร้าง test dataset"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"Zg0tWa2jaUvx"},"outputs":[],"source":["# สร้าง train loader และ test loader โดยกำหนดให้มี batch size เท่ากับ 4 และ shuffle ข้อมูล\n","# โดยที่ DataLoader คือ class สำหรับการโหลดข้อมูลในรูปแบบของ batch\n","train_loader = DataLoader(train_dataset, batch_size=4, shuffle=True) # สร้าง train loader\n","test_loader = DataLoader(test_dataset, batch_size=4, shuffle=True) # สร้าง test loader"]},{"cell_type":"markdown","metadata":{"id":"jiBIfq7haUvx"},"source":["เขียนฟังก์ชันเพื่อยกตัวอย่างภาพที่ใช้ train โดยใช้ `next(iter(train_loader))` เพื่อดึงตัวอย่างออกมา 1 batch ในด้านบนเรากำหนดให้ `batch_size` มีค่าเท่ากับ 4"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/","height":210},"executionInfo":{"elapsed":12,"status":"ok","timestamp":1688465954897,"user":{"displayName":"Korrawiz Chotayapa","userId":"12619977060284176060"},"user_tz":-420},"id":"BkCBEg1daUvx","outputId":"bf4792c8-161d-430b-a27d-4c6f0f140f12"},"outputs":[{"data":{"image/png":"iVBORw0KGgoAAAANSUhEUgAAAh8AAACwCAYAAACviAzDAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAs60lEQVR4nO2de1TVVfr/33gBr2DiCCKiaBbeMxVFLS0xMvOSdndGu8y0nKC8rFVqTfadGgdnZq3RaszWNGVNZZZNalnqGN5LRPGSSqIlqalgZohXNNm/P+bnZz37OXLgIHwOnPN+rXXW2pvn8Dmb/bmczX6e5/2EGGMMCCGEEEJcopa/B0AIIYSQ4IKLD0IIIYS4ChcfhBBCCHEVLj4IIYQQ4ipcfBBCCCHEVbj4IIQQQoircPFBCCGEEFfh4oMQQgghrsLFByGEEEJchYsPQgghhLhKlS0+5syZgzZt2qBevXro3bs3srKyquqjCCGEEFKDCKmK2i4ffPABxo4di9deew29e/fG7NmzsXDhQuTm5qJ58+Zef7ekpARHjhxB48aNERISUtlDI4QQQkgVYIzBqVOnEBMTg1q1ytjbMFVAYmKiSU1NdfqXLl0yMTExJj09vczfPXTokAHAF1988cUXX3zVwNehQ4fK/K6vg0rmwoULyM7OxrRp05yf1apVC8nJydi4caPH+4uLi1FcXOz0zf/fiJk0aRLCwsIqe3iEEEIIqQKKi4sxa9YsNG7cuMz3Vvri4/jx47h06RKioqKsn0dFRWHPnj0e709PT8cf//hHj5+HhYVx8UEIIYTUMMoTMuH3bJdp06bh5MmTzuvQoUP+HhIhhBBCqpBK3/lo1qwZateujYKCAuvnBQUFiI6O9ng/dzgIIYSQ4KLSdz5CQ0PRo0cPZGRkOD8rKSlBRkYGkpKSKvvjCCGEEFLDqPSdDwCYPHkyxo0bh549eyIxMRGzZ8/GmTNn8PDDD1/1sf/v//7v6gdIqgXeziXPc+BQ1rnkuQ4ceE8HB5VxLqtk8XHffffhxx9/xPTp05Gfn48bbrgBy5cv9whCJYQQQkjwUSWLDwBIS0tDWlpaVR2eEEIIITUUv2e7EEIIISS4qLKdD+KJEUr2RqnalylFS6o1Fy9edNoXLlywbFJET9saNGjgtOvVq2fZ9DUh+7xeCCE1GT7BCCGEEOIqXHwQQgghxFXodnGR8+fPO+21a9datvr161v96667zmm3aNGiagdGyuSXX36x+jt37rT6V6pPdCWbPpdS0VdLEtetW9fqSzE+7aKJiIhw2tdcc43nH0BK5dKlS1b/6NGjTjsmJsayVYW7q6SkxOpL11xoaGiVfz4h/oBXMiGEEEJchYsPQgghhLgKFx+EEEIIcRXGfLiIjOvQKZdffvml1e/SpYsrYyKlc+7cOae9bds2yyZTZAGga9euTlvHEGRmZpZq+/777512t27dLFubNm2svowdOXXqlGU7cOCA0/75558tW1xcnNWvU4e3vTf27NnjtHVs1rBhw5x2eHi4ZZOxG7qwZl5entWX9tq1a1u206dPO20d26OvHxkX1KhRI8smC3k2a9bMsmm16fKUQA9mTp486bR15XUZY6XPpZznvXv3WrZly5ZZ/b59+zrtVq1aWTZ5/nQsWE2FOx+EEEIIcRUuPgghhBDiKlx8EEIIIcRVAsr5q/UVzpw5Y/UbNmzotP3h4/zhhx+cth7b9ddfb/V3797ttKWGA2DHF/iC1hOQcxBIPl/9d0q86SToc7J9+3an3bJlS8um4yjkZ+q5lP7ayMhIy9a2bVunLXVgAE+NB9lv3LixZZPxB8eOHbNs2tcs/cn6OMGI9tPfeuutTlvehwCwaNEip925c2fLJuM4ZLwQALRv397qy/tdxw/p55hExv0AduyPvn5l7I/+O6RmDAAMHz7caQejlkhRUZHV37Rpk9U/e/as05bxHwBw/Phxp61jawYMGOC0c3JyLNvBgwetfuvWrZ22jjGTx5UaUADQoUMHq19TYrqC7yojhBBCiF/h4oMQQgghrlIz9mfKid5u/+6776x+p06dnHZVbU3J7blXXnnFssmtzwcffNCy6S245cuXO21ZMRUAxo8f77R1OqbeMpUpvfo4cvtQuwP0tmygIs+XToeUrhW9Lau3v2X6mze3i06Tk2mV2gWir2dv2+H79u1z2lpeXbp2AOCnn35y2lo2ntLs9jzrlPddu3Y57blz51q2p556ymlrN6pGp8xKtBtIol00FT1f7777rtXfsWOH0+7evXuFjlnTkCnV33zzjWXTz1WZtvzJJ59YtsTERKcdGxtr2eT1ou/95ORkq3/HHXc4bf2sli49ef8CwNKlS61+TXGhVd+REUIIISQg4eKDEEIIIa7CxQchhBBCXCWg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","text/plain":["
"]},"metadata":{},"output_type":"display_data"},{"name":"stdout","output_type":"stream","text":["abnormal normal abnormal abnormal\n"]}],"source":["# สร้างฟังก์ชันสำหรับการแสดงรูปภาพและ label ของข้อมูลใน dataset\n","def imshow(img):\n"," img = img / 2 + 0.5\n"," npimg = img.numpy()\n"," plt.imshow(np.transpose(npimg, (1, 2, 0)))\n"," plt.show()\n","\n","images, labels = next(iter(train_loader)) # ดึงข้อมูลจาก train loader มาเก็บไว้ในตัวแปร images และ labels\n","# iter(...) คือ function สำหรับการสร้าง iterator ของ train loader\n","# next(...) คือ function สำหรับการดึงข้อมูลจาก iterator ของ train loader มาเก็บไว้ในตัวแปร images และ labels\n","\n","SD_type = {0: \"normal\", 1: \"abnormal\"} # สร้าง dictionary สำหรับแปลง label ที่เป็นตัวเลขเป็นชื่อของ class\n","\n","imshow(torchvision.utils.make_grid(images)) # แสดงรูปภาพที่อยู่ในตัวแปร images\n","classes = (\"normal\", \"abnormal\") # กำหนดชื่อของ class ที่เรามี\n","print(\" \".join(f\"{classes[labels[j]]:5s}\" for j in range(4))) # แสดง label ของข้อมูลที่อยู่ในตัวแปร labels"]},{"cell_type":"markdown","metadata":{"id":"TSx2h4KHaUvx"},"source":["## Introduction to Convolution\n","convolution เป็นวิธีการนึ่งที่ที่ดูการเปลี่ยนแปลงของฟังก์ชั่นหนึ่ง เมื่อมีอีกฟังก์ชั่นเข้ามา ใน deep-learning convolution ถูกนำมาใช้อย่างแพร่หลายในงานที่เกี่ยวกับรูปภาพเพื่อดึงข้อมูลเชิงพื้นที่ (spatial feature) ออกมาจากรูปภาพ โดย pytorch จะใช้คำสั่ง `nn.Conv2d` ในการทำ convolution\n","โดย parameters ที่สำคัญจะประกอบไปด้วย\n","* `in_channels`: input image channel\n","* `out_channels`: output channel หลังจาก convolve แล้ว\n","* `kernel_size`: ขนาดของ filter ที่นำมา convolve กับรูปของเรา\n","* `stride`: filter จะเคลื่อนที่ทีละกี่ pixel ค่า default = 1\n","* `padding`: padding รูปภาพ ค่า default = 0\n","\n","ภาพที่ได้หลังจากการ convolution อาจจะมีขนาด/depthเปลี่ยนไปขึ้นอยู่กับ parametersซึ่งเราสามารถคำนวนขนาดของภาพที่ออกมาได้ดังนี้\n","\n","`output_size = ((input_size - kernel_size + 2 * padding) / stride) + 1`"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"elapsed":394,"status":"ok","timestamp":1688465955282,"user":{"displayName":"Korrawiz Chotayapa","userId":"12619977060284176060"},"user_tz":-420},"id":"_czkGldbaUvy","outputId":"30328401-fbb4-4a1a-dcb3-363500d24efd"},"outputs":[{"data":{"text/plain":["torch.Size([1, 3, 5, 5])"]},"execution_count":10,"metadata":{},"output_type":"execute_result"}],"source":["# สมมุติว่ามีรูปภาพ 1 ภาพ ขนาด 5x5 และมี 3 ช่องสี\n","input = torch.randn(1, 3, 5, 5)\n","input.shape"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"elapsed":27,"status":"ok","timestamp":1688465955297,"user":{"displayName":"Korrawiz Chotayapa","userId":"12619977060284176060"},"user_tz":-420},"id":"n-UVz5IuaUvy","outputId":"20bd6819-c464-4c54-d3ed-83aa981651cd"},"outputs":[{"data":{"text/plain":["torch.Size([1, 16, 3, 3])"]},"execution_count":11,"metadata":{},"output_type":"execute_result"}],"source":["conv = nn.Conv2d(in_channels=3, # จำนวนช่องสีของภาพ\n"," out_channels=16, # จำนวนช่องสีของภาพที่เราต้องการให้เป็น output\n"," kernel_size=3, # ขนาดของ filter\n"," stride=1, # ขนาดของการเคลื่อนที่ของ filter\n"," padding=0) # จำนวนช่องสีที่เราต้องการให้เพิ่มเติมเพื่อให้ขนาดของภาพเท่าเดิม\n","# output size = (5 - 3 + 2*0)/1 + 1 = 3\n","output = conv(input) # ทำการคำนวนด้วยการป้อน input ลงไปใน conv ซึ่งเป็น object ที่เราสร้างไว้ให้ทำหน้าที่เป็น convolution ในสองมิติ \n","output.shape # depth ของภาพจะเพิ่มเป็น 16และ dimension ของภาพจะเท่ากับการคำนวนคือ 3"]},{"cell_type":"markdown","metadata":{"id":"XkwGhz4xaUvy"},"source":["## Train the Model\n","\n","สร้าง model โดยใช้ `torch.nn.Module` ซึ่งเป็นตัวแบบของ Neural Network model ใน pytorch โดยมี layer ดังนี้\n","- Convolutional layer (`nn.Conv2d`) ซึ่งจะรับค่าทั้งหมด 3 ค่าคือ input depth, output depth และ kernel size ยกตัวอย่างเช่นในเลเยอร์แรก `nn.Conv2d(3, 6, 5)` จะรับ input depth ของภาพเท่ากับ 3 นั่นก็คือ RGB channel นั่นเอง และ output ออกมาเป็นภาพที่มี depth เท่ากับ 6\n","- จากนั้นก็จะส่งต่อไปให้ `nn.MaxPool2d(2, 2)` ซึ่งมี kernel size เท่ากับ 2 และ stride เท่ากับ 2 โดยจะลดขนาดของภาพเป็นครึ่งหนึ่งนั่นเอง (แต่ depth เท่าเดิม)"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"EDT0xGwNaUvy"},"outputs":[],"source":["class Net(nn.Module): # สร้าง class สำหรับการสร้างโมเดลจาก nn.Module\n"," def __init__(self): # สร้าง constructor ของ class\n"," super().__init__() # เรียกใช้ constructor ของ nn.Module\n"," self.conv1 = nn.Conv2d(3, 6, 5) # สร้าง convolution layer แรก โดยมี in_channels เท่ากับ 3 และ out_channels เท่ากับ 6 และมี kernel_size เท่ากับ 5\n"," self.pool = nn.MaxPool2d(2, 2) # สร้าง pooling layer โดยมี kernel_size เท่ากับ 2 และ stride เท่ากับ 2\n"," self.conv2 = nn.Conv2d(6, 16, 5) # สร้าง convolution layer ที่สอง โดยมี in_channels เท่ากับ 6 และ out_channels เท่ากับ 16 และมี kernel_size เท่ากับ 5\n"," self.fc1 = nn.Linear(16 * 5 * 5, 120) # สร้าง fully connected layer แรก โดยมี in_features เท่ากับ 16 * 5 * 5 และ out_features เท่ากับ 120\n"," self.fc2 = nn.Linear(120, 84) # สร้าง fully connected layer ที่สอง โดยมี in_features เท่ากับ 120 และ out_features เท่ากับ 84\n"," self.fc3 = nn.Linear(84, 10) # สร้าง fully connected layer ที่สาม โดยมี in_features เท่ากับ 84 และ out_features เท่ากับ 10\n","\n"," def forward(self, x): # สร้าง method สำหรับการทำ forward propagation\n"," x = self.pool(F.relu(self.conv1(x))) # ทำการ convolution และ pooling และ activation function (ReLu) ใน convolution layer แรก\n"," x = self.pool(F.relu(self.conv2(x))) # ทำการ convolution และ pooling และ activation function (ReLu) ใน convolution layer ที่สอง\n"," x = torch.flatten(x, 1) # ทำการ flatten ข้อมูลใน x ให้เหลือแค่ batch dimension\n"," x = F.relu(self.fc1(x)) # ทำการแปลงผ่าน fully connected และ activation function (ReLu) ใน fully connected layer แรก\n"," x = F.relu(self.fc2(x)) # ทำการแปลงผ่าน fully connected และ activation function (ReLu) ใน fully connected layer ที่สอง\n"," x = self.fc3(x) # ทำการแปลงผ่าน fully connected ใน fully connected layer สุดท้าย\n"," return x # ส่งค่า x กลับ\n","\n","\n","model = Net() # สร้าง object ของ class Net"]},{"cell_type":"markdown","metadata":{},"source":["**เราทราบได้อย่างไรว่า class ที่เราสร้างขึ้น มี network layer เชื่อมต่อกันอย่างไร**\n","
\n"," \n"," เฉลย\n"," \n"," เราสามารถดูได้จากขั้นตอนการแปลง input ผ่าน method `forward` ที่อยู่ใน class ได้ โดยจากตัวอย่างด้านบน จะพบว่า network ของเรามีการเชื่อมต่อกันแบบเรียงลำดับดังนี้\n","
    \n","
  1. conv1 + ReLu
  2. \n","
  3. conv2 + ReLu
  4. \n","
  5. fc1 + ReLu
  6. \n","
  7. fc2 + ReLu
  8. \n","
  9. fc3
  10. \n","
\n"," โดยที่ activation function นั้นจะมีหรือไม่มีหลังจากที่ทำการคำนวณใน layer แต่ละอันก็ได้\n","
"]},{"cell_type":"markdown","metadata":{"id":"yL6TjxkXaUvy"},"source":["กำหนด `hyperparameters` ที่จำเป็นต้องใช้ ได้แก่ loss function ซึ่งคือ `CrossEntropyLoss`, optimizer ซึ่งคือ `Adam` ที่รับพารามิเตอร์ของโมเดล จากนั้นเทรนโมเดลโดยใส่ชุดข้อมูลเข้าไปในโมเดล และทำการอัพเดทโมเดลด้วยการคำนวณ gradient (`loss.backward()`) และอัพเดทพารามิเตอร์ของโมเดล (`optimizer.step()`)"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"TsUTo5x0aUvz"},"outputs":[],"source":["# Define a Loss function and optimizer\n","\n","criterion = nn.CrossEntropyLoss() # กำหนด loss function ที่ใช้ในการคำนวนค่า loss เป็นค่า Cross Entropy Loss ซึ่งหมายถึง การคำนวนค่า loss โดยใช้ค่าความแตกต่างของค่าความน่าจะเป็นของคลาสที่เราต้องการกับค่าความน่าจะเป็นของคลาสอื่นๆ\n","optimizer = optim.Adam(model.parameters(), lr=0.001) # กำหนด optimizer ที่ใช้ในการปรับค่า parameter ของโมเดลให้เป็น optimizer แบบ Adam ซึ่งเป็น optimizer ที่ใช้ในการปรับค่า parameter โดยใช้การคำนวนค่า gradient ของ parameter แต่ละตัวเพื่อใช้ในการปรับค่า parameter โดยใช้ learning rate เท่ากับ 0.001"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"elapsed":14006,"status":"ok","timestamp":1688465969279,"user":{"displayName":"Korrawiz Chotayapa","userId":"12619977060284176060"},"user_tz":-420},"id":"4tXoLWriaUvz","outputId":"5791df0c-64e4-41ba-971b-ae9caa12876a"},"outputs":[{"name":"stdout","output_type":"stream","text":["[epoch: 1 ] loss: 20.075\n","[epoch: 2 ] loss: 8.571\n","[epoch: 3 ] loss: 9.308\n","[epoch: 4 ] loss: 7.529\n","[epoch: 5 ] loss: 6.981\n","[epoch: 6 ] loss: 7.147\n","[epoch: 7 ] loss: 6.895\n","[epoch: 8 ] loss: 7.352\n","[epoch: 9 ] loss: 6.986\n","[epoch: 10 ] loss: 7.029\n","[epoch: 11 ] loss: 7.027\n","[epoch: 12 ] loss: 7.004\n","[epoch: 13 ] loss: 7.241\n","[epoch: 14 ] loss: 7.126\n","[epoch: 15 ] loss: 6.847\n","[epoch: 16 ] loss: 6.681\n","[epoch: 17 ] loss: 6.850\n","[epoch: 18 ] loss: 6.481\n","[epoch: 19 ] loss: 6.609\n","[epoch: 20 ] loss: 5.753\n","[epoch: 21 ] loss: 5.363\n","[epoch: 22 ] loss: 3.350\n","[epoch: 23 ] loss: 3.134\n","[epoch: 24 ] loss: 1.701\n","[epoch: 25 ] loss: 2.908\n","[epoch: 26 ] loss: 1.856\n","[epoch: 27 ] loss: 1.559\n","[epoch: 28 ] loss: 0.711\n","[epoch: 29 ] loss: 0.615\n","[epoch: 30 ] loss: 0.605\n","[epoch: 31 ] loss: 1.207\n","[epoch: 32 ] loss: 0.896\n","[epoch: 33 ] loss: 0.525\n","[epoch: 34 ] loss: 0.756\n","[epoch: 35 ] loss: 0.229\n","[epoch: 36 ] loss: 0.151\n","[epoch: 37 ] loss: 0.324\n","[epoch: 38 ] loss: 0.378\n","[epoch: 39 ] loss: 0.846\n","[epoch: 40 ] loss: 0.418\n","[epoch: 41 ] loss: 0.150\n","[epoch: 42 ] loss: 0.042\n","[epoch: 43 ] loss: 0.023\n","[epoch: 44 ] loss: 0.033\n","[epoch: 45 ] loss: 0.032\n","[epoch: 46 ] loss: 0.022\n","[epoch: 47 ] loss: 0.028\n","[epoch: 48 ] loss: 0.022\n","[epoch: 49 ] loss: 0.017\n","[epoch: 50 ] loss: 0.021\n","[epoch: 51 ] loss: 0.014\n","[epoch: 52 ] loss: 0.017\n","[epoch: 53 ] loss: 0.018\n","[epoch: 54 ] loss: 0.011\n","[epoch: 55 ] loss: 0.014\n","[epoch: 56 ] loss: 0.014\n","[epoch: 57 ] loss: 0.007\n","[epoch: 58 ] loss: 0.013\n","[epoch: 59 ] loss: 0.008\n","[epoch: 60 ] loss: 0.011\n","[epoch: 61 ] loss: 0.006\n","[epoch: 62 ] loss: 0.011\n","[epoch: 63 ] loss: 0.007\n","[epoch: 64 ] loss: 0.010\n","[epoch: 65 ] loss: 0.006\n","[epoch: 66 ] loss: 0.006\n","[epoch: 67 ] loss: 0.007\n","[epoch: 68 ] loss: 0.008\n","[epoch: 69 ] loss: 0.007\n","[epoch: 70 ] loss: 0.007\n","[epoch: 71 ] loss: 0.007\n","[epoch: 72 ] loss: 0.007\n","[epoch: 73 ] loss: 0.007\n","[epoch: 74 ] loss: 0.002\n","[epoch: 75 ] loss: 0.002\n","[epoch: 76 ] loss: 0.006\n","[epoch: 77 ] loss: 0.004\n","[epoch: 78 ] loss: 0.006\n","[epoch: 79 ] loss: 0.005\n","[epoch: 80 ] loss: 0.005\n","[epoch: 81 ] loss: 0.005\n","[epoch: 82 ] loss: 0.003\n","[epoch: 83 ] loss: 0.003\n","[epoch: 84 ] loss: 0.003\n","[epoch: 85 ] loss: 0.003\n","[epoch: 86 ] loss: 0.002\n","[epoch: 87 ] loss: 0.003\n","[epoch: 88 ] loss: 0.002\n","[epoch: 89 ] loss: 0.004\n","[epoch: 90 ] loss: 0.002\n","[epoch: 91 ] loss: 0.003\n","[epoch: 92 ] loss: 0.002\n","[epoch: 93 ] loss: 0.003\n","[epoch: 94 ] loss: 0.003\n","[epoch: 95 ] loss: 0.003\n","[epoch: 96 ] loss: 0.003\n","[epoch: 97 ] loss: 0.002\n","[epoch: 98 ] loss: 0.001\n","[epoch: 99 ] loss: 0.002\n","[epoch: 100 ] loss: 0.002\n","Finished Training\n"]}],"source":["for epoch in range(100): # ทำการ train model โดยใช้จำนวน epoch เท่ากับ 100\n"," running_loss = 0.0 # กำหนดค่าเริ่มต้นของ running loss เท่ากับ 0\n"," for i, data in enumerate(train_loader, 0): # ทำการวน loop ในแต่ละ batch ของ train_loader\n"," inputs, labels = data # ดึงข้อมูลและ label จาก data\n"," optimizer.zero_grad() # ทำการปรับค่า gradient เป็น 0 ก่อนการคำนวนค่า gradient ของ parameter แต่ละตัว เนื่องจากการคำนวณ gradient ด้วย backpropagation จะเก็บค่า gradient ไว้ในตัวแปรเดียวกันเรื่อยๆ จนกว่าจะมีการเรียกใช้ method zero_grad() เพื่อทำการเคลียร์ค่า gradient ในตัวแปรนั้น\n"," outputs = model(inputs) # ทำการ forward propagation โดยใช้ข้อมูลใน inputs ที่ได้จาก data ที่วน loop อยู่\n"," loss = criterion(outputs, labels) # ทำการคำนวณค่า loss โดยใช้ค่า output ที่ได้จากการทำ forward propagation กับ label ที่ได้จาก data ที่วน loop อยู่\n"," loss.backward() # ทำการคำนวณค่า gradient ของ parameter แต่ละตัวโดยใช้ backpropagation\n"," optimizer.step() # ทำการปรับค่า parameter ของโมเดลด้วยค่า gradient ที่คำนวณได้\n"," running_loss += loss.item() # ทำการบวกค่า loss ในแต่ละ batch เข้ากับ running loss เพื่อใช้ในการคำนวณค่า loss ในแต่ละ epoch\n"," if i % 10 == 9: # ทำการ print ค่า loss ในแต่ละ batch ทุกๆ 10 batch\n"," print(f\"[epoch: {epoch + 1} ] loss: {running_loss / 1:.3f}\")\n"," running_loss = 0.0 # ทำการเคลียร์ค่า running loss เพื่อใช้ในการคำนวณค่า loss ใน batch ถัดไป\n","\n","print(\"Finished Training\")"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"iJWp-Za2aUvz"},"outputs":[],"source":["# Save the model\n","PATH = \"./CDT_model.pth\"\n","torch.save(model.state_dict(), PATH) # ทำการบันทึกโมเดลที่ได้จากการ train เป็นไฟล์ .pth โดย .state_dict() จะเป็นการบันทึกเฉพาะค่า parameter ของโมเดลเท่านั้น"]},{"cell_type":"markdown","metadata":{"id":"jx6txDwxaUvz"},"source":["### Test the Data\n","\n","ทดลองนำภาพตัวอย่างจาก test set มาทำนายด้วยโมเดลที่เราได้เทรนไป"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/","height":210},"executionInfo":{"elapsed":12,"status":"ok","timestamp":1688465969280,"user":{"displayName":"Korrawiz Chotayapa","userId":"12619977060284176060"},"user_tz":-420},"id":"6jP-6W-PaUvz","outputId":"d3fb40c2-679d-485d-e135-42d6abb5dfae"},"outputs":[{"data":{"image/png":"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08f6Xe/evXHw4EEAcI5BuH7Wfvvb3+KJJ57A3Xffjf79++OXv/wlpk2bhpycHAA8PlJdjkXHjh1x7Ngxq/+nn37CyZMnw+Z4nb/xOHDgAJYtW+aMegA8Plqju/mIiopCSkoKcnNznd9VVVUhNzcX6enpIdwz7xljkJ2djYULF2LFihXo3r271Z+SkoJWrVpZx2r37t04ePBgWByrUaNGYfv27di6davzM2TIEEyaNMlph/PxGT58eLXU7D179qBr164AgO7du6Njx47W8SktLcWGDRvC4vj88MMPiIy0vwJbtGiBqqoqADw+Ul2ORXp6OkpKSpCfn+88ZsWKFaiqqkJaWprn++y18zcee/fuxfLlyxEXF2f1h/vxqSbUM14vZP78+SY6Otq8/vrrZufOnebBBx807dq1M0VFRaHeNU/95je/MT6fz6xatcocPXrU+fnhhx+cx0yZMsUkJyebFStWmM2bN5v09HSTnp4ewr0OLZntYkx4H5+NGzeali1bmpkzZ5q9e/eat956y1x00UXm3//+t/OYWbNmmXbt2pkPPvjAfPXVV+a2225rtqmkWmZmpuncubOTavv++++bDh06mMcee8x5TDgdn7KyMrNlyxazZcsWA8D8/e9/N1u2bHGyNepyLMaMGWMGDRpkNmzYYNatW2d69OjRbFJJ3Y7P2bNnza233mqSkpLM1q1bre/riooK5zma8/HxV6O8+TDGmJdeeskkJyebqKgok5qaatavXx/qXfIcgAv+zJ0713nMjz/+aB566CFzySWXmIsuusjcfvvt5ujRo6Hb6RDTNx/hfnw++ugj069fPxMdHW169eplXn31Vau/qqrKPP300yYhIcFER0ebUaNGmd27d4dob71VWlpqHn30UZOcnGxat25tLrvsMvPUU09ZfyzC6fisXLnygt83mZmZxpi6HYvvv//eTJw40bRt29bExsaa++67z5SVlYXg3QSe2/EpKCio8ft65cqVznM05+PjrwhjRDk/IiIioiBrdHM+iIiIqHnjzQcRERF5ijcfRERE5CnefBAREZGnePNBREREnuLNBxEREXmKNx9ERETkKd58EBERkad480FERESe4s0HEREReYo3H0REROSp/wdhV1EZHfoQ3wAAAABJRU5ErkJggg==","text/plain":["
"]},"metadata":{},"output_type":"display_data"},{"name":"stdout","output_type":"stream","text":["GroundTruth: normal normal normal abnormal\n"]}],"source":["dataiter = iter(test_loader) # ทำการสร้าง object ของ test_loader ให้เป็น iterator\n","images, labels = next(dataiter) # ทำการดึงข้อมูลและ label จาก test_loader ที่เป็น iterator ออกมา\n","\n","# print images\n","imshow(torchvision.utils.make_grid(images)) # ทำการแสดงรูปภาพที่อยู่ใน images ที่เป็น tensor ในรูปแบบของ grid\n","print(\"GroundTruth: \", \" \".join(f\"{classes[labels[j]]:5s}\" for j in range(4))) # ทำการแสดง label ของรูปภาพที่อยู่ใน images ที่เป็น tensor ในรูปแบบของ grid"]},{"cell_type":"markdown","metadata":{"id":"sn5_zWONaUv0"},"source":["ในการใช้งานจริงเราสามารถสร้างโมเดล และอ่าน weights (หรือ parameters) ของโมเดลที่เทรนไว้แล้ว เพื่อนำมาใช้ทำนายเลยได้ ไม่จำเป็นต้องเทรนใหม่"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"elapsed":11,"status":"ok","timestamp":1688465969281,"user":{"displayName":"Korrawiz Chotayapa","userId":"12619977060284176060"},"user_tz":-420},"id":"pO30INrfaUv0","outputId":"608006a4-17fd-413d-d265-157227078645"},"outputs":[{"data":{"text/plain":[""]},"execution_count":17,"metadata":{},"output_type":"execute_result"}],"source":["model = Net() # สร้าง object ของ class Net ที่เป็นโมเดลที่เราสร้างไว้\n","model.load_state_dict(torch.load(PATH)) # ทำการโหลดค่า parameter ของโมเดลจากไฟล์ .pth ที่เราบันทึกไว้ก่อนหน้านี้"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"elapsed":8,"status":"ok","timestamp":1688465969281,"user":{"displayName":"Korrawiz Chotayapa","userId":"12619977060284176060"},"user_tz":-420},"id":"i-5-5gRmaUv0","outputId":"27c983dc-ed0a-4120-8480-47d24634dd80"},"outputs":[{"name":"stdout","output_type":"stream","text":["Predicted: normal normal normal abnormal\n"]}],"source":["# Predictions\n","outputs = model(images)\n","\n","_, predicted = torch.max(outputs, 1)\n","\n","print(\"Predicted: \", \" \".join(f\"{classes[predicted[j]]:5s}\" for j in range(4)))"]},{"cell_type":"markdown","metadata":{"id":"WK_9iPgLaUv2"},"source":["ในตัวอย่างนี้เราเพียงแค่ต้องการโชว์กระบวนการเบื้องต้นของการสร้างและเทรนโมเดล Convolutional Neural Network เท่านั่นแต่ในเทรนโมเดลจริงนั่นเรายังต้องคำนึงถึงปัจจัยอื่นๆอีกมากมาย เช่น การ overfit ของโมเดล, การปรับหน้าตาของโมเดล, การใช้ optimizer แบบอื่นๆ ที่เป็นปัจจัยในการเทรนโมเดลให้มีประสิทธิภาพ\n","\n","นอกจากนี้ปัจจุบัน CNN ได้ถูกออกแบบและพัฒนาให้มีรูปแบบที่ซับซ้อนขึ้นเพื่อให้ได้ความแม่นยำที่มากขึ้นเช่น VGG, DenseNet, ResNet รวมถึงจะมีการเทรนโมเดลเหล่านั้นกับข้อมูลขนาดใหญ่แล้วค่อยนำมา fine-tune เพื่อให้เหมาะสมกับ task ที่เราต้องการทำให้ลดเวลาในการเทรนและเพิ่มความแม่นยำให้กับโมเดลได้ โดยในบทถัดๆไปจะเป็นการยกตัวอย่างการ download โมเดลที่ซับซ้อนขึ้นที่ถูกเทรนมาแล้ว และนำมาใช้ในงานของเรา เช่น\n","* Classification\n","* Object detection\n","* Semantic Segmentation\n","\n","

\n","**ผู้จัดเตรียม code ใน tutorial**: ดร. ฐิติพัทธ อัชชะกุลวิสุทธิ์"]},{"cell_type":"markdown","metadata":{},"source":[]}],"metadata":{"accelerator":"GPU","colab":{"gpuType":"T4","provenance":[]},"kernelspec":{"display_name":"Python 3","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.10.11"}},"nbformat":4,"nbformat_minor":0}