Tutorial

C Language 101 - File I/O (EP.10)

By Arnon Puitrakul - 15 มิถุนายน 2015

C Language 101 - File I/O (EP.10)

ในที่สุดเราก็มาถึงเรื่องเกือบสุดท้ายกันแล้ว นั่นคือ File I/O หรือ การอ่านเขียนไฟล์นั่นเอง !!
คำเตือน ในเรื่องนี้ จะมีเรื่อง Pointer และ Structure เข้ามาเกี่ยวข้องฉะนั้น ถ้ายังไม่เข้าใจเรื่อง Pointer และ Structure ให้ย้อนกลับไปอ่านเรื่อง Pointer และ Structure ให้เข้าใจซะก่อน แล้วค่อยมาอ่านเรื่องนี้

การเปิดและปิดไฟล์

ในการอ่านเขียนไฟล์ในภาษา C เราจะต้องเรียกผ่าน Structure ที่ชื่อ FILE เท่านั้น (ตัวใหญ่หมดจริง ๆ นะ) ซึ่งเราก็ต้องประกาศเป็น Pointer ด้วย

FILE *a;

สังเกตได้ว่ามันจะมี parameter 2 ตัว ตัวแรกคือที่อยู่ไฟล์ แต่ในกรณีนี้คือไฟล์ที่ต้องการเปิดอยู่ที่เดียวกับที่ที่เราเรียกโปรแกรมเลยเขียนแค่ชื่อกับสกุลไฟล์
ส่วนอีกอันคือ mode ของการเปิด ซึ่ง mode ของการเปิดนั้นจะมีหลาย mode แต่ mode ที่แนะนำคือ r+ เพราะเปิดทีเดียวจะเขียนก็ได้จะอ่านก็ได้ หลักๆก็จะมี

  • r - Read อ่านอย่างเดียว
  • w - write เขียนอย่างเดียว
  • r+ - เขียนด้วยอ่านด้วย

ซึ่งในโลกความเป็นจริงๆมันจะมีบางครั้งที่ไฟล์เปิดไม่ได้ด้วยสาเหตุใดก็ตาม เราจะต้องมีวิธีรับมือกับมัน ซึ่ง fopen ถ้ามันเปิดไฟล์ไม่ได้มันจะ return ค่ากลับมาเป็น NULL
เพราะฉะนั้นวิธีเช็คคือใช้ if ในการเช็ค เช่น

if((a=fopen(“hello.txt”,”r”)) == NULL) printf(“Can’t Open File”);

จากโค๊ตด้านบนเราพยายามที่จะเปิดไฟล์ขื่อ hello แต่เราเอา if มาค่อมเพื่อเช็คว่า ตอนเปิดไฟล์ออกมา คำสั่ง fopen มัน return ค่า NULL มารึเปล่า ถ้าไม่คือไฟล์เปิดได้ แต่ถ้ามันได้ค่า NULL ออกมานั้นคือ ไฟล์นั้นเปิดไม่ได้ด้วยเหตุผลอะไรสักอย่าง อาจเพราะว่าหาไฟล์ไม่เจอ หรือติดเรื่อง permission ประการใด
แต่ ๆ ๆ มีเปิดไฟล์ ก็ต้องมีปิดไฟล์ ในการปิดไฟล์เราจะใช้คำสั่ง fclose(); ในการปิด

fclose(a);

การอ่านเขียนไฟล์

หลังจากที่เราเปิดปิดไฟล์ได้แล้ว เราจะมาอ่านไฟล์กัน
การอ่านไฟล์ทำได้เหมือนกันกับการรับข้อมูลเข้าทางหน้าจอปกติเลย เช่น
scanf ที่เราคุ้นเคยกัน เราก็แค่เปลี่ยนเป็น fscanf แต่มันจะต่างกันนิดหน่อย
fscanf(filepointer,"",destination); เช่น

fscanf(a,”%d %d %s %s”,&id,&year,name,surname);

