Tải bản đầy đủ (.pdf) (30 trang)

Tài liệu OPP C++ (phần 1) docx

Bạn đang xem bản rút gọn của tài liệu. Xem và tải ngay bản đầy đủ của tài liệu tại đây (1.26 MB, 30 trang )

1


CHƯƠNG 1
GIỚI THIỆU VỀ LẬP TRÌNH HƯỚNG ĐỐI TƯỢNG

1.1 LẬP TRÌNH HƯỚNG ĐỐI TƯỢNG (OOP) LÀ GÌ ?
Lập trình hướng đối tượng (Object-Oriented Programming, viết tắt là OOP) là một phương pháp mới trên
bước đường tiến hóa của việc lập trình máy tính, nhằm làm cho chương trình trở nên linh hoạt, tin cậy và dễ
phát triển. Tuy nhiên để hiểu được OOP là gì, chúng ta hãy bắt đầu từ lịch sử của quá trình lập trình – xem
xét OOP đã tiến hóa như thế nào.
1.1.1 Lập trình tuyến tính


Máy tính đầu tiên được lập trình bằng mã nhị phân, sử dụng các công tắt cơ khí để nạp chương trình.
Cùng với sự xuất hiện của các thiết bị lưu trữ lớn và bộ nhớ máy tính có dung lượng lớn nên các
ngôn ngữ lập trình cấp cao đầu tiên được đưa vào sử dụng . Thay vì phải suy nghĩ trên một dãy các
bit và byte, lập trình viên có thể viết một loạt lệnh gần với tiếng Anh và sau đó chươ
ng trình dịch
thành ngôn ngữ máy. Các ngôn ngữ lập trình
cấp cao đầu tiên được thiết kế để lập các chương trình làm các công việc tương đối đơn giản như tính
toán. Các chương trình ban đầu chủ yếu liên quan đến tính toán và không đòi hỏi gì nhiều ở ngôn
ngữ lập trình. Hơn nữa phần lớn các chương trình này tương đối ngắn, thường ít hơn 100 dòng.
Khi khả năng của máy tính tăng lên thì khả năng để triển khai các chươ
ng trình phức tạp hơn cũng
2

tăng lên. Các ngôn ngữ lập trình ngày trước không còn thích hợp đối với việc lập trình đòi hỏi cao
hơn. Các phương tiện cần thiết để sử dụng lại các phần mã chương trình đã viết hầu như không có
trong ngôn ngữ lập trình tuyến tính. Thật ra, một đoạn lệnh thường phải được chép lặp lại mỗi khi
chúng ta dùng trong nhiều chương trình do đó chương trình dài dòng, logic của chương trình khó


hiểu. Ch
ương trình được điều khiển để nhảy đến nhiều chỗ mà thường không có sự giải thích rõ ràng,
làm thế nào để chương trình đến chỗ cần thiết hoặc tại sao như vậy.
Ngôn ngữ lập trình tuyến tính không có khả năng kiểm soát phạm vi nhìn thấy của các dữ liệu. Mọi
dữ liệu trong chương trình đều là dữ liệu toàn cục nghĩa là chúng có thể bị sửa đổi ở b
ất kỳ phần nào
của chương trình. Việc dò tìm các thay đổi không mong muốn đó của các phần tử dữ liệu trong một
dãy mã lệnh dài và vòng vèo đã từng làm cho các lập trình viên rất mất thời gian.
1.1.2 Lập trình cấu trúc
:
Rõ ràng là các ngôn ngữ mới với các tính năng mới cần phải được phát triển để có thể tạo ra các ứng
dụng tinh vi hơn. Vào cuối các năm trong 1960 và 1970, ngôn ngữ lập trình có cấu trúc ra đời. Các
chương trình có cấu trúc được tổ chức theo các công việc mà chúng thực hiện.
Về bản chất, chương trình chia nhỏ thành các chương trình con riêng rẽ (còn gọi là hàm hay thủ tục)
thực hiện các công việc rời rạc trong quá trình lớn hơn, phức tạ
p hơn. Các hàm này được giữ càng
độc lập với nhau càng nhiều càng tốt, mỗi hàm có dữ liệu và logic riêng.Thông tin được chuyển giao
giữa các hàm thông qua các tham số, các hàm có thể có các biến cục bộ mà không một ai nằm bên
ngoài phạm vi của hàm lại có thể truy xuất được chúng. Như vậy, các hàm có thể được xem là các
chương trình con được đặt chung với nhau để xây dựng nên một ứng dụng.
Mục tiêu là làm sao cho việc triển khai các phần mềm dễ dàng hơn đối v
ới các lập trình viên mà vẫn
cải thiện được tính tin cậy và dễ bảo quản chương trình. Một chương trình có cấu trúc được hình
thành bằng cách bẻ gãy các chức năng cơ bản của chương trình thành các mảnh nhỏ mà sau đó trở
thành các hàm. Bằng cách cô lập các công việc vào trong các hàm, chương trình có cấu trúc có thể
làm giảm khả năng của một hàm này ảnh hưởng đến một hàm khác. Việc này cũng làm cho việc tách
các vấn đề trở nên d
ễ dàng hơn. Sự gói gọn này cho phép chúng ta có thể viết các chương trình sáng
sủa hơn và giữ được điều khiển trên từng hàm. Các biến toàn cục không còn nữa và được thay thế
bằng các tham số và biến cục bộ có phạm vi nhỏ hơn và dễ kiểm soát hơn. Cách tổ chức tốt hơn này

nói lên rằng chúng ta có khả năng quản lý logic của cấu trúc chương trình, làm cho việc triển khai và
bảo dưỡng chương trình nhanh hơn và hữ
u hiện hơn và hiệu quả hơn.
Một khái niệm lớn đã được đưa ra trong lập trình có cấu trúc là sự trừu tượng hóa (Abstraction).
Sự trừu tượng hóa có thể xem như khả năng quan sát một sự việc mà không cần xem xét đến các chi
tiết bên trong của nó. Trong một chương trình có cấu trúc, chúng ta chỉ cần biết một hàm đã cho có
thể làm được một công việc cụ thể gì là đủ. Còn làm thế nào mà công vi
ệc đó lại thực hiện được là
không quan trọng, chừng nào hàm còn tin cậy được thì còn có thể dùng nó mà không cần phải biết nó
thực hiện đúng đắn chức năng của mình như thế nào. Điều này gọi là sự trừu tượng hóa theo chức
năng (Functional abstraction) và là nền tảng của lập trình có cấu trúc.
Ngày nay, các kỹ thuật thiết kế và lập trình có cấu trúc được sử rộng rãi. Gần như
mọi ngôn ngữ lập
trình đều có các phương tiện cần thiết để cho phép lập trình có cấu trúc. Chương trình có cấu trúc dễ
viết, dễ bảo dưỡng hơn các chương trình không cấu trúc.
Sự nâng cấp như vậy cho các kiểu dữ liệu trong các ứng dụng mà các lập trình viên đang viết cũng
đang tiếp tục diễn ra. Khi độ phức tạp của một chương trình tăng lên, sự phụ thuộc củ
a nó vào các
kiểu dữ liệu cơ bản mà nó xử lý cũng tăng theo. Vấn đề trở rõ ràng là cấu trúc dữ liệu trong chương
trình quan trọng chẳng kém gì các phép toán thực hiện trên chúng. Điều này càng trở rõ ràng hơn khi
kích thước của chương trình càng tăng. Các kiểu dữ liệu được xử lý trong nhiều hàm khác nhau bên
trong một chương trình có cấu trúc. Khi có sự thay đổi trong các dữ liệu này thì cũng cần phải thực
hiện cả các thay đổi ở
mọi nơi có các thao tác tác động trên chúng. Đây có thể là một công việc tốn
thời gian và kém hiệu quả đối với các chương trình có hàng ngàn dòng lệnh và hàng trăm hàm trở
lên.
3

Một yếu điểm nữa của việc lập trình có cấu trúc là khi có nhiều lập trình viên làm việc theo nhóm
cùng một ứng dụng nào đó. Trong một chương trình có cấu trúc, các lập trình viên được phân công

viết một tập hợp các hàm và các kiểu dữ liệu. Vì có nhiều lập trình viên khác nhau quản lý các hàm
riêng, có liên quan đến các kiểu dữ liệu dùng chung nên các thay đổi mà lập trình viên tạo ra trên một
phần tử dữ liệu sẽ làm ảnh hưởng đến công vi
ệc của tất cả các người còn lại trong nhóm. Mặc dù
trong bối cảnh làm việc theo nhóm, việc viết các chương trình có cấu trúc thì dễ dàng hơn nhưng sai
sót trong việc trao đổi thông tin giữa các thành viên trong nhóm có thể dẫn tới hậu quả là mất rất
nhiều thời gian để sửa chữa chương trình.
1.1.3 Sự trừu tượng hóa dữ liệu
:
Sự trừu tượng hóa dữ liệu (Data abstraction) tác động trên các dữ liệu cũng tương tự như sự trừu
tượng hóa theo chức năng. Khi có trừu tượng hóa dữ liệu, các cấu trúc dữ liệu và các phần tử có thể
được sử dụng mà không cần bận tâm đến các chi tiết cụ thể. Chẳng hạn như các số dấu chấm động đã
được trừu tượng hóa trong tấ
t cả các ngôn ngữ lập trình, Chúng ta không cần quan tâm cách biểu
diễn nhị phân chính xác nào cho số dấu chấm động khi gán một giá trị, cũng không cần biết tính bất
thường của phép nhân nhị phân khi nhân các giá trị dấu chấm động. Điều quan trọng là các số dấu
chấm động hoạt động đúng đắn và hiểu được.
Sự trừu tượng hóa dữ liệu giúp chúng ta không phải bận tâm về các chi tiết không cần thiết. N
ếu lập
trình viên phải hiểu biết về tất cả các khía cạnh của vấn đề, ở mọi lúc và về tất cả các hàm của
chương trình thì chỉ ít hàm mới được viết ra, may mắn thay trừu tượng hóa theo dữ liệu đã tồn tại sẵn
trong mọi ngôn ngữ lập trình đối với các dữ liệu phức tạp như số dấu chấm động. Tuy nhiên chỉ mới
gầ
n đây, người ta mới phát triển các ngôn ngữ cho phép chúng ta định nghĩa các kiểu dữ liệu trừu
tượng riêng.
1.1.4 Lập trình hướng đối tượng
:
Khái niệm hướng đối tượng được xây dựng trên nền tảng của khái niệm lập trình có cấu trúc và sự
trừu tượng hóa dữ liệu. Sự thay đổi căn bản ở chỗ, một chương trình hướng đối tượng được thiết kế
xoay quanh dữ liệu mà chúng ta có thể làm việc trên đó, hơn là theo bản thân chức năng của chương

trình. Điều này hoàn toàn tự nhiên một khi chúng ta hiểu rằng m
ục tiêu của chương trình là xử lý dữ
liệu. Suy cho cùng, công việc mà máy tính thực hiện vẫn thường được gọi là xử lý dữ liệu. Dữ liệu
và thao tác liên kết với nhau ở một mức cơ bản (còn có thể gọi là mức thấp), mỗi thứ đều đòi hỏi ở
thứ kia có mục tiêu cụ thể, các chương trình hướng đối tượng làm tường minh mối quan hệ này.
Lập trình hướng đố
i tượng liên kết cấu trúc dữ liệu với các thao tác, theo cách mà tất cả thường nghĩ
về thế giới quanh mình. Chúng ta thường gắn một số các hoạt động cụ thể với một loại hoạt động
nào đó và đặt các giả thiết của mình trên các quan hệ đó.
Ví dụ1.1
: Chúng ta biết rằng một chiếc xe có các bánh xe, di chuyển được và có thể đổi hướng của
nó bằng cách quẹo tay lái. Tương tự như thế, một cái cây là một loại thực vật có thân gỗ và lá. Một
chiếc xe không phải là một cái cây, mà cái cây không phải là một chiếc xe, chúng ta có thể giả thiết
rằng cái mà chúng ta có thể làm được với một chiếc xe thì không thể làm được với một cái cây.
Chẳng hạn, thật là vô nghĩa khi muốn lái một cái cây, còn chi
ếc xe thì lại chẳng lớn thêm được khi
chúng ta tưới nước cho nó.
Lập trình hướng đối tượng cho phép chúng ta sử dụng các quá trình suy nghĩ như vậy với các khái
niệm trừu tượng được sử dụng trong các chương trình máy tính. Một mẫu tin (record) nhân sự có thể
được đọc ra, thay đổi và lưu trữ lại; còn số phức thì có thể được dùng trong các tính toán. Tuy vậy
không thể nào lại viết một số phức vào tập tin làm mẫu tin nhân sự và ng
ược lại hai mẫu tin nhân sự
lại không thể cộng với nhau được. Một chương trình hướng đối tượng sẽ xác định đặc điểm và hành
vi cụ thể của các kiểu dữ liệu, điều đó cho phép chúng ta biết một cách chính xác rằng chúng ta có
thể có được những gì ở các kiểu dữ liệu khác nhau.
Chúng ta còn có thể tạo ra các quan hệ giữa các kiểu dữ liệu tương tự nhưng khác nhau trong mộ
t
4

chương trình hướng đối tượng. Người ta thường tự nhiên phân loại ra mọi thứ, thường đặt mối liên

hệ giữa các khái niệm mới với các khái niệm đã có, và thường có thể thực hiện suy diễn giữa chúng
trên các quan hệ đó. Hãy quan niệm thế giới theo kiểu cấu trúc cây, với các mức xây dựng chi tiết
hơn kế tiếp nhau cho các thế hệ sau so với các thế hệ trước. Đây là phươ
ng pháp hiệu quả để tổ chức
thế giới quanh chúng ta. Các chương trình hướng đối tượng cũng làm việc theo một phương thức
tương tự, trong đó chúng cho phép xây dựng các các cơ cấu dữ liệu và thao tác mới dựa trên các cơ
cấu có sẵn, mang theo các tính năng của các cơ cấu nền mà chúng dựa trên đó, trong khi vẫn thêm
vào các tính năng mới.
Lập trình hướng đối tượng cho phép chúng ta tổ chức dữ liệu trong ch
ương trình theo một cách
tương tự như các nhà sinh học tổ chức các loại thực vật khác nhau. Theo cách nói lập trình đối tượng,
xe hơi, cây cối, các số phức, các quyển sách đều được gọi là các lớp (Class).
Một lớp là một bản mẫu mô tả các thông tin cấu trúc dữ liệu, lẫn các thao tác hợp lệ của các phần tử
dữ liệu. Khi một phần tử dữ liệu được khai báo là phầ
n tử của một lớp thì nó được gọi là một đối
tượng (Object). Các hàm được định nghĩa hợp lệ trong một lớp được gọi là các phương thức
(Method) và chúng là các hàm duy nhất có thể xử lý dữ liệu của các đối tượng của lớp đó. Một thực
thể (Instance) là một vật thể có thực bên trong bộ nhớ, thực chất đó là m
ột đối tượng (nghĩa là một
đối tượng được cấp phát vùng nhớ).
Mỗi một đối tượng có riêng cho mình một bản sao các phần tử dữ liệu của lớp còn gọi là các biến
thực thể (Instance variable). Các phương thức định nghĩa trong một lớp có thể được gọi bởi các đối
tượng của lớp đó. Điều này được gọi là gửi một thông
điệp (Message) cho đối tượng. Các thông
điệp này phụ thuộc vào đối tượng, chỉ đối tượng nào nhận thông điệp mới phải làm việc theo thông
điệp đó. Các đối tượng đều độc lập với nhau vì vậy các thay đổi trên các biến thể hiện của đối tượng
này không ảnh hưởng gì trên các biến thể hiện của các đối tượng khác và việc gửi thông điệp cho
m
ột đối tượng này không ảnh hưởng gì đến các đối tượng khác.
1.2 MỘT SỐ KHÁI NIỆM MỚI TRONG LẬP TRÌNH HƯỚNG ĐỐI TƯỢNG

