By Arnon Puitrakul - 15 มิถุนายน 2015
ในที่สุดเราก็มาถึงเรื่องเกือบสุดท้ายกันแล้ว นั่นคือ File I/O หรือ การอ่านเขียนไฟล์นั่นเอง !!
คำเตือน ในเรื่องนี้ จะมีเรื่อง Pointer และ Structure เข้ามาเกี่ยวข้องฉะนั้น ถ้ายังไม่เข้าใจเรื่อง Pointer และ Structure ให้ย้อนกลับไปอ่านเรื่อง Pointer และ Structure ให้เข้าใจซะก่อน แล้วค่อยมาอ่านเรื่องนี้
ในการอ่านเขียนไฟล์ในภาษา C เราจะต้องเรียกผ่าน Structure ที่ชื่อ FILE เท่านั้น (ตัวใหญ่หมดจริง ๆ นะ) ซึ่งเราก็ต้องประกาศเป็น Pointer ด้วย
FILE *a;
สังเกตได้ว่ามันจะมี parameter 2 ตัว ตัวแรกคือที่อยู่ไฟล์ แต่ในกรณีนี้คือไฟล์ที่ต้องการเปิดอยู่ที่เดียวกับที่ที่เราเรียกโปรแกรมเลยเขียนแค่ชื่อกับสกุลไฟล์
ส่วนอีกอันคือ mode ของการเปิด ซึ่ง mode ของการเปิดนั้นจะมีหลาย mode แต่ mode ที่แนะนำคือ 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 เพื่อเขียนไฟล์
เราเป็นคนที่อ่านกับซื้อหนังสือเยอะมาก ปัญหานึงที่ประสบมาหลายรอบและน่าหงุดหงิดมาก ๆ คือ ซื้อหนังสือซ้ำเจ้าค่ะ ทำให้เราจะต้องมีระบบง่าย ๆ สักตัวในการจัดการ วันนี้เลยจะมาเล่าวิธีการที่เราใช้ Obsidian ในการจัดการหนังสือที่เรามีกัน...
หากเราเรียนลงลึกไปในภาษาใหม่ ๆ อย่าง Python และ Java โดยเฉพาะในเรื่องของการจัดการ Memory ว่าเขาใช้ Garbage Collection นะ ว่าแต่มันทำงานยังไง วันนี้เราจะมาเล่าให้อ่านกันว่า จริง ๆ แล้วมันทำงานอย่างไร และมันมีเคสใดที่อาจจะหลุดจนเราต้องเข้ามาจัดการเองบ้าง...
ก่อนหน้านี้เราเปลี่ยนมาใช้ Zigbee Dongle กับ Home Assistant พบว่าเสถียรขึ้นเยอะมาก อุปกรณ์แทบไม่หลุดออกจากระบบเลย แต่การติดตั้งมันเข้ากับ Synology DSM นั้นมีรายละเอียดมากกว่าอันอื่นนิดหน่อย วันนี้เราจะมาเล่าวิธีการเพื่อใครเอาไปทำกัน...
เมื่อหลายวันก่อนมีพี่ที่รู้จักกันมาถามว่า เราจะโหลด CSV ยังไงให้เร็วที่สุด เป็นคำถามที่ดูเหมือนง่ายนะ แต่พอมานั่งคิด ๆ ต่อ เห้ย มันมีอะไรสนุก ๆ ในนั้นเยอะเลยนี่หว่า วันนี้เราจะมาเล่าให้อ่านกันว่า มันมีวิธีการอย่างไรบ้าง และวิธีไหนเร็วที่สุด เหมาะกับงานแบบไหน...