Socket网络编程

最近编写了一个Socket网络数据传输程序,实现服务器与客户端的数据传输及硬件控制。服务器即为PC机,客户端为TQ2440,wince6.0.

网络为PC机设这的无线热点,即TQ2440通过USB Wifi链接至PC机设置的无线局域网中。比如PC机IP地址为192.168.22.1,客户端为192.168.22.10,客户端每次启动连接时IP地址会变化,PC机最后3为一般就是1,固定不变。

下面是代码:

服务器端:

#include "stdafx.h"
#include "Sever3.h"
#include "Sever3Dlg.h"
#include "winsock2.h"
#include "winnls.h"


#ifdef _DEBUG
#define new DEBUG_NEW
#endif


#define Char2WChar(lpMultiByteStr,cbMultiByte,lpWideCharStr,cbWideChar) \
MultiByteToWideChar(CP_ACP,0,lpMultiByteStr,cbMultiByte,lpWideCharStr,cbWideChar)


#define WChar2Char(lpWideCharStr,cbWideChar,lpMultiByteStr,cbMultiByte) \
WideCharToMultiByte(CP_ACP,0,lpWideCharStr,cbWideChar,lpMultiByteStr,cbMultiByte,NULL,NULL)//用于CString转换为char[]




CString clientIP;
char IP_Client[20];




// 用于应用程序“关于”菜单项的 CAboutDlg 对话框


class CAboutDlg : public CDialog
{
public:
CAboutDlg();


// 对话框数据
enum { IDD = IDD_ABOUTBOX };


protected:
virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持


// 实现
protected:
DECLARE_MESSAGE_MAP()
};


CAboutDlg::CAboutDlg() : CDialog(CAboutDlg::IDD)
{
}


void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
CDialog::DoDataExchange(pDX);
}


BEGIN_MESSAGE_MAP(CAboutDlg, CDialog)
     


END_MESSAGE_MAP()




// CSever3Dlg 对话框








CSever3Dlg::CSever3Dlg(CWnd* pParent /*=NULL*/)
: CDialog(CSever3Dlg::IDD, pParent)
{
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}


void CSever3Dlg::DoDataExchange(CDataExchange* pDX)
{
CDialog::DoDataExchange(pDX);
}


BEGIN_MESSAGE_MAP(CSever3Dlg, CDialog)
ON_WM_SYSCOMMAND()
ON_WM_PAINT()
ON_WM_QUERYDRAGICON()
ON_MESSAGE(WM_RECVDATA,OnRecvData)
//}}AFX_MSG_MAP
ON_BN_CLICKED(IDOK, &CSever3Dlg::OnBnClickedOk)
ON_BN_CLICKED(IDC_BUTTON1, &CSever3Dlg::OnBnClickedButton1)
ON_BN_CLICKED(IDC_BUTTON2, &CSever3Dlg::OnBnClickedButton2)
ON_EN_CHANGE(IDC_EDIT1, &CSever3Dlg::OnEnChangeEdit1)
END_MESSAGE_MAP()




// CSever3Dlg 消息处理程序


BOOL CSever3Dlg::OnInitDialog()
{
CDialog::OnInitDialog();


// 将“关于...”菜单项添加到系统菜单中。


// IDM_ABOUTBOX 必须在系统命令范围内。
ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
ASSERT(IDM_ABOUTBOX < 0xF000);


CMenu* pSysMenu = GetSystemMenu(FALSE);
if (pSysMenu != NULL)
{
CString strAboutMenu;
strAboutMenu.LoadString(IDS_ABOUTBOX);
if (!strAboutMenu.IsEmpty())
{
pSysMenu->AppendMenu(MF_SEPARATOR);
pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
}
}


// 设置此对话框的图标。当应用程序主窗口不是对话框时,框架将自动
//  执行此操作
SetIcon(m_hIcon, TRUE); // 设置大图标
SetIcon(m_hIcon, FALSE); // 设置小图标


// TODO: 在此添加额外的初始化代码


InitSocket();
RECVPARAM *pRecvParam=new RECVPARAM;
pRecvParam->sock=m_socket;
pRecvParam->hwnd=m_hWnd;
HANDLE hThread1=CreateThread(NULL,0,RecvProc,(LPVOID)pRecvParam,0,NULL);    //创建socket信号接收进程
CloseHandle(hThread1);  




return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE
}