Trong phần này, chúng ta tìm hiểu các khái niệm như sự đóng gói, tính kế thừa và tính đa hình. Đây là các
khái niệm căn bản, là nền tảng tư tưởng của lập trình hướng đối tượng. Hiểu được khái niệm này, chúng ta
bước đầu tiếp cận với phong cách lập trình mới, phong cách lập trình dựa vào đối tượng làm nền tảng mà
trong đó quan điểm che dấu thông tin thông qua sư đóng gói là quan điểm trung tâm của vấn đề
.
1.2.1 Sự đóng gói (Encapsulation)

Sự đóng gói là cơ chế ràng buộc dữ liệu và thao tác trên dữ liệu đó thành một thể thống nhất, tránh
được các tác động bất ngờ từ bên ngoài. Thể thống nhất này gọi là đối tượng.
Trong một đối tượng, dữ liệu hay thao tác hay cả hai có thể là riêng (private) hoặc chung (public)
của đối tượng đó. Thao tác hay dữ liệu riêng là thuộc về đối tượng đó chỉ
được truy cập bởi các
thành phần của đối tượng, điều này nghĩa là thao tác hay dữ liệu riêng không thể truy cập bởi các
phần khác của chương trình tồn tại ngoài đối tượng. Khi thao tác hay dữ liệu là chung, các phần khác
của chương trình có thể truy cập nó mặc dù nó được định nghĩa trong một đối tượng. Các thành phần
chung của một đối tượng dùng để cung cấp một giao diện có điều khiển cho các thành thành riêng
của đối tượng.Cơ chế đóng gói là phương thức tốt để thực hiện cơ chế che dấu thông tin so với các
ngôn ngữ lập trình cấu trúc.
1.2.2 Tính kế thừa (Inheritance)

Chúng ta có thể xây dựng các lớp mới từ các lớp cũ thông qua sự kế thừa. Một lớp mới còn gọi là
lớp dẫn xuất (derived class), có thể thừa hưởng dữ liệu và các phương thức của lớp cơ sở (base
class) ban đầu. Trong lớp này, có thể bổ sung các thành phần dữ liệu và các phương thức mới vào

1.
2
những
t
xuất) c
ó

lớp đư

được g

Ví dụ 1
phẩm:
t
phần d

Các ấn
Tiếp đ
ó
và chứ
a
được đ

Các th
à
và số h
i
phẩm,
t
Với tín
h
có đượ
c
2
.3 Tính đa
h
t

hành phần
d
ó
thể có mộ
t

c hình thàn
h

i là lớp ch
a
.2: Chúng t
a
t
ạp chí và s
á

liệu tương
phẩm có th

ó
chúng ta đ

a
nhiều bài
c

t vào định
n
à

nh phần dữ
i
ệu ISBN c

t
rong khi vẫ
n
h
kế thừa, c
h
c
trong các l
h
ì
nh (Poly
m
d
ữ liệu và c
á
t
số lượng b

h
. Dạng cây
a
(parent cl
a
a
sẽ xây dự
n

á
ch. Chúng t
a
ứng với số
t

được lấy r
a

nh nghĩa ha
c
ủa các tác g
n
ghĩa của lớ
p
liệu xác địn
h

a nó. Như
v
n
có các thu

Hình 1.1:
h
úng ta khô
n
ớp dẫn xuất
m
orphism)

á
c phương t
h

t kỳ các lớ
p
của các lớp
a
ss) và các l
n
g một tập c
á
a
có thể tạo
m
t
rang,
m
ã số
a
, cất đi và
đ
i lớp dẫn xu
iả khác nha
u
p
tạp chí. T

h
cho sách

s
v
ậy chúng ta

c tính riên
g
Lớp ấn phẩ
m
n
g phải mất
c
các đặc trư
n
5
h
ức mà nó t
h
p
dẫn xuất.
Q
trông giốn
g
ớp dẫn xuất
á
c lớp mô tả
m
ột ấn phẩ
m
tra cứu, ng
à

đ
ọc. Đó là c
á
ất tên là tạp
u
. Các thàn
h

p chí cũng
c
s
ẽ bao gồm t
có thể thấy,
g
của chúng.
m
và các lớ
p
c
ông xây d

n
g cần thiết.
h
ừa hưởng t

Q
ua cơ cấu
k
g

như các câ
y
được gọi là
cho thư việ
n
m
tổng quát
b
à
y tháng xu

á
c phương t
h
chí và sách.
h
phần dữ li

c
ần có một
p
ên của (các
)
sách và tạp

p
dẫn xuất c


ng lại từ đầ

u

lớp cơ sở.
M
k
ế thừa này,
d
y
gia phả vì
t
lớp con (c
h
n
các ấn ph

b
ằng cách đ


t bản, bản
q
h
ức thực hiệ
n
Tạp chí có

u
t
ương ứn
g

p
hương thức
)
tác giả, loạ
i
chí có chu
n

a nó.
u
các lớp m

M
ỗi lớp (kể
d
ạng hình c
â
t
hế các lớp
c
h
ild class).

m. Có hai k

nh nghĩa c
á
q
uyền và nh
à

n
trên một ấ
n
tên, số ký p
h
g
với các yế
u
nữa đó là đ

i
bìa (cứng
h
n
g các đặc t
r
ư

i, chỉ cần b

cả lớp dẫn
â
y của các
c
ơ sở còn

iểu ấn
á
c thành
à

xuất bản.
n
phẩm.
h
át hành
u
tố này

t mua.
h
ay mềm)
ư
ng ấn


sung để


Đó là
k
tượng
n
phươn
g
tượng
c
xuất đị
n
với ph
ư

