揭开Socket编程的面纱

对TCP/IP、UDP、Socket编程这些词你不会很陌生吧?随着网络技术的发展,这些词充斥着我们的耳朵。那么我想问:

  1. 什么是TCP/IP、UDP?

  2. Socket在哪里呢?

  3. Socket是什么呢?

  4. 你会使用它们吗?

  什么是TCP/IP、UDP?

  TCP/IP(Transmission Control Protocol/Internet Protocol)即传输控制协议/网间协议,是一个工业标准的协议集,它是为广域网(WANs)设计的。

  UDP(User Data Protocol,用户数据报协议)是与TCP相对应的协议。它是属于TCP/IP协议族中的一种。

  这里有一张图,表明了这些协议的关系。                                                                               

                                                                        图1

       TCP/IP协议族包括运输层、网络层、链路层。现在你知道TCP/IP与UDP的关系了吧。

  Socket在哪里呢?

  在图1中,我们没有看到Socket的影子,那么它到底在哪里呢?还是用图来说话,一目了然。

图2

  原来Socket在这里。

  Socket是什么呢?

  Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口。在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议。

  你会使用它们吗?

  前人已经给我们做了好多的事了,网络间的通信也就简单了许多,但毕竟还是有挺多工作要做的。以前听到Socket编程,觉得它是比较高深的编程知识,但是只要弄清Socket编程的工作原理,神秘的面纱也就揭开了。

  一个生活中的场景。你要打电话给一个朋友,先拨号,朋友听到电话铃声后提起电话,这时你和你的朋友就建立起了连接,就可以讲话了。等交流结束,挂断电话结束此次交谈。    生活中的场景就解释了这工作原理,也许TCP/IP协议族就是诞生于生活中,这也不一定。

      

图3

  先从服务器端说起。服务器端先初始化Socket,然后与端口绑定(bind),对端口进行监听(listen),调用accept阻塞,等待客户端连接。在这时如果有个客户端初始化一个Socket,然后连接服务器(connect),如果连接成功,这时客户端与服务器端的连接就建立了。客户端发送数据请求,服务器端接收请求并处理请求,然后把回应数据发送给客户端,客户端读取数据,最后关闭连接,一次交互结束。

  在这里我就举个简单的例子,我们走的是TCP协议这条路(见图2)。例子用MFC编写,运行的界面如下:

图4

图5

  在客户端输入服务器端的IP地址和发送的数据,然后按发送按钮,服务器端接收到数据,然后回应客户端。客户端读取回应的数据,显示在界面上。

  下面是接收数据和发送数据的函数:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
int    Receive(SOCKET fd,char *szText,int len)
{
       int cnt;
       int rc;
       cnt=len;
       while(cnt>0)
       {
              rc=recv(fd,szText,cnt,0);
              if(rc==SOCKET_ERROR)
              {
                     return -1;
             }
             if(rc==0)
                     return len-cnt;
              szText+=rc;
              cnt-=rc;
       }
       return len;
}
int Send(SOCKET fd,char *szText,int len)
{
       int cnt;
       int rc;
       cnt=len;
       while(cnt>0)
       {
              rc=send(fd,szText,cnt,0);
              if(rc==SOCKET_ERROR)
              {
                     return -1;
              }
              if(rc==0)
                     return len-cnt;
              szText+=rc;
              cnt-=rc;
       }
       return len;
}

  服务器端:

  在服务器端,主要是启动Socket和监听线程。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
#define DEFAULT_PORT      2000
void CServerDlg::OnStart()
{
       sockaddr_in local;
       DWORD dwThreadID = 0;
        
       local.sin_family=AF_INET;
       //设置的端口为DEFAULT_PORT。
       local.sin_port=htons(DEFAULT_PORT);
       //IP地址设置成INADDR_ANY,让系统自动获取本机的IP地址。
       local.sin_addr.S_un.S_addr=INADDR_ANY;
  
       //初始化Socket
       m_Listening = socket(AF_INET,SOCK_STREAM,0);
       if(m_Listening == INVALID_SOCKET)
       {
              return ;
       }
       //将本地地址绑定到所创建的套接字上
       if(bind(m_Listening,(LPSOCKADDR)&local,sizeof(local)) == SOCKET_ERROR )
       {
              closesocket(m_Listening);
              return ;
       }
       //创建监听线程,这样也能响应界面上操作。
       m_hListenThread = ::CreateThread(NULL,0,ListenThread,this,0,&dwThreadID);
       m_StartBtn.EnableWindow(FALSE);
       m_StopBtn.EnableWindow(TRUE);
}

  监听线程函数:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