void CSever3Dlg::OnSysCommand(UINT nID, LPARAM lParam)
{
if ((nID & 0xFFF0) == IDM_ABOUTBOX)
{
CAboutDlg dlgAbout;
dlgAbout.DoModal();
}
else
{
CDialog::OnSysCommand(nID, lParam);
}
}


// 如果向对话框添加最小化按钮,则需要下面的代码
//  来绘制该图标。对于使用文档/视图模型的 MFC 应用程序,
//  这将由框架自动完成。


void CSever3Dlg::OnPaint()
{
if (IsIconic())
{
CPaintDC dc(this); // 用于绘制的设备上下文


SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);


// 使图标在工作区矩形中居中
int cxIcon = GetSystemMetrics(SM_CXICON);
int cyIcon = GetSystemMetrics(SM_CYICON);
CRect rect;
GetClientRect(&rect);
int x = (rect.Width() - cxIcon + 1) / 2;
int y = (rect.Height() - cyIcon + 1) / 2;


// 绘制图标
dc.DrawIcon(x, y, m_hIcon);
}
else
{
CDialog::OnPaint();
}
}


//当用户拖动最小化窗口时系统调用此函数取得光标
//显示。
HCURSOR CSever3Dlg::OnQueryDragIcon()
{
return static_cast<HCURSOR>(m_hIcon);
}




void CSever3Dlg::OnBnClickedOk()
{
// TODO: 在此添加控件通知处理程序代码
OnOK();
}


BOOL CSever3Dlg::InitSocket()
{
SetDlgItemText( IDC_EDIT1,GetLocalIP() );
m_socket=socket(AF_INET,SOCK_DGRAM,0);
if(INVALID_SOCKET==m_socket)
{
MessageBox(_T("套接字创建失败!"));
return FALSE;
}

SOCKADDR_IN addrSock;
addrSock.sin_family=AF_INET;
addrSock.sin_port=htons(7000);
addrSock.sin_addr.S_un.S_addr=htonl(INADDR_ANY);


int retval;
retval=bind(m_socket,(SOCKADDR*)&addrSock,sizeof(SOCKADDR));//Socket地址绑定
if(SOCKET_ERROR==retval)
{
closesocket(m_socket);
MessageBox(_T("绑定失败!"));
return FALSE;
}
//listen(m_socket,50);


p_socket=socket(AF_INET,SOCK_DGRAM,0);
if(INVALID_SOCKET==p_socket)
{
MessageBox(_T("套接字创建失败!"));
return FALSE;
}


SOCKADDR_IN addrSock1;
addrSock1.sin_family=AF_INET;
addrSock1.sin_port=htons(7300);
addrSock1.sin_addr.S_un.S_addr=htonl(INADDR_ANY);
int retval1;
retval1=bind(p_socket,(SOCKADDR*)&addrSock1,sizeof(SOCKADDR));//Socket地址绑定
if(SOCKET_ERROR==retval1)
{
closesocket(p_socket);
MessageBox(_T("绑定失败!"));
return FALSE;
}
    listen(p_socket,50);
return TRUE;
}


DWORD WINAPI CSever3Dlg::RecvProc(LPVOID lpParameter)
{
SOCKET sock=((RECVPARAM*)lpParameter)->sock;
HWND hwnd=((RECVPARAM*)lpParameter)->hwnd;
delete lpParameter; //视频讲述时,遗忘了释放内存的操作。sunxin


SOCKADDR_IN addrFrom;
int len=sizeof(SOCKADDR);


char recvBuf[200];
//char tempBuf[300];
int retval;


while(TRUE)
{
retval=recvfrom(sock,recvBuf,200,0,(SOCKADDR*)&addrFrom,&len);
if(SOCKET_ERROR==retval)
break;


//sprintf(tempBuf,"%s说: %s",inet_ntoa(addrFrom.sin_addr),recvBuf);
//sprintf(tempBuf,"%s",inet_ntoa(addrFrom.sin_addr));





::PostMessage(hwnd,WM_RECVDATA,0,(LPARAM)recvBuf);
}
return 0;
}


