Bài giảng Lập trình - Chương 2: Tổng quan về C/C++

pdf 83 trang Gia Huy 17/05/2022 4571
Bạn đang xem 20 trang mẫu của tài liệu "Bài giảng Lập trình - Chương 2: Tổng quan về C/C++", để 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:

  • pdfbai_giang_lap_trinh_chuong_2_tong_quan_ve_cc.pdf

Nội dung text: Bài giảng Lập trình - Chương 2: Tổng quan về C/C++

  1. Lập trình Chương 2: Tổng quan về C/C++ , HOÀNG MINHSƠNHOÀNG , 2004 1 
  2. Nội dung chương 2 2.1 Tổ chức chương trình C/C++ 2.2 Biến và các kiểu dữ liệu cơ bản 2.3 Các kiểu dữ liệu dẫn xuất trực tiếp 2.4 Định nghĩa kiểu dữ liệu mới 2.5 Các lệnh điều khiển chương trình 2.6 Hàm và thư viện Chương 2: Tổng quan về C và C++ 2
  3. 2.1 Tổ chức chương trình C/C++ Tạo dự ná dùng visual studio .net 2008 Chương 2: Tổng quan về C và C++ 3
  4. Chương trình tính giai thừa: Phiên bản C #include Lệnh tiền xử lý: Khai báo sử dụng hàm thư viện #include Khai báo hàm int gthua(int); Chương trình chính void main() { char c = 'N'; Khai báo biến Lời chú thích int N = 1; int kq; do { printf(“\n Nhap mot nguyen duong"); /* Viet ra man hinh*/ scanf("%d",&N); /* Nhap tu ban phim cho N*/ kq = gthua(N); printf(“\n Giai thua cua %d la %d", N, kq); printf(“\n Ban co muon tiep tuc khong? Y/N"); c = getch(); } while (c=='y' || c=='Y'); } int gthua(int n) { int kq = 1; while (n > 1) Định nghĩa hàm (thân hàm) kq *= n ; return kq; } Chương 2: Tổng quan về C và C++ 4
  5. Chương trình tính giai thừa: Phiên bản C++ #include #include int gthua(int); void main() { char c = 'N'; int N = 1; do { cout > N; int kq = gthua(N); cout 1) kq *= n ; return kq; } Chương 2: Tổng quan về C và C++ 5
  6. Qui tắc soạn thảo mã nguồn 1. Tên biến, tên hàm, tên kiểu mới: . Tránh sử dụng các từ khóa và tên kiểu cơ sở . Chỉ được dùng các ký tự sau: ‘A’ ’Z’, ‘a’ ’z’, ‘0’ ’9’, ‘_’ . Chỉ được bắt đầu bằng chữ cái hoặc dấu _ . Phân biệt giữa chữ hoa và chữ thường. . Nên đặt tên ngắn gọn và có ý nghĩa (có tính mô tả) 2. Sau mỗi câu lệnh có chấm phảy (;) 3. Đoạn { } được coi là nhóm lệnh, không có dấu chấm phảy sau đó, trừ khi khai báo kiểu 4. Cấu trúc mã nguồn theo kiểu phân cấp => dễ đọc 5. Bổ sung chú thích hợp lý (/*chú thích một đoạn*/ hoặc //chú thích một dòng) 6. Chia một file lớn thành nhiều file nhỏ Chương 2: Tổng quan về C và C++ 6
  7. Các từ khóa trong C auto double int struct break else long switch case enum register typedef char extern return union const float short unsigned continue for signed void default goto sizeof volatile do if static while Chương 2: Tổng quan về C và C++ 7
  8. Từ khóa trong C++ asm auto bool break case catch char class const const_cast continue default delete else extern do enum false double explicit float dynamic_cast export for friend goto if inline int long mutable namespace new operator private protected public register reinterpret_cast return short signed sizeof static static_cast struct switch template this throw true try typedef typeid typename union unsigned using virtual void volatile wchar_t while Chương 2: Tổng quan về C và C++ 8
  9. Biên dịch (compile) . Trong Visual studio .Net 2008: GọiCompile (Ctrl + F7) để biên dịch riêng rẽ hoặc Build (F6) để kết hợp biên dịch và liên kết cho toàn bộ dự án . Các kiểu lỗi biên dịch (compile error): – Lỗi cú pháp: Sử dụng tên sai qui định hoặc chưa khai báo, thiếu dấu chấm phảy ;, dấu đóng } – Lỗi kiểu: Các số hạng trong biểu thức không tương thích kiểu, gọi hàm với tham số sai kiểu – . Các kiểu cảnh báo biên dịch (warning): – Tự động chuyển đổi kiểu làm mất chính xác – Hàm khai báo có kiểu trả về nhưng không trả về – Sử dụng dấu = trong trường hợp nghi vấn là so sánh == – Chương 2: Tổng quan về C và C++ 9
  10. Liên kết (link) . Liên kết là quá trình ghép nhiều file đích (*.obj, *.lib) để tạo ra chương trình chạy cuối cùng *.exe . Trong Visual studio .Net 2008: Gọi Build (F6) . Lỗi liên kết có thể là do: – Sử dụng hàm nhưng không có định nghĩa hàm – Biến hoặc hàm được định nghĩa nhiều lần – Chương 2: Tổng quan về C và C++ 10
  11. Chạy thử và gỡ rối (debug) . Chạy thử trong Visual studio .Net 2008: Start without debugging hoặc Ctrl+F5 . Tìm lỗi: – Lỗi khi chạy là lỗi thuộc về phương pháp, tư duy, thuật toán, không phải về cú pháp – Lỗi khi chạy bình thường không được báo – Lỗi khi chạy rất khó phát hiện, vì thế trong đa số trường hợp cần tiến hành debug. . Chạy Debug trong Visual studio .Net 2008: – Chạy tới chỗ đặt cursor: F9 – Chạy từng dòng lệnh: F10 – Chạy vào trong hàm: F11 – Chạy tiếp bình thường: F5 – Xem kết quả dưới cửa sổ Output hoặc gọi QuickWatch Chương 2: Tổng quan về C và C++ 11
  12. Tổ chức bộ nhớ Hệ điều hành Các CT khác Mã chương trình Đỉnh ngăn xếp SP Dữ liệu toàn cục n Array_A Vùng nhớ tự do f Ngăn xếp k (tham biến, biến i cục bộ) count a Vùng nhớ tự do Chương 2: Tổng quan về C và C++ 12
  13. 2.2 Biến và dữ liệu . Biểu thức = dữ liệu + phép toán . Biểu diễn dữ liệu: Thông qua biến hoặc hằng số, kèm theo kiểu . Nội dung trong phần này: – Các kiểu dữ liệu cơ bản – Các phép toán áp dụng – Tương thích và chuyển đổi kiểu – Khai báo biến, phân loại biến Chương 2: Tổng quan về C và C++ 13
  14. 2.2.1 Các kiểu dữ liệu cơ bản của C/C++ Kiểu Kích cỡ thông dụng Phạm vi tối thiểu (tính bằng bit) char 8 −127 to 127 signed char 8 −127 127 unsigned char 8 0 255 int 16/32 −32767 32767 signed int 16/32 -nt- unsigned int 16/32 0 65535 short 16 −32767 32767 signed short 16 nt unsigned short 16 0 65535 long 32 −2147483647 2147483647 signed long 32 nt unsigned long 32 0 4294967295 float 32 Độ chính xác 6 chữ số double 64 Độ chính xác 10 chữ số long double 80 Độ chính xác 10 chữ số bool (C++) - - wchar_t (C++) 16 −32767 32767 Chương 2: Tổng quan về C và C++ 14
  15. Các phép toán cơ bản Phép toán Ký hiệu Kiểu nguyên Kiểu số thực Kiểu bool Gán = X X X Số học +, -, *, /, X X x +=, -=, *=, /= %, %= X x ++, X x So sánh >, =, >, >= X x Lựa chọn ? : X X X Lũy thừa? Không có! Chương 2: Tổng quan về C và C++ 15
  16. Tương thích và chuyển đổi kiểu . Những kiểu dữ liệu tương thích nhau có thể tự động chuyển đổi kiểu: – Giữa các kiểu số nguyên với nhau (lưu ý phạm vi giá trị) – Giữa các kiểu số thực với nhau (lưu ý độ chính xác) – Giữa các kiểu số nguyên và số thực (lưu ý phạm vi giá trị và độ chính xác) – Kiểu bool sang số nguyên, số thực: true => 1, false => 0 – Số nguyên, số thực sang kiểu bool: 0 => true, 0 => false . Nếu có lỗi hoặc cảnh báo => khắc phục bằng cách ép chuyển đổi kiểu – VD: i = int(5.1) % 2; //C j = (int)5.1 + 2; // C++ Chương 2: Tổng quan về C và C++ 16
  17. 2.2.2 Khai báo biến char c = 'N'; Khai báo và khởi tạo giá trị bool b = true; int kq; Chỉ khai báo, giá trị bất định double d; long count, i=0; Khai báo kết hợp, chỉ i=0 unsigned vhexa=0x00fa; Đặt giá trị đầu hexa unsigned voctal=082; Đặt giá trị đầu octal -> 66 chứ không phải 82 . C: Toàn bộ biến phải khai báo ngay đầu thân hàm . C++: Có thể khai báotại chỗ nào cần . Phân loại biến: – Biến toàn cục: Khai báo ngoài hàm, lưu giữ trong vùng nhớ dữ liệu chương trình – Biến cục bộ: Khai báo trong thân hàm, lưu giữ trong ngăn xếp (vùng nhớ tự do) – Tham biến: Khai báo trên danh sách tham số của hàm, lưu giữ trong ngăn xếp Chương 2: Tổng quan về C và C++ 17
  18. Ví dụ khai báo các loại biến Biến toàn cục int N = 1; void main() { char c = 'N'; do { cout 0:"; cin>>N; Biến cục bộ int kq = gthua(N); } while (c == 'y' || c == 'Y') } Hai biến cục bộ cùng tên ở hai phạm int gthua(int n) { vi khác nhau, int kq = 1; không liên quan gì while (n > 1) đến nhau! kq *= n ; return kq; } Tham biến Chương 2: Tổng quan về C và C++ 18
  19. 2.3 Các kiểu dữ liệu dẫn xuất trực tiếp . Kiểu liệt kê . Kiểu hằng . Kiểu con trỏ . Kiểu mảng . Kiểu tham chiếu (C++) Chương 2: Tổng quan về C và C++ 19
  20. 2.3.1 Kiểu liệt kê (enum) . Tương tự như một kiểu số nguyên. . Được dùng khi: – Cần hạn chế phạm vi sử dụng – Hoặc khi muốn sử dụng tên cụ thể thay cho hằng số nguyên . Ví dụ void main() { enum Color {Red, Green, Blue}; enum WeekDay {Mon = 2,Tue, Wed, Thu, Fri, Sat, Sun = 1 }; //Tue=3, Wed=4 int n=Red; cout<<n; //in ra số 0 Color c=Green; n=c; //n=1; } . Lưu ý: không thể gán trực tiếp một giá trị nguyên cho một biến enum mà phải ép kiểu VD: c=2;//Error c=Color(2);//OK Chương 2: Tổng quan về C và C++ 20
  21. 2.3.2 Kiểu hằng (const) . Khi muốn một biến không bị thay đổi trong suốt chương trình => khai báo hằng số void main() { const double pi = 3.1412; const int x = 1; pi = 3.14; // error x = 2; // error } Chương 2: Tổng quan về C và C++ 21
  22. 2.3.3 Kiểu con trỏ . Trong chương trình, mỗi biến chiếm một số byte nhất định và lưu ở một vị trí cụ thể trong bộ nhớ máy tính . Con trỏ là một biến mang địa chỉ của một ô nhớ (của một biến khác hoặc một hàm) . Cách khai báo: Data_type* pointer_var_name; int x = 2; int* p = &x; // p chua dia chi cua x *p = 3; // x=3 int y; p = &y; // p chua dia chi cua y *p = 5; // y = 5, x = 3 Chương 2: Tổng quan về C và C++ 22
  23. Kết luận về con trỏ . Con trỏ là một biến chứa địa chỉ byte đầu của một biến dữ liệu, được sử dụng để truy cập gián tiếp dữ liệu đó . Nếu khai báo con trỏ mà không khởi tạo, mặc định con trỏ mang một địa chỉ bất định . Để khởi tạo giá trị cho con trỏ, ta có thể cho nó trỏ vào một địa chỉ nhớ cụ thể hoặc địa chỉ của một biến khác thông qua toán tử lấy địa chỉ & . Con trỏ có thể mỗi lúc đại diện cho một biến dữ liệu khác, do đó địa chỉ màcon trỏ mang có thểthay đổi được. . Dùng toán tử * để truy nhập nội dung của biến mà con trỏ trỏ tới, có thể đọc hoặc thay đổi nội dung của biến đó. . Không bao giờ sử dụng toán tử truy nhập nội dung, nếu con trỏ chưa mang một địa chỉ ô nhớ mà chương trình có quyền kiểm soát Chương 2: Tổng quan về C và C++ 23
  24. 2.3.4 Kiểu mảng Chỉ số 0 1 2 N-1 địa chỉ đầu địa chỉ cuối . Số phần tử của mảng: N . Số lượng các phần tử của mảng (tĩnh) là cố định . Các phần tử có cùng kiểu . Các phần tử được sắp xếp kế tiếp trong bộ nhớ . Có thể truy nhập từng phần tử một cách tự do theo chỉ số hoặc theo địa chỉ Chương 2: Tổng quan về C và C++ 24
  25. Khai báo mảng . Số phần tử của mảng phải là hằng số nguyên. . Khai báo không khởi tạo: int a[5]; enum {index = 5}; double b[index]; const int N = 2; char c[N]; . Khai báo với số phần tử và khởi tạo giá trị các phần tử int d[3]= {1, 2, 3}; double e[5]= {1, 2, 3}; char f[4]= {0}; Chương 2: Tổng quan về C và C++ 25
  26. Khai báo mảng (tiếp) . Khai báo và khởi tạo giá trị các phần tử, số phần tử được tự động xác định int a[]= {1, 2, 3, 4, 5}; double b[]= {1, 2, 3}; double c[]= {0}; char s[]= {‘a’}; . Khai báo mảng nhiều chiều double M[2][3]; int X[3][]={{1,2},{3,4},{5,6}}; short T[3][2]={1,2,3,4,5,6}; Chương 2: Tổng quan về C và C++ 26
  27. Mảng đặc biệt: Chuỗi ký tự . Trong C/C++, chuỗi ký tự không phải là kiểu cơ bản, mà thực chất là một mảng . Phân biệt chuỗi ký tự thường và chuỗi ký tự kết 0 char s1[]= {‘H',‘E',‘L',‘L',‘O‘}; char s2[]= "HELLO"; wchar_t s3[]= L"HELLO"; . Đa số các hàm trong thư viện C làm việc với chuỗi ký tự kết 0 . Với C++, chuỗi ký tự được định nghĩa bằng lớp string trong thư viện chuẩn, không sử dụng byte kết 0 Chương 2: Tổng quan về C và C++ 27
  28. Con trỏ và mảng . Tên mảng là địa chỉ của phần tử đầu tiên của mảng. void main() { int a[3]; // a co 3 phan tu, gia tri chua xac dinh int* p; p = a; // p tro den a[0] p = &a[0]; // tuong tu nhu tren *p = 1; // a[0]=1 ++p; // p tro den a[1] *p = 2; // a[1]=2 p++; // p tro den a[2] *p = 3; // a[2]=3 } Chương 2: Tổng quan về C và C++ 28
  29. Kết luận về mảng . Mảng là một tập hợp các dữ liệu cùng kiểu, sắp xếp liền kề trong bộ nhớ . Số phần tử của mảng (tĩnh) là cố định (khi khai báo phải là hằng số), không bao giờ thay đổi được . Có thể truy cập các phần tử mảng thông qua tên mảng kèm theo chỉ số hoặc thông qua biến con trỏ (theo địa chỉ của từng phần tử) . Biến mảng (tĩnh) thực chất là một con trỏ hằng, mang địa chỉ của phần tử đầu tiên . Có thể đặt giá trị đầu cho các phần tử của mảng qua danh sách khởi tạo. . Không gán được hai mảng cho nhau. Nếu cần sao chép hai mảng thì phải sử dụng hàm . Không bao giờ được phép truy nhập với chỉ số nằm ngoài phạm vi, nếu N là số phần tử thì phạm vi cho phép là từ 0 N-1 . Con trỏ không phải là một mảng, nó chỉ có thể mang địa chỉ của một mảng và sử dụng để quản lý mảng (dù là động hay tĩnh) Chương 2: Tổng quan về C và C++ 29
  30. 2.3.5 Kiểu tham chiếu (C++) . Một biến tham chiếu là một biến đại diện trực tiếp cho một biến khác (dùng để làm bí danh cho biến đó và nó sử dụng vùng nhớ của biến này) . Biến tham chiếu không được cấp phát bộ nhớ, không có địa chỉ riêng. . Ý nghĩa: sử dụng chủ yếu trong truyền tham số cho hàm void main() { int a = 1; int& r = a; // r tham chieu den a r = 2; // a = 2 int& r2; // error int& r3 = 0; // error int b = 0; r = b; // r = 0, a=0 r = 3; // r = a= 3, b=0 } Chương 2: Tổng quan về C và C++ 30
  31. Hằng tham chiếu . Có thể tham chiếu đến một biến hoặc một hằng . Không cho phép dùng hằng tham chiếu để thay đổi giá trị vùng nhớ mà nó tham chiếu. . Ý nghĩa: thường được sử dụng làm đối của hàm để cho phép hàm sử dụng giá trị của các tham số trong lời gọi hàm nhưng tránh không làm thay đổi giá trị của các tham số . VD: int n=10, m; const int &r=n; n++;//OK m=2*n;//m=20 r=r+1;//Error Chương 2: Tổng quan về C và C++ 31
  32. Kết luận về biến tham chiếu . Khi khai báo phải chỉ rõ nó tham chiếu đến biến nào. . Có thể tham chiếu đến một phần tử của mảng . Không cho phép khai báo mảng tham chiếu . Có thể tham chiếu đến một hằng. Khi đó nó sẽ sử dụng vùng nhớ của hằng và nó có thể làm thay đổi giá trị chứa trong vùng nhớ này. Chương 2: Tổng quan về C và C++ 32
  33. 2.3.6 Typedef . Từ khóa typedef tạo ra một tên mới cho một kiểu dữ liệu có sẵn, không định nghĩa một kiểu mới . Ý nghĩa: đưa tên mới dễ nhớ, phù hợp với ứng dụng cụ thể. . Cú pháp: typedef Kieu_dang_co Ten_moi void main() { typedef int tuoi; /* Tạo tên mới cho kiểu int là “tuoi" */ typedef char ten[30]; /* Tạo tên mới cho kiểu char là “ten"*/ tuoi sv_tuoi, gv_tuoi; ten sv_ten, gv_ten; } Chương 2: Tổng quan về C và C++ 33
  34. 2.4 Định nghĩa kiểu dữ liệu mới . Cấu trúc (struct): – Tập hợp những dữ liệudù ng để mô tả một đối tượng, truy nhập theo tên (biến thành viên). – Thông dụng nhất trong C, ý nghĩa được mở rộng trong C++ . Hợp nhất (union): – Một tên kiểu chung cho nhiều dữ liệu khác nhau (chiếm cùng chỗ trong bộ nhớ). – Ít thông dụng trong cả C và C++ . Lớp (class): – mở rộng struct cũ – thêm những hàm thành viên. – Chỉ có trong C++ Chương 2: Tổng quan về C và C++ 34
  35. Kiểu struct . Cú pháp khai báo: struct struct_name { //khai báo các thành phần của struct type1 field1; type2 field2; } [bien_co_kieu_struct_name]; . Cách truy nhập vào các thuộc tính: Ten_bien_kieu_struct.Ten_thuoc_tinh. . Khởi tạo giá trị cho biến struct giống như với mảng, đặt các giá trị trong dấu { }, phân cách nhau bằng dấu “,”. Chương 2: Tổng quan về C và C++ 35
  36. Ví dụ minh họa struct #include struct emp { int MaSo; char ten[35]; }; void main() { struct emp e1={100,”Nguyen X”}; C++ cout<<“Eno: ” <<e1.Maso<<endl; cout<<“Ename: ”<<e1.ten; } Chương 2: Tổng quan về C và C++ 36
  37. Kết luận về struct . Cấu trúc (struct) được sử dụng để nhóm các dữ liệu liên quan mô tả một đối tượng, các dữ liệu có thể cùng hoặc khác kiểu . Định nghĩa kiểu cấu trúc bằng cách khai báo tên các biến thành viên. Định nghĩa kiểu cấu trúc chưa phải là định nghĩa các biến cụ thể, vì thế không được đặt giá trị đầu cho các biến . Kích cỡ của cấu trúc >= tổng kích cỡ các thành viên . Truy cập một biến cấu trúc: tên_biến_cấu_trúc.tên_biến_thành_viên . Các kiểu cấu trúc có thể lồng vào nhau, trong cấu trúc có thể sử dụng mảng, một mảng có thể có các phần tử là cấu trúc, v.v . Các biến có cùng kiểu cấu trúc có thể gán cho nhau, có thể sử dụng để khởi tạo cho nhau (khác hẳn với mảng) . Có thể sử dụng con trỏ để truy nhập dữ liệu cấu trúc thông qua toán tử toán tử (*.) hoặc (->) Chương 2: Tổng quan về C và C++ 37
  38. Hợp nhất (union) . Cú pháp khai báo: union union_name { //Khai báo các thành phần của union Type1 field1; Type2 field2; }; . Ví dụ: union VD { unsigned long u; unsigned char a[4]; }; void main() { VD vd1; vd1.u=0xDDCCBBAA; cout<<vd1.a[0]; cout<<vd1.a [1]; cout<<vd1.a[2]; cout<<vd1.a [3]; } Chương 2: Tổng quan về C và C++ 38
  39. Kết luận về hợp nhất . Hợp nhất (union) là một tập hợp (không có cấu trúc chặt chẽ) chứa các biến sử dụng chung ô nhớ, ở mỗi ngữ cảnh chỉ sử dụng một biến riêng biệt . Union thường được sử dụng khi dữ liệu đầu vào có thể có kiểu khác nhau . Các thành viên của một union không liên quan đến nhau, không cùng nhau tạo thành một thực thể thống nhất . Kích cỡ của union bằng kích cỡ của biến lớn nhất . Khai báo kiểu union tương tự như khai báo struct, nhưng ý nghĩa khác hẳn . Truy nhập biến thành viên cũng tương tự như struct, có thể qua biến trực tiếp hoặc qua biến con trỏ. . Union có thể chứa struct, struct có thể chứa union, union có thể chứa mảng, các phần tử của mảng có thể là union. Chương 2: Tổng quan về C và C++ 39
  40. Các lệnh điều khiển chương trình . Lệnh if else . Lệnh switch case . Lệnh while(dieu_kien) . Lệnh do while(dieu_kien) . Lệnh for . Lệnh break thoát khỏi một vòng lặp . Lệnh continue . Lệnh goto Chương 2: Tổng quan về C và C++ 40
  41. Cấu trúc if-else . Cú pháp: if ( expression ) stmt1; Hoặc if ( expression ) stmt1; else stmt2; . Khi số câu lệnh trong mỗi nhánh rẽ lớn hơn 1 thì đặt chúng trong dấu { } Chương 2: Tổng quan về C và C++ 41
  42. Ví dụ về if-else if (npoints >= 60) cout = 80 && npoints = 90) cout = 80) cout = 70) cout = 60) cout << ‘D’; else cout << ‘F’; Chương 2: Tổng quan về C và C++ 42
  43. Cấu trúc switch-case . Cú pháp: . Ví dụ: switch( expression ) switch( c ) { { case constant-expr1: case 'Y': stmt; cout<< "Yes" ; break; break; case constant-expr2: stmt; case 'N': break; cout<<"No" ; . . . break; default: default: stmt cout<< "What?" ; } } Chương 2: Tổng quan về C và C++ 43
  44. Cấu trúc switch-case ( ) . Có thể dùng nhiều nhãn case trên cùng một nhánh switch( c ) { case 'Y': case 'y': cout<< "Yes" ; break; case 'N': case 'n': /* 2 điều khiển trên cùng1 dòng */ cout<< "No"; break; default: cout<<"What?"; } Chương 2: Tổng quan về C và C++ 44
  45. Cấu trúc while . Cú pháp: while ( expression ) stmt ; . Lệnh (khối lệnh) stmt chỉ được thực hiện khi expression là đúng. . Ví dụ: x = 1; while ( (x * 5) >= (x * x) ) { cout<<x; x++; } Chương 2: Tổng quan về C và C++ 45
  46. Cấu trúc do while . Cú pháp: do stmt ; while ( expression ); . Lệnh (khối lệnh) stmt được thực hiện cho đến khi expression còn đúng. . Ví dụ: lng = 0; do { c = getchar(); ++lng; } while ( c != '\n' ); cout<< "line length is: “<<lng ; Chương 2: Tổng quan về C và C++ 46
  47. Cấu trúc for . Cú pháp: for ( init-expr; cont-expr; loop-expr ) stmt; . Thực hiện: . Tính toán init-expr nếu có . Tính toán cont-expr nếu có . Nếu cont-expr là true hoặc được bỏ qua thì thực hiện stmt, sau đó loop-expr được thực hiện . Lặp lại bước 2 đến khi cont-expr là false Chương 2: Tổng quan về C và C++ 47
  48. Cấu trúc for ( ) . Ví dụ: /* Bảng cửu chương*/ 9 times 0 is 0 const int max=9 9 times 1 is 9 for ( i = 0; i <= max; i++ ) 9 times 2 is 18 { 9 times 3 is 27 for ( j = 0; j <= max; j++ ) 9 times 4 is 36 { 9 times 5 is 45 cout<<i<< “ times”<<j<<“ is”<<i*j<“\n”; 9 times 6 is 54 } 9 times 7 is 63 } 9 times 8 is 72 9 times 9 is 81 Chương 2: Tổng quan về C và C++ 48
  49. 2.6 Hàm và thư viện Hàm là gì? . Một đơn vị tổ chức chương trình, một đoạn mã chương trình có cấu trúc để thực hiện một chức năng nhất định, cógiá trị sử dụng lại . Các hàm có quan hệ với nhau thông qua lời gọi, các biến tham số (đầu vào, đầu ra) và giá trị trả về . Cách thực hiện cụ thể một hàm phụ thuộc nhiều vào dữ kiện (tham số, đối số của hàm): – Thông thường, kết quả thực hiện hàm mỗi lần đều giống nhau nếu các tham số đầu vào như nhau – Một hàm không có tham số thì giá trị sử dụng lại rất thấp . Trong C/C++: Không phân biệt giữa thủ tục và hàm, cả đoạn mã chương trình chính cũng là hàm Chương 2: Tổng quan về C và C++ 49
  50. Ví dụ . Yêu cầu bài toán: Tính tổng một dãy số nguyên (liên tục) trong phạm vi do người sử dụng nhập. In kết quả ra màn hình. . Các nhiệm vụ: – Nhập số nguyên thứ nhất:  Yêu cầu người sử dụng nhập  Nhập số vào một biến – Nhập số nguyên thứ hai  Yêu cầu người sử dụng nhập  Nhập số vào một biến – Tính tổng với vòng lặp – Hiển thị kết quả ra màn hình Chương 2: Tổng quan về C và C++ 50
  51. Ví dụ: Cách 1 #include void main() { int a, b; char c; do { cout > a; cout > b; int tong = 0; for (int i = a; i > c; } while (c == 'y' || c == 'Y'); } Chương 2: Tổng quan về C và C++ 51
  52. Cách 2 (phân hoạch hàm) #include int Nhap(); int Tinh_tong(int,int); void Hien_thi(int a, int b, int kq); void main() { char c; do { int a = Nhap(); int b = Nhap(); int T = Tinh_tong(a,b); Hien_thi(a,b,T); cout > c; } while (c == 'y' || c == 'Y'); } Chương 2: Tổng quan về C và C++ 52
  53. Cách 2 ( ) int Nhap() { cout > N; return N; } int Tinh_tong(int a, int b) { int T = 0; for (int i = a; i <= b; ++i) T += i; return T; } void Hien_thi (int a, int b, int kq) { cout << "Tong tu " << a << " den " << b << " la " << kq << endl; } Chương 2: Tổng quan về C và C++ 53
  54. Ưu nhược điểm của phân hoạch hàm . Chương trình dễ đọc hơn => dễ phát hiện lỗi . Chương trình dễ mở rộng hơn . Có giá trị sử dụng lại . Mã nguồn dài hơn . Mã chạy lớn hơn . Chạy chậm hơn  Không phải cứ phân hoạch thành nhiều hàm là tốt, mà vấn đề nằm ở cách phân hoạch và thiết kế hàm làm sao cho tối ưu! Chương 2: Tổng quan về C và C++ 54
  55. 3.2 Khai báo và định nghĩa hàm . Định nghĩa hàm: tạo mã thực thi hàm Kiểu trả về Tên hàm Tham biến (hình thức) int Tinh_tong(int a, int b) { int T = 0; for (int i = a; i <= b; ++i) T += i; return T; } . Khai báo hàm thuần túy: không tạo mã hàm int Tinh_tong (int a, int b); Kiểu trả về Tên hàm Kiểu tham biến . Tại sao và khi nào cần khai báo hàm? Chương 2: Tổng quan về C và C++ 55
  56. Khai báo hàm và lời gọi hàm . Ý nghĩa của khai báo hàm: – Khi cần sử dụng hàm (gọi hàm) – Trình biên dịch cần lời khai báo hàm để kiểm tra lời gọi hàm đúng hay sai về cú pháp, về số lượng các tham số, kiểu các tham số và cách sử dụng giá trị trả về. int Tinh_tong(int a, int b); – Có thể khai báo hàm độc lập với việc định nghĩa hàm (tất nhiên phải đảm bảo nhất quán) . Có thể định nghĩa hàm sau lời gọi hàm . Gọihàm: yêu cầu thực thi mã hàm với tham số thực tế (tham trị) int x = 5; int k = Tinh_tong(x, 10); Tên hàm Tham số (gọi hàm) Chương 2: Tổng quan về C và C++ 56
  57. Khai báo hàm C/C++ ở đâu? . Ở phạm vi toàn cục (ngoài bất cứ hàm nào) . Một hàm phải được khai báo trước lời gọi đầu tiên trong một tệp tin mã nguồn . Nếu sử dụng nhiều hàm thì sẽ cần rất nhiều dòng mã khai báo (mất công viết, dễ sai và mã chương trình lớn lên?): – Nếu người xây dựng hàm (định nghĩa hàm) đưa sẵn tất cả phần khai báo vào trong một tệp tin => Header file (*.h, *.hx, ) thì người sử dụng chỉ cần bổ sung dòng lệnh #include – Mã chương trình không lớn lên, bởi khai báo không sinh mã! . Một hàm có thể khai báo nhiều lần tùy ý! Chương 2: Tổng quan về C và C++ 57
  58. Định nghĩa hàm ở đâu? . Ở phạm vi toàn cục (ngoài bất cứ hàm nào) . Có thể định nghĩa trong cùng tệp tin với mã chương trình chính, hoặc tách ra một tệp tin riêng. Trong Visual C++: *.c => C compiler, *.cpp => C++ compiler . Một hàm đã có lời gọi thì phải được định nghĩa chính xác 1 lần trong toàn bộ (dự án) chương trình, trước khi gọi trình liên kết (lệnh Build trong Visual C++) . Đưa tệp tin mã nguồn vào dự án, không nên: #include “xxx.cpp” . Một hàm được địnhnghĩa sẵn bằng C, C++, hợp ngữ hoặc bằng một ngôn ngữ khác và dùng trong C/C++ => Sử dụng hàm không cần mã nguồn! . Một thư viện cho C/C++ bao gồm: – Header file (thường đuôi *.h, *.hxx, , nhưng không bắt buộc) – Tệp tin mã nguồn (*.c, *.cpp, *.cxx, ) hoặc mã đích (*.obj, *.o, *.lib, *.dll, ) Chương 2: Tổng quan về C và C++ 58
  59. Tham biến hình thức và tham số thực tế int Tinh_tong (int a, int b) { } Tham biến (hình thức) int x = 5; int k = Tinh_tong (x, 10); Tham số (thực tế) int a = 2; k = Tinh_tong (a,x); Chương 2: Tổng quan về C và C++ 59
  60. Truyền giá trị int Tinh_tong (int, int); void main() { int x = 5; int k = Tinh_tong(x, 10); SP b = 10 } a = 5 SP k =k 45 // Dinh nghia ham x = 5 int Tinh_tong(int a, int b) { Ngăn xếp } Chương 2: Tổng quan về C và C++ 60
  61. Ví dụ #include void Nhap(int N) { cin >> N; } void main() { int x = 5; cout<< “Hay nhap so nguyen: ”; Nhap(x); cout << “Bay gio x la " << x; } . Kết quả: x không hề thay đổi sau đó. Chương 2: Tổng quan về C và C++ 61
  62. Truyền giá trị ( ) . Truyền giá trị là cách thông thường trong C . Tham biến chỉ nhận được bản sao của biến đầu vào (tham số thực tế) . Thay đổi tham biến chỉ làm thay đổi vùng nhớ cục bộ, không làm thay đổi biến đầu vào . Tham biến chỉ có thể mang tham số đầu vào, không chứa được kết quả (tham số ra) . Truyền giá trị trong nhiều trường hợp kém hiệu quả do mất công sao chép dữ liệu Chương 2: Tổng quan về C và C++ 62
  63. Truyền địa chỉ int Tong_mang(int* p, int N); // Goi ham void main() { SP int a[] = {1, 2, 3, 4}; k =k 10 int k = Tong_mang(a,4); N=4 p=00A0 SP } k =k 10 a[3]=4 a[2]=3 // Dinh nghia ham a[1]=2 int Tong_mang(int* p, int N) { a[0]=1 int *p2 = p + N, k = 0; 00A0 while (p < p2) k += *p++; return k; } Chương 2: Tổng quan về C và C++ 63
  64. Truyền mảng tham số? int Tong_mang(int p[4], int N); // Goi ham void main() { int a[] = {1, 2, 3, 4}; int k = Tong_mang(a,4); Bản chất giống truyền địa chỉ } // Dinh nghia ham int Tong_mang(int p[4], int N) { int *p2 = p + N, k = 0; while (p < p2) k += *p++; return k; } Chương 2: Tổng quan về C và C++ 64
  65. Thử lại ví dụ trước #include void Nhap(int* pN) { cin >> *pN; } void main() { int x = 5; cout<<“Nhap so nguyen:“; Nhap(&x); cout << “Bay gio x la " << x; } . Kết quả: x thay đổi giá trị sau đó Chương 2: Tổng quan về C và C++ 65
  66. Khi nào sử dụng truyền địa chỉ? . Khi cần thay đổi "biến đầu vào" (truy nhập trực tiếp vào ô nhớ, không qua bản sao) . Khi kích cỡ kiểu dữ liệu lớn => tránh sao chép dữ liệu vào ngăn xếp . Truyền tham số là một mảng => bắt buộc truyền địa chỉ . Lưu ý: Sử dụng con trỏ để truyền địa chỉ của vùng nhớ dữ liệu đầu vào. Bản thân con trỏ có thể thay đổi được trong hàm nhưng địa chỉ vùng nhớ không thay đổi (nội dung của vùng nhớ đó thay đổiđược) Chương 2: Tổng quan về C và C++ 66
  67. Truyền tham chiếu (C++) #include void Nhap(int& N) { cin >> N; } void main() { int x = 5; cout<<“Nhap so nguyen:“; Nhap(x); cout << “Bay gio x la " << x; } . Kết quả: x thay đổi giá trị sau đó Chương 2: Tổng quan về C và C++ 67
  68. Ví dụ hàm hoán vị . Viết một hàm hoán đổi giá trị của hai phần tử nguyên, sau đó viết chương trình ứng dụng hàm đó. . Còn cách nào giải quyết bài toán trên? Chương 2: Tổng quan về C và C++ 68
  69. Khi nào sử dụng truyền tham chiếu? . Truyền tham chiếu chỉ dùng trongC++ . Khi cần thay đổi "biến đầu vào" (truy nhập trực tiếp vào ô nhớ, không qua bản sao) . Một tham biến tham chiếu có thể đóng vai trò là đầu ra (chứa kết quả), hoặc có thể vừa là đầu vào và đầu ra . Khi kích cỡ kiểu dữ liệu lớn => tránh sao chép dữ liệu vào ngăn xếp. Chương 2: Tổng quan về C và C++ 69
  70. Kiểu trả về . Kiểu trả về gần như tùy ý. . Không thể trả về trực tiếp một mảng . Có thể trả về kiểu: – Giá trị – Con trỏ (địa chỉ) – Tham chiếu . Tuy nhiên, cần rất thận trọng với trả về địa chỉ hoặc tham chiếu: – Không bao giờ trả về con trỏ hoặc tham chiếu vào biến cục bộ – Không bao giờ trả về con trỏ hoặc tham chiếu vào tham biến truyền qua giá trị Chương 2: Tổng quan về C và C++ 70
  71. Cơ chế trả về int Tinh_tong(int a, int b) { int k = 0; for (int i=a; i <= b; ++i) k +=i; SP return k; k =45 } b = 10 a = 5 SP void main() { k k= =450 int x = 5, k = 0; x = 5 k = Tinh_tong(x,10); Ngăn xếp } 45 Chương 2: Tổng quan về C và C++ 71
  72. Trả về con trỏ . Viết hàm tìm giá trị lớn nhất của một mảng? . Viết hàm trả về địa chỉ của phần tử lớn nhất trong một mảng? . Viết chương trình ứng dụng các hàm trên. Chương 2: Tổng quan về C và C++ 72
  73. Lý do trả về con trỏ hoặc tham chiếu . Tương tự như lý do truyền địa chỉ hoặc truyền tham chiếu: – Tránh sao chép dữ liệu lớn không cần thiết – Để có thể truy cập trực tiếp và thay đổi giá trị đầu ra . Có thể trả về con trỏ hoặc tham chiếu vào đâu? – Vào biến toàn cục – Vào tham số truyền cho hàm qua địa chỉ hoặc qua tham chiếu – Nói chung: vào vùng nhớ mà còn tiếp tục tồn tại sau khi kết thúc hàm Chương 2: Tổng quan về C và C++ 73
  74. Tham số mặc định của hàm . VD: void func(int a, int b=0, int c=0) { } void main() { int x=4, y=5, z=6; func(x,y,z);//OK func(x,y);//OK func(x);//OK } Chương 2: Tổng quan về C và C++ 74
  75. Tham biến mặc định của hàm (tiếp) . Các tham biến mặc phải nằm ở cuối cùng của dãy tham số (tính từ trái sang phải) . Tham số thiếu vắng trong lời gọi hàm sẽ tương ứng với các tham biến mặc định cuối cùng (tính từ trái sang phải) Chương 2: Tổng quan về C và C++ 75
  76. Ví dụ tìm min, max của mảng . Viết một hàm đồng thời tìm giá trị lớn nhất và giá trị nhỏ nhất của một mảng. Viết một chương trình ứng dụng hàm đó? Chương 2: Tổng quan về C và C++ 76
  77. Nạp chồng tên hàm trong C++ . Trong C++ có thể xây dựng nhiều hàm có cùng tên, ví dụ: int max(int a, int b); double max(double a, double b); double max(double a, double b, double c); double max(double *seq, int n); . Mục đích củanạp chồng tên hàm: – Đơn giản hóa cho người xây dựng hàm trong việc chọn tên (thay vì maxInt, maxDouble, maxDouble3, maxDoubleSequence, ) – Đơn giản hóa cho người sử dụng hàm, chỉ cần nhớ1 tên quen thuộc thay cho nhiều tên phức tạp Chương 2: Tổng quan về C và C++ 77
  78. Ví dụ: định nghĩa các hàm max() int max(int a, int b) { // (1) return (a > b)? a : b; } double max(double a, double b) { // (2) return (a > b)? a : b; } double max(double a, double b, double c); { // (3) if (a < b) a = b; if (a < c) a = c; return a; } double max(double *seq, int n) { // (4) int i = 0, kq = seq[0]; while (i < n) { if (kq < seq[i])kq = seq[i]; ++i; } return kq; } Chương 2: Tổng quan về C và C++ 78
  79. Ví dụ: sử dụng các hàm max() int max(int a, int b); // (1) double max(double a, double b); // (2) double max(double a, double b, double c); // (3) double max(double *seq, int n); // (4) void main() { int k = max(5,7); // call? double d = max(5.0,7.0); // call? double a[] = {1,2,3,4,5,6}; d = max(d, a[1], a[2]); // call? d = max(a, 5); // call? d = max(5,7); // call? d = max(d, 5); // call? }  Đẩy trách nhiệm kiểm tra và tìm hàm phù hợp cho compiler! Chương 2: Tổng quan về C và C++ 79
  80. Một số qui tắc về nạp chồng tên hàm . Các hàm cùng tên được định nghĩa cùng trong một file/ trong một thư viện hoặc sử dụng trong cùng một chương trình phải khác nhau ít nhất về: – Số lượng các tham số, hoặc – Kiểu của ít nhất một tham số (int khác short, const int khác int, int khác int&, )  Không thể chỉ khác nhau ở kiểu trả về . Tại sao vậy? – Compiler cần có cơ sở để quyết định gọi hàm nào – Dựa vào cú pháp trong lời gọi (số lượng và kiểu các tham số thực tế) compiler sẽ chọn hàm có cú pháp phù hợp nhất – Khi cần compiler có thể tự động chuyển đổi kiểu theo chiều hướng hợp lý nhất (vd short=>int, int => double) Chương 2: Tổng quan về C và C++ 80
  81. Hàm inline trong C++ . Vấn đề: Hàm tiện dụng, nhưng nhiều khi hiệu suất không cao, đặc biệt khi mã thực thi hàm ngắn vì – Các thủ tục như nhớ lại trạng thái chương trình, cấp phát bộ nhớ ngăn xếp, sao chép tham số, sao chép giá trị trả về, khôi phục trạng thái chương trình mất nhiều thời gian – Nếu mã thực thi hàm ngắn thì sự tiện dụng không bõ so với sự lãng phí thời gian Chương 2: Tổng quan về C và C++ 81
  82. Dùng hàm inline trong C++ . Điều duy nhất cần làm là thêm từ khóa inline vào đầu dòng khai báo và định nghĩa hàm inline int max(int a, int b) { return (a > b)? a : b; } . Hàm inline khác gì hàm bình thường? – "Hàm inline" thực chất không phải là một hàm! – Khi gọi hàm thì lời gọi hàm đượcthay thế một cách thông minh bởi mã nguồn định nghĩa hàm, không thực hiện các thủ tục gọi hàm Chương 2: Tổng quan về C và C++ 82
  83. Khi nào nên dùng hàm inline . Ưu điểm của hàm inline: – Tiện dụng như hàm bình thường – Hiệu suất như viết thẳng mã, không gọi hàm – Tin cậy, an toàn. . Nhược điểm của hàm inline: – Nếu gọi hàm nhiều lần trong chương trình, mã chương trình có thể lớn lên nhiều (mã thực hiện hàm xuất hiện nhiều lần trong chương trình) – Mã định nghĩa hàm phải để mở => đưa trong header file . Lựa chọn xây dựng và sử dụng hàm inline khi: – Mã định nghĩa hàm nhỏ (một vài dòng lệnh, không chứa vòng lặp) – Yêu cầu về tốc độ đặt ra trước dung lượng bộ nhớ Chương 2: Tổng quan về C và C++ 83