我承认,我正在从事的这个项目并不是关于如何与python迈出第一步的教科书示例,但是我在这里。
设置
与RaspberryPi本地连接(以太网,插座)的ESC/POS打印机,它正在接收打印作业(通过AMQP)。
问题
连接到AMQP经纪人和打印机以及打印效果很好。我正在挣扎的是用CTYPE处理打印机状态代码。
在挖掘互联网,寻找最佳方法时,我遇到了有关位操纵的Python Wiki文章(链接),并指出建议使用CTYPES,建议使用CTYPES。它确实看起来很棒且干净,所以我立即将其应用于我的代码(以下简化版本)。
import socket
import ctypes
class PrinterStatus_bits( ctypes.LittleEndianStructure ):
_fields_ = [
("fix0", c_uint8, 1 ), # asByte & 1
("fix1", c_uint8, 1 ), # asByte & 2
("drawer", c_uint8, 1 ), # asByte & 4
("offline", c_uint8, 1 ), # asByte & 8
("fix4", c_uint8, 1 ), # asByte & 16
("recovery", c_uint8, 1 ), # asByte & 32
("feed", c_uint8, 1 ), # asByte & 64
("fix7", c_uint8, 1 ), # asByte & 128
]
class Flags( ctypes.Union ):
_anonymous_ = ("bit",)
_fields_ = [
("bit", PrinterStatus_bits),
("asByte", c_uint8 )
]
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server_address = ('123.123.123.123', 9100)
c_uint8 = ctypes.c_uint8
flags = Flags()
msg = b'x10x04x01'
print('connecting to %s port %s' % server_address)
sock.connect(server_address)
print('requesting printer status')
sock.sendall(msg)
data = sock.recv(1)
flags.asByte = ord(data)
有意识地设置1个字节接收长度,因为这是响应的固定长度。但是,事实是,这只是第一个(四个),一般状态代码。
我很想通过flags
对象获得所有这些状态代码,只是不知道如何与工会一起工作。
如果我做类似:
的事情class PrinterStatus_bits( ctypes.LittleEndianStructure ):
class PrinterOffline_bits( ctypes.LittleEndianStructure ):
class PrinterError_bits( ctypes.LittleEndianStructure ):
class PrinterPaper_bits( ctypes.LittleEndianStructure ):
class Flags( ctypes.Union ):
_anonymous_ = ("status",)
_fields_ = [
("status", PrinterStatus_bits),
("offline", PrinterOffline_bits),
("error", PrinterError_bits),
("paper", PrinterPaper_bits ),
("asByte", c_uint8 )
]
我甚至如何为相应的类分配正确的值?
P.S。所有4个状态代码始终都是1个字节长
P.P.S我不愿意使用ctypes,如果在这种情况下这不是有效的方法,我会很好,我只会循环完成所有4个响应,并将这些值分配给0/1数组,但是我的好奇心变得更好,我想找出是否可以使用这种方式
你是方式 复杂。
flags = ord(data)
drawer = (flags & 4) != 0
offline = (flags & 8) != 0
recovery = (flags & 32) != 0
feed = (flags & 64) != 0