pythonのpack unpack関数

2056 ワード

c言語はネットワーク通信で構造体データを伝送することが多く、通常はバイナリ形式であるが、pythonがcプログラムと通信するにはバイナリのデータを認識しなければならない.構造体は通常メモリの位置合わせの問題にかかわるため、c中の構造体データを認識するだけでなく、解くこともできる.
 
次の例では、pythonのstructモジュールでこの機能を実行できます.
 
python書き込みサーバ:
import socket, struct

s = socket.socket()
s.bind(('127.0.0.1', 8000))
s.listen(1)

try:
	while True:
		cli, addr = s.accept()
		data = cli.recv(100)

		print "recv %d bytes" % len(data)
		a, b, c = struct.unpack('i10sh', data)
		print a, b, c

		sdata = struct.pack('i10sh', 34, "abcdefghi\0", 65)
		cli.send(sdata)
finally:
	s.close()

 
c書き込みクライアント:
 
/* tcp_client.c */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>

typedef struct _data {
	int a;
	char b[10];
	short c;
} Data;

int main()
{
	int client_fd;
	struct sockaddr_in server_addr;

	server_addr.sin_family = AF_INET;
	server_addr.sin_port = htons(8000);
	server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
	bzero(&(server_addr.sin_zero), 8);

	client_fd = socket(AF_INET, SOCK_STREAM, 0);

	connect(client_fd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr));

	Data d;
	memset(&d, 0, sizeof(d));
	d.a = 3;
	memcpy(&d.b, "hello", 5);
	d.c = 6;
	send(client_fd, &d, sizeof(d), 0);

	char buf[200];
	bzero(buf, 200);
	recv(client_fd, buf, sizeof(buf), 0);

	Data* p = (Data*)&buf;
	printf("%d %s %d
", p->a, p->b, p->c); close(client_fd); return 0; }