BOOL CSever3Dlg::OnSend(DWORD dwIP,CString strSend)
{
SOCKADDR_IN addrTo;
addrTo.sin_family=AF_INET;
addrTo.sin_port=htons(7300);
addrTo.sin_addr.S_un.S_addr=dwIP;// 此IP为接收端所在主机的IP
    char  sendBuf[200]; 
int nLen = WideCharToMultiByte(CP_ACP, 0, strSend, -1, NULL, 0,NULL,NULL);  //cchMultiByte:指定由参数lpMultiByteStr指向的缓冲区最大值(用字节来计量)。若此值为零,函数返回lpMultiByteStr指向的目标缓冲区所必需的字节数,在这种情况下,lpMultiByteStr参数通常为NULL。 
    WideCharToMultiByte(CP_ACP, 0, strSend, -1, sendBuf, nLen,NULL,NULL);
sendto(p_socket,sendBuf,strSend.GetLength()+1,0,
(SOCKADDR*)&addrTo,sizeof(SOCKADDR));
return true;
}


LRESULT CSever3Dlg::OnRecvData(WPARAM wparam,LPARAM lparam)        //处理终端发送过来的数据
{
     char *temp= (char *)lparam;
char RecvPara[200];
 
 for(int i=0;*(temp+i);i++)
{
    RecvPara[i]=*(temp+i);
}
 
if(RecvPara[0]=='0'&& RecvPara[7]=='a')
 {
MessageBox(_T("data success")); //终端接收数据校验位正确
    switch(RecvPara[1])
    { 
          case '1': 
             for(int i=8;*(temp+i);i++)
{
   IP_Client[i-8]=*(temp+i);
}
clientIP = IP_Client;        //服务器端获取客户端IP地址
break;
 case '2':
break;
 default:
break;
    }
 }
else
    MessageBox(_T("data false"));   //终端接收数据校验位有误
 


 
     
return 0;
}
CString CSever3Dlg::GetLocalIP()
{
// 获得本机主机名

CString m_strIP,IP2;
char hostname[MAX_PATH] = {0};
gethostname(hostname,MAX_PATH);                
struct hostent FAR* lpHostEnt = gethostbyname(hostname);
if(lpHostEnt == NULL)
{
return 0;
}


// 取得IP地址列表中的第一个为返回的IP(因为一台主机可能会绑定多个IP)
LPSTR lpAddr = lpHostEnt->h_addr_list[0];      


// 将IP地址转化成字符串形式
struct in_addr inAddr;
memmove(&inAddr,lpAddr,4);
m_strIP = CString( inet_ntoa(inAddr) );        
    
return m_strIP;

}


void CSever3Dlg::OnBnClickedButton1()
{
// TODO: 在此添加控件通知处理程序代码
DWORD IP=inet_addr(IP_Client);       //inet_addr需要char *类型,而不是CString

CString strsend;
strsend="0bbbbbbbGPB.7.1";
//DWORD IP=inet_addr("192.168.137.126");
OnSend(IP,strsend);
}


void CSever3Dlg::OnBnClickedButton2()
{
// TODO: 在此添加控件通知处理程序代码


}


void CSever3Dlg::OnEnChangeEdit1()
{
// TODO:  如果该控件是 RICHEDIT 控件,则它将不会
// 发送该通知,除非重写 CDialog::OnInitDialog()
// 函数并调用 CRichEditCtrl().SetEventMask(),
// 同时将 ENM_CHANGE 标志“或”运算到掩码中。


// TODO:  在此添加控件通知处理程序代码
}

客户端:

#include "stdafx.h"
#include "CEAPP1.h"
#include "CEAPP1Dlg.h"
#include "winsock2.h"
#include "winnls.h"
#include "GPIO_Driver.h"


