免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 5510 | 回复: 19
打印 上一主题 下一主题

关于struct的一个问题,很紧急,谢谢 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2008-03-21 09:17 |只看该作者 |倒序浏览
最近需要写一个网络程序,其中网络传输部分采用了udt的代码,我把udt的c++库用swig封装成了pyd库文件,可以调用,但udt库中有一个函数,
udt.bind(UDTSOCKET u,struct  sockaddr* name, int* namelen)
这个函数需要一个struct类型的sockaddr指针。

其中sockaddr的定义:
struct sockaddr {
        u_short sa_family;              /* address family */
        char    sa_data[14];            /* up to 14 bytes of direct address */
};

广泛采用的等效定义:
//Socket address, internet style.
struct sockaddr_in {
        short   sin_family;
        u_short sin_port;
        struct  in_addr sin_addr;
        char    sin_zero[8];
};

//Internet address (old style... should be updated)
struct in_addr {
        uint32_t s_addr;
};


请问:我怎么用python的struct模块把sockaddr_in 做出来?
谢谢,很紧急,忙了两天了。

论坛徽章:
0
2 [报告]
发表于 2008-03-21 09:35 |只看该作者
在线等,等高手指点迷津

论坛徽章:
0
3 [报告]
发表于 2008-03-21 10:00 |只看该作者
继续等,怎么没有人答复,给点儿线索也行啊

论坛徽章:
0
4 [报告]
发表于 2008-03-21 11:28 |只看该作者
等待中

论坛徽章:
0
5 [报告]
发表于 2008-03-21 12:39 |只看该作者
用python的ctypes模块,可以作出来。
import ctypes
class sockaddr(ctypes.Structure):
    """
struct sockaddr {
        u_short sa_family;              /* address family */
        char    sa_data[14];            /* up to 14 bytes of direct address */
};

    """
    _fields_ = [
        ('sa_family', ctypes.c_ushort),
        ('sa_data', ctypes.c_char * 14),
        ]

name = sockaddr()

#调用函数时需要用ctypes.byref,如:
udt.bind(u,ctypes.byref(name), namelen)

对于想和外部dll交互,使用ctypes模块是一个不错的选择。
它还提供了很多其他功能,如:
from_address
addressof
...
不过我在用ctypes模块调用一个dll,而这个dll又有使用py模块时,出现了个莫名其妙的问题。

[ 本帖最后由 seewind 于 2008-3-21 12:43 编辑 ]

论坛徽章:
0
6 [报告]
发表于 2008-03-21 12:54 |只看该作者
谢谢楼上的指点,太感谢了,原来ctypes模块这么方便

论坛徽章:
0
7 [报告]
发表于 2008-03-21 13:53 |只看该作者
还是有问题,


class sockaddr(ctypes.Structure):
    _fields_=[("sa_family",ctypes.c_ushort),
              ("sa_data",ctypes.c_char*14),
              ]
   
class in_addr(ctypes.Structure):
    _fields_=[("s_addr",ctypes.c_uint32)]

class sockaddr_in(ctypes.Structure):
    _fields_=[("sin_family",ctypes.c_short),
              ("sin_port",ctypes.c_ushort),
              ("sin_addr",in_addr),
              ("sinzero",ctypes.c_char*,
              ]

my_in_addr=in_addr(struct.unpack('I',socket.inet_aton(IP))[0])
myaddr=sockaddr_in(AF_INET,
                   socket.htons(Port),
                   my_in_addr,
                   '\x00\x00\x00\x00\x00\x00\x00\x00',
                   )
myaddr_p=ctypes.pointer(myaddr)

mysocket=udt.socket(AF_INET,SOCK_STREAM,0)
udt.bind(mysocket,myaddr_p,ctypes.sizeof(myaddr))


系统显示:
Traceback (most recent call last):
  File "C:\yuan\work\udt-dll\test_server.py", line 103, in <module>
    udt.bind(mysocket,myaddr_p,ctypes.sizeof(myaddr))
TypeError: in method 'bind', argument 2 of type 'sockaddr const *'
>>>

论坛徽章:
0
8 [报告]
发表于 2008-03-21 14:41 |只看该作者
udt.bind(mysocket,myaddr_p,ctypes.sizeof(myaddr))
打错字了吧?是byref

论坛徽章:
0
9 [报告]
发表于 2008-03-21 14:44 |只看该作者
byref跟pointer效果一样,我都试过了,一样的错误

论坛徽章:
0
10 [报告]
发表于 2008-03-21 14:51 |只看该作者
主要错误原因就是,myaddr是sockaddr_in格式的,不是sockaddr格式,这两种格式内存长度是一样的,sockaddr很难赋值,所以c程序当中都是用sockaddr_in来构造,然后等调用函数的时候,再强制类型转化

附:c语言的语句
int main()
{
UDTSOCKET serv = UDT::socket(AF_INET, SOCK_STREAM, 0);

sockaddr_in my_addr;
my_addr.sin_family = AF_INET;
my_addr.sin_port = htons(9000);
my_addr.sin_addr.s_addr = INADDR_ANY;
memset(&(my_addr.sin_zero), '\0', ;

if (UDT::ERROR == UDT::bind(serv, (sockaddr*)&my_addr, sizeof(my_addr)))
{
  cout << "bind: " << UDT::getlasterror().getErrorMessage();
  return 0;
}
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP