c++chuong2
chương 2
Hàm trong C++
Chương này trình bầy những khả năng mới của C++ trong việc xây dựng và sử dụng hàm. Đó là:
36 37
+ Kiểu tham chiếu và việc truyền dữ liệu cho hàm bằng tham chiếu.
+ Đối tham chiếu hằng (const)
+ Đối có giá trị mặc định
+ Hàm trực tuyến
+ Việc định nghĩa chồng các hàm
+ Việc định nghĩa chồng các toán tử
§ 1. Biến tham chiếu (Reference variable)
1.1. Hai loại biến dùng trong C
Trước khi nói đến biến tham chiếu, chúng ta nhắc lại 2 loại biến gặp trong C là:
Biến giá trị dùng để chứa dữ liệu (nguyên, thực, ký tự, ... )
Biến con trỏ dùng để chứa địa chỉ
Các biến này đều được cung cấp bộ nhớ và có địa chỉ. Ví dụ câu lệnh khai báo:
double x , *px;
sẽ tạo ra biến giá trị kiểu double x và biến con trỏ kiểu double px. Biến x có vùng nhớ 8 byte, biến px có vùng nhớ 4 byte (nếu dùng mô hình Large). Biến x dùng để chứa giá trị kiểu double, ví dụ lệnh gán:
x = 3.14;
sẽ chứa giá trị 3.14 vào biễn x. Biến px dùng để chứa địa chỉ của một biến thực, ví dụ câu lệnh:
px = &x ;
sẽ lưu trữ địa chỉ của biễn x vào con trỏ px.
1.2. Biến tham chiếu
Trong C++ cho phép sử dụng loại biến thứ ba là biến tham chiếu. So với 2 loại biến quen biết nói trên, thì biến này có những đặc điểm sau:
+ Biến tham chiếu không được cấp phát bộ nhớ, không có địa chỉ riêng.
+ Nó dùng làm bí danh cho một biến (kiểu giá trị) nào đó và nó sử dụng vùng nhớ của biến này. Ví dụ câu lệnh:
float u, v, &r = u ;
tạo ra các biến thực u, v và biến tham chiếu thực r. Biến r không được cấp phát bộ nhớ, nó là một tên khác (bí danh) của u và nó dùng chung vùng nhớ của biến u.
Thuật ngữ: Khi r là bí danh (alias) của u thì ta nói r tham chiếu đến biến u. Như vậy 2 thuật ngữ trên được hiểu như nhau.
ý nghĩa: Khi r là bí danh của u thì r dùng chung vùng nhớ của u, dó đó :
+ Trong mọi câu lệnh, viết u hay viết r đều có ý nghĩa như nhau, vì đều truy nhập đến cùng một vùng nhớ.
+ Có thể dùng biến tham chiếu để truy nhập đến một biến kiểu giá trị.
Ví dụ:
int u, v, &r = u;
r = 10 ; // u=10
cout << u ; // in ra số 10
r++ ; // u = 11
++ u ; // r = 12
cout << r ; // in ra số 12
v = r ; // v=12
& r ; // Cho địa chỉ của u
Công dụng: Biến tham chiếu thường được sử dụng làm đối của hàm để cho phép hàm truy nhập đến các tham số biến trong lời gọi hàm.
Vài chú ý về biến tham chiếu:
a. Vì biến tham chiếu không có địa chỉ riêng, nó chỉ là bí danh của một biến kiểu giá trị nên trong khai báo phải chỉ rõ nó tham chiếu đến biến nào. Ví dụ nếu khai báo:
double &x ;
thì Trình biên dịch sẽ báo lỗi:
Reference variable ‘x’ must be initialized
b. Biến tham chiếu có thể tham chiếu đến một phần tử mảng, ví dụ:
int a[10] , &r = a[5];
r = 25 ; // a[5] = 25
c. Không cho phép khai báo mảng tham chiếu
d. Biến 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.
Ví dụ nếu khai báo:
int &s = 23 ;
thì Trình biên dịch đưa ra cảnh báo (warning):
Temporary used to initialize 's'
Tuy nhiên chương trình vẫn làm việc. Các câu lệnh dưới đây vẫn thực hiện và cho kết quả như sau:
s++;
cout << "
s= " << s; // In ra s=24
Chương trình dưới đây minh hoạ cách dùng biến tham chiếu đến một phần tử mảng cấu trúc để nhập dữ liệu và thực hiện các phép tính trên các trường của phần tử mảng cấu trúc.
#include <iostream.h>
38 39
#include <conio.h>
struct TS
{
char ht[25];
float t,l,h,td;
} ;
void main()
{
TS ts[10],&h=ts[1]; // h tham chiếu đến ts[1]
cout << "
Ho ten: " ;
cin.get(h.ht,25) ;
cout << "Cac diem toan, ly, hoa: ";
cin >> h.t >> h.l >> h.h ;
h.td = h.t + h.l + h.h ;
cout << "
Ho ten: " << ts[1].ht;
cout << "
Tong diem: " << ts[1].td;
getch();
}
1.3. Hằng tham chiếu (const)
Hằng tham chiếu được khai báo theo mẫu:
int n = 10 ;
const int &r = n;
Cũng giống như biến tham chiếu, hằng tham chiếu có thể tham chiếu đến một biến hoặc một hằng. Ví dụ:
int n = 10 ;
const int &r = n ; // Hằng tham chiếu r tham chiếu đến biến n
const int &s=123 ; //Hằng tham chiếu s tham chiếu đến hằng 123
Sự khác nhau giữa biến và hằng tham chiếu ở chỗ: Không cho phép dùng hằng tham chiếu để làm thay đổi giá trị của vùng nhớ mà nó tham chiếu.
Ví dụ:
int y = 12, z ;
const int &py=y; // Hằng tham chiếu py tham chiếu đến biến y
y++; // Đúng
z = 2*py ; // Đúng z = 26
cout << y <<" "<< py; // In ra: 13 13
py=py+1; // Sai, Trình biên dịch thông báo lỗi:
// Cannot modify a const object
Cách dùng: Hằng tham chiếu cho phép sử dụng giá trị chứa trong một vùng nhớ, nhưng không cho phép thay đổi giá trị này.
Hằng tham chiếu 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ố.
§ 2. Truyền giá trị cho hàm theo tham chiếu
2.1. Hàm trong C
Trong C chỉ có một cách truyền dữ liệu cho hàm theo giá trị :
+ Cấp phát vùng nhớ cho các đối.
+ Gán giá trị các tham số trong lời gọi hàm cho các đối sau đó hàm làm việc trên vùng nhớ của các đối chứ không liên quan gì đến các tham số.
Như vây chương trình sẽ tạo ra các bản sao (các đối) của các tham số và hàm sẽ thao tác trên các bản sao này, chứ không làm việc trực tiếp với các tham số. Phương pháp này có 2 nhược điểm chính:
Tốn kém về thời gian và bộ nhớ vì phải tạo ra các bản sao. Không thao tác trực tiếp trên các tham số, vì vậy không làm thay đổi được giá trị các tham số.
2.2. Truyền giá trị cho hàm theo tham chiếu
40 41
Trong C++ cung cấp thêm cách truyền dữ liệu cho hàm theo tham chiếu bằng cách dùng đối là biến tham chiếu hoặc đối là hằng tham chiếu. Cách này có ưu điểm:
Không cần tạo ra các bản sao của các tham số, do đó tiết kiệm bộ nhớ và thời gian chạy máy.
Hàm sẽ thao tác trực tiếp trên vùng nhớ của các tham số, do đó dễ dàng thay đổi giá trị các tham số khi cần.
2.3. Mối quan hệ giữa đối và tham số trong lời gọi hàm
Nếu đối là biến hoặc hằng tham chiếu kiểu K thì tham số (trong lời gọi hàm) phải là biến hoặc phần tử mảng kiểu K. Ví dụ:
+ Đối là biến hoặc hằng tham chiếu kiểu double, thì tham số là biến hoặc phần tử mảng kiểu double
+ Đối là biến hoặc hằng tham chiếu kiểu cấu trúc, thì tham số là biến hoặc phần tử mảng kiểu cấu trúc
2.4. Các chương trình minh hoạ
/*
Chương trình sau được tổ chức thành 3 hàm:
Nhập dẫy số double
Hoán vị 2 biến double
Sắp xếp dẫy số double theo thứ tự tăng dần
Chương trình sẽ nhập một dẫy số và in dẫy sau khi sắp xếp
*/
#include <iostream.h>
#include <conio.h>
#include <stdio.h>
void nhapds(double *a, int n)
{
for (int i=1; i<= n ; ++i)
{
cout << "
Phan tu thu " << i << " : " ;
cin >> a[i] ;
}
}
void hv(double &x, double &y)
{
double tg=x; x=y; y= tg;
}
void sapxep(double * a, int n)
{
for (int i=1; i <= n-1 ;++i)
for (int j=i+1 ; j<=n ;++j)
if (a[i] > a[j])
hv(a[i],a[j]);
}
void main()
{
double x[100];
int i, n;
cout <<"
N= ";
cin >> n;
nhapds(x,n);
sapxep(x,n);
for (i=1;i<=n;++i)
printf("
%0.1lf",x[i]);
getch();
}
/*
Chương trình sau gồm các hàm:
- Nhập dẫy cấu trúc (mỗi cấu trúc chứa dữ liệu một thí sinh)
- Hoán vị 2 biến cấu trúc
42 43
- Sắp xếp dẫy thí sinh theo thứ tự giảm của tổng điểm
- In một cấu trúc (in họ tên và tổng điểm)
Chương trình sẽ nhập dữ liệu một danh sách thí sinh, nhập điểm chuẩn và in danh sách thí sinh trúng tuyển
*/
#include <iostream.h>
#include <iomanip.h>
#include <conio.h>
struct TS
{
char ht[20];
float t,l,h,td;
} ;
void ints(const TS &ts)
{
cout << setiosflags(ios::showpoint) << setprecision(1) ;
cout << "
Ho ten: " << setw(20) << ts.ht << setw(6) << ts.td ;
}
void nhapsl(TS *ts,int n)
{
for (int i=1;i<=n;++i)
{
cout << "
Thi sinh " << i ;
cout << "
Ho ten: " ;
cin.ignore(1);
cin.get(ts[i].ht,25) ;
cout << "Cac diem toan, ly, hoa: ";
cin >> ts[i].t >> ts[i].l >> ts[i].h ;
ts[i].td = ts[i].t + ts[i].l + ts[i].h ;
}
}
void hvts(TS &ts1, TS &ts2)
{
TS tg=ts1;
ts1=ts2;
ts2=tg;
}
void sapxep(TS *ts,int n)
{
for (int i=1;i<=n-1;++i)
for (int j=i+1;j<=n;++j)
if (ts[i].td < ts[j].td)
hvts(ts[i],ts[j]);
}
void main()
{
TS ts[100];
int n,i;
clrscr();
cout << " So thi sinh: " ;
cin >> n ;
nhapsl(ts,n);
sapxep(ts,n) ;
float dc;
cout << " Diem chuan: " ;
cin >> dc;
cout << "
Danh sach trung tuyen
" ;
for (i=1;i<=n;++i)
if (ts[i].td >= dc)
44 45
ints(ts[i]);
else
break;
getch();
}
/*
Chương trình sau gồm các hàm:
Nhập một ma trận thực cấp mxn
In một ma trận thực dưới dạng bảng
Tìm phần tử lớn nhất và phần tử nhỏ nhất của dẫy số thưc;
Chương trình sẽ nhập một ma trận, in ma trận vừa nhập và in các phần tử lớn nhất và nhỏ nhất trên mỗi hàng của ma trận
*/
#include <iostream.h>
#include <iomanip.h>
#include <conio.h>
#include <stdio.h>
void nhapmt(float a[20][20], int m, int n)
{
for (int i=1 ; i<= m ; ++i)
for (int j=1; j<= n ; ++j)
{
cout << "
a[" << i << "," << j << "]= " ;
cin >> a[i][j] ;
}
}
void inmt(float a[20][20], int m, int n)
{
cout << setiosflags(ios::showpoint) << setprecision(1);
for (int i=1 ; i<= m ; ++i)
for (int j=1; j<= n ; ++j)
{
if (j==1) cout << "
" ;
cout << setw(6) << a[i][j] ;
}
}
void maxminds(float *x, int n,int &vtmax, int &vtmin)
{
vtmax = vtmin = 1 ;
for (int i=2; i<=n ; ++i)
{
if (x[i] > x[vtmax]) vtmax = i;
if (x[i] < x[vtmin]) vtmin = i;
}
}
void main()
{
float a[20][20];
int m, n;
cout <<"
So hamg va so cot ma tran: ";
cin >> m >> n;
nhapmt(a,m,n);
clrscr();
inmt(a,m,n);
float *p = (float*)a;
int vtmax, vtmin;
for (int i=1;i<=m;++i)
46 47
{
p = ((float*)a) + i*20 ;
maxminds(p , n, vtmax, vtmin) ;
printf("
Hang %d Phan tu max= %6.1f tai cot
%d",i,p[vtmax],vtmax);
printf("
Phan tu min= %6.1f tai cot %d", p[vtmin],vtmin);
}
getch();
}
§ 3. Hàm trả về các tham chiếu
Hàm có thể có kiểu tham chiếu và trả về giá trị tham chiếu. Khi đó có thể dùng hàm để truy nhập đến một biến hoặc một phần tử mảng nào đó. Dưới đây là một số ví dụ.
Ví dụ 1 trình bầy một hàm trả về một tham chiếu đến một biến toàn bộ. Do đó có thể dùng hàm để truy nhập đến biến này.
#include <iostream.h>
#include <conio.h>
int z ;
int &f() // Hàm trả về một bí danh của biến toàn bộ z
{
return z;
}
void main(void)
{
f()=50; // z = 50
cout <<"
z= " << z;
getch();
}
Ví dụ 2 trình bầy một hàm trả về bí danh của một biến cấu trúc toàn bộ. Khác với ví dụ trên, ở đây không dùng hàm một cách trực tiếp mà gán hàm cho một biến tham chiếu, sau đó dùng biến tham chiếu này để truy nhập đến biến cấu trúc toàn bộ.
#include <iostream.h>
#include <conio.h>
struct TS
{
char ht[25];
float t,l,h,td;
};
TS ts;
TS &f()
{
return ts;
}
void main()
{
TS &h=f(); // h tham chiếu đến biến ts
cout << "
Ho ten: " ;
cin.get(h.ht,25) ;
cout << "Cac diem toan, ly, hoa: ";
cin >> h.t >> h.l >> h.h ;
h.td = h.t + h.l + h.h ;
cout << "
Ho ten: " << ts.ht;
cout << "
Tong diem: " << ts.td;
getch();
48 49
}
Ví dụ 3 trình bầy một hàm trả về bí danh của một phần tử mảng cấu toàn bộ.
Hàm sẽ kiểm tra xem chỉ số mảng có vượt ra ngoài miền quy định hay không. Sau đó dùng hàm này để truy nhập đến các phần tử mảng cấu trúc.
#include <iostream.h>
#include <conio.h>
#include <stdlib.h>
struct TS
{
char ht[25];
float t,l,h,td;
};
TS *ts;
void cap_phat_bo_nho_nhapsl(int n)
{
ts = new TS[n+1] ;
if (ts==NULL)
{
cout << "Loi cap phat bo nho " ;
exit(1);
}
for (int i=1;i<=n;++i)
{
TS &h=ts[i];
cout << "
Thi sinh thu " << i ;
cout << "
Ho ten: " ;
cin.ignore(1);
cin.get(h.ht,25) ;
cout << "Cac diem toan, ly, hoa: ";
cin >> h.t >> h.l >> h.h ;
h.td = h.t + h.l + h.h ;
}
}
TS &f(int i, int n) // Cho bi danh ts[i]
{
if (i<1 || i>n)
{
cout << "Chi so mang khong hop le " ;
exit(1);
}
return ts[i];
}
void main()
{
int n, i ;
cout << "
So thi sinh : " ;
cin >> n;
cap_phat_bo_nho_nhapsl(n);
while (1)
{
cout << "
Can xem thi sinh thu may: " ;
cout << "
Chon so tu 1 den " << n << " (bam sai ket thuc CT) ";
cin >> i;
TS &h=f(i,n);
cout << "
Ho ten: " << h.ht;
cout << "
Tong diem: " << h.td;
}
}
50 51
§ 4. Đối có giá trị mặc định
4.1. Thế nào là đối mặc định
Một trong các khả năng mạnh của C++ là nó cho phép xây dựng hàm với các đối có giá trị mặc định. Thông thường số tham số trong lời gọi hàm phải bằng số đối của hàm. Mỗi đối sẽ được khởi gán giá trị theo tham số tương ứng của nó. Trong C++ cho phép tạo giá trị mặc định cho các đối. Các đối này có thể có hoặc không có tham số tương ứng trong lời gọi hàm. Khi không có tham số tương ứng, đối được khởi gán bởi giá trị mặc định.
Ví dụ hàm delay với đối số mặc định được viết theo một trong 2 cách sau:
Cách 1 (Không khai báo nguyên mẫu):
void delay(int n=1000)
{
for (int i=0 ; i<n ; ++i)
;
}
Cách 2 (Có khai báo nguyên mẫu):
void delay(int n=1000) ;
void delay(int n)
{
for (int i=0 ; i<n ; ++i)
;
}
Cách dùng:
+ Cung cấp giá trị cho đối n (Có tham số trong lời gọi hàm)
delay(5000) ; // Đối n = 5000
+ Sử dụng giá trị mặc định của đối (Không có tham số trong lời gọi)
delay() ; // Đối n = 1000
4.2. Quy tắc xây dựng hàm với đối mặc định
+ Các đối mặc định cần phải là các đối cuối cùng tính từ trái sang phải. Giả sử có 5 đối theo thứ tự từ trái sang phải là
d1, d2, d3, d4, d5
Khi đó:
nếu một đối mặc định thì phải là d5
nếu hai đối mặc định thì phải là d4, d5
nếu ba đối mặc định thì phải là d3, d4, d5
...
Các ví dụ sai:
d3 và d5 mặc định (khi đó d4 cũng phải mặc định)
d3 và d4 mặc định (khi đó d5 cũng phải mặc định)
+ Khi xây dựng hàm, nếu sử dụng khai báo nguyên mẫu, thì các đối mặc định cần được khởi gán trong nguyên mẫu, ví dụ:
// Khởi gán giá trị cho 3 đối mặc định d3, d4 và d5)
void f(int d1, float d2, char *d3=”HA NOI”,
int d4 = 100, double d5=3.14) ;
void f(int d1, float d2, char *d3, int d4, double d5)
{
// Các câu lệnh trong thân hàm
}
Không được khởi gán lại cho các đối mặc định trong dòng đầu của định nghĩa hàm. Nếu vi phạm điều này thì Chương trình dịch sẽ thông báo lỗi.
+ Khi xây dựng hàm, nếu không khai báo nguyên mẫu, thì các đối mặc định được khởi gán trong dòng đầu của định nghĩa hàm, ví dụ:
// Khởi gán giá trị cho 3 đối mặc định d3, d4 và d5)
void f(int d1, float d2, char *d3=”HA NOI”,
52 53
int d4 = 100, double d5=3.14)
{
// Các câu lệnh trong thân hàm
}
+ Giá trị dùng để khởi gán cho đối mặc đinh
Có thể dùng các hằng, các biến toàn bộ, các hàm để khởi gán cho đối mặc định, ví dụ:
int MAX = 10000;
void f(int n, int m = MAX, int xmax = getmaxx(),
int ymax = getmaxy() ) ;
4.3. Cách sử dụng hàm có đối mặc định
Lời gọi hàm cần viết theo quy định sau:
Các tham số thiếu vắng trong lời gọi hàm phải tương ứng với các đối mặc định cuối cùng (tính từ trái sang phải).
Nói cách khác: Đã dùng giá trị mặc định cho một đối (tất nhiên phải là đối mặc định) thì cũng phải sử dụng giá trị mặc định cho các đối còn lại.
Ví dụ với hàm có 3 đối mặc định:
void f(int d1, float d2, char *d3=”HA NOI”,
int d4 = 100, double d5=3.14) ;
Thì các lời gọi sau là đúng:
f(3,3.4,”ABC”,10,1.0) ; // Đầy đủ tham số
f(3,3.4,”ABC”) ; // Thiếu 2 tham số cuối
f(3,3.4) ; // Thiếu 3 tham số cuối
Các lời gọi sau là sai:
f(3) ; // Thiếu tham số cho đối không mặc định d2
f(3,3.4, ,10) ; // Đã dùng giá trị mặc định cho d3, thì cũng
// phải dùng giá trị mặc định cho d4 và d5
4.4. Các ví dụ
Hàm ht (bên dưới) dùng để hiển thị chuỗi ký tự dc trên n dòng màn hình. Các đối dc và n đều có giá trị mặc định.
#include <conio.h>
#include <iostream.h>
void ht(char *dc="HA NOI",int n=10) ;
void ht(char *dc , int n )
{
for (int i=0;i<n;++i)
cout << "
" << dc;
}
void main()
{
ht(); // In dòng chữ “HA NOI” trên 10 dòng
ht("ABC",3); // In dòng chữ “ABC” trên 3 dòng
ht("DEF"); // In dòng chữ “DEF” trên 10 dòng
getch();
}
Ví dụ dưới đây trình bầy hàm hiển thị một chuỗi str trên màn hình đồ hoạ, tại vị trí (x,y) và có mầu m. Các đối x, y và m là mặc định. Dùng các hàm getmaxx() và getmaxy() để khởi gán cho x, y. Dùng hằng RED gán cho m.
#include <conio.h>
#include <graphics.h>
void hiendc(char *str, int x=getmaxx()/2,
int y = getmaxy()/2, int m=RED);
void hiendc(char *str, int x,int y, int m)
{
54 55
int mau_ht = getcolor(); // Luu mau hien tai
setcolor(m);
outtextxy(x,y,str) ;
setcolor(mau_ht); // Khoi phuc mau hien tai
}
void main()
{
int mh=0, mode=0;
initgraph(&mh,&mode,"");
setbkcolor(BLUE);
hiendc("HELLO"); // HELLO mầu đỏ giữa màn hình
hiendc("CHUC MUNG",1,1); // CHUC MUNG mầu đỏ tại vị
// trí (1,1)
hiendc("CHAO",1,400,YELLOW); // CHAO mầu vàng tại vị
// trí (1,400)
getch();
}
Ví dụ dưới đây trình bầy hàm tính tích phân xác định gồm 3 đối: f là hàm cần tính tích phân, a và b là các cận dưới và trên (a<b). Cả 3 đối f, a và b đều mặc định. Giá trị mặc định của con trỏ hàm f là địa chỉ của hàm bp (bình phương), của a bằng 0, của b bằng 1.
#include <conio.h>
#include <iostream.h>
#include <iomanip.h>
#include <math.h>
double bp(double x);
double tp( double (*f)(double)=bp,double a=0.0, double b=1.0) ;
double bp(double x)
{
return x*x;
}
double tp(double (*f)(double), double a, double b )
{
int n=1000;
double s=0.0, h=(b-a)/n;
for (int i=0; i<n ; ++i)
s+= f(a+i*h + h) + f(a+i*h ) ;
return s*h/2;
}
void main()
{
clrscr();
cout << setiosflags(ios::showpoint) << setprecision(2);
cout << "
Tich phan tu 0 den 1 cua x*x= " << tp() ;
cout << "
Tich phan tu 0 den 1 cua exp(x)= " << tp(exp);
cout << "
Tich phan tu 0 den PI/2 cua sin(x) " <<
tp(sin,0,3.14/2);
getch();
}
§ 5. Các hàm trực tuyến (inline)
5.1. Ưu, nhược điểm của hàm
Việc tổ chức chương trình thành các hàm có 2 ưu điểm rõ rệt : Thứ nhất là chia chương trình thành các đơn vị độc lập, làm cho chương trình được tổ chức một cách khoa học dễ kiểm soát dễ phát hiện lỗi, dễ phát triển, mở rộng.
56 57
Thứ hai là giảm được kích thước chương trình, vì mỗi đoạn chương trình thực hiện nhiệm vụ của hàm được thay bằng một lời gọi hàm.
Tuy nhiên hàm cũng có nhược điểm là làm chậm tốc độ chương trình do phải thực hiện một số thao tác có tính thủ tục mỗi khi gọi hàm như: Cấp phát vùng nhớ cho các đối và biến cục bộ, truyền dữ liệu của các tham số cho các đối, giải phóng vùng nhớ trước khi thoát khỏi hàm.
Các hàm trực tuyến trong C++ cho khả năng khắc phục được nhược điểm nói trên.
5.2. Các hàm trực tuyến
Để biến một hàm thành trực tuyến ta viết thêm từ khoá
inline
vào trước khai báo nguyên mẫu hàm. Nếu không dùng nguyên mẫu thì viết từ khoá này trước dòng đầu tiên của định nghĩa hàm. Ví dụ:
inline float f(int n, float x);
float f(int n, float x)
{
// Các câu lệnh trong thân hàm
}
hoặc
inline float f(int n, float x)
{
// Các câu lệnh trong thân hàm
}
Chú ý: Trong mọi trường hợp, từ khoá inline phải xuất hiện trước các lời gọi hàm thì Trình biên dịch mới biết cần xử lý hàm theo kiểu inline.
Ví dụ hàm f trong chương trình sau sẽ không phải là hàm trực tuyến vì từ khoá inline viết sau lời gọi hàm:
#include <conio.h>
#include <iostream.h>
void main()
{
int s ;
s = f(5,6);
cout << s ;
getch();
}
inline int f(int a, int b)
{
return a*b;
}
Chú ý: Trong C++ , nếu hàm được xây dựng sau lời gọi hàm thì bắt buộc phải khai báo nguyên mẫu hàm trước lời gọi. Trong ví dụ trên, Trình biên dịch C++ sẽ bắt lỗi vì thiếu khai báo nguyên mẫu hàm f .
5.3. Cách biên dịch hàm trực tuyến
Chương trình dịch xử lý các hàm inline như các macro (được định nghĩa trong lệnh #define), nghĩa là nó sẽ thay mỗi lời gọi hàm bằng một đoạn chương trình thực hiện nhiệm vụ của hàm. Cách này làm cho chương trình dài ra, nhưng tốc độ chương trình tăng lên do không phải thực hiện các thao tác có tính thủ tục khi gọi hàm.
5.4. So sánh macro và hàm trực tuyến
Dùng macro và hàm trực tuyến đều dẫn đến hiệu quả tương tự, tuy nhiên người ta thích dùng hàm trực tuyến hơn, vì cách này đảm bảo tính cấu trúc của chương trình, dễ sử dụng và tránh được các sai sót lặt vặt thường gặp khi dùng #define (như thiếu các dấu ngoặc, dấu chấm phẩy)
5.5. Khi nào thì nên dùng hàm trực tuyến
58 59
Phương án dùng hàm trực tuyến rút ngắn được thời gian chạy máy nhưng lại làm tăng khối lượng bộ nhớ chương trình (nhất là đối với các hàm trực tuyến có nhiều câu lệnh). Vì vậy chỉ nên dùng phương án trực tuyến đối với các hàm nhỏ.
5.6. Sự hạn chế của Trình biên dịch
Không phải khi gặp từ khoá inline là Trình biên dịch nhất thiết phải xử lý hàm theo kiểu trực tuyến.
Chú ý rằng từ khoá inline chỉ là một sự gợi ý cho Trình biên dịch chứ không phải là một mệnh lệnh bắt buộc.
Có một số hàm mà các Trình biên dịch thường không xử lý theo cách inline như các hàm chứa biến static, hàm chứa các lệnh chu trình hoặc lệnh goto hoặc lệnh switch, hàm đệ quy. Trong trường hợp này từ khoá inline lẽ dĩ nhiên bị bỏ qua.
Thậm chí từ khoá inline vẫn bị bỏ qua ngay cả đối với các hàm không có những hạn chế nêu trên nếu như Trình biên dịch thấy cần thiết (ví dụ đã có quá nhiều hàm inline làm cho bộ nhớ chương trình quá lớn)
Ví dụ: Chương trình sau sử dụng hàm inline tính chu vi và diện tích của hình chữ nhật:
Phương án 1: Không khai báo nguyên mẫu. Khi đó hàm dtcvhcn phải đặt trên hàm main.
#include <conio.h>
#include <iostream.h>
inline void dtcvhcn(int a, int b, int &dt, int &cv)
{
dt=a*b;
cv=2*(a+b);
}
void main()
{
int a[20],b[20],cv[20],dt[20],n;
cout << "
So hinh chu hat: " ;
cin >> n;
for (int i=1;i<=n;++i)
{
cout << "
Nhap 2 canh cua hinh chu nhat thu " <<i<< ": ";
cin >> a[i] >> b[i] ;
dtcvhcn(a[i],b[i],dt[i],cv[i]);
}
clrscr();
for (i=1;i<=n;++i)
{
cout << "
Hinh chu nhat thu " << i << " : ";
cout << "
Do dai 2 canh= " << a[i] << " va " << b[i] ;
cout << "
Dien tich= " << dt[i] ;
cout << "
Chu vi= " << cv[i] ;
}
getch();
}
Phương án 2: Sử dụng khai báo nguyên mẫu. Khi đó từ khoá inline đặt trước nguyên mẫu.
Chú ý: Không được đặt inline trước định nghĩa hàm. Trong chương trình dưới đây nếu đặt inline trước định nghĩa hàm thì hậu quả như sau: Chương trình vẫn dịch thông, nhưng khi chạy thì chương trình bị quẩn, không thoát được.
#include <conio.h>
#include <iostream.h>
inline void dtcvhcn(int a, int b, int &dt, int &cv) ;
void main()
{
int a[20],b[20],cv[20],dt[20],n;
cout << "
So hinh chu hat: " ;
cin >> n;
for (int i=1;i<=n;++i)
60 61
{
cout << "
Nhap 2 canh cua hinh chu nhat thu " <<i<< ": ";
cin >> a[i] >> b[i] ;
dtcvhcn(a[i],b[i],dt[i],cv[i]);
}
clrscr();
for (i=1;i<=n;++i)
{
cout << "
Hinh chu nhat thu " << i << " : ";
cout << "
Do dai 2 canh= " << a[i] << " va " << b[i] ;
cout << "
Dien tich= " << dt[i] ;
cout << "
Chu vi= " << cv[i] ;
}
getch();
}
void dtcvhcn(int a, int b, int &dt, int &cv)
{
dt=a*b;
cv=2*(a+b);
}
§ 6. Định nghĩa chồng các hàm (Overloading)
6.1. Khái niệm về định nghĩa chồng
Định nghĩa chồng (hay còn gọi sự tải bội) các hàm là dùng cùng một tên để định nghĩa các hàm khác nhau. Đây là một mở rộng rất có ý nghĩa của C++.
Như đã biết, trong C và các ngôn ngữ khác (như PASCAL, FOXPRO,...) mỗi hàm đều phải có một tên phân biệt. Đôi khi đây là một sự hạn chế lớn, vì phải dùng nhiều hàm khác nhau để thực hiện cùng một công việc. Ví dụ để lấy giá trị tuyệt đối trong C cần dùng tới 3 hàm khác nhau:
int abs(int i); // Lấy giá trị tuyệt đối giá trị kiểu int
longt labs(longt l); // Lấy giá trị tuyệt đối giá trị kiểu long
double fabs(double d); // Lấy giá trị tuyệt đối giá trị kiểu double
Nhờ khả năng định nghĩa chồng, trong C++ có thể dùng chung một tên cho cả 3 hàm trên như sau:
int abs(int i) ; // Lấy giá trị tuyệt đối giá trị kiểu int
longt abs(longt l) ; // Lấy giá trị tuyệt đối giá trị kiểu long
double abs(double d) ; // Lấy giá trị tuyệt đối giá trị kiểu double
6.2. Yêu cầu về các hàm định nghĩa chồng
Khi dùng cùng một tên để định nghĩa nhiều hàm, Trình biên dịch C++ sẽ dựa vào sự khác nhau về tập đối của các hàm này để đổi tên các hàm. Như vậy, sau khi biên dịch mỗi hàm sẽ có một tên khác nhau.
Từ đó cho thấy: các hàm được định nghĩa trùng tên phải có tập đối khác nhau (về số lượng hoặc kiểu). Nếu 2 hàm hoàn toàn trùng tên và trùng đối thì Trình biên dịch sẽ không có cách nào phân biệt được. Ngay cả khi 2 hàm này có kiểu khác nhau thì Trình biên dịch vẫn báo lỗi. Ví dụ sau xây dựng 2 hàm cùng có tên là f và cùng có một đối nguyên a, nhưng kiểu hàm khác nhau. Hàm thứ nhất kiểu nguyên (trả về a*a), hàm thứ hai kiểu void (in giá trị a). Chương trình sẽ bị thông báo lỗi khi biên dịch (bạn hãy thử xem sao)
#include <conio.h>
#include <iostream.h>
int f(int a);
void f(int a);
int f(int a)
{
return a*a;
}
void f(int a)
62 63
{
cout << "
" << a ;
}
void main()
{
int b=f(5);
f(b);
getch();
}
6.3. Sử dụng các hàm định nghĩa chồng
Khi gặp một lời gọi, Trình biên dịch sẽ căn cứ vào số lượng và kiểu của các tham số để gọi hàm có đúng tên và đúng bộ đối số tương ứng. Ví dụ:
abs(123); // Tham số kiểu int, gọi hàm int abs(int i) ;
abs(123L); // Tham số kiểu long, gọi hàm long abs(long l);
abs(3.14); //Tham số kiểu double, gọi hàm double abs(double d);
Khi không có hàm nào có bộ đối cùng kiểu với bộ tham số (trong lời gọi), thì Trình biên dịch sẽ chọn hàm nào có bộ đối gần kiểu nhất (phép chuyển kiểu dễ dàng nhất). Ví dụ:
abs(‘A’) ; // Tham số kiểu char, gọi hàm int abs(int i) ;
abs(3.14F); // Tham số kiểu float, gọi hàm double abs(double d);
6.4. Nên sử dụng phép định nghĩa chồng các hàm như thế nào
Như đã nói ở trên, khi xây dựng cũng như sử dụng các hàm trùng tên, Trình biên dịch C++ đã phải suy đoán và giải quyết nhiều trường hợp khá nhập nhằng. Vì vậy không nên lạm dụng quá đáng khả năng định nghĩa chồng, vì điều đó làm cho chương trình khó kiểm soát và dễ dẫn đến sai sót. Việc định nghĩa chồng sẽ hiệu quả hơn nếu được sử dụng theo các lời khuyên sau:
+ Chỉ nên định nghĩa chồng các hàm thực hiện những công việc như nhau nhưng trên các đối tượng có kiểu khác nhau. Ví dụ trong chương trình cần xây dựng các hàm: cộng 2 ma trận vuông kiểu double, cộng 2 ma trận vuông kiểu int, cộng 2 ma trân chữ nhật kiểu double, cộng 2 ma trận chữ nhật kiểu int, thì 4 hàm trên nên định nghĩa chồng (đặt cùng tên).
+ Nên dùng các phép chuyển kiểu (nếu cần) để bộ tham số trong lời gọi hoàn toàn trùng kiểu với bộ đối số của một hàm được định nghĩa chồng. Vì như thế mới tránh được sự nhập nhằng cho Trình biên dịch và Trình biên dịch sẽ chọn đúng hàm cần gọi.
6.5. Lấy địa chỉ các hàm trùng tên
Giả sử có 4 hàm đều có tên là tinh_max được khai báo như sau:
int tinh_max(int a, int b, int c) ; // Max của 3 số nguyên
double tinh_max(double a, double b, double c); // Max của 3 số // thực
int tinh_max(int *a, int n) ; // Max của một dẫy số nguyên
double tinh_max(double *a, int n) ; //Max của một dẫy số thực
Vấn đề đặt ra là làm thế nào lấy được địa chỉ của mỗi hàm. Câu trả lời như sau:
Để lấy địa chỉ của một hàm, ta khai báo một con trỏ hàm có kiểu và bộ đối như hàm cần lấy địa chỉ. Sau đó gán tên hàm cho con trỏ hàm. Ví dụ:
int (*f1)(int , int, int );
f1 = tinh_max ; // Lấy địa chỉ của hàm thứ nhất
double (*f2)(double , double, double);
f2 = tinh_max ; // Lấy địa chỉ của hàm thứ hai
int (*f3)(int *, int );
f3 = tinh_max ; // Lấy địa chỉ của hàm thứ ba
double (*f4)(double *, int );
f4 = tinh_max ; // Lấy địa chỉ của hàm thứ tư
6.6. Các ví dụ
64 65
Ví dụ 1: Chương trình giải bài toán tìm max của một dẫy số nguyên và max của một dẫy số thực. Trong chươmg trình có 6 hàm. Hai hàm dùng để nhập dẫy số nguyên và dẫy số thực có tên chung là nhapds. Bốn hàm: tính max 2 số nguyên, tính max 2 số thực, tính max của dẫy số nguyên, tính max của dẫy số thực được đặt chung một tên là max.
#include <conio.h>
#include <iostream.h>
#include <iomanip.h>
void nhapds(int *x, int n);
void nhapds(double *x, int n);
int max(int x, int y);
double max(double x, double y);
int max(int *x, int n);
double max(double *x, int n);
void nhapds(int *x, int n)
{
for (int i=1;i<=n;++i)
{
cout << "Phan tu " << i << " = " ;
cin >> x[i] ;
}
}
void nhapds(double *x, int n)
{
for (int i=1;i<=n;++i)
{
cout << "Phan tu " << i << " = " ;
cin >> x[i] ;
}
}
int max(int x, int y)
{
return x>y?x:y ;
}
double max(double x, double y)
{
return x>y?x:y ;
}
int max(int *x, int n)
{
int s=x[1];
for (int i=2;i<=n;++i)
s = max(s,x[i]);
return s;
}
double max(double *x, int n)
{
double s=x[1];
for (int i=2;i<=n;++i)
s = max(s,x[i]);
return s;
}
void main()
{
int a[20] , n , ni, nd, maxi ;
double x[20] , maxd ;
clrscr();
cout << "
So phan tu nguyen ni = " ;
cin >> ni ;
cout << "Nhap day so nguyen
" ;
nhapds(a,ni);
cout << "
So phan tu thuc nd = " ;
66 67
cin >> nd ;
cout << "Nhap day so thuc
" ;
nhapds(x,nd);
maxi = max(a,ni);
maxd = max(x,nd);
cout << "
Max cua day nguyen = " << maxi ;
cout << "
Max cua day thuc = " << maxd ;
getch();
}
Ví dụ 2:
Chương trình sau thực hiện phép nhân ma trận:
D = A*B*C
trong đó A, B là các ma trận vuông, C là ma trận chữ nhật. Trong chương trình có 3 cặp hàm trùng tên để thực hiện 3 nhiệm vụ (nhưng trên 2 đối tượng khác nhau là ma trận vuông và chữ nhật): Nhập ma trận, nhân 2 ma trận và in ma trân.
#include <conio.h>
#include <iostream.h>
#include <iomanip.h>
typedef int MT[20][20];
void nhapmt(MT a,char *ten, int m, int n);
void inmt(MT a,char *ten, int m, int n);
void nhanmt(MT a,MT b, MT c, int m, int n, int p);
void nhapmt(MT a,char *ten, int n);
void inmt(MT a,char *ten, int n);
void nhanmt(MT a,MT b, MT c, int n);
void nhapmt(MT a, char *ten, int m, int n)
{
for (int i=1;i<=m;++i)
for (int j=1;j<=n;++j)
{
cout << "
" << ten <<"[" << i << "," << j << "]= " ;
cin >> a[i][j];
}
}
void nhapmt(MT a,char *ten, int n)
{
nhapmt(a,ten,n,n) ;
}
void inmt(MT a,char *ten, int m, int n)
{
cout << "
Ma tran: " << ten;
for (int i=1;i<=m;++i)
{
cout << "
" ;
for (int j=1;j<=n;++j)
cout << setw(6) << a[i][j];
}
}
void inmt(MT a,char *ten, int n)
{
inmt(a,ten,n,n) ;
}
void nhanmt(MT a,MT b, MT c, int m, int n, int p)
{
for (int i=1;i<=m;++i)
for (int j=1;j<=p;++j)
{
c[i][j]=0;
68 69
for (int k=1;k<=n;++k)
c[i][j] += a[i][k] * b[k][j];
}
}
void nhanmt(MT a,MT b, MT c, int n)
{
nhanmt(a,b,c,n,n, n) ;
}
void main()
{
MT a,b,c,d; // d= abc
MT u;
clrscr();
nhapmt(a,"A",2);
nhapmt(b,"B",2);
nhapmt(c,"C",2,3);
nhanmt(a,b,u,2);
nhanmt(u,c,d,2,2,3);
inmt(a,"A",2);
inmt(b,"B",2);
inmt(u,"U = A*B",2);
inmt(c,"C",2,3);
inmt(d,"D = U*C",2,3);
getch();
}
§ 7. Định nghĩa chồng các toán tử
7.1. Các phép toán trong C và C++
Trong C và C++ có khá nhiều các phép toán dùng để thực hiện các thao tác trên các kiểu dữ liệu chuẩn. Ví dụ các phép số học: + - * / áp dụng cho các kiểu dữ liệu nguyên, thực. Phép lấy phần dư % áp dụng đối với kiểu nguyên.
7.2. Thực hiện các phép toán trên các kiểu dữ liệu không chuẩn trong C
Việc thực hiện các phép toán trên các đối tượng tự định nghĩa (như mảng, cấu trúc) là nhu cầu bắt buộc của thực tế. Chẳng hạn cần thực hiện các phép số học trên số phức, trên phân số, trên đa thức, trên véc tơ, trên ma trận. Để đáp ứng yêu cầu này, ta sử dụng các hàm trong C. Ví dụ sau đây là một chương trình C gồm các hàm nhập phân số, in phân số và thực hiện các phép cộng trừ nhân chia phân số. Chương trình sẽ nhập 5 phân số: p, q, z, u, v và tính phân số s theo công thức:
s = (p – q*z)/(u + v)
#include <conio.h>
#include <stdio.h>
#include <math.h>
typedef struct
{
int a,b;
} PS;
void nhap(PS *p);
void in(PS p);
int uscln(int x, int y);
PS rutgon(PS p);
PS cong(PS p1, PS p2);
PS tru(PS p1, PS p2);
PS nhan(PS p1, PS p2);
PS chia(PS p1, PS p2);
void nhap(PS *p)
{
int t, m;
70 71
printf("
Tu va mau: ");
scanf("%d%d", &t, &m);
p->a = t; p->b = m;
}
void in(PS p)
{
printf(" %d/%d",p.a,p.b);
}
int uscln(int x, int y)
{
x=abs(x); y=abs(y);
if (x*y==0) return 1;
while (x!=y)
if (x>y) x-=y;
else y-=x;
return x;
}
PS rutgon(PS p)
{
PS q;
int x;
x=uscln(p.a,p.b);
q.a = p.a / x ;
q.b = p.b / x ;
return q;
}
PS cong(PS p1, PS p2)
{
PS q;
q.a = p1.a*p2.b + p2.a*p1.b;
q.b = p1.b * p2.b ;
return rutgon(q);
}
PS tru(PS p1, PS p2)
{
PS q;
q.a = p1.a*p2.b - p2.a*p1.b;
q.b = p1.b * p2.b ;
return rutgon(q);
}
PS nhan(PS p1, PS p2)
{
PS q;
q.a = p1.a * p2.a ;
q.b = p1.b * p2.b ;
return rutgon(q);
}
PS chia(PS p1, PS p2)
{
PS q;
q.a = p1.a * p2.b ;
q.b = p1.b * p2.a ;
return rutgon(q);
}
void main()
{
PS p, q, z, u, v ;
72 73
PS tu,mau, s;
printf("
Nhap phan so p: "); nhap(&p);
printf("
Nhap phan so q: ");nhap(&q);
printf("
Nhap phan so z: ");nhap(&z);
printf("
Nhap phan so u: ");nhap(&u);
printf("
Nhap phan so v: ");nhap(&v);
tu = nhan(q,z);
tu = tru(p,tu) ;
mau = cong(u,v) ;
s = chia(tu,mau);
printf(“
Phan so s = “); in(s);
getch();
}
Nhận xét: Việc sử dụng các hàm để thực hiện các phép tính không được tự nhiên và tỏ ra dài dòng. Ví dụ để thực hiện một công thức
s = (p - q*z)/(u + v)
phải dùng 2 biến trung gian và 4 lời gọi hàm. Câu hỏi đặt ra là có cách nào để chỉ cần viết đúng công thức toán học, mà vẫn nhận được kết quả mong muốn hay không?
Trong C++ có thể đáp ứng được mong muốn này bằng cách sử dụng các phép toán chuẩn của nó cho các kiểu dữ liệu tự định nghĩa (mảng, cấu trúc, ...). Nói cách khác C++ cho phép dùng các phép toán để định nghĩa các hàm, mà ta thường gọi là định nghĩa chồng các toán tử (hay còn gọi: Sự tải bội các toán tử).
7.3. Cách định nghĩa chồng các toán tử
7.3.1.Tên hàm toán tử: Gồm từ khoá operator và tên phép toán, ví dụ:
operator+ (định nghĩa chồng phép +)
operator- (định nghĩa chồng phép -)
7.3.2. Các đối của hàm toán tử:
a. Với các phép toán có 2 toán hạng, thì hàm toán tử cần có 2 đối. Đối thứ nhất ứng với toán hạng thứ nhất, đối thứ hai ứng với toán hạng thứ hai. Do vậy, với các phép toán không giao hoán (như phép-) thì thứ tự đối là rất quan trọng.
Ví dụ các hàm toán tử cộng , trừ phân số được khai báo như sau:
struct PS
{
int a; // Tử số
int b; // Mẫu số
} ;
PS operator+(PS p1, PS p2); // p1 + p2
PS operator-(PS p1, PS p2); // p1 - p2
PS operator*(PS p1, PS p2); // p1 * p2
PS operator/(PS p1, PS p2); // p1 / p2
b. Với các phép toán có một toán hạng, thì hàm toán tử có một đối. Ví dụ hàm toán tử đổi dấu ma trận (đổi dấu tất cả các phần tử của ma trận) được khai báo như sau:
struct MT
{
double a[20][20] ; // Mảng chứa các phần tử ma trận
int m ; // Số hàng ma trận
int n ; // Số cột ma trân
} ;
MT operator-(MT x) ;
7.3.3. Thân của hàm toán tử: Viết như thân của hàm thông thường. Ví dụ hàm đổi dấu ma trận có thể được định nghĩa như sau:
struct MT
{
74 75
double a[20][20] ; // Mảng chứa các phần tử ma trận
int m ; // Số hàng ma trận
int n ; // Số cột ma trân
} ;
MT operator-(MT x)
{
MT y;
for (int i=1; i<= m ;++i)
for (int j=1; j<= n ;++j)
y[i][j] = - x[i][j] ;
return y;
}
7.4. Cách dùng hàm toán tử
Có 2 cách dùng:
Cách 1: Dùng như một hàm thông thường bằng cách viết lời gọi.
Ví dụ:
PS p, q, u, v ;
u = operator+(p, q) ; // u = p + q
v = operator-(p, q) ; // v = p - q
Cách 2: Dùng như phép toán của C++ .
Ví dụ:
PS p, q, u, v ;
u = p + q ; // u = p + q
v = p - q ; // v = p - q
Chú ý: Khi dùng các hàm toán tử như phép toán của C++ ta có thể kết hợp nhiều phép toán để viết các công thức phức tạp. Cũng cho phép dùng dấu ngoặc tròn để quy định thứ tự thực hiện các phép tính. Thứ tự ưu tiên của các phép tính vẫn tuân theo các quy tắc ban đầu của C++ . Chẳng hạn các phép * và / có thứ ưu tiên cao hơn so với các phép + và -
Ví dụ:
PS p, q, u, v, s1, s2 ;
s1 = p*q - u/v ; // s1 = (p*q)
s2 = (p - q)/(u + v) ; // s2 = (p - q)/(u + v)
§ 8. Các ví dụ về định nghĩa chồng toán tử
Ví dụ 1: Trong ví dụ này ngoài việc sử dụng các hàm toán tử để thực hiện 4 phép tính trên phân số, còn định nghĩa chồng các phép toán << và >> để xuất và nhập phân số (xem chi tiết trong chương 7).
Hàm operator<< có 2 đối kiểu ostream& và PS (Phân số). Hàm trả về giá trị kiểu ostream&. Hàm được khai báo như sau:
ostream& operator<< (ostream& os, PS p);
Tượng tự hàm operator>> được khai báo như sau:
istream& operator>> (istream& is,PS &p);
Dưới đây sẽ chỉ ra cách xây dựng và sử dụng các hàm toán tử. Chúng ta cũng sẽ thấy việc sử dụng các hàm toán tử rất tự nhiên, ngắn gọn và tiện lợi.
Chương trình dưới đây có nội dung như chương trình trong §6.2, nhưng thay các hàm bằng các hàm toán tử.
#include <conio.h>
#include <iostream.h>
#include <math.h>
typedef struct
{
int a,b;
} PS;
ostream& operator<< (ostream& os, PS p);
istream& operator>> (istream& is,PS &p);
76 77
int uscln(int x, int y);
PS rutgon(PS p);
PS operator+(PS p1, PS p2);
PS operator-(PS p1, PS p2);
PS operator*(PS p1, PS p2);
PS operator/(PS p1, PS p2);
ostream& operator<< (ostream& os, PS p)
{
os << p.a << '/' << p.b ;
return os;
}
istream& operator>> (istream& is,PS &p)
{
cout << "Nhap tu va mau: " ;
is >> p.a >> p.b ;
return is;
}
int uscln(int x, int y)
{
x=abs(x); y=abs(y);
if (x*y==0) return 1;
while (x!=y)
if (x>y) x-=y;
else y-=x;
return x;
}
PS rutgon(PS p)
{
PS q;
int x;
x=uscln(p.a,p.b);
q.a = p.a / x ;
q.b = p.b / x ;
return q;
}
PS operator+(PS p1, PS p2)
{
PS q;
q.a = p1.a*p2.b + p2.a*p1.b;
q.b = p1.b * p2.b ;
return rutgon(q);
}
PS operator-(PS p1, PS p2)
{
PS q;
q.a = p1.a*p2.b - p2.a*p1.b;
q.b = p1.b * p2.b ;
return rutgon(q);
}
PS operator*(PS p1, PS p2)
{
PS q;
q.a = p1.a * p2.a ;
q.b = p1.b * p2.b ;
return rutgon(q);
}
PS operator/(PS p1, PS p2)
{
PS q;
q.a = p1.a * p2.b ;
q.b = p1.b * p2.a ;
return rutgon(q);
78 79
}
void main()
{
PS p, q, z, u, v ;
PS s;
cout <<"
Nhap cac PS p, q, z, u, v:
" ;
cin >> p >> q >> z >> u >> v ;
s = (p - q*z) / (u + v) ;
cout << "
Phan so s = " << s;
getch();
}
Ví dụ 2: Chương trình đưa vào các hàm toán tử:
operator- có một đối dùng để đảo dấu một đa thức
operator+ có 2 đối dùng để cộng 2 đa thức
operator- có 2 đối dùng để trừ 2 đa thức
operator* có 2 đối dùng để nhân 2 đa thức
operator^ có 2 đối dùng để tính giá đa thức tại x
operator<< có 2 đối dùng để in đa thức
operator>> có 2 đối dùng để nhập đa thức
Chương trình sẽ nhập 4 đa thức: p, q, r, s. Sau đó tính đa thức:
f = -(p+q)*(r-s)
Cuối cùng tính giá trị f(x), với x là một số thực nhập từ bàn phím.
#include <conio.h>
#include <iostream.h>
#include <math.h>
struct DT
{
double a[20]; // Mang chua cac he so da thuc a0, a1,...
int n ; // Bac da thuc
} ;
ostream& operator<< (ostream& os, DT d);
istream& operator>> (istream& is,DT &d);
DT operator-(const DT& d);
DT operator+(DT d1, DT d2);
DT operator-(DT d1, DT d2);
DT operator*(DT d1, DT d2);
double operator^(DT d, double x); // Tinh gia tri da thuc
ostream& operator<< (ostream& os, DT d)
{
os << " - Cac he so (tu ao): " ;
for (int i=0 ; i<= d.n ; ++i)
os << d.a[i] <<" " ;
return os;
}
istream& operator>> (istream& is, DT &d)
{
cout << " - Bac da thuc: " ;
cin >> d.n;
cout << "Nhap cac he so da thuc:
" ;
for (int i=0 ; i<= d.n ; ++i)
{
cout << "He so bac " << i << " = " ;
is >> d.a[i] ;
}
return is;
}
DT operator-(const DT& d)
{
DT p;
80 81
p.n = d.n;
for (int i=0 ; i<=d.n ; ++i)
p.a[i] = -d.a[i];
return p;
}
DT operator+(DT d1, DT d2)
{
DT d;
int k,i;
k = d1.n > d2.n ? d1.n : d2.n ;
for (i=0; i<=k ; ++i)
if (i<=d1.n && i<=d2.n)
d.a[i] = d1.a[i] + d2.a[i];
else if (i<=d1.n)
d.a[i] = d1.a[i];
else
d.a[i] = d2.a[i];
i=k;
while (i>0 && d.a[i]==0.0) --i;
d.n = i;
return d ;
}
DT operator-(DT d1, DT d2)
{
return (d1 + (-d2));
}
DT operator*(DT d1, DT d2)
{
DT d;
int k, i, j;
k = d.n = d1.n + d2.n ;
for (i=0; i<=k; ++i) d.a[i] = 0;
for (i=0 ; i<= d1.n ; ++i)
for (j=0 ; j<= d2.n ; ++j)
d.a[i+j] += d1.a[i]*d2.a[j] ;
return d;
}
double operator^(DT d, double x)
{
double s=0.0 , t=1.0;
for (int i=0 ; i<= d.n ; ++i)
{
s += d.a[i]*t;
t *= x;
}
return s;
}
void main()
{
DT p,q,r,s,f;
double x,g;
clrscr();
cout <<"
Nhap da thuc P " ; cin >> p;
cout <<"
Nhap da thuc Q " ; cin >> q;
cout <<"
Nhap da thuc R " ; cin >> r;
cout <<"
Nhap da thuc S " ; cin >> s;
cout << "
Nhap so thuc x: " ; cin >> x;
f = -(p+q)*(r-s);
82 83
g = f^x;
cout << "
Da thuc f " << f ;
cout << "
x = " << x;
cout << "
f(x) = " << g;
getch();
}
§ 9. Các bài toán về ma trận và véc tơ
Trong mục này sẽ xét các ma trận thực vuông cấp n và các véc tơ thực cấp n. Chúng được biểu diễn thông qua các kiểu cấu trúc MT và VT:
struct MT
{
double a[20][20] ; // Mang a chứa các phần tử ma trận
int n ; // Cấp ma trận
} ;
struct VT
{
double b[20]; // Mang chua cac phan tu cua vec to
int n ; // Cap vec to
} ;
Để xử lý ma trận và véc tơ, chúng ta xây dựng 9 hàm toán tử:
ostream& operator<< (ostream& os, const MT& x); // In ma trận
ostream& operator<< (ostream& os, const VT& v); // In véc tơ
istream& operator>> (istream& is,MT& x); // Nhập ma trận
istream& operator>> (istream& is, VT &v); // Nhập véc tơ
MT operator+(const MT& x1, const MT& x2); // Cộng 2 ma trận
MT operator-(const MT& x1, const MT& x2); // Trừ 2 ma trận
MT operator*(const MT& x1, const MT& x2); // Nhân 2 ma trận
VT operator*(const MT& x, const VT& v); // Nhân ma trận véc tơ
MT operator!(MT x); // Nghịch đảo ma trận
Thuật toán cho 8 hàm toán tử đầu tương đối quen thuộc không có gì phải bàn. Để nghịch đảo ma trận có nhiều cách, ở đây chúng ta dùng phương pháp Jordance như sau. Giả sử cần nghịch đảo ma trận x cấp n. Ta dùng thêm ma trận đơn vị y. Sau đó thực hiện đồng thời các phép tính trên cả x và y sao cho x trở thành đơn vị. Kết quả y chính là nghịch đảo của x. Thuật toán được tiến hành trên n bước. Nội dung của bước k (k = 1,...,n) như sau:
Tìm chỉ số r ( k <= r <= n) sao cho
abs(x[r,k]) = max { abs(x[i,k] với i = k,...,n }
Nếu abs(x[r,k]) = 0 thì ma trận không có nghịch đảo và thuật toán kết thúc giữa chừng.
Hoán vị hàng k với hàng r trong cả 2 ma trận x và y.
Chia hàng k của cả x và y cho tg = x[k,k] (mục đích làm cho x[k,k] = 1).
Biến đổi để cột k của x trơ thành véc tơ đơn vị bằng cách làm cho các phần tử x[i,k] = 0 (với i khác k). Muốn vậy ta thực hiện các phép tính sau trên cả x và y:
(hàng i) = (hàng i) - x[i,k]*(hàng k) , với mọi i khác k
Nội dung chương trình là nhập 4 ma trận X, Y, R, S và véc tơ u. Sau đó tính véc tơ v theo công thức:
v = ((X + Y)*(R - S))-1u
Như sẽ thấy trong hàm main() dưới đây, nhờ các hàm toán tử mà câu lệnh tính v được viết gần giống như công thức toán học nêu trên.
/* Chương trình */
#include <conio.h>
#include <iostream.h>
#include <iomanip.h>
#include <math.h>
struct MT
{
84 85
double a[20][20]; // Mang chua cac phan tu ma tran
int n ; // Cap ma tran
} ;
struct VT
{
double b[20]; // Mang chua cac phan tu cua vec to
int n ; // Cap vec to
} ;
ostream& operator<< (ostream& os, const MT& x);
ostream& operator<< (ostream& os, const VT& v);
istream& operator>> (istream& is,MT& x);
istream& operator>> (istream& is, VT &v);
MT operator+(const MT& x1, const MT& x2);
MT operator-(const MT& x1, const MT& x2);
MT operator*(const MT& x1, const MT& x2);
VT operator*(const MT& x, const VT& v);
MT operator!(MT x); // Tinh ma tran nghich dao
ostream& operator<< (ostream& os, const MT& x)
{
os << setprecision(2) << setiosflags(ios::showpoint);
for (int i=1 ; i<= x.n ; ++i)
{
os << "
" ;
for (int j=1; j<=x.n; ++j)
os << setw(6) << x.a[i][j] ;
}
os << "
" ;
return os;
}
ostream& operator<< (ostream& os, const VT& v)
{
os << setprecision(2) << setiosflags(ios::showpoint);
for (int i=1 ; i<= v.n ; ++i)
os << setw(6) << v.b[i] ;
os << "
" ;
return os;
}
istream& operator>> (istream& is, MT& x)
{
cout << " - Cap ma tran: " ;
is >> x.n;
cout << "Nhap cac phan tu :
" ;
for (int i=1 ; i<= x.n ; ++i)
for (int j=1; j<=x.n; ++j)
{
cout << "PT hang " << i << " cot " << j << " = " ;
is >> x.a[i][j] ;
}
return is;
}
istream& operator>> (istream& is, VT& v)
{
cout << " - Cap vec to: " ;
is >> v.n;
cout << "Nhap cac phan tu :
" ;
for (int i=1 ; i<= v.n ; ++i)
{
cout << "Phan tu thu " << i << " = " ;
is >> v.b[i] ;
86 87
}
return is;
}
MT operator+(const MT& x1, const MT& x2)
{
if (x1.n!=x2.n)
{
cout << "
Khong thuc hien duoc phep cong vi 2 MT khong cung cap";
getch();
return x1;
}
else
{
MT x;
int i, j, n;
n = x.n = x1.n ;
for (i=1; i<=n; ++i)
for (j=1; j<=n ;++j)
x.a[i][j] = x1.a[i][j] + x2.a[i][j] ;
return x;
}
}
MT operator-(const MT& x1, const MT& x2)
{
if (x1.n!=x2.n)
{
cout << "
Khong thuc hien duoc phep tru vi 2 MT khong cung cap";
getch();
return x1;
}
else
{
MT x;
int i, j, n;
n = x.n = x1.n;
for (i=1; i<=n; ++i)
for (j=1; j<=n ;++j)
x.a[i][j] = x1.a[i][j] - x2.a[i][j] ;
return x;
}
}
MT operator*(const MT& x1, const MT& x2)
{
if (x1.n!=x2.n)
{
cout << "
Khong thuc hien duoc phep nhan vi 2 MT khong cung cap";
getch();
return x1;
}
else
{
MT x;
int n, i, j,k;
n = x.n = x1.n;
for (i=1; i<=n; ++i)
for (j=1; j<=n ;++j)
{
88 89
x.a[i][j] = 0.0 ;
for (k=1 ; k<=n; ++k)
x.a[i][j] += x1.a[i][k]*x2.a[k][j] ;
}
return x;
}
}
VT operator*(const MT& x, const VT& v)
{
if (x.n != v.n)
{
cout << "
Cap ma tran khac cap vec to, phep nhan vo nghia";
getch();
return v;
}
else
{
VT u; int n;
n = u.n = v.n ;
for (int i=1; i <=n ; ++i)
{
u.b[i] = 0;
for (int j=1; j<=n; ++j)
u.b[i] += x.a[i][j]*v.b[j];
}
return u;
}
}
MT operator!(MT x)
{
MT y;
int i,j,k,r,n;
double tg;
n = y.n = x.n ;
for (i=1 ; i<=n ; ++i)
for (j=1 ; j<=n ; ++j)
if (i==j) y.a[i][j] = 1;
else y.a[i][j] = 0;
for (k=1; k<=n; ++k)
{
r=k;
for (i=k+1; i<=n; ++i)
if (abs(x.a[i][k]) > abs(x.a[r][k]) ) r = i;
if (abs(x.a[r][k]) < 1.0E-8)
{
cout << "
Ma tran suy bien, khong co nghich dao" ;
getch();
return x;
}
/* Hoan vi hang r va hang k */
for (j=1 ; j<=n ; ++j)
{
tg = x.a[k][j];
x.a[k][j] = x.a[r][j];
90 91
x.a[r][j] = tg;
tg = y.a[k][j];
y.a[k][j] = y.a[r][j];
y.a[r][j] = tg;
}
/* Chia hang k cho a[k,k] */
tg = x.a[k][k] ;
for (j=1 ; j<=n ; ++j)
{
x.a[k][j] /= tg;
y.a[k][j] /= tg;
}
/* Khu cot k : lam cho a[i,k] = 0 voi i != k */
for (int i=1; i<= n ; ++i)
if (i != k)
{
tg = x.a[i][k] ;
for (j=1 ; j<=n ; ++j)
{
x.a[i][j] -= tg*x.a[k][j] ;
y.a[i][j] -= tg*y.a[k][j] ;
}
}
}
return y;
}
void main()
{
MT x,y,r,s;
VT u,v;
clrscr();
cout <<"
Nhap ma tran X " ; cin >> x;
cout <<"
Nhap ma tran Y " ; cin >> y;
cout <<"
Nhap ma tran R " ; cin >> r;
cout <<"
Nhap ma tran S " ; cin >> s;
cout <<"
Nhap vec to u " ; cin >> u;
v = !((x+y)*(r-s))*u ;
cout << "
Vec to v = xu " << v ;
getch();
}
92
Bạn đang đọc truyện trên: Truyen247.Pro