#ifdef _DEBUG
#define new DEBUG_NEW
#endif






#define Char2WChar(lpMultiByteStr,cbMultiByte,lpWideCharStr,cbWideChar) \
MultiByteToWideChar(CP_ACP,0,lpMultiByteStr,cbMultiByte,lpWideCharStr,cbWideChar)


#define WChar2Char(lpWideCharStr,cbWideChar,lpMultiByteStr,cbMultiByte) \
WideCharToMultiByte(CP_ACP,0,lpWideCharStr,cbWideChar,lpMultiByteStr,cbMultiByte,NULL,NULL)//用于CString转换为char[]


DWORD RData,BUFCON,BUFOUT,BUFIN;
struct GPIO
      {
         DWORD GPIOCON;
         DWORD GPIOIN;
DWORD GPIOOUT;
       };
 struct GPIO  //GPA={gpacon,NULL,gpaout },
         GPB={gpbcon,gpbin,gpbout },
 GPC={gpccon,gpcin,gpcout },
 GPD={gpdcon,gpdin,gpdout },
 GPE={gpecon,gpein,gpeout },
         GPF={gpfcon,gpfin,gpfout },
 GPG={gpgcon,gpgin,gpgout },
 GPH={gphcon,gphin,gphout },
 GPJ={gpjcon,gpjin,gpjout };      //define and choose GPIO








// CCEAPP1Dlg 对话框


CCEAPP1Dlg::CCEAPP1Dlg(CWnd* pParent /*=NULL*/)
: CDialog(CCEAPP1Dlg::IDD, pParent)
{
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}


void CCEAPP1Dlg::DoDataExchange(CDataExchange* pDX)
{
CDialog::DoDataExchange(pDX);
}


BEGIN_MESSAGE_MAP(CCEAPP1Dlg, CDialog)
#if defined(_DEVICE_RESOLUTION_AWARE) && !defined(WIN32_PLATFORM_WFSP)
ON_WM_SIZE()
#endif
//}}AFX_MSG_MAP
ON_MESSAGE(WM_RECVDATA,OnRecvData)
ON_BN_CLICKED(IDC_SEND_IP, &CCEAPP1Dlg::OnBnClickedSendIp)
ON_BN_CLICKED(IDC_SEND_DATA, &CCEAPP1Dlg::OnBnClickedSendData)

END_MESSAGE_MAP()




// CCEAPP1Dlg 消息处理程序


BOOL CCEAPP1Dlg::OnInitDialog()
{
CDialog::OnInitDialog();


// 设置此对话框的图标。当应用程序主窗口不是对话框时,框架将自动
//  执行此操作
SetIcon(m_hIcon, TRUE); // 设置大图标
SetIcon(m_hIcon, FALSE); // 设置小图标


// TODO: 在此添加额外的初始化代码
gpiodriver=CreateFile(L"GIO1:",GENERIC_READ | GENERIC_WRITE, 0,NULL,OPEN_EXISTING,0,NULL );//gpiodriver init

if(!gpiodriver)
MessageBox(L"打开GPIO设备失败!");   


InitSocket();
RECVPARAM *pRecvParam=new RECVPARAM;
pRecvParam->sock=P_socket;
pRecvParam->hwnd=m_hWnd;
HANDLE hThread1=CreateThread(NULL,0,RecvProc,(LPVOID)pRecvParam,0,NULL);    //创建socket信号接收进程
CloseHandle(hThread1);  
SetDlgItemText(IDC_SHOW_IP,GetLocalIP());



return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE
}


#if defined(_DEVICE_RESOLUTION_AWARE) && !defined(WIN32_PLATFORM_WFSP)
void CCEAPP1Dlg::OnSize(UINT /*nType*/, int /*cx*/, int /*cy*/)
{
if (AfxIsDRAEnabled())
{
DRA::RelayoutDialog(
AfxGetResourceHandle(), 
this->m_hWnd, 
DRA::GetDisplayMode() != DRA::Portrait ? 
MAKEINTRESOURCE(IDD_CEAPP1_DIALOG_WIDE) : 
MAKEINTRESOURCE(IDD_CEAPP1_DIALOG));
}
}
#endif


