主题:WINSOCKET 聊天程序
wuchengwei
[专家分:1650] 发布于 2006-04-08 11:30:00
// Socket Client.cpp : 定义应用程序的入口点。
//
#include "stdafx.h"
#include "windows.h"
#include "winsock2.h"
#include "ws2tcpip.h"
#include "Socket Client.h"
#pragma comment(lib,"Ws2_32.lib")
#pragma warning(disable:4312)
#pragma warning(disable:4267)
#pragma warning(disable:4244)
#define MAX_LOADSTRING 100
#define ID_RECVEDIT 1
#define ID_SENDEDIT 2
#define ID_SENDBTN 3
#define SERVER_PORT 4500
#define ADDR_MAX_LEN 16
#define WM_SOCKET (WM_USER+ 1)
//全局变量
HINSTANCE hInst;
WNDPROC SendProc;
WNDPROC RecvProc;
TCHAR szTitle[MAX_LOADSTRING];
TCHAR szWindowClass[MAX_LOADSTRING];
CHAR szServerAddr[ADDR_MAX_LEN];
// 此代码模块中包含的函数的前向声明:
ATOM MyRegisterClass(HINSTANCE hInstance);
BOOL InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK Dlg1Proc(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK SendEditProc(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK RecvEditProc(HWND, UINT, WPARAM, LPARAM);
int APIENTRY _tWinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPTSTR lpCmdLine,
int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine);
// TODO: 在此放置代码。
MSG msg;
HACCEL hAccelTable;
// 初始化全局字符串
LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
LoadString(hInstance, IDC_SOCKETCLIENT, szWindowClass, MAX_LOADSTRING);
MyRegisterClass(hInstance);
// 执行应用程序初始化:
if (!InitInstance (hInstance, nCmdShow))
{
return FALSE;
}
hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_SOCKETCLIENT));
// 主消息循环:
while (GetMessage(&msg, NULL, 0, 0))
{
if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
return (int) msg.wParam;
}
//
// 函数: MyRegisterClass()
//
// 目的: 注册窗口类。
//
// 注释:
//
// 仅当希望
// 此代码与添加到 Windows 95 中的“RegisterClassEx”
// 函数之前的 Win32 系统兼容时,才需要此函数及其用法。调用此函数十分重要,
// 这样应用程序就可以获得关联的
// “格式正确的”小图标。
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
WNDCLASSEX wcex;
wcex.cbSize = sizeof(WNDCLASSEX);
wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = WndProc;
wcex.cbClsExtra = 0;
wcex.cbWndExtra = 0;
wcex.hInstance = hInstance;
wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_SOCKETCLIENT));
wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
wcex.lpszMenuName = MAKEINTRESOURCE(IDC_SOCKETCLIENT);
wcex.lpszClassName = szWindowClass;
wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));
return RegisterClassEx(&wcex);
}
//
// 函数: InitInstance(HINSTANCE, int)
//
// 目的: 保存实例句柄并创建主窗口
//
// 注释:
//
// 在此函数中,我们在全局变量中保存实例句柄并
// 创建和显示主程序窗口。
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
HWND hWnd;
hInst = hInstance; // 将实例句柄存储在全局变量中
hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
if (!hWnd)
{
return FALSE;
}
ShowWindow(hWnd, nCmdShow);
UpdateWindow(hWnd);
return TRUE;
}
回复列表 (共13个回复)
沙发
wuchengwei [专家分:1650] 发布于 2006-04-08 11:32:00
//
// 函数: About(HWND, UINT, WPARAM, LPARAM)
//
// 目的: 处理"About"对话框消息。
//
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
UNREFERENCED_PARAMETER(lParam);
switch (message)
{
case WM_INITDIALOG:
return (INT_PTR)TRUE;
case WM_COMMAND:
if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
{
EndDialog(hDlg, LOWORD(wParam));
return (INT_PTR)TRUE;
}
break;
}
return (INT_PTR)FALSE;
}
//
// 函数: Dlg1Proc(HWND, UINT, WPARAM, LPARAM)
//
// 目的: 处理输入对话框消息。
//
INT_PTR CALLBACK Dlg1Proc(HWND hDlg1, UINT message, WPARAM wParam, LPARAM lParam)
{
HWND hWndDlgEdit;
switch (message)
{
case WM_INITDIALOG:
return (INT_PTR)TRUE;
case WM_COMMAND:
if (LOWORD(wParam) == IDOK )
{
hWndDlgEdit = GetDlgItem(hDlg1, IDC_EDIT1);
GetWindowTextA(hWndDlgEdit, szServerAddr, GetWindowTextLength(hWndDlgEdit) +1);
EndDialog(hDlg1, LOWORD(wParam));
return (INT_PTR)TRUE;
}
if(LOWORD(wParam) == IDCANCEL)
{
MessageBox(hDlg1, TEXT("请按确定键确认退出!"), TEXT("Eixt"), MB_OK),
SendMessage(GetParent(hDlg1), WM_COMMAND, MAKELONG(IDM_EXIT, 0), (LPARAM)hDlg1);
EndDialog(hDlg1, LOWORD(wParam));
return (INT_PTR)TRUE;
}
break;
}
return (INT_PTR)FALSE;
}
//
// 函数: SendEditProc(HWND, UINT, WPARAM, LPARAM)
//
// 目的: 处理SendEdit窗口框消息。(接受CTRL+ENTER发送消息)
//
LRESULT CALLBACK SendEditProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
if(message == WM_KEYDOWN && wParam == VK_RETURN && (GetKeyState(VK_CONTROL) < 0))
SendMessage(GetParent(hWnd), WM_COMMAND, MAKELONG(ID_SENDBTN, BN_CLICKED), (LPARAM)hWnd);
return CallWindowProc(SendProc, hWnd, message, wParam, lParam);
}
//
// 函数: RecvEditProc(HWND, UINT, WPARAM, LPARAM)
//
// 目的: 处理RecvEdit窗口框消息。
//
LRESULT CALLBACK RecvEditProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
HDC hDC;
if(message == WM_CREATE)
{
hDC = GetDC(hWnd);
SetTextColor(hDC, RGB(0, 0, 255));
ReleaseDC(hWnd, hDC);
}
return CallWindowProc(RecvProc, hWnd, message, wParam, lParam);
}
怎么发不了了。。。
还有主窗口的消息处理函数
板凳
fucker [专家分:680] 发布于 2006-04-08 12:47:00
你这段代码里哪有 socket 操作?发不了贴大段代码估计也没人帮你看,自己
先调试,把具体问题确定下来再问。
3 楼
wuchengwei [专家分:1650] 发布于 2006-04-08 17:33:00
不知道怎么搞的,发不了了
4 楼
wuchengwei [专家分:1650] 发布于 2006-04-08 17:35:00
//
// 函数: WndProc(HWND, UINT, WPARAM, LPARAM)
//
// 目的: 处理主窗口的消息。
//
// WM_COMMAND - 处理应用程序菜单
// WM_PAINT - 绘制主窗口
// WM_DESTROY - 发送退出消息并返回
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
HINSTANCE hInstance;
HDC hDC;
static HWND hWndRecvEdit, hWndSendEdit, hWndSendBtn;
static struct sockaddr_in client;
static SOCKET remote_server;
static TCHAR szRecv[1024];
static TCHAR szSend[1024];
static int cxChar, cyChar;
int wmId, wmEvent;
WORD wEvent, wError;
WSADATA WSAData;
static TEXTMETRIC tm;
switch (message)
{
case WM_CREATE:
hInstance = (HINSTANCE)GetWindowLong(hWnd, GWL_HINSTANCE);
hDC = GetDC(hWnd);
GetTextMetrics(hDC, &tm);
cxChar = tm.tmAveCharWidth;
cyChar = tm.tmHeight +tm.tmExternalLeading;
ReleaseDC(hWnd, hDC);
//创建两个Edit控件子窗口
hWndRecvEdit = CreateWindow(TEXT("edit"), NULL, WS_CHILD|WS_VISIBLE|WS_HSCROLL|WS_VSCROLL|WS_BORDER|WS_DISABLED
|ES_MULTILINE|ES_NOHIDESEL|ES_AUTOVSCROLL|ES_AUTOHSCROLL|ES_LEFT, 0, 0, 0, 0, hWnd, (HMENU)ID_RECVEDIT, hInstance, NULL);
RecvProc = (WNDPROC)SetWindowLong(hWndRecvEdit, GWL_WNDPROC, (LPARAM)RecvEditProc);
hWndSendEdit = CreateWindow(TEXT("edit"), NULL, WS_CHILD|WS_VISIBLE|WS_HSCROLL|WS_VSCROLL|WS_BORDER
|ES_MULTILINE|ES_AUTOVSCROLL|ES_AUTOHSCROLL|ES_LEFT, 0, 0, 0, 0, hWnd, (HMENU)ID_SENDEDIT, hInstance, NULL);
SendProc = (WNDPROC)SetWindowLong(hWndSendEdit, GWL_WNDPROC, (LPARAM)SendEditProc);
hWndSendBtn = CreateWindow(TEXT("button"), TEXT("发送"), WS_CHILD|WS_VISIBLE|BS_PUSHBUTTON, 0, 0, 0, 0, hWnd,
(HMENU)ID_SENDBTN, hInstance, NULL);
EnableWindow(hWndSendBtn, TRUE);
DialogBoxParam(hInst, (LPCTSTR)IDD_DIALOG1, hWnd, Dlg1Proc, NULL);
//初始化套接字环境
if(WSAStartup(MAKEWORD(2, 0), &WSAData))
{
MessageBox(hWnd, TEXT("WinSock Startup Error!"), TEXT("Client"), MB_ICONERROR|MB_OK);
break;
}
//创建本地套接字,用于与服务器进行数据传输
remote_server = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if(remote_server == INVALID_SOCKET)
{
MessageBox(hWnd, TEXT("WinSock Creation Error!"), TEXT("Client"), MB_ICONERROR|MB_OK);
WSACleanup();
break;
}
//选择程序需要处理的网络事件
if(SOCKET_ERROR == WSAAsyncSelect(remote_server, hWnd, WM_SOCKET, FD_ACCEPT|FD_READ))
{
MessageBox(hWnd, TEXT("WSAAsyncSelect Error!"), TEXT("Client"), MB_ICONERROR|MB_OK);
closesocket(remote_server);
WSACleanup();
break;
}
//请求服务器建立连接
client.sin_family = AF_INET;
client.sin_port = htons(SERVER_PORT);
client.sin_addr.S_un.S_addr = inet_addr(szServerAddr);
connect(remote_server, (sockaddr*)&client, sizeof(client));
if(WSAGetLastError() != WSAEWOULDBLOCK)
{
MessageBox(hWnd, TEXT("Connect Error: Would Block!"), TEXT("Client"), MB_ICONERROR|MB_OK);
closesocket(remote_server);
break;
}
break;
5 楼
wuchengwei [专家分:1650] 发布于 2006-04-08 17:35:00
case WM_SIZE:
MoveWindow(hWndRecvEdit, 0, 0, LOWORD(lParam), HIWORD(lParam)/2, TRUE);
MoveWindow(hWndSendEdit, 0, HIWORD(lParam)/2, LOWORD(lParam),
HIWORD(lParam)/2 -(cyChar +tm.tmExternalLeading), TRUE);
MoveWindow(hWndSendBtn, 0, HIWORD(lParam)-(cyChar +tm.tmExternalLeading),
LOWORD(lParam), cyChar +tm.tmExternalLeading, TRUE);
UpdateWindow(hWnd);
break;
case WM_SOCKET:
//获取事件代码和错误代码
wEvent = WSAGETSELECTEVENT(lParam);
wError = WSAGETSELECTERROR(lParam);
switch (wEvent)
{
case FD_CONNECT:
if(wError)
{
MessageBox(hWnd, TEXT("Connect Error!"), TEXT("Client"), MB_ICONERROR|MB_OK);
break;
}
EnableWindow(hWndSendEdit, TRUE);
break;
case FD_READ:
if(wError)
{
MessageBox(hWnd, TEXT("Read Error!"), TEXT("Client"), MB_ICONERROR|MB_OK);
break;
}
recv(remote_server, (CHAR*)szRecv, sizeof(szRecv) -1, 0);
SetWindowText(hWndRecvEdit, szRecv);
break;
}
break;
case WM_SETFOCUS:
SetFocus(hWndSendEdit);
break;
case WM_COMMAND:
wmId = LOWORD(wParam);
wmEvent = HIWORD(wParam);
// 分析菜单选择:
switch (wmId)
{
case ID_SENDEDIT:
if(wmEvent == EN_CHANGE)
GetWindowText(hWndSendEdit, szSend, GetWindowTextLength(hWndSendEdit) +1);
break;
case ID_SENDBTN:
if(wmEvent == BN_CLICKED)
send(remote_server, (CHAR*)szSend, (_tcslen(szSend)+1)*sizeof(TCHAR)/sizeof(CHAR), 0);
SetWindowText(hWndSendEdit, TEXT(""));
break;
case IDM_ABOUT:
DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
break;
case IDM_EXIT:
closesocket(remote_server);
WSACleanup();
DestroyWindow(hWnd);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
break;
case WM_DESTROY:
//关闭本地套接字,并清除套接字环境
closesocket(remote_server);
WSACleanup();
PostQuitMessage(0);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
}
6 楼
wuchengwei [专家分:1650] 发布于 2006-04-08 17:36:00
以上是客户端程序
以下是服务器端程序
// Socket Server.cpp : 定义应用程序的入口点。
//
#include "stdafx.h"
#include "Socket Server.h"
#include "windows.h"
#include "winsock2.h"
#include "ws2tcpip.h"
#include "winuser.h"
#pragma comment(lib,"user32.lib")
#pragma comment(lib,"Ws2_32.lib")
#pragma warning(disable:4312)
#pragma warning(disable:4267)
#pragma warning(disable:4244)
#define MAX_LOADSTRING 100
#define ID_RECVEDIT 1
#define ID_SENDEDIT 2
#define ID_SENDBTN 3
#define SERVER_PORT 4500
#define WM_SOCKET (WM_USER+ 1)
// 全局变量:
HINSTANCE hInst; // 当前实例
TCHAR szTitle[MAX_LOADSTRING]; // 标题栏文本
TCHAR szWindowClass[MAX_LOADSTRING]; // 主窗口类名
WNDPROC SendProc;
WNDPROC RecvProc;
// 此代码模块中包含的函数的前向声明:
ATOM MyRegisterClass(HINSTANCE hInstance);
BOOL InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK SendEditProc(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK RecvEditProc(HWND, UINT, WPARAM, LPARAM);
int APIENTRY _tWinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPTSTR lpCmdLine,
int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine);
// TODO: 在此放置代码。
MSG msg;
HACCEL hAccelTable;
// 初始化全局字符串
LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
LoadString(hInstance, IDC_SOCKETSERVER, szWindowClass, MAX_LOADSTRING);
MyRegisterClass(hInstance);
// 执行应用程序初始化:
if (!InitInstance (hInstance, nCmdShow))
{
return FALSE;
}
hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_SOCKETSERVER));
// 主消息循环:
while (GetMessage(&msg, NULL, 0, 0))
{
if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
return (int) msg.wParam;
}
//
// 函数: MyRegisterClass()
//
// 目的: 注册窗口类。
//
// 注释:
//
// 仅当希望
// 此代码与添加到 Windows 95 中的“RegisterClassEx”
// 函数之前的 Win32 系统兼容时,才需要此函数及其用法。调用此函数十分重要,
// 这样应用程序就可以获得关联的
// “格式正确的”小图标。
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
WNDCLASSEX wcex;
wcex.cbSize = sizeof(WNDCLASSEX);
wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = WndProc;
wcex.cbClsExtra = 0;
wcex.cbWndExtra = 0;
wcex.hInstance = hInstance;
wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_SOCKETSERVER));
wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
wcex.lpszMenuName = MAKEINTRESOURCE(IDC_SOCKETSERVER);
wcex.lpszClassName = szWindowClass;
wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));
return RegisterClassEx(&wcex);
}
//
// 函数: InitInstance(HINSTANCE, int)
//
// 目的: 保存实例句柄并创建主窗口
//
// 注释:
//
// 在此函数中,我们在全局变量中保存实例句柄并
// 创建和显示主程序窗口。
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
HWND hWnd;
hInst = hInstance; // 将实例句柄存储在全局变量中
hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
if (!hWnd)
{
return FALSE;
}
ShowWindow(hWnd, nCmdShow);
UpdateWindow(hWnd);
return TRUE;
}
7 楼
wuchengwei [专家分:1650] 发布于 2006-04-08 17:37:00
//SendEdit窗口消息处理函数(接受CTRL+ENTER发送消息)
LRESULT CALLBACK SendEditProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
if(message == WM_KEYDOWN && wParam == VK_RETURN && (GetKeyState(VK_CONTROL) < 0))
SendMessage(GetParent(hWnd), WM_COMMAND, MAKELONG(ID_SENDBTN, BN_CLICKED), (LPARAM)hWnd);
return CallWindowProc(SendProc, hWnd, message, wParam, lParam);
}
//RecvEdit窗口消息处理函数
LRESULT CALLBACK RecvEditProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
HDC hDC;
if(message == WM_CREATE)
{
hDC = GetDC(hWnd);
SetTextColor(hDC, RGB(0, 0, 255));
ReleaseDC(hWnd, hDC);
}
return CallWindowProc(RecvProc, hWnd, message, wParam, lParam);
}
// “关于”框的消息处理程序。
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
UNREFERENCED_PARAMETER(lParam);
switch (message)
{
case WM_INITDIALOG:
return (INT_PTR)TRUE;
case WM_COMMAND:
if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
{
EndDialog(hDlg, LOWORD(wParam));
return (INT_PTR)TRUE;
}
break;
}
return (INT_PTR)FALSE;
}
8 楼
wuchengwei [专家分:1650] 发布于 2006-04-08 17:37:00
//
// 函数: WndProc(HWND, UINT, WPARAM, LPARAM)
//
// 目的: 处理主窗口的消息。
//
// WM_COMMAND - 处理应用程序菜单
// WM_PAINT - 绘制主窗口
// WM_DESTROY - 发送退出消息并返回
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
HINSTANCE hInstance;
static HWND hWndRecvEdit, hWndSendEdit, hWndSendBtn;
static struct sockaddr_in server, from;
static SOCKET local_server, local_listen;
static TCHAR szRecv[1024];
static TCHAR szSend[1024];
static int cxChar, cyChar;
static TEXTMETRIC tm;
HDC hDC;
int nSize;
int wmId, wmEvent;
int nError;
WORD wEvent, wError;
WSADATA WSAData;
switch (message)
{
case WM_CREATE:
hInstance = (HINSTANCE)GetWindowLong(hWnd, GWL_HINSTANCE);
hDC = GetDC(hWnd);
GetTextMetrics(hDC, &tm);
cxChar = tm.tmAveCharWidth;
cyChar = tm.tmHeight +tm.tmExternalLeading;
ReleaseDC(hWnd, hDC);
//创建两个Edit控件子窗口和一个确认发送按钮
hWndRecvEdit = CreateWindow(TEXT("edit"), NULL, WS_CHILD|WS_VISIBLE|WS_HSCROLL|WS_VSCROLL|WS_BORDER|WS_DISABLED
|ES_MULTILINE|ES_NOHIDESEL|ES_AUTOVSCROLL|ES_AUTOHSCROLL|ES_LEFT, 0, 0, 0, 0, hWnd, (HMENU)ID_RECVEDIT, hInstance, NULL);
RecvProc = (WNDPROC)SetWindowLong(hWndRecvEdit, GWL_WNDPROC, (LPARAM)RecvEditProc);
hWndSendEdit = CreateWindow(TEXT("edit"), NULL, WS_CHILD|WS_VISIBLE|WS_HSCROLL|WS_VSCROLL|WS_BORDER
|ES_MULTILINE|ES_AUTOVSCROLL|ES_AUTOHSCROLL|ES_LEFT, 0, 0, 0, 0, hWnd, (HMENU)ID_SENDEDIT, hInstance, NULL);
SendProc = (WNDPROC)SetWindowLong(hWndSendEdit, GWL_WNDPROC, (LPARAM)SendEditProc);
hWndSendBtn = CreateWindow(TEXT("button"), TEXT("发送"), WS_CHILD|WS_VISIBLE|BS_PUSHBUTTON, 0, 0, 0, 0, hWnd,
(HMENU)ID_SENDBTN, hInstance, NULL);
EnableWindow(hWndSendBtn, TRUE);
//初始化套接字环境
if(nError = WSAStartup(MAKEWORD(2, 0), &WSAData))
{
MessageBox(hWnd, TEXT("WinSock Startup Error!"), TEXT("Sever"), MB_ICONERROR|MB_OK);
break;
}
//建立本地服务套接字
local_server = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if(local_server == INVALID_SOCKET)
{
MessageBox(hWnd, TEXT("WinSock Creation Error!"), TEXT("Sever"), MB_ICONERROR|MB_OK);
WSACleanup();
break;
}
//选择程序需要处理的网络事件
if(SOCKET_ERROR == WSAAsyncSelect(local_server, hWnd, WM_SOCKET, FD_ACCEPT|FD_READ))
{
MessageBox(hWnd, TEXT("WSAAsyncSelect Error!"), TEXT("Sever"), MB_ICONERROR|MB_OK);
closesocket(local_server);
WSACleanup();
break;
}
//设置本地端口地址
server.sin_family = AF_INET; //协议族
server.sin_port = htons(SERVER_PORT); //服务器端口
if(SOCKET_ERROR == bind(local_server, (struct sockaddr *)&server, sizeof(server)))
{
MessageBox(hWnd, TEXT("Bind Error!"), TEXT("Sever"), MB_ICONERROR|MB_OK);
closesocket(local_server);
WSACleanup();
break;
}
//监听
listen(local_server, 5);
break;
case WM_SIZE:
MoveWindow(hWndRecvEdit, 0, 0, LOWORD(lParam), HIWORD(lParam)/2, TRUE);
MoveWindow(hWndSendEdit, 0, HIWORD(lParam)/2, LOWORD(lParam),
HIWORD(lParam)/2 -(cyChar +tm.tmExternalLeading), TRUE);
MoveWindow(hWndSendBtn, 0, HIWORD(lParam)-(cyChar +tm.tmExternalLeading),
LOWORD(lParam), cyChar +tm.tmExternalLeading, TRUE);
UpdateWindow(hWnd);
break;
9 楼
wuchengwei [专家分:1650] 发布于 2006-04-08 17:37:00
case WM_SOCKET:
//获取事件代码和错误代码
wEvent = WSAGETSELECTEVENT(lParam);
wError = WSAGETSELECTERROR(lParam);
switch (wEvent)
{
case FD_ACCEPT:
if(wError)
{
MessageBox(hWnd, TEXT("Accept Error!"), TEXT("Sever"), MB_ICONERROR|MB_OK);
break;
}
nSize = sizeof(from);
local_listen = accept(local_server, (struct sockaddr *)&from, &nSize);
break;
case FD_READ:
if(wError)
{
MessageBox(hWnd, TEXT("Read Error!"), TEXT("Sever"), MB_ICONERROR|MB_OK);
break;
}
recv(local_listen, (CHAR*)szRecv, sizeof(szRecv) -1, 0);
SetWindowText(hWndRecvEdit, szRecv);
break;
}
break;
case WM_SETFOCUS:
SetFocus(hWndSendEdit);
break;
case WM_COMMAND:
wmId = LOWORD(wParam);
wmEvent = HIWORD(wParam);
// 分析菜单选择:
switch (wmId)
{
case ID_SENDEDIT:
if(wmEvent == EN_CHANGE)
GetWindowText(hWndSendEdit, szSend, GetWindowTextLength(hWndSendEdit) +1);
break;
case ID_SENDBTN:
if(wmEvent == BN_CLICKED)
send(local_listen, (CHAR*)szSend, (_tcslen(szSend)+1)*sizeof(TCHAR)/sizeof(CHAR), 0);
SetWindowText(hWndSendEdit, TEXT(""));
SendMessage(hWndSendEdit, WM_KEYDOWN, VK_BACK, 1);
break;
case IDM_ABOUT:
DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
break;
case IDM_EXIT:
closesocket(local_server);
closesocket(local_listen);
WSACleanup();
DestroyWindow(hWnd);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
break;
case WM_DESTROY:
//关闭本地服务器套接字,并清除套接字环境
closesocket(local_server);
closesocket(local_listen);
WSACleanup();
PostQuitMessage(0);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
}
10 楼
wuchengwei [专家分:1650] 发布于 2006-04-08 17:46:00
程序基本上是正确的,当然,你没有相关的头文件是无法运行的,在我自己的机器上先运行服务器端,再启动客户端,在客户端的“输入服务器IP地址”的对话框中输入我的局域网IP,则可以传输数据,若改成实际IP,则不能传输,请问,怎么办?协议问题吗?
请高手帮忙改改,需要文件的可以加我QQ
还有,我要在得到“输入服务器IP地址”中的服务器IP,如何判断该IP的主机是否处于活动状态,以便当主机处于非活动状态时,弹出提示框并中断程序
我来回复