ngh
ĩ
a c
h
Ví dụ 1
nhiên p
cuối cù
n
(dựa tr
ê
sách (d

lấy ra
m
riêng c
h
sách.
K
hai lớp
Tính đ
a
sự. Khi
tương

nhận t
h
chạy.
Đ
xảy ra
k

k
hả năng để
c
n
hận thông
đ
g
thức nào đ
ó
c
ủa lớp cơ s

n
h nghĩa lại
ư
ơng thức n
à
h
o lớp dẫn
x
.3: Xét lại
v
hương phá
p
n
g giống nh
ê
n một bản t
r


a t
r
ên hệ t
h
m
ột tạp chí
h
h
o tạp chí, c
ò
K
ết quả là ch

dẫn xuất có
a
hình dựa t
r
các phươn
g

ng với phư
ơ
h
ông điệp lú
c
Đ
iều này gọi
k
hi chương
t

c
ho một thô
n
đ
iệp. Khi m

ó
mà nó thừ
a

, sẽ dùng p
h
một phươn
g
à
y, khi được
x
uất.
v
í dụ 1.2, ch
ú
lấy ra cho t
au. Khi phả
i
r
a cứu) như
n
h
ống phiếu l
ư

h
ay một cuố
n
ò
n khi lấy r
a

cần một
t
ê
n
liên quan,
m
r
ên sự nối k
ế
g
thức kiểu
đ
ơ
ng thức nà
o
c
đó là thuộ
c
là sự kết nố
i
tr
ình đang t
h

n
g điệp có t
h

t lớp dẫn x
u
a
hưởng từ l

h
ương thức
đ
g
thức thừa
h
gởi tới một

ú
ng ta thấy
r
ạp chí có k
h
i
lấy ra tạp
c
n
g khi lấy r
a
ư
u trữ). Tín

h
n
sách. Khi
l
a
một cuốn
s
n
phương th

m
ặc dù việc
t
ế
t (Binding),
đ
a hình đượ
c
o
sẽ được g

c
lớp nào, d
o
i
muộn (Lat
e
h
ực hiện.
6

h
ể thay đổi
c
u
ất được tạo

p cơ sở củ
a
đ
ã định ngh
ĩ
h
ưởng từ lớ
p
đối tượng c


r
ằng cả tạp c
h
h
ác so với p
h
c
hí, thì phải
s
a
sách thì lại
h
đa hình ch

o
l
ấy ra một tạ
p
s
ách thì nó s


c duy nhất
t
hực hiện c

đó là quá tr
c
sử dụng th
ì

i. Hàm cụ t
h
o
đó hàm đ
ư

e
binding) h
c
ách thực hi

ra, nó có th


a
nó. Một th
ô
ĩ
a cho nó tr
o
p
cơ sở của
n

a lớp dẫn
x

h
í và và sác
h
h
ương pháp
l
s
ử dụng ph
ư
phải sử dụ
n
o
phép chú
n
p
chí nó sẽ
d


dụng phư
ơ
được dùng
c

a phương t
h
ình gắn mộ
t

ì
trình biên
d
h
ể được gọi

c gọi chỉ x
á
ay kết nối l
ú

n của nó th
e

thay đổi c
á
ô
ng điệp kh
i

o
ng lớp cơ s

n
ó thì một th
ô
x
uất sẽ gọi p
h

h
đều phải c
l
ấy ra cho s
á
ư
ơng pháp lấ
y
n
g phương p
h
n
g ta xác địn
h
d
ùng phươn
g
ơ
ng thức lấy
c

ho cả hai c
ô
h
ức đó thay
đ
t
phương th

d
ịch chưa th

sẽ tuỳ thuộ
c
á
c định đượ
c
ú
c chạy (Ru
n
e
o lớp cụ th

á
ch thực hiệ
n
i
được gởi đ

. Nếu một l
ô

ng điệp có
h
ương thức
đ

ó khả năng
l
á
ch, mặc dù
y
ra riêng c
h
h
áp lấy ra ri
ê
h
một phươ
n
g
thức lấy r
a
ra tương ứ
n
ô
ng việc tiế
n
đ
ổi
t
ùy theo


c với một h
à

xác định h
à
c
vào việc p
h
c
vào lúc ch
ư
n
time bindi
n

của đối
n
các
ến một đối
ớp dẫn
cùng tên
đ
ã định

l
ấy ra. Tuy
kết quả
h
o tạp chí

ê
ng cho
n
g thức để
a
dành
n
g với
n
hành trên
t
ừng lớp.
à
m thực
à
m nào
h
ần tử
ư
ơng trình
n
g) vì nó



7

Hình 1.2: Minh họa tính đa hình đối với lớp ấn phẩm và các lớp dẫn xuất của nó.
1.3 CÁC NGÔN NGỮ VÀ VÀI ỨNG DỤNG CỦA OOP
Xuất phát từ tư tưởng của ngôn ngữ SIMULA67, trung tâm nghiên cứu Palo Alto (PARC) của hãng

XEROR đã tập trung 10 năm nghiên cứu để hoàn thiện ngôn ngữ OOP đầu tiên với tên gọi là
Smalltalk. Sau đó các ngôn ngữ OOP lần lượt ra đời như Eiffel, Clos, Loops, Flavors, Object Pascal,
Object C, C++, Delphi, Java…
Chính XEROR trên cơ sở ngôn ngữ OOP đã đề ra tư tưởng giao diện biểu tượng trên màn hình (icon
base screen interface), kể từ đó Apple Macintosh cũng như Microsoft Windows phát triển giao diện
đồ họa như ngày nay. Trong Microsoft Windows, tư tưởng OOP được th
ể hiện một cách rõ nét nhất
đó là "chúng ta click vào đối tượng", mỗi đối tượng có thể là control menu, control menu box, menu
bar, scroll bar, button, minimize box, maximize box, … sẽ đáp ứng công việc tùy theo đặc tính của
đối tượng. Turbo Vision của hãng Borland là một ứng dụng OOP tuyệt vời, giúp lập trình viên không
quan tâm đến chi tiết của chương trình gia diện mà chỉ cần thực hiện các nội dung chính của vấn đề.
2.1 LỊCH SỬ CỦA C++
Vào những năm đầu thập niên 1980, người dùng biết C++ với tên gọi "C with Classes" được mô tả trong hai
bài báo của Bjarne Stroustrup (thuộc AT&T Bell Laboratories) với nhan đề "Classes: An Abstract Data
Type Facility for the C Language" và "Adding Classes to C : AnExercise in Language Evolution". Trong
công trình này, tác giả đã đề xuất khái niệm lớp, bổ sung việc kiểm tra kiểu tham số của hàm, các chuyển
đổi kiểu và một số mở rộng khác vào ngôn ngữ C. Bjarne Stroustrup nghiên cứu mở rộng ngôn ngữ C nhằm
đạt đến một ngôn ngữ mô phỏng (simulation language) với những tính năng hướng
đối tượng.
Trong năm 1983, 1984, ngôn ngữ "C with Classes" được thiết kế lại, mở rộng hơn rồi một trình biên dịch ra
đời. Và chính từ đó, xuất hiện tên gọi "C++". Bjarne Stroustrup mô tả ngôn ngữ C++ lần đầu tiên trong bài
báo có nhan đề "Data Abstraction in C". Sau một vài hiệu chỉnh C++ được công bố rộng rãi trong quyển
"The C++ Programming Language" của Bjarne Stroustrup xuất hiện đánh dấu sự hiện diện thực sự của C++,
người lập tình chuyên nghiệp từ đây đ
ã có một ngôn ngữ đủ mạnh cho các dữ án thực tiễn của mình.
Về thực chất C++ giống như C nhưng bổ sung thêm một số mở rộng quan trọng, đặc biệt là ý tưởng về đối
tượng, lập trình định hướng đối tượng.Thật ra các ý tưởng về cấu trúc trong C++ đã xuất phát vào các năm
1970 từ Simula 70 và Algol 68. Các ngôn ngữ này đã đưa ra các khái niệm về lớp và đơn thể. Ada là một
ngôn ng
ữ phát triển từ đó, nhưng C++ đã khẳng định vai trò thực sự của mình.