bool WINAPI CCEAPP1Dlg::InitSocket(void)
{
S_socket=socket(AF_INET,SOCK_DGRAM,0);
if(INVALID_SOCKET==S_socket)
{
MessageBox(TEXT("Socket creation failed!"));
return FALSE;
}
SOCKADDR_IN addrSock;
addrSock.sin_family=AF_INET;
addrSock.sin_port=htons(7000);
addrSock.sin_addr.S_un.S_addr=htonl(INADDR_ANY);


int retval;
retval=bind(S_socket,(SOCKADDR*)&addrSock,sizeof(SOCKADDR));
if(SOCKET_ERROR==retval)
{
closesocket(S_socket);
MessageBox(TEXT("Binding failed!"));
return FALSE;
}

P_socket=socket(AF_INET,SOCK_DGRAM,0);
if(INVALID_SOCKET==P_socket)
{
MessageBox(TEXT("Socket creation failed!"));
return FALSE;
}
SOCKADDR_IN addrSock1;
addrSock1.sin_family=AF_INET;
addrSock1.sin_port=htons(7300);
addrSock1.sin_addr.S_un.S_addr=htonl(INADDR_ANY);


int retval1;
retval1=bind(P_socket,(SOCKADDR*)&addrSock1,sizeof(SOCKADDR));
if(SOCKET_ERROR==retval1)
{
closesocket(P_socket);
MessageBox(TEXT("Binding failed!"));
return FALSE;
}
listen(P_socket,50);
return TRUE;
}


bool CCEAPP1Dlg::OnSend(DWORD dwIP,CString strSend)                                                //自定义的函数
{
SOCKADDR_IN addrTo;
addrTo.sin_family=AF_INET;
addrTo.sin_port=htons(7000);
addrTo.sin_addr.S_un.S_addr=dwIP;// 此IP为接收端所在主机的IP
    char  sendBuf[200]; 
int nLen = WideCharToMultiByte(CP_ACP, 0, strSend, -1, NULL, 0,NULL,NULL);  //cchMultiByte:指定由参数lpMultiByteStr指向的缓冲区最大值(用字节来计量)。若此值为零,函数返回lpMultiByteStr指向的目标缓冲区所必需的字节数,在这种情况下,lpMultiByteStr参数通常为NULL。 
    WideCharToMultiByte(CP_ACP, 0, strSend, -1, sendBuf, nLen,NULL,NULL);
sendto(S_socket,sendBuf,strSend.GetLength()+1,0,
(SOCKADDR*)&addrTo,sizeof(SOCKADDR));
return true;
}




DWORD WINAPI CCEAPP1Dlg::RecvProc(LPVOID lpParameter)
{
SOCKET sock=((RECVPARAM*)lpParameter)->sock;
HWND hwnd=((RECVPARAM*)lpParameter)->hwnd;
delete lpParameter; //视频讲述时,遗忘了释放内存的操作。sunxin


SOCKADDR_IN addrFrom1;
int len=sizeof(SOCKADDR);
    char recvBuf[200];

//char tempBuf[300];
int retval;


while(TRUE)
{
retval=recvfrom(sock,recvBuf,20,0,(SOCKADDR*)&addrFrom1,&len);
if(SOCKET_ERROR==retval)
break;


//sprintf(tempBuf,"%s说: %s",inet_ntoa(addrFrom.sin_addr),recvBuf);
//sprintf(tempBuf,"%s",inet_ntoa(addrFrom.sin_addr));





::PostMessage(hwnd,WM_RECVDATA,0,(LPARAM)recvBuf);
}
return 0;                                                                                         //wm recvdata是自定义的
}


