Giáo trình Lập trình hệ thống chat đơn giản bằng Winsock trong môi trường lập trình Visual C++

pdf 49 trang huongle 7390
Bạn đang xem 20 trang mẫu của tài liệu "Giáo trình Lập trình hệ thống chat đơn giản bằng Winsock trong môi trường lập trình Visual C++", để tải tài liệu gốc về máy bạn click vào nút DOWNLOAD ở trên

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

  • pdfgiao_trinh_lap_trinh_he_thong_chat_don_gian_bang_winsock_tro.pdf

Nội dung text: Giáo trình Lập trình hệ thống chat đơn giản bằng Winsock trong môi trường lập trình Visual C++

  1. LẬP TRÌNH HỆ THỐNG CHAT ĐƠN GIẢN BẰNG WINSOCK TRONG MÔI TRƯỜNG LẬP TRÌNH VISUAL C++
  2. Bài thực hành 2 LP TRÌNH HN THRNG CHAT ĐƠN GI6N B>NG WINSOCK TRONG MÔI TRƯNG LP TRÌNH VISUAL C++
  3. Nội dung 4.1 Giới thiệu môi trường lập trình Visual C++ 6.0 4.2 Lập trình Winsock trong VC++ 4.3 ThiGt kGng dang m5ng MiniChat 4.4 Hiện thực chương trình MiniChatServer 4.5 HiOn thc chương trình MiniChatClient
  4. Giới thiệu môi trường lập trình Visual C++ 6.0 (VC++) • Là môi trường lập trình C++ cho phép thiết kế trực quan giao diện. •Các ứng dụng được tổ chức theo dạng project, một project chứa các file khác nhau về mã chương trình, giao diện, các file header • Có nhiều loại ứng dụng trong VC++. Chương này giới thiệu về ứng dụng MFC
  5. Tạo mới mộtproject Dùng menu File Æ New. Hộp thoại như bên dưới xuất hiện Ở tag projecst, chọn loại ứng dụng là MFC AppWizard (exe). Ở phần location, chọn thư mục để chứa project. Gõ tên project và chọnOK
  6. Hiệu chỉnh các thông số Bước thứ nhất chọn loại ứng dụng, chọn dạng Dialog based như hình bên. Nhấn button Next để tiếp tục
  7. Hiệu chỉnh các thông số Bước2,chọn các đặc tính của ứng dụng như hình +Phải chọn checkbox Windows Sockets Nhấn button Next để tiếp tục
  8. Hiệu chỉnh các thông số Bước3,chọn các chức năng hỗ trợ như hình vẽ Nhấn button Next để tiếp tục
  9. Hiệu chỉnh các thông số Bước4:xác nhận các thông số đã chọn. Có thể qua lại các bước trước đó để hiệu chỉnh bằng button Back. Chọn button Finish để kết thúc
  10. Hiệu chỉnh các thông số Bước cuối cùng: xác nhận và chọnOK để bắt đầu lập trình
  11. Giao diện của môi trường VC++ Công cụ Controls Cửa sổ Cửa sổ Workspace chính Cửa sổ Output
  12. Thiết kế giao diện • Để thiết kế giao diện, ta cần dùng thanh công cụ Controls (right-click vào các thanh công cụ,chọn Controls như hình bên) •Các đối tượng giao diện thường dùng: – Static Text –Edit Box – Button –Listbox
  13. Vẽ các đối tượng giao diện •Mở Dialog cần vẽ các đối tượng giao diện (Ở cửa sổ Workspace, chọn chế độ ResourseView, click chọn thư mục dialog, chọn Dialog tương ứng) •Muốn vẽ đối tượng giao diện nào click vào đối tượng giao diện đó, đưa trỏ chuột vào Dialog để vẽ (dùng cơ chế Drag chuột, vừa nhấn chuột trái vừa kéo)
  14. Thiết lập thuộc tính cho các đối tượng giao diện • Right-click vào đối tượng giao diện và chọn Properties •IDlà thuộc tính tên nhận dạng của đối tượng giao diện •Tuỳ mỗi loại đối tượng giao diện có các thuộc tính riêng
  15. Thiết lập thuộc tính cho các đối tượng giao diện •Thiết lập caption (Nội dung hiển thị lên phần tử giao diện) cho đối tượng giao diệnButtonvà Static Text như hình bên dưới
  16. Khai báo biến và định nghĩa hàm • Trong cửa sổ workspace, chọn tab ClassView, right-click vào class C*Dlg, menu hiển thị như hình vẽ bên •Chọn chức năng Add Member Variable •Chức năng này cũng dùng tương tự cho việc định nghĩa hàm
  17. Khai báo biến và định nghĩa hàm • Khai báo biến như hình trên: đánh kiểu biến, tên biến và tầm vực của biến rồi nhấn OK • Định nghĩa hàm như hình bên dưới: kiểu trả về, tên hàm và các thông số,tầm vực truy xuất
  18. Gán biến cho đối tượng giao diện •Mỗi đối tượng giao diện đều có thể truy xuất thông qua biến được định nghĩa •Chọn menu View -> ClassWinzard -> Member Variables •Chọn đối tượng giao diện tương ứng (nhờ vào ID đã đặt), click button Add Variable) • Đặt tên biến, loại biến (Control hoặc Value) và kiểu dữ liệu
  19. Gán biến cho đối tượng giao diện
  20. Thiết lập-lấy giá trị phần tử giao diệnEdit BoxvàStatic Text •Thiết lập: – Gán giá trị cho biến tương ứng. – Dùng lệnh: UpdateData(FALSE); •Lấy giá trị: – Dùng lệnh: UpdateData(TRUE); – Giá trị được truyền cho biến tương ứng của phần tử giao diện Ví dụ: m_mes=m_mes+"Accepted a connection!\r\n"; UpdateData(FALSE);
  21. Thêm - loại giá trị cho phần tử giao diện Listbox • Thêm vào ListBox: – Dùng phương thức AddString(String) của đối tượng điều khiển ListBox •Loại phần tử ra khỏi ListBox: – Dùng phương thức RemoveString(int index) của đối tượng điều khiển •Lấy index của một phần tử nào, ta cần phải quản lý danh sách của Listbox
  22. Tạo hàm xử lý sự kiện cho button • Khi người sử dụng click chuột vào button nào trên giao diện, hệ thống sẽ sinh ra sự kiện BN_CLICKED cho đối tượng đó. •Người lập trình phải viết mã để xử lý sự kiện đó. • Để tạo hàm xử lý sự kiện, ta có thể double-click vào button, VC++ sẽ đề nghị tên hàm, nhấnOK để bắt đầu viết mã
  23. Tạo hàm xử lý sự kiện cho button Có thể dùng ClassWinzard, chọn đối tượng Button, chọn message BN_CLICKED và nhấnAdd Function
  24. Lập trình Winsock trong VC++ • Phân tích và thiết kế giao diện cần thiết cho ứng dụng mạng. •Thiết lập các biến dữ liệu tương ứng với các phần tử giao diện Listbox, Edit box, Static Text • Đặt các biến dùng cho lập trình socket •Viết mã lệnh trình tự các hàm như đã trình bày ở chương 3
  25. Lập trình Winsock trong VC++ •Hàm được gọi đầu tiên khi ứng dụng khởi tạo là OnInitDialog(): chúng ta có thể viết hàm khởi tạo socket, bind, listen, accept trong hàm này •Có thể tạo các button để xử lý gọi các hàm nêu trên. • Nên tạo các hàm để xử lý sự kiện và các lệnh tương ứng
  26. Lập trình Winsock trong VC++ •Tạo hàm WindowProc để xử lý các sự kiện mạng: –Cửa sổ ClassWinzard, chọn ID là C*Dlg, Messages là WindowProc và click button Add Function – Click button Edit code để viết mã
  27. Lập trình Winsock trong VC++ •Một số messages cần quan tâm khi lập trình mạng trong VC++ – WM_CLOSE: xảy ra khi người sử dụng đóng chương trình – WM_KEYUP: xảy ra khi người sử dụng nhả một phím, có thể dùng để detect phím Enter •Việc sử lý các messages này cũng cần phải tạo hàm xử lý tương ứng như slide trước
  28. ThiGt kGng dang m5ng MiniChat • Ứng dụng MiniChat có hai chương trình MiniChatServer và MiniChatClient • Trong hệ thống, chỉ có một chương trình server và nhiều chương trình client đang chạy. •Chương trình client gởi dữ liệu đến cho chương trình server để yêu cầu thông tin hoặc gởi thông tin => Định nghĩa các loại dữ liệu gởi
  29. Định nghĩa các loại dữ liệu gởi •Dữ liệu gởi của client cho server: – Tham gia vào chat room: LOGIN:nickname* • nickname là tên của người sử dụng dùng để chat, không được trùng với các nickname khác –Gởi message cho toàn bộ chat room: PUBLIC: nicknamesender:message* –Gởi message cho riêng một user: PRIVATE:nicknamesender:nicknamereciever: message* – Thoát khỏi chat room: QUIT*
  30. Định nghĩa các loại dữ liệu gởi •Dữ liệu gởi từ server cho client: – Danh sách các user (nickname) có trong chatroom: LIST[:nickname]+* •Ví dụ: LIST:cuc:mai:lan:dao* – Message cho toàn bộ user: PUBLIC:nicknamesender:message* – Message cho riêng một user: PRIVATE:nicknamesender:message*
  31. Định nghĩa các loại dữ liệu gởi •Dữ liệu gởi từ server cho client: –Một user login vào: USERL:nickname* – Đãxử lý yêu cầu đúng: +OK –Các lỗi: •-100: Unknown command  -106: Login already •-101: Not login •-102: Nickname existed •-103: Nickname not exist •-104: Cannot send the message •-105: Not accept null nickname
  32. Thiết kế sơ đồ chức năng của ứng dụng MiniChatServer •Chương trình server mở socket và lắng nghe kết nối từ các client. Khi có dữ liệu đến, server phân tích dữ liệu thuộc dạng nào và xử lý tương ứng: Dữ liệu đến LOGIN* PUBLIC* PRIVATE* QUIT* –Nếu không thuộc các định dạng trên thì gởi lện –100: Bad request cho client
  33. Thiết kế sơ đồ chức năng của ứng dụng MiniChatServer •Với mỗi loại dữ liệu sẽ xử lý tương ứng: – LOGIN* Phân tích lấy Nickname đã Đ Gởi thông báo lỗi nickname có trong danh –102: Nickname existed sách? cho client Đ Dữ liệu đúng định dạng S Thêm nickname và địa chỉ S socket vào danh sách. Gởi thông báo lỗi Gởi lệnh LIST:* cho client –100: Bad request này. Gởi thông tin cho tât cho client cả các user lệnh USERL:nickname*
  34. Thiết kế sơ đồ chức năng của ứng dụng MiniChatServer •Xử lý dữ liệu PUBLIC*: Phân tích lấy Gởi thông tin +OK cho dữ liệu Lấy thông tin client đã gởitin. nicknamesender Gởi thông tin cho tât cả . Đã có trong các địa chỉ socket trong danh sách? danh sách các user Dữ liệu đúng Đ định dạng S Gởi thông báo lỗi S –101: Not login Gởi thông báo lỗi cho client –100: Bad request cho client
  35. Thiết kế sơ đồ chức năng của ứng dụng MiniChatServer •Xử lý dữ liệu PRIVATE*: Phân tích lấy Lấy thông dữ liệu Lấy thông tin Đ tin nicknamesender nicknamere Đ . Đã có trong ciever. Đã danh sách? có trong danh sách? Dữ liệu đúng Đ định dạng S S Gởi thông báo lỗi S Gởi thông báo lỗi –101: Not login –100: Bad request cho client cho client Tìm địa chỉ socket của user trong danh sách và gởi thông tin Gởi thông báo lỗi PRIVATE:nicknamesender:message* –103: Nickname not exist cho client này. Gởi thông tin +OK cho cho client client đã gởi tin
  36. Thiết kế sơ đồ chức năng của ứng dụng MiniChatServer •Xử lý dữ liệuQUIT*: Phân tích lấy dữ liệu Tìm địachỉ socket này có trong danh sách các user? Dữ liệu đúng Đ Gởi thông tin +OK cho client định dạng này. Xoá thông tin user khỏi S danhsáchuser. Gởilệnh S Gởi thông báo lỗi LIST[:nickname]* cho tấtcả client có trong danh sách user. Gởi thông báo lỗi –101: Not login –100: Bad request cho client cho client
  37. Thiết kế sơ đồ chức năng của ứng dụng MiniChatClient •Chương trình client tạo socket, lấy các thông tin từ giao diện của người sử dụng để kết nối đến server. •Nếu kết nối thành công, lấy thông tin nickname để gởi dữ liệu LOGIN:nickname* đến server và chờ nhận dữ liệu về. Dữ liệu về có hai dạng: –nếu bắt đầu bằng ký hiệu ‘-’ có nghĩa là bị lỗi, phân tích lỗi tương ứng để thông báo cho user –Nếu là LIST* có nghĩa là đăng nhập thành công, phân tích danh sách các nickname để hiển thị cho user
  38. Thiết kế sơ đồ chức năng của ứng dụng MiniChatClient •User gởi message vào chat room, có hai dạng: –Chỉ gởi cho một user: tạo lệnh PRIVATE* và gởi cho server. Chờ nhận dữ liệu về: • +OK: tiếp tục các quá trình khác • -xxx: lỗi giao thức, phân tích lỗi và thông báo –Gởi cho toàn bộ chat room: tạo lệnh PUBLIC* và gởi cho server. Chờ nhận dữ liệu về: • +OK: tiếp tục quá trình khác • -xxx: lỗi giao thức, phân tích lỗi và thông báo
  39. Thiết kế sơ đồ chức năng của ứng dụng MiniChatClient • User thoát khỏi chat room hoặc tắt chương trình –Gởi lệnh QUIT* cho server và chờ nhận dữ liệu về: •+OK:tiếp tục các quá trình khác • -xxx: lỗi giao thức, phân tích lỗi và thông báo •Nhận dữ liệu bất kỳ từ server: phân tích dữ liệu thuộc một trong các dạng: USERL*, USERQ*, PRIVATE*, PUBLIC*, nếu không thuộc các dạng này thì thông báo lỗi. Đối với mỗi dạng dữ liệu sẽ được sử lý như slide kế
  40. Thiết kế sơ đồ chức năng của ứng dụng MiniChatClient •Nếu là lệnh USERL*: – Phân tích lấy nickname – Thêm nickname vào danh sách các user •Nếu là lệnh USERQ*: – Phân tích lấy nickname – Xoá nickname khỏi danh sách các user, nếu có lỗi thì báo lỗi cho user •Nếu là lệnh PRIVATE* hoặcPUBLIC* –Hiển thị thông tin này cho user
  41. Hiện thực chương trình Editbox, MiniChatServer ID=IDD_EDIT_PORT, biến tương ứng: Thiết kế giao int m_port; diện như hình StaticText, bên và đặtcác ID=IDD_STATIC_STATUS , caption =‘’, biến tương biếntương ứng CString m_status; ứng cho các phầntử giao Listbox, ID=IDD_LIST_USER, diện Multiline, scroll, biến Edit box, tương ứng: ID=IDD_EDIT_MESSAGE, CListBox m_users; Multiline, scroll, biến tương ứng: CString m_message;
  42. Hiện thực chương trình MiniChatServer • Định nghĩa kiểu dữ liệu record để lưu danh sách các user (đầu file C*Dlg.h): typedef struct T_UserRecord { char name[20]; SOCKET socket; int state; struct T_UserRecord* next; } T_UserRecord; • Khai báo các biến: (dùng chức năng Add Member Variable) SOCKET ServerSocket; char temp_message[128]; T_UserRecord *UserRecordList; T_UserRecord *tempUserRecord;
  43. Hiện thực chương trình MiniChatServer • Khai báo các hằng số (file Resource.h) – #define MSG_LENGTH 256 – #define WSA_ACCEPT 1006 – #define WSA_RDCLOSE 1007 – #define CONNECTED 2000 – #define LOGIN 2001 – #define CHAT 2002 •Lập trình theo các bước sau: –Tạo hàm xử lý sự kiện khi người dùng click chuột vào Button Listen, hàm OnButtonListen(): lần lượt gọi các lệnh socket, bind, listen, và WSASyncSelect
  44. Hiện thực chương trình MiniChatServer • Trình tự: –Tạo hàm WindowProc() và viết mã lệnh: • Tùy theo loại message sẽ gọi hàm tương ứng để xử lý (hai loại message định nghĩa là WSA_ACCEPT, WSA_RDCLOSE). – Định nghĩa và viết mã lệnh hàm OnAccept( ): xử lý sự kiện FD_ACCEPT, message WSA_ACCEPT khi có yêu cầu kết nối – Định nghĩa và viết mã lệnh hàm Process( ): xử lý sự kiện FD_READ, message WSA_RDCLOSE khi có dữ liệu gởi đến từ client
  45. Hiện thực chương trình MiniChatServer • Trình tự: – Trong hàm Process(): phân tích định dạng của dữ liệu đến xử lý tương ứng: • LOGIN*: hàm Login( ) • PUBLIC*: hàm SendPublic( ) • PRIVATE*: hàm SendPrivate( ) • QUIT*: hàm Logout( ) – Định nghĩa và viết mã lệnh cho lần lượt các hàm trên theo sơ đồ khối đã thiết kế
  46. Hiện thực chương trình MiniChatClient Thiết kế giao diện Static Text, của chương trình CString m_status client như hình vẽ Editbox, Editbox, CString m_host int port Editbox, Listbox, Editbox, CString m_message CListBox CString m_nickname m_list_members Editbox, CString user Editbox, CString m_data
  47. Hiện thực chương trình MiniChatClient • Khai báo các biến: (dùng chức năng Add Member Variable) SOCKET ClientSocket; char temp_message[128]; int chat_status; CString data; • Khai báo các hằng số (file Resource.h) – #define NOTLOGIN 2000 – #define LOGIN 2001 – #define QUIT 2002 – #define CHAT 2003 – #define WSA_RDREAD 3000
  48. Hiện thực chương trình MiniChatClient •Lập trình theo các bước sau: –Tạo hàm xử lý sự kiện khi người dùng click chuột vào Button Login, hàm OnButtonLogin(): lần lượt gọi các lệnh socket, connect, và WSASyncSelect để chờ nhận sự kiện mạng –Tạo hàm WindowProc() và viết mã lệnh: •Chương trình client chỉ có một message (WSA_RDREAD) cho hai sự kiện FR_READ và FD_CLOSE, với mỗi sự kiện ta thực hiện lệnh gọi hàm tương ứng – Định nghĩa và viết mã lệnh hàm Process( ): xử lý sự kiện FD_READ, message WSA_RDREAD khi có dữ liệu từ server gởi đến
  49. Hiện thực chương trình MiniChatClient •Trình tự: – Định nghĩa và viết mã lệnh lần lượt các hàm Login, ResponseLogin, Send, Communicate, DisplayUserList –Tạo hàm xử lý sự kiện :OnSelchangeListMember, OnButtonSend, OnButtonLogout