什么是网络编程?
网络编程最主要的工作就是在发送端把信息通过规定好的协议进行组装包,在接收端按照规定好的协议把包进行解析,从而提取出对应的信息,达到通信的目的。
计算机与计算机之间的通信要解决那些问题?
通信协议:通信协议就是指信息的交流方式
常见的通信协议:
tcp :TCP(Transmission Control Protocol 传输控制协议)是一种面向连接的、可靠的、基于字节流的传输层通信协议,由IETF的RFC 793定义。
udp :UDP 是User Datagram Protocol的简称, 中文名是用户数据报协议,是OSI(Open System Interconnection,开放式系统互联) 参考模型中一种无连接的传输层协议,提供面向事务的简单不可靠信息传送服务,IETF RFC 768是UDP的正式规范。UDP在IP报文的协议号是17。
http:超文本传输协议(HTTP,HyperText Transfer Protocol)是互联网上应用最为广泛的一种网络协议。
ftp:FTP 是File Transfer Protocol(文件传输协议)的英文简称,而中文简称为“文传协议”。用于Internet上的控制文件的双向传输。(本地)
poo3:POP3,全名为“Post Office Protocol - Version 3”,即“邮局协议版本3”。是TCP/IP协议族中的一员,由RFC1939 定义。本协议主要用于支持使用客户端远程管理在服务器上的电子邮件。提供了SSL加密的POP3协议被称为POP3S。
如何去实现 计算机与计算机之间的通信?
首先我们要知道 那个是主叫方 那个是被叫方 然后就可以创建他们之间的通信协议了网络编程我们常用的就是socket
什么是 Socket?
Socket又称"套接字",应用程序通常通过"套接字"向网络发出请求或者应答网络请求,使主机间或者一台计算机上的进程间可以通讯。
Socket 对象(内建)方法
函数 | 描述 |
---|---|
服务器端套接字 | |
s.bind() | 绑定地址(host,port)到套接字, 在AF_INET下,以元组(host,port)的形式表示地址。 |
s.listen() | 开始TCP监听。backlog指定在拒绝连接之前,操作系统可以挂起的最大连接数量。该值至少为1,大部分应用程序设为5就可以了。 |
s.accept() | 被动接受TCP客户端连接,(阻塞式)等待连接的到来 |
客户端套接字 | |
s.connect() | 主动初始化TCP服务器连接,。一般address的格式为元组(hostname,port),如果连接出错,返回socket.error错误。 |
s.connect_ex() | connect()函数的扩展版本,出错时返回出错码,而不是抛出异常 |
公共用途的套接字函数 | |
s.recv() | 接收TCP数据,数据以字符串形式返回,bufsize指定要接收的最大数据量。flag提供有关消息的其他信息,通常可以忽略。 |
s.send() | 发送TCP数据,将string中的数据发送到连接的套接字。返回值是要发送的字节数量,该数量可能小于string的字节大小。 |
s.sendall() | 完整发送TCP数据,完整发送TCP数据。将string中的数据发送到连接的套接字,但在返回之前会尝试发送所有数据。成功返回None,失败则抛出异常。 |
s.recvform() | 接收UDP数据,与recv()类似,但返回值是(data,address)。其中data是包含接收数据的字符串,address是发送数据的套接字地址。 |
s.sendto() | 发送UDP数据,将数据发送到套接字,address是形式为(ipaddr,port)的元组,指定远程地址。返回值是发送的字节数。 |
s.close() | 关闭套接字 |
s.getpeername() | 返回连接套接字的远程地址。返回值通常是元组(ipaddr,port)。 |
s.getsockname() | 返回套接字自己的地址。通常是一个元组(ipaddr,port) |
s.setsockopt(level,optname,value) | 设置给定套接字选项的值。 |
s.getsockopt(level,optname[.buflen]) | 返回套接字选项的值。 |
s.settimeout(timeout) | 设置套接字操作的超时期,timeout是一个浮点数,单位是秒。值为None表示没有超时期。一般,超时期应该在刚创建套接字时设置,因为它们可能用于连接的操作(如connect()) |
s.gettimeout() | 返回当前超时期的值,单位是秒,如果没有设置超时期,则返回None。 |
s.fileno() | 返回套接字的文件描述符。 |
s.setblocking(flag) | 如果flag为0,则将套接字设为非阻塞模式,否则将套接字设为阻塞模式(默认值)。非阻塞模式下,如果调用recv()没有发现任何数据,或send()调用无法立即发送数据,那么将引起socket.error异常。 |
s.makefile() | 创建一个与该套接字相关连的文件 |
被叫方:
import socket
import threading #创建服务套接字 serversockt ss=socket.socket(socket.AF_INET,socket.SOCK_STREAM) print("建立服务套接字") #服务套接字 绑定端口号,绑定主机 ss.bind(("localhost",9999)) print("绑定端口和主机") ss.listen(5) #接受客户套接字请求 print("开始等待客户的请求") c=ss.accept();#线程阻塞 print("某个客户连接了我") # msg=c[0].recv(1024) # print(msg.decode())#接受的数据要解码 # c[0].send("要约么?".encode())def myrecv(c): while True: msg=c[0].recv(1024) print(msg.decode()) threading._start_new_thread(myrecv,(c,)) while True: msg=input() c[0].send(msg.encode())
主叫方:
import socket import threading #建立客户套接字 c=socket.socket(socket.AF_INET,socket.SOCK_STREAM) print("创建客户套接字") c.connect(("localhost",9999))#连接对方,不成功抛出异常 print("连接成功") #send 发送的数据要编码 # c.send("黄震".encode()) # msg=c.recv(1024) # print(msg.decode()) def myrecv(c): while True: msg=c.recv(1024) print(msg.decode()) threading._start_new_thread(myrecv,(c,)) while True: msg=input() c.send(msg.encode())
以上的代码是实现两个控制台之间聊天的,当然如果要和别人连接的话, 需要主叫方连接到被叫方的地址localhost,并且保证在同一个局域网,以上代码是我在我的电脑上实现的所以我都是用的本机的电脑地址
当然 如果我们要实现多人通信的话,我们需要去创建一个第三方的服务,在这个第三方的服务中呢我们就可以吧每个人发来的信息转发个其他的人了,我们这个第三方的服务需要用到线程 实例代码如下:
''' 设计多人聊天转发消息的服务 要有两个线程 一个在收消息 一个在把收到的消息发出去 ''' import socket import threading ss=socket.socket(socket.AF_INET,socket.SOCK_STREAM) ss.bind(("0.0.0.0",9999)) ss.listen(10) msg=None lock=threading.Lock() con=threading.Condition(lock) def server_recv(c,a): global msg while True: msg1=c.recv(3000) con.acquire() msg=msg1 msg=str(a)+msg1.decode() con.notify_all() con.release() def server_send(c,a): global msg while True: con.acquire() con.wait() con.release() c.send(msg.encode()) while True: c,a=ss.accept() threading._start_new_thread(server_recv,(c,a)) threading._start_new_thread(server_send,(c,a))
当然如果我们让其他的人都可以连接到我们呢就需要把我们的主机的地址改成我0.0.0.0,这样的权限就可以去让其他的人去连接了,当然我这只是简单的实例,一些socket的用法 用兴趣的可以去深入的了解下。