DWORD WINAPI CServerDlg::ListenThread(LPVOID lpparam)
{
       CServerDlg* pDlg = (CServerDlg*)lpparam;
       if(pDlg == NULL)
              return 0;
  
       SOCKET  Listening = pDlg->m_Listening;
       //开始监听是否有客户端连接。
       if(listen(Listening,40) == SOCKET_ERROR)
       {
              return 0;
       }
       char szBuf[MAX_PATH];
       //初始化
       memset(szBuf,0,MAX_PATH);
       while(1)
       {
              SOCKET ConnectSocket;
              sockaddr_in    ClientAddr;
              int                  nLen = sizeof(sockaddr);
              //阻塞直到有客户端连接,不然多浪费CPU资源。
              ConnectSocket = accept(Listening,(sockaddr*)&ClientAddr,&nLen);
              //都到客户端的IP地址。
              char *pAddrname = inet_ntoa(ClientAddr.sin_addr);
              pDlg->Receive(ConnectSocket,szBuf,100);
              //界面上显示请求数据。
              pDlg->SetRequestText(szBuf);
              strcat(szBuf," :我是老猫,收到(");
              strcat(szBuf,pAddrname);
              strcat(szBuf,")");
              //向客户端发送回应数据
              pDlg->Send(ConnectSocket,szBuf,100);
       }
       return 0;
}

  服务器端一直在监听是否有客户端连接,如有连接,处理客户端的请求,给出回应,然后继续监听。

  客户端:

  客户端的发送函数:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
#define DEFAULT_PORT      2000
void CClientDlg::OnSend()
{
       DWORD dwIP = 0;      
       TCHAR szText[MAX_PATH];
       memset(szText,0,MAX_PATH);
       m_IP.GetWindowText(szText,MAX_PATH);
       //把字符串形式的IP地址转成IN_ADDR结构需要的形式。
       dwIP = inet_addr(szText);
       m_RequestEdit.GetWindowText(szText,MAX_PATH);
  
       sockaddr_in local;
       SOCKET socketTmp;
       //必须是AF_INET,表示该socket在Internet域中进行通信
       local.sin_family=AF_INET;
       //端口号
       local.sin_port=htons(DEFAULT_PORT);
       //服务器的IP地址。
       local.sin_addr.S_un.S_addr=dwIP;
        
       初始化Socket
       socketTmp=socket(AF_INET,SOCK_STREAM,0);
       //连接服务器
       if(connect(socketTmp,(LPSOCKADDR)&local,sizeof(local)) < 0)
       {
              closesocket(socketTmp);
              MessageBox("连接服务器失败。");
              return ;
       }
       //发送请求,为简单只发100字节,在服务器端也规定100字节。
       Send(socketTmp,szText,100);
       //读取服务器端返回的数据。
       memset(szText,0,MAX_PATH);
       //接收服务器端的回应。
       Receive(socketTmp,szText,100);
  
       TCHAR szMessage[MAX_PATH];
       memset(szMessage,0,MAX_PATH);
       strcat(szMessage,szText);
       //界面上显示回应数据。
       m_ReplyBtn.SetWindowText(szMessage);
       closesocket(socketTmp);
}

  客户端就一个函数完成了一次通信。在这里IP地址为何用127.0.0.1呢?使用这个IP地址,服务器端和客户端就能运行在同一台机器上,这样调试方便多了。当然你可以在你朋友的机器上运行Server程序(本人在局域网中测试过),在自己的机器上运行Client程序,当然输入的IP地址就该是你朋友机器的IP地址了。

  简单的理论和实践都说了,现在Socket编程不神秘了吧?希望对你有些帮助。               

转载于:https://www.cnblogs.com/Blog-Yang/p/3332516.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/433113.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

flash地址 html,STM32 Flash读写;Flash地址对应的存储内容及方式

