Con trỏ trong lập trình là 1 định nghĩa hơi khó nhằn đối với các bạn mới học về C++. Không ngoa khi nói rằng C++ khó vì có con trỏ. Tuy nhiên ví như làm chủ được con trỏ, bạn có thể hiểu và thao tác với dữ liệu trong bộ nhớ máy tính, và những kiến thức liên quan mà bạn học được thông qua con trỏ cũng rất hữu ích cho việc học các ngôn ngữ hướng đối tượng sau này như Java chẳng hạn.
Bạn đang xem: Bài tập con trỏ trong c
Hãy cùng khám phá con trỏ trong C + + là gì, quan hệ giữa con trỏ và địa chỉ trong C + +, cấu trúc, vai trò và cách sử dụng con trỏ sau bài học kinh nghiệm này nhé .
Danh Mục Bài Viết
IX. Nhập Xuất Mảng Bằng Con Trỏ Trong C++X. Con Trỏ Và Tham Chiếu Trong C++XI. Ý Nghĩa Con Trỏ This Trong C++XII. Con Trỏ Void Trong C++XIV. Ép Kiểu Con Trỏ Trong C++
Tóm Tắt
I. Khai Báo Con Trỏ Trong C++
IX. Nhập Xuất Mảng Bằng Con Trỏ Trong C + + X. Con Trỏ Và Tham Chiếu Trong C + + XI. Ý Nghĩa Con Trỏ This Trong C + + XII. Con Trỏ Void Trong C + + XIV. Ép Kiểu Con Trỏ Trong C + +Để khai báo con trỏ trong C + +, chung ta sử dụng với cấu trúc ngữ pháp sau đây :type * p ; Trong đó type là kiểu tài liệu của con trỏ, và p là tên con trỏ. Lưu ý là kiểu tài liệu của con trỏ phải giống với kiểu tài liệu của tài liệu cần lưu địa chỉ trong con trỏ .Ví dụ, tất cả chúng ta khai báo con trỏ tên p với kiểu int như sau :int * p ; Lưu ý là những cách viết sau đây cũng được gật đầu khi khai báo con trỏ trong C + + :int * p ; int * p ;
II. Cấp Phát Bộ Nhớ Cho Con Trỏ Trong C++
Ngôn ngữ C++ hỗ trợ ba loại cấp phát bộ nhớ cơ bản, hai loại trong số đó bạn đã được học ở các bài học trước:Khai Báo Con Trỏ Trong C + + Ngôn ngữ C + + tương hỗ ba loại cấp phép bộ nhớ cơ bản, hai loại trong số đó bạn đã được học ở những bài học kinh nghiệm trước :
1. Cấp phát bộ nhớ tĩnh (Static memory allocation):
Xảy ra trên các biến tĩnh và biến toàn cục.Vùng nhớ của các loại biến này được cấp phát một lần khi chương trình bắt đầu chạy và vẫn tồn tại trong suốt thời gian tồn tại của chương trình.Kích thước của biến/mảng nên được biết tại thời điểm biên dịch chương trình.Xảy ra trên những biến tĩnh và biến toàn cục. Vùng nhớ của những loại biến này được cấp phép một lần khi chương trình mở màn chạy và vẫn sống sót trong suốt thời hạn sống sót của chương trình. Kích thước của biến / mảng nên được biết tại thời gian biên dịch chương trình .
2. Cấp phát bộ nhớ tự động (Automatic memory allocation):
Xảy ra trên các tham số hàm và biến cục bộ.Vùng nhớ của các loại biến này được cấp phát khi chương trình đi vào khối lệnh và được giải phóng khi khối lệnh bị thoát.Kích thước của biến/mảng phải được biết tại thời điểm biên dịch chương trình.Xảy ra trên những tham số hàm và biến cục bộ. Vùng nhớ của những loại biến này được cấp phép khi chương trình đi vào khối lệnh và được giải phóng khi khối lệnh bị thoát. Kích thước của biến / mảng phải được biết tại thời gian biên dịch chương trình .
3. Cấp phát bộ nhớ động (Dynamic memory allocation) sẽ được kể tới trong bài học này.
Trong tổng thể những trường hợp, cấp phép bộ nhớ tĩnh và tự động hóa hoàn toàn có thể phân phối tốt những ý kiến đề nghị của chương trình. Tuy nhiên, ta cùng xem ví dụ bên dưới :Ví dụ : Chúng ta cần sử dụng một chuỗi để lưu tên của người dùng, nhưng tất cả chúng ta không biết tên của họ dài bao nhiêu cho đến khi họ nhập tên. Hoặc tất cả chúng ta cần tàng trữ list nhân viên cấp dưới trong một công ty, nhưng tất cả chúng ta không biết trước được công ty đó sẽ có bao nhiêu nhân viên cấp dưới .Đối với cấp phép bộ nhớ tĩnh và tự động hóa, kích cỡ của biến / mảng phải được biết tại thời gian biên dịch chương trình. Vì vậy, điều tốt nhất tất cả chúng ta hoàn toàn có thể làm là nỗ lực đoán một size tối đa của những biến đó :char name <3 0 > ; / / tên không quá 30 ký tựStaff staff < 500 > ; / / công ty không quá 500 nhân viênKhuyết điểm của cách khai báo trên :+ Gây tiêu tốn lãng phí bộ nhớ nếu những biến không thực sự sử dụng hết size khi khai báo. Ví dụ : nếu công ty chỉ có 100 nhân viên cấp dưới, tất cả chúng ta có 400 vùng nhớ nhân viên cấp dưới không được sử dụng tới .+ Thứ hai, hầu hết những biến thường thì ( gồm có mảng tĩnh ) được cấp phép trong một phần bộ nhớ gọi là ngăn xếp ( stack ). Kích thước bộ nhớ stack cho một chương trình khá nhỏ ( khoảng chừng 1M b với Visual Studio ), nếu nhu yếu cấp phép vùng nhớ vượt quá số lượng này, chương trình sẽ bị đóng bởi hệ quản lý với lỗi stack overflow .char byte < 1000000 * 2 > ; / / khoảng chừng 2M b bộ nhớ => lỗi stack overflow + Thứ ba, điều gì xảy ra nếu công ty có 600 nhân viên cấp dưới, trong khi mảng staff chỉ có 500 thành phần. Lúc này, chương trình sẽ bị số lượng giới hạn bởi size được khai báo khởi đầu .Để xử lý những hạn chế trên, cấp phép bộ nhớ động được sinh ra .
III. Con Trỏ Trong C++ Dùng Để Làm Gì
Cấp Phát Bộ Nhớ Cho Con Trỏ Trong C + +Tác dụng của con trỏ trong c chính là để lưu giữ địa chỉ của tài liệu trong bộ nhớ máy tính, và bằng cách truy vấn vào địa chỉ này, tất cả chúng ta hoàn toàn có thể lấy được giá trị của tài liệu tại đó .Ngoài ra thì giá trị của con trỏ cũng là một số ít, nên tất cả chúng ta cũng hoàn toàn có thể thực thi những phép tính toán với con trỏ, ví dụ như cộng thêm hoặc hoặc trừ đi một số lượng đơn vị chức năng .Do đó, con trỏ trong C sẽ được dùng để làm 1 trong 2 việc làm sau đây trong chương trình :Thao tác với địa chỉ bằng các phép tính toán với số được lưu trong nóThao tác với giá trị tại địa chỉ mà nó lưu mà thôi.
IV. Mảng Con Trỏ Trong C++
Thao tác với địa chỉ bằng những phép tính toán với số được lưu trong nóThao tác với giá trị tại địa chỉ mà nó lưu mà thôi. Con Trỏ Trong C + + Dùng Để Làm GìTrước khi tất cả chúng ta hiểu về khái niệm mảng những con trỏ, tất cả chúng ta xem xét ví dụ sau, mà sử dụng một mảng gồm 3 số integer :# include using namespace std ; const int MAX = 3 ; int main ( ) { int var = { 10, 100, 200 } ; for ( int i = 0 ; i Khi code trên được biên dịch và thực thi, nó cho tác dụng sau :Gia tri cua var < 0 > = 10G ia tri cua var < 1 > = 100G ia tri cua var < 2 > = 200C ó một trường hợp khi tất cả chúng ta muốn duy trì một mảng, mà hoàn toàn có thể lưu giữ những con trỏ tới một kiểu tài liệu int hoặc char hoặc bất kể kiểu nào khác. Sau đây là khai báo một mảng của những con trỏ tới một integer :int * contro ; Nó khai báo contro như thể một mảng những con trỏ MAX kiểu integer. Vì thế, mỗi thành phần trong contro, giờ đây giữ một con trỏ tới một giá trị int. Ví dụ sau sử dụng 3 số integer, mà sẽ được lưu giữ trong một mảng những con trỏ như sau :# include using namespace std ; const int MAX = 3 ; int main ( ) { int var = { 10, 100, 200 } ; int * contro ; for ( int i = 0 ; i Khi code trên được biên dịch và thực thi, nó cho hiệu quả sau :Gia tri cua var < 0 > = 10G ia tri cua var < 1 > = 100G ia tri cua var < 2 > = 200B ạn hoàn toàn có thể sử dụng một mảng những con trỏ tới ký tự để lưu giữ một list những chuỗi như sau :# include using namespace std ; const int MAX = 4 ; int main ( ) { char * tensv = { ” Nguyen Thanh Tung “, ” Tran Minh Chinh “, ” Ho Ngoc Ha “, ” Hoang Minh Hang “, } ; for ( int i = 0 ; i Chạy chương trình C + + trên sẽ cho tác dụng như hình sau :
V. Con Trỏ Hàm Trong C++
Mảng Con Trỏ Trong C + +Con trỏ hàm là một biến tàng trữ địa chỉ của một hàm, trải qua biến đó, ta hoàn toàn có thể gọi hàm mà nó trỏ tới .Cú pháp khai báo con trỏ hàm 🙁 * ) ( ) ;int ( * fcnPtr ) ( int ) ; / / con trỏ hàm nhận vào 1 biến kiểu int và trả về kiểu intvoid ( * fcnPtr ) ( int, int ) ; / / con trỏ hàm nhận vào 2 biến kiểu int và trả về kiểu voidChú ý : Dấu ngoặc ( ) quanh * fcnPtr là bắt buộc .
VI. Con Trỏ Trong Class C++
Một con trỏ tới một lớp trong C + + được triển khai theo cách giống hệt như một con trỏ tới một cấu trúc ; và để truy vấn những thành viên của một con trỏ tới một lớp bạn sử dụng toán tử truy vấn thành viên trong C + + là toán tử ->, như khi bạn triển khai với những con trỏ tới cấu trúc. Cũng như với toàn bộ con trỏ, bạn phải khai báo con trỏ trước khi sử dụng nó .Bạn thử ví dụ sau để hiểu khái niệm con trỏ tới một lớp trong C + + :# include using namespace std ; class Box { public : / / phan dinh nghia Constructor Box ( double dai = 1.0, double rong = 1.0, double cao = 1.0 ) { cout theTich ( ) theTich ( ) Biên dịch và chạy chương trình C + + trên sẽ cho tác dụng sau :
VII. Con Trỏ Trong Struct C++
Con Trỏ Trong Class C + +Chào những bạn đang theo dõi khóa học lập trình trực tuyến ngôn từ C + +. Chúng ta cùng liên tục tìm hiểu và khám phá về kiểu tài liệu tự định nghĩa trải qua từ khóa struct mà ngôn từ C + + tương hỗ. Trong bài học kinh nghiệm này, mình sẽ trình diễn về kiểu struct khi sử dụng tích hợp với con trỏ .Như những bạn đã học trong bài trước, sau khi tất cả chúng ta tự định nghĩa một struct, compiler sẽ coi tên gọi của struct đó như thể một kiểu tài liệu. Điều này có nghĩa khi tất cả chúng ta sử dụng những kiểu tài liệu built-in để tạo ra những biến, tham chiếu hoặc con trỏ thì tất cả chúng ta cũng hoàn toàn có thể sử dụng kiểu struct để tạo ra biến struct, tham chiếu struct và con trỏ kiểu struct ( Pointer to struct ) .Pointer to structĐầu tiên, tất cả chúng ta cùng định nghĩa một kiểu tài liệu theo ý muốn. Dưới đây, mình định nghĩa một kiểu tài liệu có tên là Letter :struct Letter { } ; Trong struct Letter mình chưa định nghĩa những trường tài liệu, lúc này Letter là một kiểu tài liệu rỗng. Nhưng ngôn từ C + + vẫn đặt size của kiểu Letter này là 1 bytes .Mục đích là để bảo vệ địa chỉ của 2 biến được tạo ra sẽ có địa chỉ khác nhau. Tuy nhiên, định nghĩa ra một struct rỗng không có tính năng gì trong chương trình, tất cả chúng ta cùng thêm vào một số ít trường tài liệu cho Letter :struct Letter { char from < 50 > ; char to < 50 > ; } ; Một lá thư sẽ có thông tin về người gửi và người nhận, nên mình thêm vào 2 trường tài liệu kiểu C-style string dùng để lưu thông tin mà người dùng điền vào một lá thư .Mình vừa định nghĩa xong một kiểu tài liệu mới để ship hàng cho chương trình của mình. Bây giờ tất cả chúng ta cùng tạo ra một đơn vị chức năng từ kiểu tài liệu trên ( mình thao tác luôn trong hàm main ) :int main ( ) { Letter myLetter ; return 0 ; } Với mỗi biến kiểu Letter được tạo ra, chương trình sẽ nhu yếu cấp phép 100 bytes ( 50 bytes cho trường tài liệu from và 50 bytes cho trường tài liệu to ), và chắc như đinh rồi, biến đó sẽ có một địa chỉ xác lập được trải qua toán tử address-of .int main ( ) { Letter myLetter ; std :: cout Ở đoạn chương trình trên, mình in ra địa chỉ của biến myLetter, đồng thời in ra luôn địa chỉ của trường tài liệu from của biến myLetter. Kết quả cho thấy 2 địa chỉ được in ra có giá trị trọn vẹn giống nhau. Điều này có nghĩa địa chỉ của trường tài liệu tiên phong trong một biến struct cũng là địa chỉ của biến struct đó .Các bạn hoàn toàn có thể liên hệ struct với mảng một chiều trong C / C + +, khi mảng một chiều mà tập hợp những thành phần có cùng kiểu tài liệu được bảo phủ bởi tên mảng một chiều và địa chỉ của mảng một chiều cũng là địa chỉ của thành phần tiên phong trong mảng, một biến struct sẽ gồm có tập hợp những trường tài liệu mà địa chỉ của biến struct sẽ là địa chỉ của trường tài liệu được khai báo tiên phong trong struct .Và như những bạn cũng đã học về con trỏ ( Pointer ), kiểu tài liệu của con trỏ dùng để xác lập kiểu tài liệu của vùng nhớ mà con trỏ hoàn toàn có thể trỏ đến. Vậy thì để cho con trỏ trỏ đến một địa chỉ của biến kiểu struct, tất cả chúng ta cần có một con trỏ cùng kiểu struct với biến được trỏ đến .Letter myLetter ; Letter * pLetter = và myLetterDù size của kiểu tài liệu struct có lớn bao nhiêu, biến con trỏ cũng chỉ có kích cỡ 4 bytes trên hệ quản lý 32 bits và size 8 bytes trên hệ quản lý 64 bits ( đủ để trỏ đến hàng loạt địa chỉ trên bộ nhớ ảo ) .Access struct membersTrong bài học kinh nghiệm trước, những bạn đã biết cách truy vấn đến những trường tài liệu của những biến struct trải qua thành viên selection operator ( dấu chấm ). Nhưng khi sử dụng Pointer to struct, thành viên selection operator được sử dụng dưới cách viết khác. Để phân biệt sự khác nhau khi sử dụng thành viên selection operator cho biến struct thường thì và một Pointer to struct, những bạn cùng xem ví dụ bên dưới :struct BankAccount { __int64 accountNumber ; __int64 balance ; } ; int main ( ) { BankAccount myAccount = { 123456789, 50 } ; / / USD 50 BankAccount * pAccount = và myAccount std :: cout accountNumber balance Như những bạn thấy, hiệu quả của việc truy xuất giá trị trải qua tên biến struct và con trỏ kiểu struct là trọn vẹn giống nhau, và chúng đều dùng toán tử thành viên selection. Tuy nhiên, để phân biệt biến con trỏ và biến thường thì, biến con trỏ kiểu struct sẽ truy vấn đến những trường tài liệu trong vùng nhớ bằng toán tử ( -> ). Hai toán tử này cùng tên, chỉ khác nhau về cách trình diễn .
Một số nhầm lần khi sử dụng struct và Pointer to struct
Khi mới tìm hiểu và khám phá về Pointer to struct, những bạn hoàn toàn có thể bị nhầm lẫn giữa cách khởi tạo hoặc gán giá trị cho biến struct thường thì và biến con trỏ struct .struct BankAccount { __int64 accountNumber ; __int64 balance ; } ; int main ( ) { BankAccount myAccount = { 12345, 50 } ; BankAccount * pAccount = { 12345, 50 } ; / / error return 0 ; } Đoạn chương trình trên báo lỗi vì biến con trỏ chỉ nhận giá trị là địa chỉ. Tuy nhiên, lỗi này hoàn toàn có thể thấy thuận tiện vì Visual studio đưa ra thông tin lỗi ngay. Dưới đây là cách gán giá trị đúng khi mình sử dụng toán tử dereference cho biến con trỏ struct để đổi khác giá trị bên trong vùng nhớ :struct BankAccount { __int64 accountNumber ; __int64 balance ; } ; int main ( ) { BankAccount myAccount = { 0, 0 } ; BankAccount * pAccount = và myAccount * pAccount = { 12345, 50 } ; std :: cout accountNumber balance Hoặc một cách khác là tất cả chúng ta cấp phép vùng nhớ cho biến con trỏ struct, và dereference đến đó để gán giá trị cho nó :BankAccount * pAccount = new BankAccount ; * pAccount = { 12345, 50 } ; Và những bạn chú ý quan tâm khi sử dụng biến kiểu con trỏ struct thì tất cả chúng ta sử dụng toán tử thành viên selection này ( -> ). Có một số ít bạn nhầm lẫn giữa biến con trỏ struct và trường tài liệu kiểu con trỏ. Ví dụ :struct BankAccount { char * name ; __int64 accountNumber ; __int64 balance ; } ; Mình thêm vào struct một trường tài liệu kiểu con trỏ char nhưng việc truy xuất đến trường tài liệu này không có gì đổi khác khi mình sử dụng biến struct thường thì .BankAccount myAccount = { ” Le Tran Dat “, 12345, 50 } ; std :: cout Sẽ phức tạp hơn một chút ít khi những bạn sử dụng những nested struct. Ví dụ :struct BankAccount { Date registrationDate ; __int64 accountNumber ; __int64 balance ; } ; int main ( ) { BankAccount * pAccount = new BankAccount ; * pAccount = { { 2, 5, năm nay }, 12345, 50 } ; std :: cout registrationDate. year Như những bạn thấy, từ biến con trỏ pAccount truy xuất vào những trường tài liệu bên trong thì mình dùng toán tử ( -> ), nhưng trường tài liệu Date trong struct BankAccount là biến thường thì, nên mình dùng dấu chấm để truy xuất tài liệu ngày đăng kí .
VIII. Con Trỏ Null Trong C++
Con Trỏ Trong Struct C + +Con trỏ NULL trong C + + là một hằng với một giá trị là 0 được định nghĩa trong một vài thư viện chuẩn, gồm iostream .# include using namespace std ; int main ( ) { int * ptr = NULL ; cout Kết quả :Gia tri cua contro la 0T rên hầu hết những hệ quản lý, những chương trình không được phép truy vấn bộ nhớ tại địa chỉ 0, vì bộ nhớ đó được dự trữ bởi hệ quản lý và điều hành. Tuy nhiên, địa chỉ bộ nhớ 0 có ý nghĩa đặc biệt quan trọng, nó chỉ ra rằng con trỏ không được trỏ tới một vị trí ô nhớ hoàn toàn có thể truy vấn. Nhưng theo qui ước, nếu một con trỏ chứa giá trị 0, nó được xem như thể không trỏ tới bất kể thứ gì .Để kiểm tra một con trỏ null trong C + +, bạn hoàn toàn có thể sử dụng lệnh if như sau :if ( contro ) / / true neu contro khong la NULLif ( ! contro ) / / true neu contro la NULL
IX. Nhập Xuất Mảng Bằng Con Trỏ Trong C++
Bằng cách sử dụng con trỏ mảng, chúng ta có thể chỉ định vị trí các phần tử trong mảng, cũng như là truy cập và lấy giá trị của các phần tử đó.Con Trỏ Null Trong C + + Bằng cách sử dụng con trỏ mảng, tất cả chúng ta hoàn toàn có thể chỉ định vị trí những thành phần trong mảng, cũng như là truy vấn và lấy giá trị của những thành phần đó .Ứng dụng điều này, tất cả chúng ta cũng hoàn toàn có thể nhập xuất mảng bằng con trỏ trong C + + như sau .
Nhập mảng bằng con trỏ trong C++
Trong bài Nhập xuất mảng trong C + + tất cả chúng ta đã biết cách tạo hàm nhập trực tiếp những giá trị từ bàn phím vào mảng như sau :
/*Tạo hàm nhập mảng 1 chiều trong C++*/void input_array(int array<>, int length){ //array: tên mảng //length: độ dài mảng for (short i = 0; i > array;}Trong đó array và length lần lượt là tên và độ dài (số phần tử) của mảng cần nhập.
Để nhập mảng bằng con trỏ trong C + + tất cả chúng ta chỉ cần sử dụng giá trị con trỏ thay cho mảng, và thay vì dùng index để chỉ định vị trí nhập tài liệu, thì tất cả chúng ta sẽ sử dụng trực tiếp giá trị con trỏ để chỉ định vị trí cần nhập. Chúng ta viết hàm nhập mảng bằng con trỏ trong C + + như sau :/ * Tạo hàm nhập mảng bằng con trỏ trong C + + * / void input_array ( int array < >, int length ) { / / array : tên mảng / / length : độ dài mảng for ( short i = 0 ; i > * ( array + i ) ; } Lưu ý tại đây tất cả chúng ta sử dụng lệnh cin để nhập tài liệu vào mảng, nên cần chỉ định bản thân tài liệu mà con trỏ chỉ đến bằng cách thêm dấu hoa thị * trước con trỏ .Tuy nhiên nếu sử dụng hàm scanf được thừa kế từ ngôn từ C để nhập mảng bằng con trỏ trong C + +, thay vì chỉ định tài liệu thì tất cả chúng ta cần chỉ định địa chỉ của tài liệu đã được lưu vào con trỏ như sau :/ * Tạo hàm nhập mảng bằng con trỏ trong C * / void input_array ( int * array, int length ) { / / array : tên mảng / / length : độ dài mảng for ( short i = 0 ; i
Xuất mảng bằng con trỏ trong C++
Trong bài xuất xuất mảng trong C++ chúng ta đã biết cách tạo hàm xuất trực tiếp các giá trị của mảng như sau:Trong bài xuất xuất mảng trong C + + tất cả chúng ta đã biết cách tạo hàm xuất trực tiếp những giá trị của mảng như sau :/ * Tạo hàm xuất mảng 1 chiều trong C + + * / void show_array ( int array < >, int length ) { / / array : tên mảng / / length : độ dài mảng for ( short i = 0 ; i Trong đó array và length lần lượt là tên và độ dài ( số thành phần ) của mảng cần xuất .Để xuất mảng bằng con trỏ trong C + + tất cả chúng ta chỉ cần sử dụng con trỏ thay cho mảng, và thay vì truy xuất giá trị những thành phần của mảng bằng index thì tất cả chúng ta sẽ dùng tên con trỏ và dấu hoa thị để xuất giá trị đó. Chúng ta viết hàm xuất mảng bằng con trỏ trong C + + như sau :/ * Tạo hàm xuất mảng bằng con trỏ trong C + + * / void show_array ( int array < >, int length ) { / / array : tên mảng / / length : độ dài mảng for ( short i = 0 ; i
Chương trình mẫu nhập xuất mảng bằng con trỏ trong C++
Dưới đây là chương trình mẫu sử dụng các hàm trên để nhập xuất mảng bằng con trỏ trong C++:
Dưới đây là chương trình mẫu sử dụng các hàm trên để nhập xuất mảng bằng con trỏ trong C++:
# include using namespace std ; / * Tạo hàm nhập mảng bằng con trỏ trong C + + * / void input_array ( int array < >, int length ) { / / array : tên mảng / / length : độ dài mảng for ( short i = 0 ; i > * ( array + i ) ; } / * Tạo hàm xuất mảng bằng con trỏ trong C + + * / void show_array ( int array < >, int length ) { / / array : tên mảng / / length : độ dài mảng for ( short i = 0 ; i > Nhap so phan tu : ” ; cin >> n ; int array, * p ; cout > Nhap phan tu : \ n ” ) ; input_array ( array, n ) ; / * Xuất mảng bằng con trỏ trong C + + * / cout > Mang vua nhap : \ n ” ) ; show_array ( array, n ) ; } Kết quả :
X. Con Trỏ Và Tham Chiếu Trong C++
C và C++ hỗ trợ con trỏ cái mà khác với hầu hết các ngôn ngữ lập trình khác. Các ngôn ngữ khác bao gồm C ++, Java, Python, Ruby, Perl và PHP đều hỗ trợ tham chiếu.Nhập Xuất Mảng Bằng Con Trỏ Trong C + + C và C + + tương hỗ con trỏ cái mà khác với hầu hết những ngôn từ lập trình khác. Các ngôn từ khác gồm có C + +, Java, Python, Ruby, Perl và PHP đều tương hỗ tham chiếu .Nhìn vẻ bên ngoài, cả hai tham chiếu và con trỏ đều rất giống nhau, cả hai đều được sử dụng để có một biến phân phối quyền truy vấn cho một biến khác. Với việc cả hai đều phân phối nhiều năng lực giống nhau, người ta thường không rõ điều gì độc lạ giữa những chính sách khác nhau này. Trong bài viết này, tôi sẽ cố gắng nỗ lực minh họa sự độc lạ giữa con trỏ và tham chiếu .Con trỏ : Con trỏ là một biến chứa địa chỉ bộ nhớ của một biến khác. Một con trỏ cần được deference với toán tử * để truy vấn vào vị trí bộ nhớ mà nó trỏ tới .Tham chiếu : Một biến tham chiếu là một bí danh, tức là một tên khác của một biến đã sống sót. Một tham chiếu, giống như một con trỏ, cũng được hiện thựcbằng cách tàng trữ địa chỉ của một đối tượng người tiêu dùng. Một tham chiếu hoàn toàn có thể được coi là một con trỏ hằng ( đừng nhầm với một con trỏ đến một giá trị không đổi ! ) với tính năng tự động hóa chuyển hướng, tức là trình biên dịch sẽ vận dụng toán tử * cho bạn .int i = 3 ; / / Một con trỏ đến biến i / / ( tàng trữ địa chỉ của i ) int * ptr = và i / / Một tham chiếu ( bí danh ) của i.int và ref = i ;
Sự khác nhau giữa Con trỏ và Tham chiếu
Khởi tạoMột con trỏ có thế được khai báo và khởi tạo đồng thời hoặc trên nhiều dòng .C + + int a = 10 ; int * p = và a hayint * p ; p = và aTrong khi tham chiếu phải được khởi tạo cùng lúc với khai báo .int a = 10 ; int và p = a ; / / OKnhưngint và pp = a ; / / NOKChú ý : Sự độc lạ này hoàn toàn có thể đổi khác từ trình biên dịch này sang trình biên dịch khác. Sự độc lạ trên là so với Turbo IDE .
Gán lại giá trị
Một con trỏ hoàn toàn có thể được gán lại. Thuộc tính này có ích cho việc tiến hành những cấu trúc tài liệu như list link, cây, v.v. Xem ví dụ sau :int a = 5 ; int b = 6 ; int * p ; p = và ap = và bTrong khi đó, tham chiếu không gán lại mà phải được gán ngay lúc khởi tạo .
int a = 5;int b = 6;int &p = a;int &p = b; //Tại dòng này sẽ hiện lỗi là “không được khai báo nhiều lần” //Tuy nhiên, lệnh như dưới đây thì ok.int &q=p;Địa chỉ bộ nhớ
Một con trỏ có địa chỉ bộ nhớ và size riêng của nó trên ngăn xếp trong khi một tham chiếu san sẻ cùng địa chỉ bộ nhớ ( với biến gốc ) nhưng cũng chiếm một số ít khoảng trống trên ngăn xếp .
int &p = a;cout Giá trị NULL
Con trỏ hoàn toàn có thể được gán NULL trực tiếp, trong khi tham chiếu không hề. Các ràng buộc tương quan đến tham chiếu ( không NULL, không gán lại ) bảo vệ rằng những hoạt động giải trí cơ bản không rơi vào trường hợp ngoại lệ .
Con trỏ đến con trỏ
Bạn hoàn toàn có thể có những con trỏ đến con trỏ phân phối nhiều Lever chuyển hướng bổ trợ. Trong khi đó, những tham chiếu chỉ cung ứng một mức chuyển hướng .
//Con trỏ,int a = 10;int *p;int **q; //Con trỏ đến con trỏ.p = &aq = &p //Đối với tham chiếu, int &p = a;int &&q = p; //Lỗi, không có tham chiếu của tham chiếu.Các phép tính toán học
Các phép toán số học khác nhau hoàn toàn có thể được triển khai trên con trỏ trong khi không có thứ gọi là Số học tham chiếu. ( Nhưng bạn hoàn toàn có thể lấy địa chỉ của một đối tượng người tiêu dùng được trỏ bởi một tham chiếu và thực thi số học con trỏ trên đó như và obj + 5 )
Khi nào sử dụng cái nào
Hiệu suất trọn vẹn giống nhau, vì những tham chiếu được tiến hành bên trong dưới dạng con trỏ. Tuy nhiên, bạn vẫn hoàn toàn có thể ghi nhớ một số ít điểm để quyết định hành động khi nào sử dụng cái gì :
Sử dụng tham chiếu:
Trong các tham số hàm và kiểu trả về.Trong những tham số hàm và kiểu trả về .
Sử dụng con trỏ:
Sử dụng con trỏ nếu cấn tính toán con trỏ số học hoặc NULL. Ví dụ đối với mảng (Lưu ý rằng truy cập mảng được thực hiện bằng cách sử dụng số học con trỏ).Để triển khai các cấu trúc dữ liệu như danh sách liên kết, cây, v.v. và các thuật toán của chúng vì để trỏ ô khác nhau, chúng ta phải sử dụng khái niệm con trỏ.
XI. Ý Nghĩa Con Trỏ This Trong C++
Con trỏ this trong C++
Sử dụng con trỏ nếu cấn đo lường và thống kê con trỏ số học hoặc NULL. Ví dụ so với mảng ( Lưu ý rằng truy vấn mảng được thực thi bằng cách sử dụng số học con trỏ ). Để tiến hành những cấu trúc tài liệu như list link, cây, v.v. và những thuật toán của chúng vì để trỏ ô khác nhau, tất cả chúng ta phải sử dụng khái niệm con trỏ. Con Trỏ Và Tham Chiếu Trong C + +This là một con trỏ đặc biệt quan trọng dùng để trỏ đến địa chỉ của đối tượng người dùng hiện tại. Như vậy để truy vấn đến những thuộc tính, phương pháp của đối tượng người dùng hiện tại thì ta sẽ sử dụng con trỏ this. Hãy xem ví dụ dưới đây .Ví dụ# include using namespace std ; class NhanVien { int msnv ; string ten ; int tuoi ; public : void setData ( int msnv, string ten, int tuoi ) { this -> msnv = msnv ; this -> ten = ten ; this -> tuoi = tuoi ; } void showData ( ) { cout ten msnv tuoi Trong ví dụ này mình đã tạo ra ba thuộc tính để tàng trữ thông tin của một nhân viên cấp dưới đó là : manv, ten, tuoi. Ngoài ra mình có tạo thêm phương pháp setData ( ) dùng để gán tài liệu cho sinh viên, và showData ( ) dùng để hiển thị tài liệu .Trong phương pháp setData ( ) mình đã sử dụng từ khóa this -> ten_thuoc_tinh để triển khai phép gán tài liệu cho những thuộc tính, còn ở phương pháp showData ( ) mình cũng sử dụng cú pháp tựa như để hiển thị tài liệu của những thuộc tính. Như vậy hiệu quả của từ khóa this chính là một con trỏ và trỏ đến địa chỉ của đối tượng người dùng hiện tại .Câu hỏi đặt ra là đối tượng người dùng hiện tại tại là gì ? Để hiểu rõ hơn thì hãy xem đoạn code sử dụng class trên như sau :Ví dụint main ( ) { / / Nhan vien 1 NhanVien n1 = NhanVien ( ) ; n1. setData ( 111231, ” Nguyen Van A “, 24 ) ; n1. showData ( ) ; / / Nhan vien 2 NhanVien n2 = NhanVien ( ) ; n2. setData ( 111232, ” Nguyen Van B “, 25 ) ; n2. showData ( ) ; return 0 ; } Trong ví dụ này mình đã tạo ra hai đối tượng người dùng sinh viên đó là n1 và n2, và con trỏ this của n1 sẽ trỏ đến chính đối tượng người dùng n1, con trỏ this của n2 sẽ trỏ đến chính đối tượng người tiêu dùng n2, đây ta gọi là đối tượng người dùng hiện tại .Lưu ý : Trong những phương pháp thông thường ( không phải hàm khởi tạo ) nếu bạn sử dụng tên của biến thì sẽ có hai trường hợp xảy ra .Nếu biến đó không tôn tại trong phương thức mà nó lại trùng với tên thuộc tính thì mặc nhiên nó sẽ hiểu đó là thuộc tính.Nếu biến đó có khai báo trong phương thức thì ta sẽ hiểu đó là biến bình thường, không phải là thuộc tính.
Một ví dụ khác về con trỏ this
Nếu biến đó không tôn tại trong phương pháp mà nó lại trùng với tên thuộc tính thì mặc nhiên nó sẽ hiểu đó là thuộc tính. Nếu biến đó có khai báo trong phương pháp thì ta sẽ hiểu đó là biến thông thường, không phải là thuộc tính .Bạn hãy xem ví dụ dưới đây, đây là một ví dụ mình viết lại ở phần 1 và có một chút ít đổi khác .Ví dụ# include using namespace std ; class NhanVien { int msnv ; string ten ; int tuoi ; public : NhanVien ( int msnv, string ten, int tuoi ) { cout
XII. Con Trỏ Void Trong C++
Để tìm hiều về con trỏ void trong C++, trước hết bạn cần phải nắm vững các kiến thức cơ bản về con trỏ void. Đừng lo lắng vì reset1010.com đã chuẩn bị cho bạn trong các bài viết sau đây:
Con trỏ void trong C++ là gì
Ý Nghĩa Con Trỏ This Trong C + + Để tìm hiều về con trỏ void trong C + +, trước hết bạn cần phải nắm vững những kiến thức và kỹ năng cơ bản về con trỏ void. Đừng lo ngại vì reset1010.com đã sẵn sàng chuẩn bị cho bạn trong những bài viết sau đây :Trong số những kiểu con trỏ, bạn hoàn toàn có thể xác lập một con trỏ hơi khác thường được gọi là con trỏ void .Giống như những loại con trỏ khác trong C + + thì con trỏ void cũng được sử dụng để tàng trữ địa chỉ của một tài liệu trong bộ nhớ máy tính .Tuy nhiên điều đặc biệt quan trọng ở đây là, với kiểu tài liệu mà con trỏ void lưu giữ địa chỉ thì chương trình hoàn toàn có thể truy vấn đến địa chỉ của tài liệu đó, nhưng không xác lập được kiểu của nó. Nói cách khác thì con trỏ void được sử dụng để lưu giữ địa chỉ của những kiểu tài liệu không sống sót kiểu tài liệu .
Khai báo con trỏ void trong C++
Cách khai báo con trỏ void trong C + + cũng tựa như như với những loại con trỏ khác, tất cả chúng ta viết kiểu void, rồi dấu hoa thị *, và ở đầu cuối là tên con trỏ void như sau :void * pdata ; Cách viết này cũng tựa như như với những kiểu con trỏ int hay char ví dụ điển hình :char * pCharData ; / / Con trỏ kiểu charint * pIntData ; / / Con trỏ kiểu intChúng ta cũng hoàn toàn có thể triển khai những thao tác như gán địa chỉ vào con trỏ, hoặc là in địa chỉ được gán vào con trỏ void tương tự như như những loại con trỏ khác trong C + +. Ví dụ :# include using namespace std ; int main ( ) { char data = ” A ” ; / / Khai báo biến data void * pdata = và data / / Khai báo con trỏ void và gán địa chỉ của biến data vào con trỏ cout Tuy nhiên, không giống như với những kiểu con trỏ khác thì tất cả chúng ta lại không hề triển khai những thao tác với giá trị của biến mà con trỏ void trỏ đến, và cho nên vì thế cũng không hề biết được kiểu của tài liệu đó là gì. Ví dụ, tất cả chúng ta không hề đọc được giá trị của biến trải qua con trỏ, vì lỗi sau đây sẽ xảy ra :# include using namespace std ; int main ( ) { char data = ” A ” ; / / Khai báo biến data void * pdata = và data / / Khai báo con trỏ void và gán địa chỉ của biến data vào con trỏ cout Vậy chẳng phải con trỏ void trong C + + rất là vô dụng hay sao ? Tất nhiên là không phải rồi, vì tất cả chúng ta sẽ cần tới con trỏ void trong những trường hợp như dưới đây :
Sử dụng con trỏ void trong C++
Con trỏ void trong C + + sẽ được sử dụng trong những trường hợp đặc biệt quan trọng sau đây :
Con trỏ vạn năng giúp lưu giữ tất cả các loại giữ liệu trong C++
Một điều dễ hiểu là do con trỏ void không sống sót kiểu của tài liệu mà nó đang chỉ đến, nên nó có năng lực gật đầu và lưu giữ địa chỉ của toàn bộ những loại giữ liệu khác nhau trong C + +. Đây là điều mà những con trỏ khác trong C + + không làm được. Ví dụ như con trỏ kiểu int thì chỉ đồng ý lưu địa chỉ của tài liệu kiểu int, còn con trỏ kiểu char thì cũng chỉ hoàn toàn có thể đồng ý lưu giữ địa chỉ của kiểu giữ liệu char. Nhưng với con trỏ void, void chấp hết ^ _ .Ví dụ đơn cử, con trỏ void trong C + + dưới đây hoàn toàn có thể lưu giữ địa chỉ của tổng thể những loại giữ liệu mà không sợ lỗi xảy ra trong chương trình .
#include using namespace std;int main(){ //Khai báo biến char data1; short data2; long data3; double data4; // Khai báo con trỏ void void * pdata; //Gán địa chỉ của các loại dữ liệu với nhiều kiểu vào con trỏ void pdata = &data1 // pdata –> data1 pdata = &data2 // pdata –> data2 pdata = &data3 // pdata –> data3 pdata = &data4 // pdata –> data4 return 0;}Con trỏ giúp cố ý ẩn kiểu dữ liệu
Vì con trỏ C + + không được cho phép tất cả chúng ta đọc kiểu tài liệu cũng như truy vấn vào tài liệu tại địa chỉ mà nó lưu giữ, nên con trỏ void có vai trò vô cùng quan trong khi tất cả chúng ta muốn ẩn kiểu tài liệu nào đó trong chương trình .Đây là một thuật toán vô cùng phức tạp yên cầu lượng kiến thức và kỹ năng khá cao, chỉ dành cho những bạn thực sự pro và muốn khám phá sâu về C + + mà thôi .
XIII. Bài Tập Về Con Trỏ Trong C++
Con Trỏ Void Trong C + +Trong chủ đề này, tất cả chúng ta cùng làm một số ít bài tập về Con trỏ trong C + + .Bài tập 1Sử dụng con trỏ trong C + +, bạn hãy viết một chương trình C + + để nhận tài liệu từ người dùng và tìm giá trị lớn nhất của một tập dữ liệu nội bộ .Lời giảiDưới đây là chương trình C + + để giải bài tập trên. Mình sử dụng một hàm mà nhận mảng những giá trị tài liệu và kích cỡ của nó. Hàm này trả về con trỏ mà trỏ tới giá trị lớn nhất .
#include #include using namespace std;int *findMax(int arr<>,int n); int main(){ int n,i,*p; cout>n; int arr; for(i=0;i>arr; } p=findMax(arr,n); coutChạy chương trình C++ trên sẽ cho kết quả như hình sau:
Bài Tập Về Con Trỏ Trong C + +Bài tập 2Viết một chương trình C + + để nhận 5 giá trị nguyên từ bàn phím. 5 giá trị này sẽ được tàng trữ trong một mảng bởi sử dụng một con trỏ. Sau đó, in những thành phần của mảng trên màn hình hiển thị .Lời giảiDưới đây là chương trình C + + để giải bài tập trên .# include # includeusing namespace std ; int main ( ) { int arr < 5 >, i ; int * p = arr ; cout > * p >> * ( p + 1 ) >> * ( p + 2 ) >> * ( p + 3 ) >> * ( p + 4 ) ; coutChạy chương trình C + + trên sẽ cho hiệu quả như hình sau :Bài Tập Về Con Trỏ Trong C + +Sửa đổi giải thuật trên để in những thành phần của mảng theo thứ tự đảo ngược bởi sử dụng một con trỏ .# include # includeusing namespace std ; int main ( ) { int arr < 5 >, i ; int * p = arr ; cout > * p >> * ( p + 1 ) >> * ( p + 2 ) >> * ( p + 3 ) >> * ( p + 4 ) ; cout = 0 ; i — ) coutChạy chương trình C + + trên sẽ cho hiệu quả như hình sau :
XIV. Ép Kiểu Con Trỏ Trong C++
Bài Tập Về Con Trỏ Trong C + +Ép kiểu trong C + + là việc gán giá trị của một biến có kiểu tài liệu này tới biến khác có kiểu tài liệu khác .Cú pháp 🙁 type ) value ; Ví dụ :float c = 35.8 f ; int b = ( int ) c + 1 ; Trong ví dụ trên, tiên phong giá trị dấu phảy động c được đổi thành giá trị nguyên 35. Sau đó nó được cộng với 1 và hiệu quả là giá trị 36 được lưu vào b .
Phân loại ép kiểu trong C++
Trong C + +, có hai loại ép kiểu tài liệu :Nới rộng (widening): Là quá trình làm tròn số từ kiểu dữ liệu có kích thước nhỏ hơn sang kiểu có kích thước lớn hơn. Kiểu biến đổi này không làm mất thông tin.Thu hẹp (narrowwing): Là quá trình làm tròn số từ kiểu dữ liệu có kích thước lớn hơn sang kiểu có kích thước nhỏ hơn. Kiểu biến đổi này có thể làm mất thông tinNới rộng ( widening ) : Là quy trình làm tròn số từ kiểu tài liệu có size nhỏ hơn sang kiểu có size lớn hơn. Kiểu đổi khác này không làm mất thông tin. Thu hẹp ( narrowwing ) : Là quy trình làm tròn số từ kiểu tài liệu có size lớn hơn sang kiểu có size nhỏ hơn. Kiểu biến hóa này hoàn toàn có thể làm mất thông tin
1. Nới rộng (widening)
Ép Kiểu Con Trỏ Trong C + +
Nới rộng (widening): Là quá trình làm tròn số từ kiểu dữ liệu có kích thước nhỏ hơn sang kiểu có kích thước lớn hơn. Kiểu biến đổi này không làm mất thông tin. Ví dụ chuyển từ int sang float. Chuyển kiểu loại này có thế được thực hiện ngầm định bởi trình biên dịch.
Xem thêm: Bài Tập Phân Biệt Hiện Tại Đơn Và Hiện Tại Tiếp Diễn Có Đáp Án
Ví dụ :# include using namespace std ; int main ( ) { int i = 100 ; long l = i ; / / khong yeu cau chi dinh ep kieu float f = l ; / / khong yeu cau chi dinh ep kieu cout Kết quả :
Giá trị Int: 100Giá trị Long: 100Giá trị Float: 1002.Thu hẹp (narrowwing)
Ép Kiểu Con Trỏ Trong C++
Thu hẹp (narrowwing): Là quá trình làm tròn số từ kiểu dữ liệu có kích thước lớn hơn sang kiểu có kích thước nhỏ hơn. Kiểu biến đổi này có thể làm mất thông tin như ví dụ ở trên. Chuyển kiểu loại này không thể thực hiện ngầm định bởi trình biên dịch, người dùng phải thực hiện chuyển kiểu tường minh.
Ví dụ :# include using namespace std ; int main ( ) { double d = 100.04 ; long l = ( long ) d ; / / yeu cau chi dinh kieu du lieu ( long ) int i = ( int ) l ; / / yeu cau chi dinh kieu du lieu ( int ) cout Kết quả :
Source: https://final-blade.com
Category: Kiến thức Internet