LRESULT CCEAPP1Dlg::OnRecvData(WPARAM wparam,LPARAM lparam)        //处理终端发送过来的数据
{
     
char *temp= (char *)lparam;
char RecvPara[200];
 
 for(int i=0;i<200;i++)
{
    RecvPara[i]=*(temp+i);
}
 
if(RecvPara[0]==0 && RecvPara[7]=='b' )
{
MessageBox(_T("data success")); //终端接收数据校验位正确
         GPIOOUTPUT1(GPB,5,0);
         GPIOOUTPUT1(GPB,6,0);
}   
 
//else
   // MessageBox(_T("data false"));   //终端接收数据校验位有误
 


 
     
return 0;

}




void CCEAPP1Dlg::OnBnClickedSendIp()
{
// TODO: 在此添加控件通知处理程序代码
int i,j;
char contrl[8] ="01aaaaa";          //!!!第7位为‘\0’结束符号,会代入strsend中,必须将此位用有效值替代,否则sendstr默认到此结束,致使第7位以后数据在转移过程中丢失!!!
char data_ip[20],data_ServerIP[20];
CString IPstr =GetLocalIP();
//data_ip =IPstr.GetBuffer(IPstr.GetLength());
WChar2Char(IPstr,-1,data_ip,20);
char sendstr[200];
    for(int i=0;i<8;i++)
{
  sendstr[i]=contrl[i];
}
sendstr[7] ='a';
for(int i=8;i<200;i++)
{
  sendstr[i]=data_ip[i-8];
}
strsend1 = sendstr;          //将校验码及客户端IP信息打包


for(i=0,j=0;j<3;i++)
{
   data_ServerIP[i] = data_ip[i];
if(data_ServerIP[i] == '.')
j++;
}
data_ServerIP[i] = '1';
    
//DWORD IP=inet_addr("121.249.4.238");
    DWORD IP=inet_addr(data_ServerIP);
OnSend(IP,strsend1);


}


void CCEAPP1Dlg::OnBnClickedSendData()
{
// TODO: 在此添加控件通知处理程序代码
}
CString CCEAPP1Dlg::GetLocalIP()
{
// 获得本机主机名

CString m_strIP,IP2;
char hostname[MAX_PATH] = {0};
gethostname(hostname,MAX_PATH);                
struct hostent FAR* lpHostEnt = gethostbyname(hostname);
if(lpHostEnt == NULL)
{
return 0;
}


// 取得IP地址列表中的第一个为返回的IP(因为一台主机可能会绑定多个IP)
LPSTR lpAddr = lpHostEnt->h_addr_list[0];      


// 将IP地址转化成字符串形式
struct in_addr inAddr;
memmove(&inAddr,lpAddr,4);
m_strIP = CString( inet_ntoa(inAddr) );        
    
return m_strIP;

}
BYTE CCEAPP1Dlg::GPIOINPUT1(struct GPIO GP,USHORT NUM )      //the function is used for GPIO INPUT BETTER
 {
    

BUFCON &=~(3<<(2*NUM));
    DeviceIoControl(gpiodriver,GP.GPIOCON,&BUFCON,4,NULL,0,NULL,NULL); 
    DeviceIoControl(gpiodriver,GP.GPIOIN,NULL,0,&BUFOUT,4,NULL,NULL); 
    memcpy(&RData,&BUFOUT,4);
if(RData&(1<<NUM))
  return 1;
else
  return 0;


 }


VOID CCEAPP1Dlg::GPIOOUTPUT1(struct GPIO GP,USHORT NUM,BYTE PINSTATE )   //the function is used for GPIO OUTPUT better
{
    
    
if(GP.GPIOIN!=NULL)
  BUFCON=(BUFCON &~(3<<(2*NUM)))|(1<<(2*NUM));
else
       BUFCON &=~(1<<NUM);


if(PINSTATE)
BUFIN |=(1<<NUM);
else 
BUFIN &=~(1<<NUM);
    DeviceIoControl(gpiodriver,GP.GPIOCON,&BUFCON,4,NULL,0,NULL,NULL); 
    DeviceIoControl(gpiodriver,GP.GPIOOUT,&BUFIN,4,NULL,0,NULL,0); 




}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值