概念&#xff1a;计算机中最小的信息单位是bit&#xff0c;也就是一个二进制位&#xff0c;8个bit组成一个Byte&#xff0c;也就是1个字节&#xff0c;1个存储单元存放1个字节&#xff0c;每个存储单元对应一个32位(bit)地址&#xff0c;所以重要的话说三遍&#xff1a;对于32b…

socket中的函数遇见EINTR的处理

这几天&#xff0c;写服务器代码过程当中&#xff0c;遇见EINRT信号的问题&#xff0c;我是借鉴 《unp 》&#xff0c;采用continue或者goto again循环解决的。但是感觉这个还是很有必要记录一下。网络上查找到的信息很多。下面是我查找到的和EINTR有关的介绍&#xff1a;1 ht…

vue 如何获取图片的原图尺寸_公众号封面图片尺寸是多少?如何在公众号里制作封面图?...

公众号文章封面图是自己动手制作还是直接使用网络图片呢&#xff1f;在刚开始接触公众号运营时&#xff0c;我的大部分插图和封面图都是直接选择网络图片&#xff0c;后来才发现&#xff0c;自己制作的封面图更能传达文章的内容&#xff0c;阅读效果更好。其实很多做公众号的人…

主机名排序

主机名排序 主机名由多级域名组成&#xff0c;自右向左&#xff0c;依次是顶级域名、二级域名、三级域名…..以此类推 例&#xff0c;主机名&#xff1a;google.com.hkhk是顶级域名 com是二级域名 google是三级域名 现在我们需要实现一个主机名的排序功能 排序规则 1&#xff…

《计算机应用》实践考核,《管理系统中计算机应用》实践性环节考核方案

一、参考教材&#xff1a;《管理系统中计算机应用》&#xff0c;汪星明、周山芙主编&#xff0c;武汉大学出版社2004年版。二、考核时间&#xff1a;50分钟。三、考核内容及分数分布&#xff1a;1、Windows资源管理器的综合应用(10分)1)启动资源管理器。2)熟练掌握文件(夹)的建…

信号中断 与 慢系统调用

1. 术语 1.1. 慢系统调用&#xff08;Slow system call&#xff09; 该术语适用于那些可能永远阻塞的系统调用。永远阻塞的系统调用是指调用永远无法返回&#xff0c;多数网络支持函数都属于这一类。如&#xff1a;若没有客户连接到服务器上&#xff0c;那么服务器的accept调用…

delete hive_Hive高级调优

Hive调优策略Hive作为大数据领域常用的数据仓库组件&#xff0c;在设计和开发阶段需要注意效率。影响Hive效率的不仅仅是数据量过大;数据倾斜、数据冗余、job(小文件多)或I/O过多、MapReduce分配不合理等因素都对Hive的效率有影响。对Hive的调优既包含对HiveQL语句本身的优化&a…

联想计算机如何设置用户名和密码,联想电脑怎样设密码?联想电脑设置密码方法步骤【图文】...

现代 人最重要的是什么呢?在笔者看来是隐私。如今由于我们的社会比较发达&#xff0c;而信息传播的速度有非常广泛&#xff0c;加上各种隐私被盗取或者偷窥&#xff0c;让现代人的生活总是充斥着不安&#xff0c;所以我们各类的电子设备诸如手机&#xff0c;电脑加密是非常有必…

linux中errno使用

当linux中的C api函数发生异常时,一般会将errno变量(需include errno.h)赋一个整数值,不同的值表示不同的含义,可以通过查看该值推测出错的原因&#xff0c;在实际编程中用这一招解决了不少原本看来莫名其妙的问题。但是errno是一个数字&#xff0c;代表的具体含义还要到errno.…

unity3d 捕获系统日志,来处理一些问题

注册系统日志回调&#xff0c;根据日志内容和类型处理一些特殊问题 using UnityEngine; using System.Collections;public class SetupVerification : MonoBehaviour {public string message "";private bool badSetup false;void Awake (){Application.RegisterL…

em算法 实例 正态分布_EM算法解GMM