2.2 CÁC MỞ RỘNG CỦA C++
2.2.1 Các từ khóa mới của C++
Để bổ sung các tính năng mới vào C, một số từ khóa (keyword) mới đã được đưa vào C++ ngoài các
từ khóa có trong C. Các chương trình bằng C nào sử dụng các tên trùng với các từ khóa cần phải thay
đổi trước khi chương trình được dịch lại bằng C++. Các từ khóa mới này là :
asm catch class delete friend inline
new operator private protected public template
this throw try virtual

2.2.2 Cách ghi chú thích

C
+
b
i
ê
2.
2
+
+ chấp nhậ
n
ê
n dịch sẽ b


1
2
3
4
5

6
7
8
9
1
1



Mọi th

minh h

chạy ví
Nói ch
u
được d
ù
2
.3 Dòng nh
n
hai kiểu c
h

qua mọi th

1
: /*
2
: Chương

3
: */
4
: #inclu
d
5
: int mai
6
: {

7
: int
8
: for(
9
:
c
1
0: ret
u
1
1: }


nằm giữa
/

a cách chú
dụ 2.1
, kết

q
u
ng, kiểu ch
ú
ù
ng cho các
ập/xuất ch
u
h
ú thích. Cá
c

nằm giữa
/
trình in
c
d
e <iostre
a
n()

I;

I = 0; I
<
c
out<<I<<"
\
u
rn 0;



/
*…*/ từ dò
n
thích thứ h
a
q
uả ở hình
2
ú
thích /*…
*
chú thích
m
u
ẩn
c
lập trình vi
/
*…*/.
c
ác s

từ
a
m.h>
<
10 ; ++
\

n";
/
/ In
n
g 1 đến dò
n
a
i. Đó là các
h
2
.1.
Hình 2.
1
*
/ được dùn
g
m
ột dòng.
8
ên bằng C
đ

d
CT2_1.CPP

0 đến 9.
I)// 0 -
9
ra 0 - 9
n

g 3 đều đư

h
chú thích
b
1
: Kết quả c

g
cho các k
h
đ
ã quen với
c
d
ụ 2.1: Tron
g
9


c chương t
r
b
ắt đầu bằn
g

a ví dụ 2.1
h
ối chú thíc
h

c
ách chú thí
c
g
chương trì
n

r
ình bỏ qua.
g
// ở dòng 8
h
lớn gồm n
h
c
h bằng /*

n
h sau :

Chương trì
n
và dòng 9.
C
h
iều dòng, c
ò

*/. Trình
n

h này còn
C
húng ta

ò
n kiểu //

Trong
c
Trong
C
dữ liệu
Ví dụ
2

1
2
3
4
5
6
7
8
9
1
1
1




Để thự
c
(inserti
o
(consol
cout
h
a
nhập/x
u
c
hương t
r
ìn
h
C
++ chúng
t
thông qua
h
2
.2: Chương
1
: #inclu
d
2
: int mai
3
: {


4
: int
5
: cout
6
: cin
>
7
: cout
8
: cin
>
9
: cout
1
0: cout
1
1: ret
u
1
2: }

c
hiện dòng
x
o
n operator
)
e input) kết
a

y
cin
, chún
g
u
ất ở chươn
g
h
C, chúng t
a
t
a có thể dù
n
h
ai biến đối
t
trình nhập
v
d
e <iostre
a
n()

X, Y;

<< "Nhap
v
>
>X;


<< "Nhap
v
>
>Y;

<<"Tong c
u
<<"Hieu c
u
u
rn 0;


x
uất chúng
t
)

<<
như ở c
á
hợp với toá
n
g
ta phải ké
o
g
8. Chúng t
a
a

thường sử
n
g dòng nhậ
p
t
ượng của d
ò
v
ào hai số.
T
a
m.h>
v
ao mot so
v
ao mot so
u
a chung:"
u
a chung:"
t
a sử dụng b
á
c dòng 5,
7
n
tử trích (e
x
o
file iostrea

m
a
chạy ví d

Hình 2.
2
9
dụng các hà
p
/xuất chuẩ
n
ò
ng (stream
T
ính tổng và
CT2_2.CPP

X:";

Y:";

<<X+Y<<"\
n
<<X-Y<<"\
n
iến
cout
(c
o
7

, 9 và 10. C
ò
x
traction op
e
m
.h như dò
n

2.2 , kết q
u
2
: Kết quả c

m nhập/xuấ
t
n
(standard i
n
object) là
c
o
hiệu của ha
i
n
";

n
";


o
nsole outpu
t
ò
n dòng nh

e
rator)
>>
n
h
n
g 1. Chúng
u
ả ở hình 2.2

a ví dụ 2.2
t
dữ liệu là
p
n
put/output
o
ut

cin
.
i
số vừa nhậ
p


t
)

kết hợp v


p chúng ta
s
h
ư ở các dò
n
ta sẽ tìm hi

.
p
rintf() và s
c
stream) để
n

p
.


i toán tử c
h
s
ử dụng biế
n

n
g 6 và 8. K
h

u kỹ về dò
n
c
anf().
n
hập/xuất

h
èn
n

cin
h
i sử dụng
n
g



2.
2
2
.4 Cách ch
u
Hình t
h

đổi kiể
u

1
2
3
4
5
6
7
8
9
1
1


Chúng
t
u
yển đổi ki

h
ức chuyển
đ
u
giống như
1
: #inclu
d
2

: int mai
3
: {

4
: int
5
: lon
g
6
: lon
g
7
: cout
8
: cout
9
: cout
1
0: ret
u
1
1: }

t
a chạy ví d

H

u dữ liệu

đ
ổi kiểu tro
n
m
ột lệnh g

d
e <iostre
a
n()

X = 200;

