Python学习笔记八

类的高级用法

多态:  

     在其他语言,使用的是类的继承。

    在python中,不需要指定数据类型。

基于TCP协议的socket通信实现:

  类似于打电话的情景。

  服务端: 

  1.买手机

  2.插卡

  3.开机

  4.等待电话链接

  5.收消息

  6.发消息

  7.挂电话

  8.关手机

  具体代码实现如下:

import socket #导入一个socket模块
server1=socket.socket(socket.AF_INET,socket.SOCK_STREAM)    #买手机,基于socket的网络通信,使用的是TCP协议
server1.bind(('127.0.0.1',8080))   #插卡,指定服务器的IP地址和端口号
server1.listen(5)      #开机,设置监听等待数
conn,client_addr=server1.accept()      #等待电话链接
while True:
    recv_data=conn.recv(1024)             #收消息,限制单个消息的最大数为1024字节
    print(recv_data)
    conn.send(recv_data.upper())          #发消息

conn.close()       #挂手机
server1.close()         #关机

  客户端:

  1.买手机

  2.打电话

  3.发消息

  4.收消息

  具体代码实现:

import socket   
client=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
client.connect(('127.0.0.1',8080))      #打电话

while True:
    message=input('数据>>').strip()    
    client.send(message.encode('utf-8'))    #发消息
    recv=client.recv(1024)          #收消息
    print(recv)

  上面的服务端和客户端都添加了while True,实现了通信循环。

  但是现在有个问题:就是如果客户端输入为空,客户端会卡住。

  分析这个问题:

  要么就是客户端没有发送出去,要么就是客户端没有收到。

  结果:因为客户端发送为空的消息,但是服务器并没有收到该消息,所以客户端也不会收到空消息,所以会被卡住。

     所以需要在客户端上设置,用户不可以发送为空的消息。

  修改后的代码:

  

import socket
client=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
client.connect(('127.0.0.1',8080))      #打电话

while True:
    message=input('数据>>').strip()
    if not message:continue
    client.send(message.encode('utf-8'))    #发消息
    recv=client.recv(1024)          #收消息
    print(recv)

 

加上链接循环:

import socket #导入一个socket模块
server1=socket.socket(socket.AF_INET,socket.SOCK_STREAM)    #买手机,基于socket的网络通信,使用的是TCP协议
server1.bind(('127.0.0.1',8080))   #插卡,指定服务器的IP地址和端口号
server1.listen(5)      #开机,设置监听等待数
while True:
    conn,client_addr=server1.accept()      #等待电话链接
    while True:
        try:
            recv_data=conn.recv(1024)        #收消息,限制单个消息的最大数为1024字节
       if not recv_date:break
print(recv_data) conn.send(recv_data.upper()) #发消息 except ConnectionResetError: break conn.close() #挂手机 server1.close() #关机
前提是:客户端单方面断开链接
如果服务端是windows系统,就添加一个try错误处理,如果服务端是linux,就不会有异常,服务端一直在接受空,所以需要加一个判断,如果为空就退出通信循环

 命令的结果:如何获取

  os.system('命令'): 这种方法只是执行了一下命令,并返回一个值。可以通过这个值,判断命令是否执行成功。

  subprocess.Popen('命令',shell=True):这个方法是执行的结果是内存地址。它可以指定执行的结果放到某个位置。

  subprocess.Popen('命令',shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)

import subprocess
res=subprocess.Popen('dir',shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)
res_stdout=res.stdout.read()
res_stderr=res.stderr.read()
print(res_stdout.decode('gbk'))
res_stdout1=res.stdout.read()
print('jieguo',res_stdout1.decode('gbk'))
#管道读取一次以后,管道就为空了。因为取走了。
结果如下:

C:\Users\yangjianbo\AppData\Local\Programs\Python\Python36\python.exe C:/Users/yangjianbo/PycharmProjects/untitled/第八课面向对象高级/subprocess模块.py
驱动器 C 中的卷是 Windows
卷的序列号是 58DC-A9CB


C:\Users\yangjianbo\PycharmProjects\untitled\第八课面向对象高级 的目录


