Giáo trình Quản trị mạng (Phần 2)

pdf 51 trang phuongnguyen 5990
Bạn đang xem 20 trang mẫu của tài liệu "Giáo trình Quản trị mạng (Phần 2)", để tải tài liệu gốc về máy bạn click vào nút DOWNLOAD ở trên

Tài liệu đính kèm:

  • pdfgiao_trinh_quan_tri_mang_phan_2.pdf

Nội dung text: Giáo trình Quản trị mạng (Phần 2)

  1. CHƯƠNG 4: CẤU TRÚC VÒNG LẶP Mã chương/ bài:MH18-04 Mục tiêu: - Trình bày ý nghĩa của vòng lặp ; - Trình bày cú pháp, công dụng của lệnh for, while, do while; - Trình bày ý nghĩa và cách sử dụng lệnh break, continue; - Giải một số bài toán sử dụng lệnh for, while, do while ; - Sử dụng được các vòng lặp lồng nhau. - Thực hiện các thao tác an toàn với máy tính. 4.1. Lệnh for Mục tiêu: Hiểu cú pháp và có thể vận dụng vòng lặp for để giải quyết bài toán. Cú pháp : for ( biểu thức 1; biểu thức 2; biểu thức 3) Lệnh hoặc khối lệnh ; Giải thích : + Biểu thức 1: khởi tạo giá trị ban đầu cho biến điều khiển. + Biểu thức 2: là quan hệ logic thể hiện điều kiện tiếp tục vòng lặp. + Biểu thức 3: phép gán dùng thay đổi giá trị biến điều khiển. Lưu ý : + Từ khóa for phải viết bằng chữ thường + Nếu là khối lệnh thì phải đặt trong dấu { } + Biểu thức 1, 2, 3 phải phân cách bằng dấu chấm phẩy (;) + Nếu biểu thức 2 không có, vòng for được xem là luôn luôn đúng. Muốn thoát khỏi vòng lặp for phải dùng một trong 3 lệnh break, goto hoặc return.
  2. + Với mỗi biểu thức có thể viết thành một dãy biểu thức con phân cách nhau bởi dấu phẩy. Khi đó các biểu thức con được xác định từ trái sang phải. Tính đúng sai của dãy biểu thức con trong biểu thức thứ 2 được xác định bởi biểu thức con cuối cùng. + Trong thân for (khối lệnh) có thể chứa một hoặc nhiều cấu trúc điều khiển khác. + Khi gặp lệnh break, cấu trúc lặp sâu nhất sẽ thoát ra. + Trong thân for có thể dùng lệnh goto để thoát khỏi vòng lặp đến vị trí mong muốn. + Trong thân for có thể sử dụng return để trở về một hàm nào đó. + Trong thân for có thể sử dụng lệnh continue để chuyển đến đầu vòng lặp (bỏ qua các câu lệnh còn lại trong thân). Ví dụ : Nhập n và tính tổng S = 1 + 2 + + n #include #include void main() { int i, n, s = 0; printf("Nhap vao so n: "); scanf("%d", &n); i = 0; for(i = 0; i<=n; i++) s = s + i; //hoac s += i; printf("Tong: %d", s); getch(); } Bài tập : Dùng vòng lặp for để : 1. Viết chương trình tính tổng bình phương các số lẻ từ 1 đến N. (N nhập vào từ bàn phím) . 2. Viết chương trình nhập vào N rồi tính giai thừa của N. (N nhập vào từ bàn phím). 3. Hãy làm theo yêu cầu sau:
  3. -Viết chương trình kiểm tra n có phải là số nguyên tố hay không, với số n được nhập vào từ bàn phím -Viết chương trình nhập vào số n và in ra các số nguyên tố có từ 1-> n, đếm có bao nhiêu số nguyên tố như vậy. 4. Tính các tổng sau: S=1+2+3+ +n S= 12+22+3+2+ +n2 S=1/1+1/2+1/3+ +1/n 5. Viết chương trình in ra các số từ 1 đến 100 theo dạng sau: 1 2 10 11 12 13 20 92 93 100 Hướng dẫn 3. Thuật toán: - Khai báo biến n, i - Nhập số n - Thuật toán kiểm tra n có phải là số nguyên tố không: o Cho i chạy từ 2 đến n . Nếu n%i = 0 thì thoát; o Nếu i = n thì in ra màn hình n là số nguyên tố o Ngược lại, n không phải là số nguyên tố Thuật toán: - Khai báo biến n, i, j - Nhập số n - Tìm các số nguyên tố có được từ 1->n: o Khởi tạo biến đếm count=0 o Cho i chạy từ 2 đến n . Cho j chạy từ 2 đến i Nếu i%j = 0 thì thoát . Nếu j=i thì: i là số nguyên tố count=count+1;
  4. - In giá trị count và các số nguyên tố tìm được ra màn hình Chương trình: 4. Thuật toán: - Khai báo biến S, i, n - Nhập n từ bàn phím - Tính tổng S: o S=0 o Cho i chạy từ 1 đến n . S=S+i - Xuất S ra màn hình Thuật toán: - Khai báo biến S, i, n - Nhập n từ bàn phím - Tính tổng S: o S=0 o Cho i chạy từ 1 đến n . S=S+i*i hoặc S=S+pow(i,2) - Xuất S ra màn hình Thuật toán: - Khai báo biến S, i, n - Nhập n từ bàn phím - Tính tổng S: o S=0 o Cho i chạy từ 1 đến n . S=S+1/i - Xuất S ra màn hình 5: Thuật toán: - Khai báo biến i - Tạo bảng số: o Cho i chạy từ 1 đến 100 làm . Nếu i%10= =1 thì in xuống dòng và i ra màn hình . Ngược lại, in giá trị i ra màn hình 4.2. Lệnh break Mục tiêu: Hiểu và biết cách sử dụng lệnh break trong chương trình. Thông thường lệnh break dùng để thoát khỏi vòng lặp không xác định điều kiện dừng hoặc người dùng muốn dừng vòng lặp theo điều kiện do người dùng chỉ định. Việc dùng lệnh break để thoát khỏi vòng lặp thường sử dụng phối hợp
  5. với lệnh if. Lệnh break dùng trong for, while, do while, switch. Lệnh break thoát khỏi vòng lặp chứa nó. Ví dụ: Viết chương trình đọc từ bàn phím một số nguyên n (1≤n≤10) rồi đưa ra tiếng Anh của số đó. Chẳng hạn, nếu gõ vào n = 4 thì in ra Four. #include #include #include main() { clrscr(); //khai bao bien int so; char *kq; //nhap tuoi printf("nhap so bat ky nam trong khoang [1,10]: "); scanf("%d",&so); ///chuyen so vua nhap sang tieng anh switch(so) { case 1: kq="One";break; case 2: kq="Two";break; case 3: kq="Three";break; case 4: kq="Four";break; case 5: kq="Five";break; case 6: kq="Six";break; case 7: kq="Seven";break; case 8: kq="Eight";break; case 9: kq="Nine";break; case 10: kq="Ten";break; } printf("Trong tieng Anh\n"); printf("%d : %s",so,kq); getch(); } Sử dụng lệnh break trong switch để nhảy bỏ các câu lệnh kế tiếp còn lại. 4.3. Lệnh continue Mục tiêu: Hiểu và biết cách sử dụng lệnh break trong chương trình. Được dùng trong vòng lặp for, while, do while. Khi lệnh continue thi hành thì quyền điều khiển sẽ trao qua cho biểu thức điều kiện của vòng lặp gần nhất. Nghĩa là lộn ngược lên đầu vòng lặp, tất cả những lệnh đi sau trong vòng lặp chứa continue sẽ bị bỏ qua không thi hành.
  6. 4.4. Lệnh while Mục tiêu: Hiểu và biết cách sử dụng vòng lặp while khi giải quyết bài toán. Cú pháp : while (biểu thức) Khối lệnh; Giải thích : + Biểu thức: có thể là một biểu thức hoặc nhiều biểu thức con. Nếu là nhiều biểu thức con thì cách nhau bởi dấu phẩy (,) và tính đúng sai của biểu thức được quyết định bởi biểu thức con cuối cùng. + Trong thân while (khối lệnh) có thể chứa một hoặc nhiều cấu trúc điều khiển khác. + Trong thân while có thể sử dụng lệnh continue để chuyển đến đầu vòng lặp (bỏ qua các câu lệnh còn lại trong thân). + Muốn thoát khỏi vòng lặp while tùy ý có thể dùng các lệnh break, goto, return như lệnh for. Chú ý: Từ khóa while phải viết bằng chữ thường Nếu khối lệnh bao gồm từ 2 lệnh trở lên thì phải đặt trong dấu { } Ví dụ: Viết chương trình nhập vào một số n và in ra tổng các số đó #include main() { int n,s; printf("\n Chuong trinh tinh tong cac so tu nhien tu 1 den"); scanf("%d",&n); s=0; while(n>0) { s=s+n; n=n-1; } printf(“\nTong tim duoc la: %d”,s);
  7. getch(); } Bài tập: Dùng vòng lặp while để: 1. Viết chương trình tính tổng bình phương các số lẻ từ 1 đến N. 2. Viết chương trình nhập vào N rồi tính giai thừa của N. 3. Hãy làm theo yêu cầu sau: -Viết chương trình kiểm tra n có phải là số nguyên tố hay không, với số n được nhập vào từ bàn phím -Viết chương trình nhập vào số n và in ra các số nguyên tố có từ 1-> n, đếm có bao nhiêu số nguyên tố như vậy. 4. Tính các tổng sau: S=1+2+3+ +n S= 12+22+32+ +n2 S=1/1+1/2+1/3+ +1/n 5. Viết chương trình in ra các số từ 1 đến 100 theo dạng sau: 1 2 10 11 12 13 20 92 93 100 4.5. Lệnh do while Mục tiêu: Hiểu và biết cách sử dụng vòng lặp do while khi giải quyết bài toán.  Cú pháp: do khối lệnh; while (biểu thức);
  8.  Giải thích: + Biểu thức: có thể là một biểu thức hoặc nhiều biểu thức con. Nếu là nhiều biểu thức con thì cách nhau bởi dấu phẩy (,) và tính đúng sai của biểu thức được quyết định bởi biểu thức con cuối cùng. + Trong thân do while (khối lệnh) có thể chứa một hoặc nhiều cấu trúc điều khiển khác. + Trong thân do while có thể sử dụng lệnh continue để chuyển đến đầu vòng lặp (bỏ qua các câu lệnh còn lại trong thân). + Muốn thoát khỏi vòng lặp do while tùy ý có thể dùng các lệnh break, goto, return. Chú ý: Từ khóa do, while phải viết bằng chữ thường Nếu khối lệnh bao gồm từ 2 lệnh trở lên thì phải đặt trong dấu { } Ví dụ : Viết chương trình nhập vào một số n và in ra tổng các số đó (sinh viên tự làm) Bài tập: Dùng vòng lặp do while để: 1. Viết chương trình tính tổng bình phương các số lẻ từ 1 đến N. 2. Viết chương trình nhập vào N rồi tính giai thừa của N. 3. Hãy làm theo yêu cầu sau: -Viết chương trình kiểm tra n có phải là số nguyên tố hay không, với số n được nhập vào từ bàn phím -Viết chương trình nhập vào số n và in ra các số nguyên tố có từ 1-> n, đếm có bao nhiêu số nguyên tố như vậy. 4. Tính các tổng sau: S=1+2+3+ +n S= 12+22+32+ +n2
  9. S=1/1+1/2+1/3+ +1/n 5. Viết chương trình in ra các số từ 1 đến 100 theo dạng sau: 1 2 10 11 12 13 20 92 93 100 4.6. Vòng lặp lồng nhau Mục tiêu: Biết cách kết hợp để các vòng lặp lồng nhau khi giải quyết bài toán có tính phức tạp. Các lệnh lặp for, while, do while có thể lồng vào chính nó, hoặc lồng vào lẫn nhau. Nếu không cần thiết không nên lồng vào nhiều cấp dễ gây nhầm lẫn khi lập trình cũng như kiểm soát chương trình. Ví dụ : Vẽ hình chữ nhật đặc bằng các dấu '*' #include #include void main(void) { int i, ij, idai, irong; printf("Nhap vao chieu dai: "); scanf("%d", &idai); printf("Nhap vao chieu rong: "); scanf("%d", &irong); for (i = 1; i <= irong; i++) { for (ij = 1; ij <= idai; ij++) //in mot hang voi chieu dai dau * printf("*"); printf("\n"); } getch(); } Bài tập : Sử dụng vòng lặp lồng nhau để : 1. Viết chương trình in các số nguyên tố từ 1 đến n.(n được nhập từ bàn phím) 2. Viết chương trình tính :
  10. S = 1 + 2! + 3! + +n! 4.7. So sánh sự khác nhau của các vòng lặp. Mục tiêu: Hiểu và biết ở trường hợp nào thì dùng vòng lặp nào để giải quyết các bài toán. - Vòng lặp for thường sử dụng khi biết được số lần lặp xác định. - Vòng lặp thường while, do while sử dụng khi không biết rõ số lần lặp. - Khi gọi vòng lặp while, do while, nếu biểu thức sai vòng lặp while sẽ không được thực hiện lần nào nhưng vòng lặp do while thực hiện được 1 lần Số lần thực hiện ít nhất của while là 0 và của do while là 1. 4.8. Bài tập thực hành 1. Viết chương trình in ra bảng mã ASCII 2. Viết chương trình nhập vào một số nguyên rồi in ra tất cả các ước số của số đó. 3. Viết chương trình vẽ một tam giác cân bằng các dấu * 4. Viết chương trình tính tổng nghịch đảo của N số nguyên đầu tiên theo công thức : S = 1 + 1/2 + 1/3 + + 1/N 5. Viết chương trình tìm USCLN, BSCNN của 2 số. 6. Viết chương trình vẽ một tam giác cân rỗng bằng các dấu *. 7. Viết chương trình vẽ hình chữ nhật rỗng bằng các dấu *. 8. Viết chương trình nhập vào một số và kiểm tra xem số đó có phải là số nguyên tố hay không? 9. Viết chương trình tính xn với x, n được nhập vào từ bàn phím. 10. Viết chương trình lặp lại nhiều lần công việc nhập một ký tự và in ra mã ASCII của ký tự đó, khi nào nhập số 0 thì dừng.
  11. CHƯƠNG 5: HÀM Mã chương: MH18-05 Ý nghĩa: Chương trình viết bằng ngôn ngữ C gồm 1 dãy các hàm trong đó có 1 hàm chính là main và chương trình bắt đầu từ main. Mục tiêu: - Trình bày khái niệm và cách khai báo về hàm ; - Trình bày được cách truyền tham số, tham biến, tham trị ; - Sử dụng biến cục bộ, toàn cục trong hàm ; - Sử dụng tiền xử lý #define. - Thực hiện các thao tác an toàn với máy tính. 5.1. Các ví dụ về hàm Mục tiêu: - Trình bày khái niệm và cách khai báo về hàm ; - Thực hiện các thao tác an toàn với máy tính. 5.1.1. Khái niệm về hàm Trong những chương trình lớn, có thể có những đoạn chương trình viết lặp đi lặp lại nhiều lần, để tránh rườm rà và mất thời gian khi viết chương trình; người ta thường phân chia chương trình thành nhiều module, mỗi module giải quyết một công việc nào đó. Các module như vậy gọi là các chương trình con. Một tiện lợi khác của việc sử dụng chương trình con là ta có thể dễ dàng kiểm tra xác định tính đúng đắn của nó trước khi ráp nối vào chương trình chính và do đó việc xác định sai sót để tiến hành hiệu đính trong chương trình chính sẽ thuận lợi hơn. Trong C, chương trình con được gọi là hàm. Hàm trong C có thể trả về kết quả thông qua tên hàm hay có thể không trả về kết quả.
  12. Hàm có hai loại: hàm chuẩn và hàm tự định nghĩa. Trong chương này, ta chú trọng đến cách định nghĩa hàm và cách sử dụng các hàm đó. Một hàm khi được định nghĩa thì có thể sử dụng bất cứ đâu trong chương trình. Trong C, một chương trình bắt đầu thực thi bằng hàm main. Ví dụ 1: Ta có hàm max để tìm số lớn giữa 2 số nguyên a, b như sau: int max(int a, int b) { return (a>b) ? a:b; } Ví dụ 2: Ta có chương trình chính (hàm main) dùng để nhập vào 2 số nguyên a,b và in ra màn hình số lớn trong 2 số #include #include int max(int a, int b) { return (a>b) ? a:b; } void main() { int a, b, c; printf("\n Nhap vao 3 so a, b,c "); scanf("%d%d%d",&a,&b,&c); printf("\n So lon la %d",max(a, max(b,c))); getch(); }  Hàm thư viện Hàm thư viện là những hàm đã được định nghĩa sẵn trong một thư viện nào đó, muốn sử dụng các hàm thư viện thì phải khai báo thư viện trước khi sử dụng bằng lệnh: #include Một số thư viện: alloc.h assert.h bcd.h bios.h complex.h conio.h ctype.h dir.h dirent.h dos.h errno.h fcntl.h float.h fstream.h grneric.h graphics.h io.h iomanip.h iostream.h limits.h
  13. locale.h malloc.h math.h mem.h process.h setjmp.h share.h signal.h stdarg.h stddef.h stdio.h stdiostr.h stdlib.h stream.h string.h strstrea.h sys\stat.h sys\timeb.h sys\types.h time.h values.h Ý nghĩa của một số thư viện thường dùng: 1. stdio.h: Thư viện chứa các hàm vào/ ra chuẩn (standard input/output). Gồm các hàm printf(), scanf(), getc(), putc(), gets(), puts(), fflush(), fopen(), fclose(), fread(), fwrite(), getchar(), putchar(), getw(), putw() 2. conio.h: Thư viện chứa các hàm vào ra trong chế độ DOS (DOS console). Gồm các hàm clrscr(), getch(), getche(), getpass(), cgets(), cputs(), putch(), clreol(), 3. math.h: Thư viện chứa các hàm tính toán gồm các hàm abs(), sqrt(), log(). log10(), sin(), cos(), tan(), acos(), asin(), atan(), pow(), exp(), 4. alloc.h: Thư viện chứa các hàm liên quan đến việc quản lý bộ nhơ. Gồm các hàm calloc(), realloc(), malloc(), free(), farmalloc(), farcalloc(), farfree(), 5. io.h: Thư viện chứa các hàm vào ra cấp thấp. Gồm các hàm open(), _open(), read(), _read(), close(), _close(), creat(), _creat(), creatnew(), eof(), filelength(), lock(), 6. graphics.h: Thư viện chứa các hàm liên quan đến đồ họa. Gồm initgraph(), line(), circle(), putpixel(), getpixel(), setcolor(), Muốn sử dụng các hàm thư viện thì ta phải xem cú pháp của các hàm và sử dụng theo đúng cú pháp (xem trong phần trợ giúp của Turbo C).  Hàm người dùng Hàm người dùng là những hàm do người lập trình tự tạo ra nhằm đáp ứng nhu cầu xử lý của mình. 5.1.2. Các ví dụ về hàm Ví dụ 1:
  14. 1 #include 2 #include 3 4 5 // khai bao prototype void line(); 6 7 // ham in 1 dong dau 8 void line() 9 { 10 int i; 11 for(i = 0; i < 19; i++) 12 printf("*"); 13 printf("\n"); 14 } 15 16 void main(void) 17 { 18 line(); 19 printf("* Minh hoa ve ham *"); 20 line(); 21 getch(); 22 } Giải thích chương trình Dòng 8 đến dòng 14: định nghĩa hàm line, hàm này không trả về giá trị, thực hiện công việc in ra 19 dấu sao. Dòng 5: khai báo prototype, sau tên hàm phải có dấu chấm phẩy Trong hàm line có sử dụng biến i, biến i là biến cục bộ chỉ sử dụng được trong phạm vi hàm line. Dòng 18 và 20: gọi thực hiện hàm line. * Trình tự thực hiện chương trình Không có dấu chấm phẩy sau tên hàm, phải có cặp dấu ngoặc ( ) sau tên hàm nếu hàm không có tham số truyền vào. Phải có dấu chấm phẩy sau tên hàm
  15. khai báo prototype. Nên khai báo prototype cho dù hàm được gọi nằm trước hay sau câu lệnh gọi nó. Ví dụ 2: 1 #include 2 #include 3 4 // khai bao prototype int power(int, int); 5 6 // ham tinh so mu 7 int power(int ix, int in) 8 { 9 int i, ip = 1; 10 for(i = 1; i <= in; i++) 11 ip *= ix; 12 return ip; 13 } 14 15 void main(void) 16 { 17 printf("2 mu 2 = %d.\n", power(2, 2)); 18 printf("2 mu 3 = %d.\n", power(2, 3)); 19 getch(); 20 } Kết quả in ra màn hình 2 mu 2 = 4. 2 mu 3 = 8. Giải thích chương trình Hàm power có hai tham số truyền vào là ix, in có kiểu int và kiểu trả về cũng có kiểu int. Dòng 13: return ip: trả về giá trị sau khi tính toán Dòng 18: đối mục 2 và 3 có kiểu trả về là int sau khi thực hiện gọi power. Hai tham số ix, in của hàm power là dạng truyền tham trị.
  16. * Trình tự thực hiện chương trình trả về giá trị kiểu int để xuất ra màn hình Quy tắc đặt tên hàm giống tên biến, hằng Mỗi đối số cách nhau = dấu phẩy kèm theo kiểu dữ liệu tương ứng. Ví dụ 3: 1 #include 2 #include 3 // khai bao prototype 4 void time(int & , int &); 5 // ham doi phut thanh gio:phut 6 void time(int &ig, int &ip) 7 { 8 ig = ip / 60; 9 ip %= 60; 10 } 11 void main(void) 12 { 13 int igio, iphut; 14 printf("Nhap vao so phut : "); 15 scanf("%d", &iphut); time(igio, iphut); 16 printf("%02d:%02d\n", igio, iphut); 17 getch(); 18 }  Kết quả in ra màn hình Nhap vao so phut: 185 03:05  Giải thích chương trình Hàm time có hai tham số truyền vào là ix, in có kiểu int. 2 tham số này có toán tử địa chỉ & đi trước cho biết 2 tham số này là dạng truyền tham biến.
  17. 5.2. THAM SỐ DẠNG THAM BIẾN VÀ THAM TRỊ Mục tiêu: - Trình bày được cách truyền tham số, tham biến, tham trị ; - Thực hiện các thao tác an toàn với máy tính. 5.2.1. Tham số dạng tham trị Mặc nhiên, việc truyền tham số cho hàm trong C là truyền theo giá trị; nghĩa là các giá trị thực (tham số thực) không bị thay đổi giá trị khi truyền cho các tham số hình thức Ví dụ 1: Giả sử ta muốn in ra nhiều dòng, mỗi dòng 50 ký tự nào đó. Để đơn giản ta viết một hàm, nhiệm vụ của hàm này là in ra trên một dòng 50 ký tự nào đó. Hàm này có tên là InKT. #include #include void InKT(char ch) { int i; for(i=1;i<=50;i++) printf(“%c”,ch); printf(“\n”); } int main() { char c = ‘A’; InKT(‘*’); /* In ra 50 dau * */ InKT(‘+’); InKT(c); return 0;
  18. } Chú ý: - Trong hàm InKT ở trên, biến ch gọi là tham số hình thức được truyền bằng giá trị (gọi là tham trị của hàm). Các tham trị của hàm coi như là một biến cục bộ trong hàm và chúng được sử dụng như là dữ liệu đầu vào của hàm. - Khi chương trình con được gọi để thi hành, tham trị được cấp ô nhớ và nhận giá trị là bản sao giá trị của tham số thực. Do đó, mặc dù tham trị cũng là biến, nhưng việc thay đổi giá trị của chúng không có ý nghĩa gì đối với bên ngoài hàm, không ảnh hưởng đến chương trình chính, nghĩa là không làm ảnh hưởng đến tham số thực tương ứng. Ví dụ 2: Ta xét chương trình sau đây: #include #include int hoanvi(int a, int b) { int t; t=a; /*Đoạn này hoán vị giá trị của 2 biến a, b*/ a=b; b=t; printf("\Ben trong ham a=%d , b=%d",a,b); return 0; } void main() { int a, b; clrscr(); printf("\n Nhap vao 2 so nguyen a, b:"); scanf("%d%d",&a,&b); printf("\n Truoc khi goi ham hoan vi a=%d, b=%d", a, b); hoanvi(a,b); printf("\n Sau khi goi ham hoan vi a=%d ,b=%d",a,b); getch(); } Kết quả thực hiện chương trình:  Giải thích: - Nhập vào 2 số 6 và 5 (a=6, b=5)
  19. - Trước khi gọi hàm hoán vị thì a=6, b=5 - Bên trong hàm hoán vị a=5, b=6 - Khi ra khỏi hàm hoán vị thì a=6, b=5 5.2.2. Bài tập thực hành về tham trị 1. Viết chương trình giải phương trình bậc nhất ax+b=0. Hướng dẫn: - Xây dựng hàm giải phương trình bậc nhất: void giaiptb1(int a, int b) - Truyền tham trị cho hàm giaiptb1(int a, int b) - Gọi hàm giaiptb1(int a, int b) vừa xây dựng thực hiện trong chương trình chính 2. Viết chương trình tính tổng các số nguyên chẵn từ 1 đến n. Hướng dẫn: - Xây dựng hàm tính tổng các số nguyên chẵn từ 1 đến n: int tinhtongchan (int n) - Truyền tham trị cho hàm int tinhtongchan (int n) - Gọi hàm tinhtongchan (int n) vừa xây dựng thực hiện trong chương trình chính 5.2.3. Tham số dạng tham biến Trong đoạn chương trình trên, nếu ta muốn sau khi kết thúc chương trình con giá trị của a, b thay đổi thì ta phải đặt tham số hình thức là các con trỏ, còn tham số thực tế là địa chỉ của các biến. Lúc này mọi sự thay đổi trên vùng nhớ được quản lý bởi con trỏ là các tham số hình thức của hàm thì sẽ ảnh hưởng đến vùng nhớ đang được quản lý bởi tham số thực tế tương ứng (cần để ý rằng vùng nhớ này chính là các biến ta cần thay đổi giá trị). Người ta thường áp dụng cách này đối với các dữ liệu đầu ra của hàm. Ví dụ: Xét chương trình sau đây: #include #include long hoanvi(long *a, long *b) /* Khai báo tham số hình thức *a, *b là các con trỏ kiểu long */
  20. { long t; t=*a; /*gán nội dung của x cho t*/ *a=*b; /*Gán nội dung của b cho a*/ *b=t; /*Gán nội dung của t cho b*/ printf("\n Ben trong ham a=%ld , b=%ld",*a,*b); /*In ra nội dung của a, b*/ return 0; } void main() { long a, b; clrscr(); printf("\n Nhap vao 2 so nguyen a, b:"); scanf("%ld%ld",&a,&b); printf("\n Truoc khi goi ham hoan vi a=%ld, b=%ld", a,b); hoanvi(&a,&b); /* Phải là địa chỉ của a và b */ printf("\n Sau khi goi ham hoan vi a=%ld, b=%ld", a,b); getch(); } Kết quả thực hiện chương trình sau.  Giải thích: - Nhập vào 2 số 5, 6 (a=5, b=6) - Trước khi gọi hàm hoanvi thì a=5, b=6 - Trong hàm hoanvi (khi đã hoán vị) thì a=6, b=5 - Khi ra khỏi hàm hoán vị thì a=6, b=6 5.2.4. Bài tập thực hành 1. Viết chương trình giải phương trình bậc nhất ax+b=0 (a#0). Hướng dẫn: - Xây dựng hàm giải phương trình bậc nhất: float giaiptb1(int a, int b, float *x) - Truyền tham trị cho hàm giaiptb1(int a, int b, float *x) - Gọi hàm giaiptb1(int a, int b, float *x) vừa xây dựng thực hiện trong chương trình chính
  21. 2. Viết chương trình tính tổng các số nguyên chẵn từ 1 đến n. Hướng dẫn: - Xây dựng hàm tính tổng các số nguyên chẵn từ 1 đến n: int tinhtongchan (int n, int *S) - Truyền tham biến cho hàm int tinhtongchan (int n, int *S) - Gọi hàm int tinhtongchan (int n, int *S) vừa xây dựng thực hiện trong chương trình chính 5.3. SỬ DỤNG BIẾN TOÀN CỤC Mục tiêu: Sử dụng biến cục bộ, toàn cục trong hàm 5.3.1. Sử dụng biến toàn cục 1 #include 2 #include 3 4 // khai bao prototype void oddeven( ); 5 void negative(); 6 7 //khai bao bien toan cuc int inum; 8 9 void main(void) 10 { 11 printf("Nhap vao 1 so nguyen : "); 12 scanf("%d", &inum); 13 oddeven(); 14 negative(); 15 getch(); 16 } 17 18 // ham kiem tra chan le void oddeven() 19 { 20 if (inum % 2) 21 printf("%d la so le.\n", inum); 22 else 23 printf("%d la so chan.\n", inum); 24 } 25 //ham kiem tra so am void negative() 26 { 27 if (inum < 0) 28 printf("%d la so am.\n", inum); 29 else 30 printf("%d la so duong.\n", inum); 31 } Kết quả in ra màn hình Nhap vao 1 so nguyen: 3
  22. 3 la so le. 3 la so duong.  Giải thích chương trình Chương trình trên gồm 2 hàm oddeven và negative, 2 hàm này bạn thấy không có tham số để truyền biến inum vào xử lý nhưng vẫn cho kết quả đúng. Do chương trình sử dụng biến inum toàn cục (dòng.9) nên biến này có ảnh hưởng đến toàn bộ chương trình mỗi khi gọi và sử dụng nó. Xét tình huống sau: Giả sử trong hàm negative ta khai báo biến inum có kiểu int như sau: void negative() { int inum; . } Khi đó chương trình sẽ cho kết quả sai! Do các câu lệnh trong hàm negative sử dụng biến inum sẽ sử dụng biến inum khai báo trong hàm negative và lúc này biến inum toàn cục không có tác dụng đối với các câu lệnh trong hàm này. Biến inum khai báo trong hàm negative chỉ có ảnh hưởng trong phạm vi hàm và chu trình sống của nó bắt đầu từ lúc gọi hàm đến khi thực hiện xong. Cẩn thận khi đặt tên biến, xác định rõ phạm vi của biến khi sử dụng để có thể dễ dàng kiểm soát chương trình. Ví dụ 6: #include #include #define PI 3.14 // khai bao prototype float area(); //khai bao bien toan cuc float frad; void main(void) { printf("Nhap vao ban kinh hinh cau : "); scanf("%f", &frad); printf("Dien tich hinh cau: %10.3f.\n", area());
  23. getch(); } // ham tinh dien tich hinh cau float area() { return (4*PI*frad*frad); } Kết quả in ra màn hình Nhap vao ban kinh hinh cau: 3.2 Dien tich hinh cau: 128.614 5.3.2. Bài tập thực hành Bài 1: Viết hàm kiểm tra số nguyên n có phải là số nguyên tố hay không? Bài 2: Viết hàm kiểm tra số nguyên n có phải là số chính phương hay không? (25=52, 16=42, 9=32; các số 25, 16, 9 là những số chính phương) Bài 3: Viết hàm tính n! (n là số nguyên nhập vào từ bàn phím) Bài 4: Viết thủ tục tìm UCLN và BCNN của hai số nhập vào từ bàn phím Bài 5: Viết hàm, thủ tục tìm số lớn nhất trong hai số. Áp dụng hàm này vào việc tìm số lớn nhất trong ba số nhập vào từ bàn phím Hướng dẫn Bài 1: Thuật toán: Viết hàm (function) ktra_ngto(int x): kiểm tra x có phải là số nguyên tố hay không Khai báo biến i, ktra Cho i chạy từ 2 đến x Nếu x mod i = 0 thì thoát Nếu x=i thì ktra=1
  24. Ngược lại, ktra=0 Hàm ktra_ngto trả về giá trị ktra Chương trình chính: Khai báo biến n, kq Nhập số n Gọi hàm: kq = ktra_ngto(n) Nếu kq = = 1 thì n là số nguyên tố Ngược lại, n không phải là số nguyên tố Bài 2: Thuật toán: Viết hàm (function) ktra_chinhphuong(int x): kiểm tra x có phải là số chính phương hay không Khai báo biến i, ktra Cho i chạy từ 1 đến x (nếu dùng sqrt(x) thì không cần lệnh thoát) Nếu i*i = x thì ktra = 1 Thoát Nếu ktra= =1 thì x là số chính phương Ngược lại, x không phải là số chính phương Chương trình chính: Khai báo biến n, kq Nhập số n Gọi hàm: kq = ktra_chinhphuong(n) Bài 3:
  25. Thuật toán: Viết hàm (function) Giaithua(int n):giai thừa của n Khai báo biến I, GT Cho i chạy từ 1 đến n GT=1 GT=GT*i Trả về kết quả tìm được cho hàm Giaithua Chương trình chính: Khai báo biến n Nhập số n Gọi hàm: kq = Giaithua(n) Bài 4: Thuật toán: Viết hàm (function) USCLN(a:integer, b:integer): integer :tìm ước số chung lớn nhất của hai số a và b Lấy trị tuyệt đối hai số a và b Chừng nào (a 0) làm Nếu a>b thì a=a-b Ngược lại, b=b-a Nếu a = 0 thì b là USCLN Ngược lại, a là USCLN Viết hàm (function) BSCNN(a:integer, b:integer):integer tìm bội số chung nhỏ nhất của hai số a và b BSCNN= (a*b)/USCLN(a,b) Chương trình chính:
  26. Khai báo biến a, b, US, BS Nhập hai số a,b Gọi hàm: US = USCLN(a, b) BS = BSCNN(a, b) Xuất BS, US ra màn hình Bài 5: Thuật toán: Viết hàm (function) solonnhat(a:integer, b:integer):integer tìm số lớn nhất trong hai số a và b Khai báo biến max Gán max = a Nếu max<b thì max=b Hàm solonnhat(a,b) trả về giá trị max Chương trình chính: Khai báo biến a, b, c, max Nhập hai số a, b, c Gọi hàm: max = solonnhat(a, b) max = solonnhat(max, c) Xuất max ra màn hình 5.4. Dùng dẫn hướng #define Mục tiêu: Sử dụng tiền xử lý #define.
  27. Sau đây là một vài ví dụ dùng dẫn hướng #define để định nghĩa hàm đơn giản #define AREA_CIRCLE (frad) (4*PI*frad*frad) //tinh dien tich hinh cau #define SUM (x, y) (x + y) //cong 2 so #define SQR (x) (x*x) //tinh x binh phuong #define MAX(x, y) (x > y) ? x : y //tim so lon nhat giua x va y #define ERROR (s) printf("%s.\n", s) //in thong bao voi chuoi s Ví dụ 7: Trong ví dụ 6 xóa từ dòng 20 đến dòng 24, xóa dòng 6, 7; thêm dòng AREA_CIRCLE (frad) (4*PI+frad*frad) vào sau dòng 5. Sửa dòng printf("Dien tich hinh cau: %10.3f.\n", area()); thành printf("Dien tich hinh cau: %10.3f.\n", AREA_CIRCLE(frad)); Chạy lại chương trình, quan sát và nhận xét kết quả. Ví dụ 8: 1 #include 2 #include 3 #define MAX(x, y) (x > y) ? x : y 4 void main(void) 5 { 6 float a = 4.5, b = 6.1; 7 printf("So lon nhat la: %5.2f.\n", MAX(a, b)); 8 getch(); 9 } Kết quả in ra màn hình So lon nhat la: 6.10 Thêm vào dòng 8 giá trị c = 10 Sửa lại dòng 9: MAX(a, b) thành MAX(MAX(a, b), c) Chạy lại chương trình, quan sát và nhận xét kết quả
  28. - 72 - CHƯƠNG 6: MẢNG VÀ CHUỖI Mã chương/ bài:MH18-06 Ý nghĩa: Trong C, mảng được dùng để biểu thị một cấu trúc của một dãy nhiều giá trị có cùng một kiểu được xếp thứ tự. Mục tiêu: - Trình bày được ý nghĩa, cách khai báo mảng, chuỗi; - Nhập xuất mảng, chuỗi; - Khởi tạo mảng chuỗi; - Trình bày một số kỹ thuật thao tác trên mảng, chuỗi; - Vận dụng được mảng làm tham số cho hàm; - Giải một số bài toán sử dụng kiểu mảng, chuỗi. - Thực hiện các thao tác an toàn với máy tính. 6.1. GIỚI THIỆU KIỂU DỮ LIỆU “KIỂU MẢNG” TRONG C Mục tiêu: Trình bày được ý nghĩa, cách khai báo mảng Mảng là một tập hợp các phần tử cố định có cùng một kiểu, gọi là kiểu phần tử. Kiểu phần tử có thể là có các kiểu bất kỳ: ký tự, số, chuỗi ký tự ; cũng có khi ta sử dụng kiểu mảng để làm kiểu phần tử cho một mảng (trong trường hợp này ta gọi là mảng của mảng hay mảng nhiều chiều). Ta có thể chia mảng làm hai loại: Mảng một chiều và mảng nhiều chiều. Mảng là kiểu dữ liệu được sử dụng rất thường xuyên. Chẳng hạn người ta cần quản lý một danh sách họ và tên của khoảng 100 sinh viên trong một lớp. Nhận thấy rằng mỗi họ và tên để lưu trữ ta cần một biến kiểu chuỗi, như vậy 100 họ và tên thì cần khai báo 100 biến kiểu chuỗi. Nếu khai báo như thế này thì đoạn khai báo cũng như các thao tác trên các họ tên sẽ rất dài dòng và rắc rối. Vì thế, kiểu dữ liệu mảng giúp ích ta trong trường hợp này; chỉ cần khai báo một biến, biến này có thể coi như là tương đương với 100 biến chuỗi ký
  29. - 73 - tự; đó là một mảng mà các phần tử của nó là chuỗi ký tự. Hay như để lưu trữ các từ khóa của ngôn ngữ lập trình C, ta cũng dùng đến một mảng để lưu trữ chúng. 6.2. MẢNG MỘT CHIỀU Mục tiêu: - Trình bày được ý nghĩa, cách khai báo mảng một chiều - Nhập xuất mảng một chiều - Khởi tạo mảng một chiều - Trình bày một số kỹ thuật thao tác trên mảng một chiều; - Vận dụng được mảng làm tham số cho hàm - Giải một số bài toán sử dụng kiểu mảng một chiều - Thực hiện các thao tác an toàn với máy tính. Nếu xét dưới góc độ toán học, mảng một chiều giống như một vector. Mỗi phần tử của mảng một chiều có giá trị không phải là một mảng khác. 6.2.1. Khai báo 6.2.1.1. Khai báo mảng với số phần tử xác định (khai báo tường minh) Cú pháp: Ý nghĩa: - Tên mảng: đây là một cái tên đặt đúng theo quy tắc đặt tên của danh biểu. Tên này cũng mang ý nghĩa là tên biến mảng. - Số phần tử: là một hằng số nguyên, cho biết số lượng phần tử tối đa trong mảng là bao nhiêu (hay nói khác đi kích thước của mảng là gì). - Kiểu: mỗi phần tử của mảng có dữ liệu thuộc kiểu gì. - Ở đây, ta khai báo một biến mảng gồm có số phần tử phần tử, phần tử thứ nhất là tên mảng [0], phần tử cuối cùng là tên mảng[số phần tử -1] Ví dụ: int a[10]; /* Khai báo biến mảng tên a, phần tử thứ nhất là a[0], phần tử cuối cùng là a[9].*/
  30. - 74 - Ta có thể coi mảng a là một dãy liên tiếp các phần tử trong bộ nhớ như sau: Vị trí 0 1 2 3 4 5 6 7 8 9 Tên phần tử Hình 1: Hình ảnh mảng a trong bộ nhớ 6.2.1.2. Khai báo mảng với số phần tử không xác định (khai báo không tường minh) Cú pháp: Khi khai báo, không cho biết rõ số phần tử của mảng, kiểu khai báo này thường được áp dụng trong các trường hợp: vừa khai báo vừa gán giá trị, khai báo mảng là tham số hình thức của hàm.  Vừa khai báo vừa gán giá trị Cú pháp: []= {Các giá trị cách nhau bởi dấu phẩy} Nếu vừa khai báo vừa gán giá trị thì mặc nhiên C sẽ hiểu số phần tử của mảng là số giá trị mà chúng ta gán cho mảng trong cặp dấu {}. Chúng ta có thể sử dụng hàm sizeof() để lấy số phần tử của mảng như sau: Số phần tử=sizeof(tên mảng)/ sizeof(kiểu)  Khai báo mảng là tham số hình thức của hàm Trong trường hợp này ta không cần chỉ định số phần tử của mảng là bao nhiêu. 6.2.2. Truy xuất từng phần tử của mảng Mỗi phần tử của mảng được truy xuất thông qua Tên biến mảng theo sau là chỉ số nằm trong cặp dấu ngoặc vuông [ ]. Chẳng hạn a[0] là phần tử đầu tiên của mảng a được khai báo ở trên. Chỉ số của phần tử mảng là một biểu thức mà giá trị là kiểu số nguyên. Với cách truy xuất theo kiểu này, Tên biến mảng[Chỉ số] có thể coi như là một biến có kiểu dữ liệu là kiểu được chỉ ra trong khai báo biến mảng. Ví dụ 1: int a[10]; Trong khai báo này, việc truy xuất các phần tử được chỉ ra trong hình 1. Chẳng hạn phần tử thứ 2 (có vị trí 1) là a[1] Ví dụ 2: Vừa khai báo vừa gán trị cho 1 mảng 1 chiều các số nguyên. In mảng số nguyên này lên màn hình.
  31. - 75 - Giả sử ta đã biết số phần tử của mảng là n; việc hiển thị 1 giá trị số nguyên lên màn hình ta cần sử dụng hàm printf() với định dạng %d, tổng quát hóa lên nếu muốn hiển thị lên màn hình giá trị của n số nguyên, ta cần gọi hàm printf() đúng n lần. Như vậy trong trường hợp này ta sử dụng 1 vòng lặp để in ra giá trị các phần tử. Ta có đoạn chương trình sau: #include #include void main() { int n,i,j,tam; int dayso[]={66,65,69,68,67,70}; clrscr(); n=sizeof(dayso)/sizeof(int); /*Lấy số phần tử*/ printf("\n Noi dung cua mang "); for (i=0;i #include void main() { unsigned int N; unsigned int Du; unsigned int NhiPhan[20],K=0,i; printf("Nhap vao so nguyen N= ");scanf("%d",&N); do { Du=N % 2; NhiPhan[K]=Du; /* Lưu số dư vào mảng ở vị trí K*/ K++; /* Tăng K lên để lần kế lưu vào vị trí kế*/ N = N/2; } while(N>0); printf("Dang nhi phan la: "); for(i=K-1;i>=0;i )
  32. - 76 - printf("%d",NhiPhan[i]); getch(); } Ví dụ 4: Nhập vào một dãy n số và sắp xếp các số theo thứ tự tăng. Đây là một bài toán có ứng dụng rộng rãi trong nhiều lĩnh vực. Có rất nhiều giải thuật sắp xếp. Một trong số đó được mô tả như sau: Đầu tiên đưa phần tử thứ nhất so sánh với các phần tử còn lại, nếu nó lớn hơn một phần tử đang so sánh thì đổi chỗ hai phần tử cho nhau. Sau đó tiếp tục so sánh phần tử thứ hai với các phần tử từ thứ ba trở đi cứ tiếp tục như vậy cho đến phần tử thứ n-1. Chương trình sẽ được chia thành các hàm Nhap (Nhập các số), SapXep (Sắp xếp) và InMang (In các số); các tham số hình thức của các hàm này là 1 mảng không chỉ định rõ số phần tử tối đa, nhưng ta cần có thêm số phần tử thực tế được sử dụng của mảng là bao nhiêu, đây là một giá trị nguyên. #include #include void Nhap(int a[],int N) { int i; for(i=0; i a[j]) { t=a[i]; a[i]=a[j]; a[j]=t; } }
  33. - 77 - void main() { int b[20], N; printf("So phan tu thuc te cua mang N= "); scanf("%d",&N); Nhap(b,N); printf("Mang vua nhap: "); InMang(b,N); SapXep(b,N); /* Gọi hàm sắp xếp*/ printf("Mang sau khi sap xep: "); InMang(b,N); getch(); } Kết quả chạy chương trình có thể là: 6.2.3. Bài tập thực hành  Mục đích yêu cầu Làm quen với kiểu dữ liệu có cấu trúc trong C, kiểu mảng. Thực hiện các bài tập trong phần nội dung bằng cách kết hợp kiểu dữ liệu mảng, các kiểu dữ liệu đã học và các phần đã học trong các bài tập trước.  Nội dung 1. Viết chương trình nhập vào một dãy n số thực a[0], a[1], , a[n-1], sắp xếp dãy số theo thứ tự từ lớn đến nhỏ. In dãy số sau khi sắp xếp. 2. Viết chương trình sắp xếp một mảng theo thứ tự tăng dần sau khi đã loại bỏ các phần tử trùng nhau. 3. Viết chương trình nhập vào một mảng, hãy xuất ra màn hình: - Phần tử lớn nhất của mảng. - Phần tử nhỏ nhất của mảng. - Tính tổng của các phần tử trong mảng .
  34. - 78 - 4. Viết chương trình nhập vào một dãy các số theo thứ tự tăng, nếu nhập sai quy cách thì yêu cầu nhập lại. In dãy số sau khi đã nhập xong. Nhập thêm một số mới và chèn số đó vào dãy đã có sao cho dãy vẫn đảm bảo thứ tự tăng. In lại dãy số để kiểm tra. 5. Viết chương trình nhập vào một mảng số tự nhiên. Hãy xuất ra màn hình: - Dòng 1 : gồm các số lẻ, tổng cộng có bao nhiêu số lẻ. - Dòng 2 : gồm các số chẵn, tổng cộng có bao nhiêu số chẵn. - Dòng 3 : gồm các số nguyên tố. - Dòng 4 : gồm các số không phải là số nguyên tố. 6. Viết chương trình tính tổng bình phương của các số âm trong một mảng các số nguyên. 7. Viết chương trình thực hiện việc đảo một mảng một chiều. Ví dụ : 1 2 3 4 5 7 9 10 đảo thành 10 9 7 5 4 3 2 1 . 6.3. MẢNG NHIỀU CHIỀU Mục tiêu: - Trình bày được ý nghĩa, cách khai báo mảng hai chiều - Nhập xuất mảng hai chiều - Khởi tạo mảng hai chiều - Trình bày một số kỹ thuật thao tác trên mảng hai chiều; - Vận dụng được mảng làm tham số cho hàm - Giải một số bài toán sử dụng kiểu mảng hai chiều - Thực hiện các thao tác an toàn với máy tính. Mảng nhiều chiều là mảng có từ hai chiều trở lên. Điều đó có nghĩa là mỗi phần tử của mảng là một mảng khác.
  35. - 79 - Người ta thường sử dụng mảng nhiều chiều để lưu các ma trận, các tọa độ 2 chiều, 3 chiều Phần dưới đây là các vấn đề liên quan đến mảng 2 chiều; các mảng 3, 4, chiều thì tương tự (chỉ cần tổng quát hóa lên). 6.3.1. Khai báo 6.3.1.1. Khai báo mảng 2 chiều tường minh Cú pháp: Ví dụ: Người ta cần lưu trữ thông tin của một ma trận gồm các số thực. Lúc này ta có thể khai báo một mảng 2 chiều như sau: float m[8][9]; /* Khai báo mảng 2 chiều có 8*9 phần tử là số thực*/ Trong trường hợp này, ta đã khai báo cho một ma trận có tối đa là 8 dòng, mỗi dòng có tối đa là 9 cột. Hình ảnh của ma trận này được cho trong hình 2: Dòng\Cột 0 1 2 3 4 5 6 7 8 0 m[0][0] m[0][1] m[0][2] m[0][3] m[0][4] m[0][5] m[0][6] m[0][7] m[0][8] 1 m[1][0] m[1][1] m[1][2] m[1][3] m[1][4] m[1][5] m[1][6] m[1][7] m[1][8] 2 m[2][0] m[2][1] m[2][2] m[2][3] m[2][4] m[2][5] m[2][6] m[2][7] m[2][8] 3 m[3][0] m[3][1] m[3][2] m[3][3] m[3][4] m[3][5] m[3][6] m[3][7] m[3][8] 4 m[4][0] m[4][1] m[4][2] m[4][3] m[4][4] m[4][5] m[4][6] m[4][7] m[4][8] 5 m[5][0] m[5][1] m[5][2] m[5][3] m[5][4] m[5][5] m[5][6] m[5][7] m[5][8] 6 m[6][0] m[6][1] m[6][2] m[6][3] m[6][4] m[6][5] m[6][6] m[6][7] m[6][8] 7 m[7][0] m[7][1] m[7][2] m[7][3] m[7][4] m[7][5] m[7][6] m[7][7] m[7][8] 6.3.1.2. Khai báo mảng hai chiều không tường minh Để khai báo mảng 2 chiều không tường minh, ta vẫn phải chỉ ra số phần tử của chiều thứ hai (chiều cuối cùng). Cú pháp: Cách khai báo này cũng được áp dụng trong trường hợp vừa khai báo, vừa gán trị hay đặt mảng 2 chiều là tham số hình thức của hàm.
  36. - 80 - 6.3.2. Truy xuất từng phần tử của mảng hai chiều Ta có thể truy xuất một phần tử của mảng hai chiều bằng cách viết ra tên mảng theo sau là hai chỉ số đặt trong hai cặp dấu ngoặc vuông. Chẳng hạn ta viết m[2][3]. Với cách truy xuất theo cách này, Tên mảng[Chỉ số 1][Chỉ số 2] có thể coi là 1 biến có kiểu được chỉ ra trong khai báo biến mảng. Ví dụ 1: Viết chương trình cho phép nhập 2 ma trận a, b có m dòng n cột, thực hiện phép toán cộng hai ma trận a,b và in ma trận kết quả lên màn hình. Trong ví dụ này, ta sẽ sử dụng hàm để làm ngắn gọn hơn chương trình của ta. Ta sẽ viết các hàm: nhập 1 ma trận từ bàn phím, hiển thị ma trận lên màn hình, cộng 2 ma trận. #include #include void Nhap(int a[][10],int M,int N) { int i,j; for(i=0;i<M;i++) for(j=0; j<N; j++) { printf("Phan tu o dong %d cot %d: ",i,j); scanf("%d",&a[i][j]); } } void InMaTran(int a[][10], int M, int N) { int i,j; for(i=0;i<M;i++) { for(j=0; j< N; j++) printf("%d ",a[i][j]); printf("\n"); } } /* Cong 2 ma tran A & B ket qua la ma tran C*/ void CongMaTran(int a[][10],int b[][10],int M,int N,int c[][10]) { int i,j; for(i=0;i<M;i++) for(j=0; j<N; j++)
  37. - 81 - c[i][j]=a[i][j]+b[i][j]; } int main() { int a[10][10], b[10][10], M, N; int c[10][10];/* Ma tran tong*/ printf("So dong M= "); scanf("%d",&M); printf("So cot M= "); scanf("%d",&N); printf("Nhap ma tran A\n"); Nhap(a,M,N); printf("Nhap ma tran B\n"); Nhap(b,M,N); printf("Ma tran A: \n"); InMaTran(a,M,N); printf("Ma tran B: \n"); InMaTran(b,M,N); CongMaTran(a,b,M,N,c); printf("Ma tran tong C:\n"); InMaTran(c,M,N); getch(); return 0; } Ví dụ 2: Nhập vào một ma trận 2 chiều gồm các số thực, in ra tổng của các phần tử trên đường chéo chính của ma trận này. Ta nhận thấy rằng giả sử ma trận a có M dòng, N cột thì các phần tử của đường chéo chính là các phần tử có dạng: a[i][i] với i ∈ [0 min(M,N)-1]. #include #include int main() { float a[10][10], T=0; int M, N, i,j, Min; clrscr(); printf("Ma tran co bao nhieu dong? "); scanf("%d",&M); printf("Ma tran co bao nhieu cot? ");scanf("%d",&N); for(i=0;i<M;i++) for(j=0; j<N; j++) { printf("Phan tu o dong %d cot %d: ",i,j); scanf("%f",&a[i][j]); } printf("Ma tran vua nhap: \n"); for(i=0;i<M;i++) { for(j=0; j< N; j++) printf("%.2f ",a[i][j]); printf("\n"); }
  38. - 82 - Min=(M>N) ? N: M; /* Tìm giá trị nhỏ nhất của M & N*/ for(i=0;i<Min;i++) T=T+a[i][i]; printf("Tong cac phan tu o duong cheo chinh la: %f", T); getch(); return 0; } 6.3.3. Bài tập thực hành 1. Viết chương trình nhập vào một ma trận (mảng hai chiều) các số nguyên, gồm m hàng, n cột. In ma trận đó lên màn hình. Nhập một số nguyên khác vào và xét xem có phần tử nào của ma trận trùng với số này không ? Ở vị trí nào ? Có bao nhiêu phần tử ? 2. Viết chương trình để chuyển đổi vị trí từ dòng thành cột của một ma trận (ma trận chuyển vị) vuông 4 hàng 4 cột. Sau đó viết cho ma trận tổng quát cấp m*n. Ví dụ: 1 2 3 4 1 2 9 1 2 5 5 8 2 5 4 5 9 4 2 0 3 5 2 8 1 5 8 6 4 8 0 6 2. Viết chương trình nhập vào hai ma trận A có cấp m, k và B có cấp k, n. In hai ma trận lên màn hình. Tích hai ma trận A và B là ma trận C được tính bởi công thức: cij= ai1*b1j + ai2 *b2j + ai3 *b3j + + aik *bkj (i=0,1,2, m-1;j=0,1,2 n-1) Tính ma trận tích C và in kết quả lên màn hình. 3. Xét ma trận A vuông cấp n, các phần tử a[i, i] ( i= 1 n ) được gọi là đường chéo chính của ma trận vuông A. Ma trận vuông A được gọi là ma trận tam giác nếu tất cả các phần tử dưới đường chéo chính đều bằng 0. Định thức của ma trận tam giác bằng tích các phần tử trên đường chéo chính. Ta có thể chuyển một ma trận vuông bất kỳ về ma trận tam giác bằng thuật toán:
  39. - 83 - - Xét cột i (i =0,1 n-2) - Trong cột i xét các phần tử a[k,i] ( k=i+1 n-1) + Nếu a[k,i]=0 thì tăng k lên xét phần tử khác + Nếu a[k,i] <> 0 thì làm như sau: Nhân toàn bộ hàng k với - a[i,i]/a[k,i] Lấy hàng i cộng vào hàng k sau khi thực hiện phép nhân trên. Đổi chỗ hai hàng i và k cho nhau Nhân toàn bộ hàng k với -1 sau khi đã đổi chỗ với hàng i Tăng k lên xét phần tử khác. Viết chương trình tính định thức cấp n thông qua các bước nhập ma trận, in ma trận, đưa ma trận về dạng tam giác, in ma trận tam giác, in kết quả tính định thức. 4. Viết chương trình thực hiện việc trộn hai dãy có thứ tự thành một dãy có thứ tự. Yêu cầu không được trộn chung rồi mới sắp thứ tự. Khi trộn phải tận dụng được tính chất đã sắp của hai dãy con. 5. Viết chương trình nhập vào hai ma trận A và B có cấp m, n. In hai ma trận lên màn hình. Tổng hai ma trận A và B là ma trận C được tính bởi công thức: cij= aij +bij ( i=0,1,2, m-1; j=0,1,2 n-1) Tính ma trận tổng C và in kết quả lên màn hình 6.4. CHUỖI Mục tiêu: - Trình bày được ý nghĩa, cách khai báo chuỗi; - Nhập xuất chuỗi; - Khởi tạo mảng chuỗi;
  40. - 84 - - Trình bày một số kỹ thuật thao tác trên chuỗi; - Giải một số bài toán sử dụng kiểu chuỗi. - Thực hiện các thao tác an toàn với máy tính. 6.4.1. KHÁI NIỆM Chuỗi ký tự là một dãy gồm các ký tự hoặc một mảng các ký tự được kết thúc bằng ký tự ‘\0’ (còn được gọi là ký tự NULL trong bảng mã Ascii). Các hằng chuỗi ký tự được đặt trong cặp dấu nháy kép “”. 6.4.2. KHAI BÁO 6.4.2.1. Khai báo theo mảng Cú pháp: char [Chiều dài tối đa] Ví dụ: Trong chương trình, ta có khai báo: char Ten[12]; Trong khai báo này, bộ nhớ sẽ cung cấp 12+1 bytes để lưu trữ nội dung của chuỗi ký tự Ten; byte cuối cùng lưu trữ ký tự ‘\0’ để chấm dứt chuỗi. Ghi chú: - Chiều dài tối đa của biến chuỗi là một hằng nguyên nằm trong khoảng từ 1 ến 255 bytes. - Chiều dài tối đa không nên khai báo thừa để tránh lãng phí bộ nhớ, nhưng cũng không nên khai báo thiếu. 6.4.2.2. Khai báo theo con trỏ Cú pháp: char * Ví dụ: Trong chương trình, ta có khai báo: char *Ten; Trong khai báo này, bộ nhớ sẽ dành 2 byte để lưu trữ địa chỉ của biến con trỏ Ten đang chỉ đến, chưa cung cấp nơi để lưu trữ dữ liệu. Muốn có chỗ để lưu trữ dữ liệu, ta phải gọi đến hàm malloc() hoặc calloc() có trong “alloc.h”, sau đó mới gán dữ liệu cho biến.
  41. - 85 -  Vừa khai báo vừa gán giá trị Cú pháp: char []= Ví dụ: #include #include int main() { char Chuoi[]="Mau nang hay la mau mat em” ; printf("Vua khai bao vua gan trị : %s”,Chuoi) ; getch(); return 0; } * Ghi chú: Chuỗi được khai báo là một mảng các ký tự nên các thao tác trên mảng có thể áp dụng đối với chuỗi ký tự. 6.4.3. CÁC THAO TÁC TRÊN CHUỖI KÝ TỰ 6.4.3.1. Nhập xuất chuỗi  Nhập chuỗi từ bàn phím Để nhập một chuỗi ký tự từ bàn phím, ta sử dụng hàm gets() Cú pháp: gets( ) Ví dụ: char Ten[20]; gets(Ten); Ta cũng có thể sử dụng hàm scanf() để nhập dữ liệu cho biến chuỗi, tuy nhiên lúc này ta chỉ có thể nhập được một chuỗi không có dấu khoảng trắng. Ngoài ra, hàm cgets() (trong conio.h) cũng được sử dụng để nhập chuỗi.  Xuất chuỗi lên màn hình Để xuất một chuỗi (biểu thức chuỗi) lên màn hình, ta sử dụng hàm puts(). Cú pháp: puts( ) Ví dụ: Nhập vào một chuỗi và hiển thị trên màn hình chuỗi vừa nhập. #include #include
  42. - 86 - #include int main() { char Ten[12]; printf("Nhap chuoi: ");gets(Ten); printf("Chuoi vua nhap: ");puts(Ten); getch(); return 0; } Ngoài ra, ta có thể sử dụng hàm printf(), cputs() (trong conio.h) để hiển thị chuỗi lên màn hình. 6.4.3.2. Một số hàm xử lý chuỗi (trong string.h)  Cộng chuỗi - Hàm strcat() Cú pháp: char *strcat(char *des, const char *source) Hàm này có tác dụng ghép chuỗi nguồn vào chuỗi đích. Ví dụ: Nhập vào họ lót và tên của một người, sau đó in cả họ và tên của họ lên màn hình. #include #include #include void main() { char HoLot[30], Ten[12]; printf("Nhap Ho Lot: "); gets(HoLot); printf("Nhap Ten: ");gets(Ten); strcat(HoLot,Ten); /* Ghep Ten vao HoLot*/ printf("Ho ten la: ");puts(HoLot); getch(); }  Xác định độ dài chuỗi - Hàm strlen() Cú pháp: int strlen(const char* s) Ví dụ: Sử dụng hàm strlen xác định độ dài một chuỗi nhập từ bàn phím. #include #include #include void main() {
  43. - 87 - char Chuoi[255]; int Dodai; printf("Nhap chuoi: ");gets(Chuoi); Dodai = strlen(Chuoi) printf("Chuoi vua nhap: ");puts(Chuoi); printf(“Co do dai %d”,Dodai); getch(); }  Đổi một ký tự thường thành ký tự hoa - Hàm toupper() Hàm toupper() (trong ctype.h) được dùng để chuyển đổi một ký tự thường thành ký tự hoa. Cú pháp: char toupper(char c)  Đổi chuỗi chữ thường thành chuỗi chữ hoa, hàm strupr() Hàm struppr() được dùng để chuyển đổi chuỗi chữ thường thành chuỗi chữ hoa, kết quả trả về của hàm là một con trỏ chỉ đến địa chỉ chuỗi được chuyển đổi. Cú pháp: char *strupr(char *s) Ví dụ: Viết chương trình nhập vào một chuỗi ký tự từ bàn phím. Sau đó sử dụng hàm strupr() để chuyển đổi chúng thành chuỗi chữ hoa. #include #include #include int main() { char Chuoi[255],*s; printf("Nhap chuoi: ");gets(Chuoi); s=strupr(Chuoi) ; printf(“Chuoi chu hoa: ”);puts(s); getch(); return 0; }  Đổi chuỗi chữ hoa thành chuỗi chữ thường, hàm strlwr() Muốn chuyển đổi chuỗi chữ hoa thành chuỗi toàn chữ thường, ta sử dụng hàm strlwr(), các tham số của hàm tương tự như hàm strupr() Cú pháp: char *strlwr(char *s)
  44. - 88 -  Sao chép chuỗi, hàm strcpy() Hàm này được dùng để sao chép toàn bộ nội dung của chuỗi nguồn vào chuỗi đích. Cú pháp: char *strcpy(char *Des, const char *Source) Ví dụ: Viết chương trình cho phép chép toàn bộ chuỗi nguồn vào chuỗi đích. #include #include #include int main() { char Chuoi[255],s[255]; printf("Nhap chuoi: ");gets(Chuoi); strcpy(s,Chuoi); printf(“Chuoi dich: ”);puts(s); getch(); return 0; }  Sao chép một phần chuỗi, hàm strncpy() Hàm này cho phép chép n ký tự đầu tiên của chuỗi nguồn sang chuỗi đích. Cú pháp: char *strncpy(char *Des, const char *Source, size_t n)  Trích một phần chuỗi, hàm strchr() Để trích một chuỗi con của một chuỗi ký tự bắt đầu từ một ký tự được chỉ định trong chuỗi cho đến hết chuỗi, ta sử dụng hàm strchr(). Cú pháp : char *strchr(const char *str, int Ghi chú: - Nếu ký tự đã chỉ định không có trong chuỗi, kết quả trả về là NULL. - Kết quả trả về của hàm là một con trỏ, con trỏ này chỉ đến ký tự c được tìm thấy đầu tiên trong chuỗi str.  Tìm kiếm nội dung chuỗi, hàm strstr() Hàm strstr() được sử dụng để tìm kiếm sự xuất hiện đầu tiên của chuỗi s2 trong chuỗi s1.
  45. - 89 - Cú pháp: char *strstr(const char *s1, const char *s2) Kết quả trả về của hàm là một con trỏ chỉ đến phần tử đầu tiên của chuỗi s1 có chứa chuỗi s2 hoặc giá trị NULL nếu chuỗi s2 không có trong chuỗi s1. Ví dụ: Viết chương trình sử dụng hàm strstr() để lấy ra một phần của chuỗi gốc bắt đầu từ chuỗi “hoc”. #include #include #include int main() { char Chuoi[255],*s; printf("Nhap chuoi: ");gets(Chuoi); s=strstr(Chuoi,”hoc”); printf(“Chuoi trich ra: ”);puts(s); getch(); return 0; }  So sánh chuỗi, hàm strcmp() Để so sánh hai chuỗi theo từng ký tự trong bảng mã Ascii, ta có thể sử dụng hàm strcmp(). Cú pháp: int strcmp(const char *s1, const char *s2) Hai chuỗi s1 và s2 được so sánh với nhau, kết quả trả về là một số nguyên (số này có được bằng cách lấy ký tự của s1 trừ ký tự của s2 tại vị trí đầu tiên xảy ra sự khác nhau). - Nếu kết quả là số âm, chuỗi s1 nhỏ hơn chuỗi s2. - Nếu kết quả là 0, hai chuỗi bằng nhau. - Nếu kết quả là số dương, chuỗi s1 lớn hơn chuỗi s2.  So sánh chuỗi, hàm stricmp() Hàm này thực hiện việc so sánh trong n ký tự đầu tiên của 2 chuỗi s1 và s2, giữa chữ thường và chữ hoa không phân biệt. Cú pháp: int stricmp(const char *s1, const char *s2) Kết quả trả về tương tự như kết quả trả về của hàm strcmp()
  46. - 90 -  Khởi tạo chuỗi, hàm memset() Hàm này được sử dụng để đặt n ký tự đầu tiên của chuỗi là ký tự c. Cú pháp: memset(char *Des, int c, size_t n)  Đổi từ chuỗi ra số, hàm atoi(), atof(), atol() (trong stdlib.h) Để chuyển đổi chuỗi ra số, ta sử dụng các hàm trên. Cú pháp : int atoi(const char *s) : chuyển chuỗi thành số nguyên long atol(const char *s) : chuyển chuỗi thành số nguyên dài float atof(const char *s) : chuyển chuỗi thành số thực Nếu chuyển đổi không thành công, kết quả trả về của các hàm là 0. Ngoài ra, thư viện string.h còn hỗ trợ các hàm xử lý chuỗi khác, ta có thể đọc thêm trong phần trợ giúp. 6.4.4. Bài tập thực hành  Mục đích yêu cầu Đi sâu vào kiểu dữ liệu chuỗi và các phép toán trên chuỗi.  Nội dung 1. Viết chương trình nhập một chuỗi ký tự từ bàn phím, xuất ra màn hình mã Ascii của từng ký tự có trong chuỗi. 2. Viết chương trình nhập một chuỗi ký tự từ bàn phím, xuất ra màn hình chuỗi đảo ngược của chuỗi đó. Ví dụ đảo của “abcd egh” là “hge dcba”. 3. Viết chương trình nhập một chuỗi ký tự và kiểm tra xem chuổi đó có đối xứng không. Ví dụ : Chuỗi ABCDEDCBA là chuỗi đối xứng. 4. Nhập vào một chuỗi bất kỳ, hãy đếm số lần xuất hiện của mỗi loại ký tự. 5. Viết chương trình nhập vào một chuỗi.
  47. - 91 - - In ra màn hình từ bên trái nhất và phần còn lại của chuỗi. Ví dụ: “Nguyễn Văn Minh” in ra thành: Nguyễn Văn Minh - In ra màn hình từ bên phải nhất và phần còn lại của chuỗi. Ví dụ: “Nguyễn Văn Minh” in ra thành: Minh Nguyễn Văn 6. Viết chương trình nhập vào một chuỗi rồi xuất chuỗi đó ra màn hình dưới dạng mỗi từ một dòng. Ví dụ: “Nguyễn Văn Minh” In ra : Nguyễn Văn Minh 7. Viết chương trình nhập vào một chuỗi, in ra chuỗi đảo ngược của nó theo từng từ. Ví dụ : chuỗi “Nguyễn Văn Minh” đảo thành “Minh Văn Nguyễn” 8. Viết chương trình đổi số tiền từ số thành chữ. 9. Viết chương trình nhập vào họ và tên của một người, cắt bỏ các khoảng trống không cần thiết (nếu có), tách tên ra khỏi họ và tên, in tên lên màn hình. Chú ý đến trường hợp cả họ và tên chỉ có một từ. 10. Viết chương trình nhập vào họ và tên của một người, cắt bỏ các khoảng trắng bên phải, trái và các khoảng trắng không có nghĩa trong chuỗi. In ra màn hình toàn bộ họ tên người đó dưới dạng chữ hoa, chữ thường. 11. Viết chương trình nhập vào một danh sách họ và tên của n người theo kiểu chữ thường, đổi các chữ cái đầu của họ, tên và chữ lót của mỗi người thành chữ hoa. In kết quả lên màn hình.
  48. - 92 - 12. Viết chương trình nhập vào một danh sách họ và tên của n người, tách tên từng người ra khỏi họ và tên rồi sắp xếp danh sách tên theo thứ tự từ điển. In danh sách họ và tên sau khi đã sắp xếp.
  49. - 93 - TÀI LIỆU THAM KHẢO [1]. Tiến Sĩ Lê Mạnh Thạnh, Giáo trình môn lập trình C, Nhà xuất bản giáo dục, năm 2000. [2]. Nguyễn Linh Giang, Nguyễn Xuân Thực, Lê Văn Thái, Giáo trình kỹ thuật lập trình C, Nhà xuất bản giáo dục, Năm 2005 [3]. GS. Phạm Văn Ất chủ biên, ThS. Nguyễn Hiếu Cường, ThS.Đỗ Văn Tuấn, Lê Trường Thông, Giáo trình kỹ thuật lập trình C, Nhà xuất bản Hồng Đức, Năm 2009 [4]. GS. Phạm Văn Ất, Kỹ Thuật Lập Trình C - Cơ Sở Và Nâng Cao, Nhà xuất bản giao thông vận tải, Năm 2006 [5]. HanoiAptech Computer Education Center , Giáo trình lập trình C căn bản [6]. Đại học Cần Thơ , Giáo trình lập trình C căn bản
  50. - 94 - DANH SÁCH BAN BIÊN SOẠN GIÁO TRÌNH DẠY NGHỀ TRÌNH ĐỘ CAO ĐẲNG Tên giáo trình: Lập trình C Tên nghề: Quản trị mạng 1. Bà Trần Thị Hà Khuê Chủ nhiệm 2. Bà Võ Thị Ngọc Tú Thành viên 3. Ông Dương Hiển Tú Thành viên
  51. - 95 - DANH SÁCH HỘI ĐỒNG NGHIỆM THU GIÁO TRÌNH DẠY NGHỀ TRÌNH ĐỘ TRUNG CẤP, CAO ĐẲNG 1. Ông (bà) Chủ tịch 2. Ông (bà) Phó chủ tịch 3. Ông (bà) Thư ký 4. Ông (bà) Thành viên 5. Ông(bà) Thành viên 6. Ông(bà) Thành viên 7. Ông(bà) Thành viên 8. Ông(bà) Thành viên 9. Ông(bà) Thành viên