g
Y = (lon
g
g
Z = long
(
<< "X = "
<
<< "Y = "
<
<< "Z = "
<
u
rn 0;


2.3 , kết q

u
H
ình 2.3: D
ò
n
g C tương
đ

i hàm.
a
m.h>
g
) X;
/
/Ch
(
X);
/
/ Ch
<
<X<<"\
n
";
<
<Y<<"\
n
";
<
<Z<<"\
n

";
u
ả ở hình 2.
4
10
ò
ng nhập/xu

đ
ối tối nghĩa
,

CT2_3.CPP

uy

n đổi
k
uy

n đ ổi



4
.

t dữ liệu
,
vì vậy C+

+

d
k
iểu theo
c
kiểu theo
+
trang bị th
ê
d
ụ 2.3:
c
ách của
C
cách mới

ê
m
m
ột các
h
C

của C++


h
chuyển


2.
2
2
.5 Vị trí kh
Trong
c
ngay đ

hàm và
phép c
h
ngay đ

Ví dụ
2

1
2
3
4
5
6
7
8
9
1
1
1
1
1

1
1
1
1
ai báo biến
c
hương t
r
ìn
h

u của phạ
m
các khai bá
o
h
úng ta khai

u của phạ
m
2
.4: Chương
1
: #inclu
d
2
: int mai
3
: {


4
: int
5
: cout
6
: cin
>
7
: int
8
: cout
9
: cin
>
1
0: ch
a
1
1: co
u
1
2: ci
n
1
3: swi
1
4: {

1
5:

1
6:
1
7:
1
8:
h
C đòi hỏi t

m
vi đó. Điề
u
o
cục bộ ph

báo linh ho

m
vi), chúng
t
trình mô ph
d
e <iostre
a
n()

X;

<< "Nhap
v

>
>X;

Y;

<< "Nhap
v
>
>Y;

a
r Op;

u
t<<"Nhap
v
n
>>Op;

tch(Op)

case ‘+’:

cout<<
"
break;

case ‘-’:

Hình 2.

4

t cả các kh
a
u
này có ngh
ĩ

i được tiến

t bất kỳ vị
t
t
a xen kẽ vi

ỏng một má
y
a
m.h>
v
ao so thu
v
ao so thu
v
ao toan t
"
Ket qua:"
11
4
: Kết quả c


a
i báo bên t
r
ĩ
a là tất cả c
á
hành trước
t
t
rí nào trong

c khai báo
d
y
tính đơn
g
CT2_4.CPP
nhat:";

hai:";

u (+-
*
/):
"
<<X+Y<<"\
n

a ví dụ 2.3

r
ong
m
ột ph

á
c khai báo
t
t
ất cả các lệ
n

m
ột phạm
v
d
ữ liệu với c
g
iản
"
;

n
";


m vi cho t
r
ư
t

oàn cục ph

n
h thực hiệ
n
v
i cho t
r
ước
ác câu lệnh
ư
ớc phải đư


i đặt trước
t
n
. Ngược lại
(không nhấ
t
thực hiện.


c đặt ở
t
ất cả các
C++ cho
t
thiết phải




2.
2
1
2
2
2
2
2
2
2
2
2
2
3
3
3
3
3



Trong
c
ta chạy
Khi kh
a
kể từ v


2
.6 Các biế
n
Trong
A
dùng
#
d
1
9:
2
0:
2
1:
2
2:
2
3:
2
4:
2
5:
2
6:
2
7:
2
8:
2
9:

3
0:
3
1:
3
2: }

3
3: ret
3
4: }

c
hương t
r
ìn
h
ví dụ 2.4, k
ế
a
i báo một b

trí nó xuất
h
n
const
A
NSI C, m
u
d

efine
thì tạ
o
cout<<
"
break;

case ‘*’:

cout<<
"
break;

case ‘/’:

if (Y)

cou
t
else

cou
t
break;

default :

cout<<
"
urn 0;



h
chúng ta x
e
ế
t quả ở hìn
h
iến trong c
h
h
iện. Vì vậ
y
u
ốn định ng
h
o
ra các hằn
g
"
Ket qua:"
"
Ket qua:"
t
<<"Ket qu
t
<<"Khong
"
Khong hie
e

n kẻ khai b
á
h
2.5.
Hình 2.
5
h
ương trình,
b
y
chúng ta k
h
h
ĩa một hằng
g
không có
c
12
<<X-Y<<"\
n
<<long(X)
*
a:"<<floa
t
the chia
d
u toan tu
á
o biến với
l

5
: Kết quả c

b
iến đó sẽ c
ó
h
ông thể sử
d
có kiểu nh

c
hứa thông
t
n
";
*
Y<<"\n";

t
(X)/Y<<"\
n
d
uoc!" <<"
\
nay!"<<"\
n
l
ệnh thực hi



a ví dụ 2.4
ó
hiệu lực t
r
d
ụng một bi
ế

t định thì c
h
t
in về kiểu).
n
";

\
n
"; 9; 9
;
n
";



n ở dòng 4
r
ong phạm v
ế
n được kh

a
h
úng ta dùn
g
Trong C++
,
;



đến dòng 1
2
i của chươn
g
a
i báo bên d
ư
g
biến
const
,
các biến
c
o
2
. Chúng

g
trình đó
ư