2017/11/08 22:10 <DIR> .
2017/11/08 22:10 <DIR> ..
2017/11/05 12:25 19 b.py
2017/11/08 21:35 355 socket客户端.py
2017/11/08 21:35 764 socket服务器.py
2017/11/08 22:10 417 subprocess模块.py
2017/11/05 12:10 0 __init__.py
2017/11/08 21:50 <DIR> __pycache__
2017/11/07 23:32 352 客户端2.py
2017/11/07 23:32 352 客户端3.py
2017/11/07 23:32 352 客户端4.py
2017/11/07 23:32 352 客户端5.py
2017/11/07 23:32 352 客户端6.py
2017/11/07 23:34 352 客户端7.py
11 个文件 3,667 字节
3 个目录 458,471,485,440 可用字节

第二次读取管道的命令结果==========》

可以看到结果:第一次的结果是正确执行命令后的结果,第二次的命令执行结果就为空了。

   


命令执行错误的结果,会放到stderr的管道中。
import subprocess
res=subprocess.Popen('啊啊啊',shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)
res_stdout=res.stdout.read()
res_stderr=res.stderr.read()
print(res_stdout.decode('gbk'))
res_stdout1=res.stdout.read()
print('第二次读取管道的命令结果==========》',res_stdout1.decode('gbk'))
print(res_stderr.decode('gbk'))
结果:

C:\Users\yangjianbo\AppData\Local\Programs\Python\Python36\python.exe C:/Users/yangjianbo/PycharmProjects/untitled/第八课面向对象高级/subprocess模块.py


第二次读取管道的命令结果==========》
'啊啊啊' 不是内部或外部命令,也不是可运行的程序
或批处理文件。

stdin参数:用来把第一个命令执行的结果输入到第二个命令中。  

 

简单SSH程序:

服务端代码:

import socket #导入一个socket模块
import subprocess
server1=socket.socket(socket.AF_INET,socket.SOCK_STREAM)    #买手机,基于socket的网络通信,使用的是TCP协议
server1.bind(('127.0.0.1',8080))   #插卡,指定服务器的IP地址和端口号
server1.listen(5)      #开机,设置监听等待数
while True:
    conn,client_addr=server1.accept()      #等待电话链接
    while True:
        try:
            cmd=conn.recv(1024)        #收消息,限制单个消息的最大数为1024字节
            if not cmd :break
            cmd=cmd.decode('utf-8')
            cmd_res=subprocess.Popen(cmd,shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)
            stdout_res=cmd_res.stdout.read()
            stderr_res=cmd_res.stderr.read()   # 返回的结果就是bytes类型,所以不需要再decode了。
            conn.send(stdout_res+stderr_res)          #发消息
        except ConnectionResetError:
            break
    conn.close()       #挂手机
server1.close()         #关机

客户端代码:

import socket
client=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
client.connect(('127.0.0.1',8080))      #打电话

while True:
    cmd=input('数据>>').strip()
    if not cmd:continue
    client.send(cmd.encode('utf-8'))    #发消息
    recv=client.recv(1024).decode('gbk')       #收消息    因为输出在windows系统下,所以客户端的编码应该为gbk.
   print(recv) client.close()

 

关于SSH远程程序执行cd命令的问题:

  以一个变量为标准,当执行cd命令的时候,把这个变量的值加上你要切换的目录,这才是整个完整的路径。

关于SSH远程程序执行top命令的问题:

  时时刷新top命令,其实是服务端在定期刷新一下,返回结果给客户端。不可能实现实时刷新的效果,肯定是在定期刷新的。

 

1024: 收包的大小限制,最大为1024字节。只是限制收消息的大小。这个值最大设置为8096,再大就没有必要了。

解决方法:

  把客户端接收的包大小调大,但是这种方法不合适,因为无法预测服务端返回的结果大小。

  而且收到的数据都存放在内存中,太消耗资源了。

 

TCP粘包现象:

  TCP为了避免网络中的大量小包,会使用一个nagle算法,将包粘在一起发送。数据量小,时间间隔小,TCP会发生粘包现象。

  为了解决粘包的问题,协议会采用一种固定的方式:报头+数据

struct模块:

  把数字转换成bytes类型。

  struct.pack('i',后面的数字)   i代表打包后的结果是4个bytes,打包的数字就是整型数字。

  struct.unpack('i',后面的数字)

最终版:

  制作报头

    header_dic={}

  序列化:

    打包报头

  把报头大小打包成固定长度。

  

 

  

 

 

 

 

  

  

 

 

  

  

 

 

 

 

  

 

posted @ 2017-11-05 09:50  奋斗史  阅读(156)  评论(0编辑  收藏  举报