เช่นเดียวกับ การเขียนไฟล์ ก็เหมือนกับ printf ที่เคยใช้ แต่เปลี่ยนเป็น fprintf แทน
fprintf(filepointer,"",destination); เช่น

 fprintf(a,”%d %d %s %s\n”,id,year,name,surname);

แต่คุณเคยประสบปัญหาแบบนี้มั้ย ?

สมมุติว่า ในไฟล์เรามีคำว่า Hello World แล้วเราอ่านไฟล์ไปแล้ว cursor มันไปอยู่ที่ W แล้วเราต้องการที่จะเขียนคำว่า Hi ต่อ แต่ถ้าเราใช้วิธีการเขียนไฟล์แบบเมื่อกี้ที่เรารู้มา มันก็จะออกมาเป็น Hello WHiorld เลยทันที เพราะฉะนั้นเราจึงมีวิธีการเขียนไฟล์อีกแบบนึง เราเรียกมันว่า Randomly Accessed File
แต่ในการที่จะใช้วิธีนี้ ตอนเราเปิดไฟล์เราจะต้องเติม b ไปในโหมดการเปิดด้วยเช่น rb,wb,rb+ เป็นต้น และอีกอย่างส่วนใหญ่เราจะใช้ structure มารองรับเพื่อความง่าย
ในการอ่านไฟล์แบบ Randomly Accessed File เราจะต้องใช้อีกคำสั่งนึงนั่นคือ fread();

fread(Structure_Name,Number of Read,1,fileptr);

อย่างที่บอกว่า ในการอ่านเขียนแบบนี้ เราจะใช้ Structure เข้ามาช่วย เพราะฉะนั้นใน fread() อันแรกมันเลยต้องการ Structure ที่จะเอาข้อมูลที่อ่านลงมาเก็บ และถัดไปเป็น จำนวนครั้งที่ต้องการอ่านเข้ามา และสุดท้ายคือ FILE Pointer ที่เราได้ประกาศไว้
ส่วนการเขียวไฟล์ ก็เหมือนกับอ่านไฟล์เลย แต่เปลี่ยนชื่อคำสั่งจาก fread(); เป็น fwrite() แทน และ Parameter เหมือนกันเลย เราลองมาดูตัวอย่างกันเลย

struct student
{
    int id;
    char name[20];
    char surname[20];
};

student ict[20];
fread (ict,sizeof(student),20,file_ptr); //1. read file
fwrite(ict,sizeof(student),20,file_ptr); //2. write file

ก่อนอื่นเลย เราก็สร้างพิมพ์เขียวของ Structure Student ขึ้นมา และเราก็เอาพิมพ์เขียวมันมาสร้างเป็น Array ของ Structure ชื่อ ict
ถัดมา เราก็มาอ่านไฟล์โดยใช้คำสั่ง fread เราจะอ่านลงไปใน Structure ชื่อ ict จำนวน เป็นขนาดของ Array ของ Structure หรือ พูดอีกแบบนึงคือ อ่านจนครบ 20 เต็ม Array นั่นเอง ผ่าน FILE pointer ที่ชื่อว่า file_ptr
และสุดท้าย เราก็มาเขียนไฟล์ ก็เหมือนกับอ่านไฟล์เลย เราจะอ่านไฟล์จาก Structure ชื่อ ict จำนวน 20 ครั้งลงไปในไฟล์ผ่าน FILE Pointer ชื่อ file_ptr

แล้วเราจะเลือกวิธีไหนดีล่ะ ?