看了很多介绍EM算法的文章&#xff0c;但是他们都没有代码&#xff0c;所以在这里写出来。Jensen 不等式参考期望最大算法Jensen不等式在优化理论中大量用到&#xff0c;首先来回顾下凸函数和凹函数的定义。假设 是定义域为实数的函数&#xff0c;如果对于所有的 &#xff0c; …

html5 webview,HTML5+学习历程之webview经典案例

看了这么多app&#xff0c;其实基本布局使用的最多的无非两种&#xff0c;如下图&#xff1a;类似微信类似QQ在这里小编简单说下这两种布局简单的实现思路&#xff0c;当然如果你还有更好的方法&#xff0c;请在下面留言&#xff0c;让更多人知道你更好的方法&#xff01;第一种…

C函数的实现(strcpy,atoi,atof,itoa,reverse)

在笔试面试中经常会遇到让你实现C语言中的一些函数比如strcpy&#xff0c;atoi等 1. atoi 把字符串s转换成数字 int Atoi( char *s ) {int num 0, i 0;int sign 1;for( i0; isspace(s[i]); i );sign (s[i] -)? -1:1;if( s[i] || s[i] - )i;for( ;isdigit(s[i]); i ){n…

bbb sdk6 ll_rw_block分析

ll_rw_block是文件系统对下访问实际的块设备驱动的接口&#xff0c;应用程序对实际文件&#xff08;非设备文件&#xff09;的操作&#xff0c;最终都是 通过文件系统来调用ll_rw_block来操作实际的存储设备的。 当然ll_rw_block的实际作用远非一个接口那么简单&#xff0c;他…

wifi 小米pro 驱动 黑苹果_搞定小米黑苹果自带WIF,又可省一个USB接口了

首先声明我的是小米笔记本PRO版本的&#xff0c;其他版本的没有经过测试&#xff0c;但理论都是没有问题的&#xff0c;其他版本的朋友&#xff0c;喜欢折腾的话&#xff0c;可以试试&#xff01;自用版本关于小米笔记本安装黑苹果&#xff0c;网上一直都有很多链接&#xff0c…

教师资格证计算机考察知识点,教师资格证考试信息技术常考知识点同步练习题.docx...

教师资格证考试信息技术常考知识点同步练习题一、信息的定义及特征( 一) 信息定义信息是通过文字、数字、图像、图形、声音、视频等方式进行传播的内容。说明&#xff1a;信息定义考查的方式有两类&#xff1a;一类是选出四个选项中是信息的 ; 另一类是判断选择题&#xff0c;选…

machine learning for hacker记录(4) 智能邮箱(排序学习推荐系统)

本章是上一章邮件过滤技术的延伸&#xff0c;上一章的内容主要是过滤掉垃圾邮件&#xff0c;而这里要讲的是对那些正常的邮件是否可以加入个性化元素&#xff0c;由于每个用户关心的主题并非一样&#xff08;有人喜欢技术类型的邮件或者购物促销方便的内容邮件等&#xff09;。…

代理模式 委派模式 策略模式_策略模式

在策略模式(Strategy Pattern)中&#xff0c;一个类的行为或其算法可以在运行时更改。这种类型的设计模式属于行为型模式。在策略模式中&#xff0c;我们创建表示各种策略的对象和一个行为随着策略对象改变而改变的 context 对象。策略对象改变 context 对象的执行算法。介绍意…

云南计算机专业知识真题,2014年云南省事业单位考试专计算机专业知识模拟真题.doc...

2014年云南省事业单位考试专计算机专业知识模拟真题1 在Word中替换的快捷键是____。A、CTRLFB、CTRLHC、CTRLSD、CTRLP2 在Word中打印的快捷键是____。A、CTRLFB、CTRLHC、CTRLOD、CTRLP3 在Word中打开新文档的快捷键是____。A、CTRLFB、CTRLHC、CTRLOD、CTRLP4 在Word中&#…

bbb mmc_blk_probe 分析

bbb 的 emmc驱动在drivers\mmc\card\block.c&#xff0c;其mmc_dirver结构体如下&#xff0c; 根据以往平台总线驱动模型的经验来看的话&#xff0c;内核里应该有mmc_devices结构体&#xff0c;并且 其name也为"mmcblk"&#xff0c;这样其probe函数将被调用&#x…