ới nó.
(vì nếu
o
nst
linh
13

hoạt hơn một cách đáng kể:
C++ xem const cũng như #define nếu như chúng ta muốn dùng hằng có tên trong chương trình.
Chính vì vậy chúng ta có thể dùng const để quy định kích thước của một mảng như đoạn mã sau:
const int ArraySize = 100;
int X[ArraySize];
Khi khai báo một biến const trong C++ thì chúng ta phải khởi tạo một giá trị ban đầu nhưng đối với
ANSI C thì không nhất thiết phải làm như vậy (vì trình biên dịch ANSI C tự
động gán trị zero cho
biến const nếu chúng ta không khởi tạo giá trị ban đầu cho nó).
Phạm vi của các biến const giữa ANSI C và C++ khác nhau. Trong ANSI C, các biến const được
khai báo ở bên ngoài mọi hàm thì chúng có phạm vi toàn cục, điều này nghĩa là chúng có thể nhìn
thấy cả ở bên ngoài file mà chúng được định nghĩa, trừ khi chúng được khai báo là static. Nhưng
trong C++, các biến const được hiểu mặc định là static.
2.2.7 Về struct, union và enum

Trong C++, các struct và union thực sự các các kiểu class. Tuy nhiên có sự thay đổi đối với C++.
Đó là tên của struct và union được xem luôn là tên kiểu giống như khai báo bằng lệnh typedef vậy.
Trong C, chúng ta có thể có đoạn mã sau :
struct Complex
{
float Real;
float Imaginary;
};

…………………
struct Complex C;
Trong C++, vấn đề trở nên đơn giản hơn:
struct Complex
{
float Real;
float Imaginary;
};
…………………
Complex C;

2.
2
Quy đị
n
nhận c
ú
vào C+
chung
m
xuất tr

Cả hai
N
kiểu
u
n
2
.8 Toán tử
Toán t


bị che
b

1
2
3
4
5
6
7
8
9


Chúng
t
n
h này cũng
ú
pháp cũ.
+ gọi là uni
o
m
ột vùng đị
a

c tiếp bằng
union
{

int
N
floa
t
};
N
um và Va
l
n
ion
có tên,
c
Num = 12;
Value = 30.
định phạm

định phạm
b
ởi phạm vi
1
: #inclu
d
2
: int X
=
3
: int mai
4
: {


5
: int
6
: cout
7
: cout
8
: ret
u
9
: }

t
a chạy ví d

áp dụng ch
o

o
n nặc danh
a
chỉ bộ nh

tên của chú
n
N
um;
t
Value;
l

ue đều dùn
g
c
ác trường
c
56;
vi
vi (scope r
e
hiện thời.
d
e <iostre
a
=
5;

n()

X = 16;

<< "Bien
X
<< "Bien
X
u
rn 0;



2.5, kết q

u
o
cả
union
v

(anonymou
s

. Một union
n
g. Chẳng h

g
chung
m
ột
c
ủa union nặ
e
solution op
e

a
m.h>
X
ben tron
X
ben ngoa
u

ả ở hình 2.
6
14
v
à
enum
. T
u

s
union). N
ó
nặc danh k
h

n như đoạ
n
vị trí và kh
ô
c danh thì đ
ư
e
rator) ký hi


V
CT2_5.CPP
g = "<<X<
<
i = "<<::

X
6

u
y nhiên để t
ư
Một
k
ó
chỉ khai bá
h
ông có
t
ên
t
n

m
ã sau:
ô
ng gian bộ
n
ư
ợc truy xu


u là
::
, nó
đ

V
í dụ 2.5 :
<
"\n";

X
<<"\n";

ư
ơng thích
v
k
iểu
union
đ
o một loạt c
t
ag, các tr
ư

n
hớ. Tuy n
h

t trực tiếp,
c
đư
ợc dùng t
r


v
ới C, C++
v
đ
ặc biệt đư

ác t
r
ường(fi

ng có thể đ
ư
h
iên không g
c
hẳng hạn n
h
r
uy xuất mộ
t

v
ẫn chấp

c thêm
eld) dùng
ư
ợc truy
iống như
h

ư sau:
t
phần tử

2.
2
Toán t

để khai
được d
ù
2
.9 Toán tử
Trong
c
viện n
h
phát độ
hơn rất
động t
h
Trong
C

định phạm
báo lớp ch

ù
ng để phân
new và del

e
c
ác chương
t
h
ư
malloc()
,
ng bộ nhớ
b
nhiều so v

h
eo lối cổ đi

int *P;
P = malloc(
if (P==NU
L
prin
t
else
{
*P
=
prin
t
free
(
}

C
++, chúng
t
int *P;
P = new int
;
if (P==NU
L
cou
t
else
vi còn đượ
c

của các ph
ư
biệt các thà
n
e
te
t
rình C, tất
c
calloc()

f
b
ằng cách d
ù


i các hàm t
h

n của C.
sizeof(int));
L
L)
t
f("Khong c
o
=
290;
t
f("%d\n", *
(
P);
t
a có thể vi
ế
;

L
L)
t
<<"Khong
c
Hình 2.
6
c
dùng trong

ư
ơng thức đ
a
n
h phần trù
n
c
ả các cấp p
h
f
ree()
. C++
đ
ù
ng hai toán
t
h
ư viện của
C
o
n du bo nh
o
P);
ế
t lại đoạn c
h
c
on du bo n
h
15

6
: Kết quả c

các định n
g
a
ng được đị
n
n
g tên của c
á
h
át động bộ
n
đ
ịnh nghĩa
m
t

new

d
C
. Đoạn ch
ư
o
de cap ph
a
h
ương trình

t
h
o de cap p
h

a ví dụ 2.5
g
hĩa hàm củ
a
n
h nghĩa đó
.
á
c lớp cơ sở
n
hớ đều đư

m
ột phương
elete
. Sử d

ư
ơng trình s
a
a
t\n");
t
rên như sa
u

h
at\n";
a
các phươn
g
.
Toán tử đị
n
khác nhau.

c xử lý thô
n
thức mới đ


ng hai toán
t
a
u dùng để c

u
:
g
thức trong
n
h phạm vi
c
n
g qua các h


thực hiện
v
t
ử này sẽ li
n

p phát vùn
g

các lớp,
c
òn có thể
àm thư
v
iệc cấp
n
h hoạt
g
nhớ
16

{
*P = 290;
cout<<*P<<"\n";
delete P;
}
Chúng ta nhận thấy rằng, cách viết của C++ sáng sủa và dễ sử dụng hơn nhiều. Toán tử new thay thế
cho hàm malloc() hay calloc() của C có cú pháp như sau :
new type_name
new ( type_name )

new type_name initializer
new ( type_name ) initializer
Trong đó :
type_name: Mô tả kiểu dữ liệu được cấp phát. Nếu kiểu dữ liệu
mô tả phức tạp, nó có thể được đặt bên trong các dấu ngoặc.
initializer: Giá trị
khởi động của vùng nhớ được cấp phát.
Nếu toán tử new cấp phát không thành công thì nó sẽ trả về giá trị NULL.
Còn toán tử delete thay thế hàm free() của C, nó có cú pháp như sau :
delete pointer
delete [] pointer
Chúng ta có thể vừa cấp phát vừa khởi động như sau :
int *P;
P = new int(100);
if (P!=NULL)
{
cout<<*P<<"\n";
delete P;

Để cấp
Chú ý
:
chẳng
h
Ví dụ
2
xếp ch
ú

1

2
3
4
}
else
cout<<"Kh
o
phát một m

int *P;
P = new int
[
if (P!=NUL
{
for(int I = 0
P[I]
=
for(
I
cou
t
delete []P;
}
else
cout<<"Kh
o
Đối với việ
c
h
ạn đoạn ch

ư
int *P;
P = new (in
t
2
.6: Chương
ú
ng.
1
: #inclu
d
2
: #inclu
d
3
: #inclu
d
4
: int mai
o
ng con du
b

ng, chúng
t
[
10]; //Cấp
p
L)
;I<10;++)

=
I;
I
= 0;I<10;+
+
t
<<P[I]<<"\
n
o
ng con du
b
c
cấp phát
m
ư
ơng t
r
ình s
a
t
[10])(3); //
S
trình tạo m

d
e <iostre
a
d
e <time.h
>

d
e <stdlib
.
n()

b
o nho de c
a
t
a làm như s
a
p

t
mảng 1
0
+
)
n
";
b
o nho de c
a
m
ảng chúng t
a
u là sai :
S
ai !!!


t mảng độn
g
a
m.h>
>

.
h>
17
a
p phat\n";
a
u :
0
số nguyên
a
p phat\n";
a không thể
g
, khởi độn
g
CT2_6.CPP
vừa cấp ph
á
g
mảng này
v
á
t vừa khởi
đ

v
ới các giá t
r
đ
ộng giá trị
c
r
ị ngẫu nhiê
n
c
ho chúng,
n và sắp
18

5: {
6: int N;
7: cout<<"Nhap vao so phan tu cua mang:";
8: cin>>N;
9: int *P=new int[N];
10: if (P==NULL)
11: {
12: cout<<"Khong con bo nho de cap phat\n";
13: return 1;
14: }
15: srand((unsigned)time(NULL));
16: for(int I=0;I<N;++I)
17: P[I]=rand()%100; //Tạo các số ngẫu nhiên từ 0 đến 99
18: cout<<"Mang truoc khi sap xep\n";
19: for(I=0;I<N;++I)
20: cout<<P[I]<<" ";

21: for(I=0;I<N-1;++I)
22: for(int J=I+1;J<N;++J)
23: if (P[I]>P[J])
24: {
25: int Temp=P[I];
26: P[I]=P[J];
27: P[J]=Temp;
28: }
29: cout<<"\nMang sau khi sap xep\n";
30: for(I=0;I<N;++I)
31: cout<<P[I]<<" ";

3
3
3



Chúng
t
Ví dụ
2
xem m

Gọi X
l
chính l
à

1

2
3
2: del
3
3: ret
3
4: }

t
a chạy ví d

2
.7: Chương

ng hai chiề
u
l
à mảng hai
c
à
A[k] thì k
=
1
: #inclu
d
2
: #inclu
d
ete []P;


urn 0;



2.6, kết q
u
trình cộng
h
u
như mảng
Hình 2.8:
c
hiều có kíc
=
i*n + j
d
e <iostre
a
d
e <conio.
h
u
ả ở hình 2.
7
Hình 2.
7
h
ai ma trận t
r
một chiều

n
Mảng hai c
h
h thước m
d

a
m.h>
h
>
19
7

7
: Kết quả c

r
ong đó mỗi
n
hư hình 2.8
h
iều có thể
x
d
òng và n cộ
t
Ch
ú
CT2_7.CPP


a ví dụ 2.6
ma trận đư

x
em như mả
n
t
.A là mảng
ú
ng ta có c
h


c cấp phát
đ
n
g một chiề
u
một chiều
t
ư
h
ương trình
n

đ
ộng.Chún
g
u
.

ư
ơng ứng.N
ế
n
hư sau :

g
ta có thể
ế
u X[i][j]
20

3: //prototype
4: void AddMatrix(int * A,int *B,int*C,int M,int N);

5: int AllocMatrix(int **A,int M,int N);
6: void FreeMatrix(int *A);
7: void InputMatrix(int *A,int M,int N,char Symbol);
8: void DisplayMatrix(int *A,int M,int N);
9:
10: int main()
11: {
12: int M,N;
13: int *A = NULL,*B = NULL,*C = NULL;
14:
15: clrscr();
16: cout<<"Nhap so dong cua ma tran:";
17: cin>>M;
18: cout<<"Nhap so cot cua ma tran:";
19: cin>>N;

20: //Cấp phát vùng nhớ cho ma trận A
21: if (!AllocMatrix(&A,M,N))
22: { //endl: Xuất ra kí tự xuống dòng (‘\n’)
23: cout<<"Khong con du bo nho!"<<endl;
24: return 1;
25: }
26: //Cấp phát vùng nhớ cho ma trận B
27: if (!AllocMatrix(&B,M,N))
28: {
29: cout<<"Khong con du bo nho!"<<endl;
21

30: FreeMatrix(A);//Giải phóng vùng nhớ A
31: return 1;

32: }
33: //Cấp phát vùng nhớ cho ma trận C
34: if (!AllocMatrix(&C,M,N))
35: {
36: cout<<"Khong con du bo nho!"<<endl;
37: FreeMatrix(A);//Giải phóng vùng nhớ A
38: FreeMatrix(B);//Giải phóng vùng nhớ B
39: return 1;
40: }
41: cout<<"Nhap ma tran thu 1"<<endl;
42: InputMatrix(A,M,N,'A');
43: cout<<"Nhap ma tran thu 2"<<endl;
44: InputMatrix(B,M,N,'B');
45: clrscr();
46: cout<<"Ma tran thu 1"<<endl;

47: DisplayMatrix(A,M,N);
48: cout<<"Ma tran thu 2"<<endl;
49: DisplayMatrix(B,M,N);
50: AddMatrix(A,B,C,M,N);
51: cout<<"Tong hai ma tran"<<endl;
52: DisplayMatrix(C,M,N);
53: FreeMatrix(A);//Giải phóng vùng nhớ A
54: FreeMatrix(B);//Giải phóng vùng nhớ B
55: FreeMatrix(C);//Giải phóng vùng nhớ C
56: return 0;
22

57: }
68: //Cộng hai ma trận
69: void AddMatrix(int *A,int *B,int*C,int M,int N)
70: {
71: for(int I=0;I<M*N;++I)
72: C[I] = A[I] + B[I];
73: }
74: //Cấp phát vùng nhớ cho ma trận
75: int AllocMatrix(int **A,int M,int N)
76: {
77: *A = new int [M*N];
78: if (*A == NULL)
79: return 0;
80: return 1;
81: }
82: //Giải phóng vùng nhớ
83: void FreeMatrix(int *A)
84: {

85: if (A!=NULL)
86: delete [] A;
87: }
88: //Nhập các giá trị của ma trận
89: void InputMatrix(int *A,int M,int N,char Symbol)
90: {
91: for(int I=0;I<M;++I)
92: for(int J=0;J<N;++J)
93 {

9
9
9
9
1
1
1
1
1
1
1
1
k
1
1
1
1
1




Chúng
t
9
4:
9
5:
9
6: }

9
7: }

1
00: //Hiể
1
01: void
1
02: {

1
03: fo
1
04: {

1
05:
1
06: {
1

07:
k
y tu

1
08:
1
09:
1
10: co
1
11: }

1
12: }

t
a chạy ví d
u
cout<<S
y
cin>>A[
I
n thị ma
t
DisplayMa
t
r(int I=0
;
for(int J

=

out.wid
t
cout<<A
[
}

ut<<endl;


u
2.7 , kết q
u
y
mbol<<"["
I
*N+J];

t
rận

t
rix(int *
;
I<M;++I)

=
0;J<N;++
J

t
h(7);
/
/Hi
[
I*N+J];

u
ả ở hình 2.
9
23
<<I<<"]["
<
A,int M,i
n
J
)

en thi ca
n
9

<
<J<<"]=";
n
t N)

n
h le phai voi chie
u


u
dai 7


Một cá
c
Toán t

một hà
m
đặc biệ
t
đến hà
m
c
h khác để
c
int
*
int
*
for(
i
{
}
//Th


dele

t
dele
t


new
còn c
ó
m
xử lý lỗi
d
t
. Khi toán t

m
được chỉ
b
c
ấp phát mả
n
*
* A = new
i
*
Tmp = ne
w
i
nt I=0;I<M
;
A[I]=T

m
Tmp+=
N
ao tác trên
m

…………

t
e [] *A;
t
e [] A;
ó

m
ột thuận
d
o ng
ư
ời dù
n


new
được
b
ởi con trỏ
n
Hình 2.
9

n
g hai chiều
i
nt *[M];
w
int[M*N];
;
++I)
m
p;
N
;
m
ảng hai chi


lợi khác, đó
n
g tự định n
g
sử dụng để
n
ày. Định ng
h
24
9
: Kết quả c

A
gồm

M
d
ò
ều A
là tất cả cá
c
g
hĩa. C++ c
ó
cấp phát độ
n
h
ĩa của con

a ví dụ 2.7
ò
ng và
N
cộ
t
c
lỗi cấp ph
á
ó
định nghĩ
a
n
g và một l

trỏ này như

t
như sau:
á
t động đều
c
a
một con tr


i xảy ra do
c
sau:
c
ó thể bắt đ
ư

(pointer) tr
c
ấp phát, C
+


ư
ợc bằng
ỏ đến hàm
+
+ tự gọi




typede
f
pvf _n
e
Điều n
à
giá trị.
S
có thể
b
Ví dụ
2

1
2
3
4
5
6
7
8
9
1
1
1
1
1
1
1
1

1
1
2
2
2



f
void (*pv
f
e
w_handler
à
y có ngh
ĩ
a
l
S
au khi chú
n
b
ắt được tất
c
2
.8:
1
: #inclu
d
2

: #inclu
d
3
: #inclu
d
4
:

5
: void M
y
6
:

7
: unsigne
8
: void m
a
9
: {

1
0: int
1
1:
_
ne
1
2: fo

r
1
3:
1
4:

1
5: }

1
6:

1
7: void
M
1
8: {

1
9: co
u
2
0: co
u
2
1: exi
2
2: }

f

)();
(pvf p);
l
à con trỏ
_
n
n
g ta định n
g
c
ả các lỗi d
o
d
e <iostre
a
d
e <stdlib
.
d
e <new.h>
y
Handler()
;
d long I
=
a
in()

*A;


w
_
handler
r
( ; ; ++I
)
A = new i
n
M
yHandler(
)
u
t<<"Lan c
a
u
t<<"Khong
t(1);




n
ew_handle
g
hĩa hàm n
h
o
cấp phát đ

a

m.h>
.
h>
;

=
0; 9;

= MyHandl
)

n
t;

)

a
p phat th
con du bo
25


r
là con trỏ
t
h
ư vậy và gá
n

ng.

CT2_8.CPP

er;

u "<<I<<e
n
nho!"<<e
n


t
rỏ đến hàm
n
địa chỉ củ
a

n
dl;

n
dl;



không có t
h
a
nó cho
_n
e





h
am số và k
h
e
w_handler




h
ông trả về
chúng ta




×