วิธีแรกจะเหมาะกับการอ่านเขียนไฟล์แบบ ธรรมดาเลย คือ ไม่มีข้อมูลอะไรเยอะ อาจจะแค่อยากจะเขียนคำลงไปอะไรแบบนี้ แต่อีกแบบนึง Randomly Accessed File เราจำเป็นต้องใช้ Structure ซึ่งมันเหมาะกับการอ่านเขียนไฟล์ที่เป็นลำดับ ๆ เหมือนในตัวอย่างที่เป็นทะเบียนนักศึกษาเลย ซึ่งนั่นก็แล้วแต่ แล้วล่ะ ว่าเวลาเราเจอปัญหาจริง ๆ เราจะเลือกใช้แบบไหน
สรุปสุดท้ายแล้ว ไม่รู้จะสรุปอะไรดี เอาเป็นว่า การอ่านเขียนไฟล์จะมี 2 แบบล่ะกัน แบบแรกจะเป็นการอ่านและเขียนแบบเรียงไปเรื่อย เหมือนกับเราพิมพ์งานที่จะมี Cursor วิ่งไปเรื่อย ๆ เมื่อเราอ่านหรือเขียนมัน ถ้าเราเขียนมันก็จะเขียนลงไปที่ตำแหน่งของ Cursor ที่มันวิ่งอยู่ คำสั่งที่ใช้จะเหมือนกับเรารับค่า ส่งค่าที่หน้าจอเลย แค่เราเติมตัว f เข้าไป ส่วนอีกแบบ หรือเรียกว่า Randomly Accessed File จะเป็นการอ่านเขียนไฟล์แบบ Random เลยแต่เวลาเราอ่านหรือเขียนเราจะต้องใช้ Structure เข้ามาช่วย มันจะเหมาะกับการอ่านเขียนข้อมูลที่ทำออกมาในรูปของลำดับ ๆ มากกว่า คำสั่งที่ใช้คือ fread(); เพื่ออ่านไฟล์ และ fwrite เพื่อเขียนไฟล์

Read Next...

ติดตั้ง Zigbee Dongle บน Synology NAS กับ Home Assistant

ติดตั้ง Zigbee Dongle บน Synology NAS กับ Home Assistant

ก่อนหน้านี้เราเปลี่ยนมาใช้ Zigbee Dongle กับ Home Assistant พบว่าเสถียรขึ้นเยอะมาก อุปกรณ์แทบไม่หลุดออกจากระบบเลย แต่การติดตั้งมันเข้ากับ Synology DSM นั้นมีรายละเอียดมากกว่าอันอื่นนิดหน่อย วันนี้เราจะมาเล่าวิธีการเพื่อใครเอาไปทำกัน...

โหลด CSV วิธีไหนเร็วที่สุด ?

โหลด CSV วิธีไหนเร็วที่สุด ?

เมื่อหลายวันก่อนมีพี่ที่รู้จักกันมาถามว่า เราจะโหลด CSV ยังไงให้เร็วที่สุด เป็นคำถามที่ดูเหมือนง่ายนะ แต่พอมานั่งคิด ๆ ต่อ เห้ย มันมีอะไรสนุก ๆ ในนั้นเยอะเลยนี่หว่า วันนี้เราจะมาเล่าให้อ่านกันว่า มันมีวิธีการอย่างไรบ้าง และวิธีไหนเร็วที่สุด เหมาะกับงานแบบไหน...

ฮาวทู Sync Obsidian Note ด้วย LiveSync Plugin

ฮาวทู Sync Obsidian Note ด้วย LiveSync Plugin

ตั้งแต่ใช้ Obsidian เป็น App จด Note หลัก ชอบวิธีการจัดการมาก ๆ เมื่อใช้งานร่วมกับ Plugin ต่าง ๆ ยิ่งทำให้การใช้งานยืดหยุ่นมากกว่าเดิมอีกเยอะมาก ๆ ติดอยู่เรื่องเดียวคือ การใช้งานในหลาย ๆ อุปกรณ์ แต่เราเจอ Plugin ตัวนึงที่น่าสนใจมาก ๆ คือ LiveSync วันนี้เราจะมาเล่าวิธีการติดตั้ง และใช้งานมันกัน...

การสร้าง SSD Storage Pool บน Synology DSM

การสร้าง SSD Storage Pool บน Synology DSM

สำหรับคนที่ใช้ Synology NAS บางรุ่นจะมีช่อง M.2 สำหรับเสียบ NVMe SSD โดยพื้นฐาน Synology บอกว่ามันสำหรับการทำ Cache แต่ถ้าเราต้องการเอามันมาทำเป็น Storage ละ มันจะทำได้มั้ย วันนี้เราจะมาเล่าวิธีการทำกัน...