添加新的python类型:TypeError:不能设置内置/扩展类型的属性

时间:2015-06-22 17:03:42

标签: python python-c-api python-c-extension

下面正确编译的python-c代码

#include <Python.h>
#include <structmember.h>

struct rangerr {
        long    min;
        long    max;
};

//Python type to represent rangerr
struct py_rangerr {
    PyObject_HEAD
    struct rangerr range;
};
// * get & set methods for py_rangerr

static PyObject * py_rangerr_min_get(struct py_rangerr *self) {
    self->range.min  = 1;
        return PyLong_FromLong(self->range.min);
}
static PyObject * py_rangerr_min_set(struct py_rangerr *self) {
printf("Setter called");
self->range.min  = 1;
}


static PyObject * py_rangerr_max_get(struct py_rangerr *self) {
    self->range.max  = 10;
        return PyLong_FromLong(self->range.max);
}


//* GetSet method definition for py_rangerr
static PyGetSetDef py_rangerr_getset[] = {
    {"min",(getter)py_rangerr_min_get, (setter)py_rangerr_min_set, "min",NULL},
    {"max",(getter)py_rangerr_max_get, NULL, "max",NULL},

    /* Sentinel */
    {NULL},
};

/******************************************************************************
 */
static void py_rangerr_dealloc(struct py_rangerr *self) { 
        self->ob_type->tp_free((PyObject *)self);               
}

static PyTypeObject py_rangerr_type = {  
    PyObject_HEAD_INIT(NULL)    
     .tp_name       = "rangerr",        
    .tp_basicsize = sizeof(struct py_rangerr),
    .tp_dealloc   = (destructor) py_rangerr_dealloc,  
    .tp_flags     = Py_TPFLAGS_DEFAULT,                        
    .tp_alloc     = PyType_GenericAlloc,                       
    .tp_doc       = "rangerr",                                 
    .tp_getset    = py_rangerr_getset,                
};

void
initrangerr(void)
{
   PyObject* mod;

 mod = Py_InitModule3("rangerr", NULL, "An extension with a type.");
   if (mod == NULL) {
      return;
   }
  py_rangerr_type.tp_new = PyType_GenericNew;
  if (PyType_Ready(&py_rangerr_type) < 0){
      return;
   }


 Py_INCREF(&py_rangerr_type);
   PyModule_AddObject(mod, "rangerr", (PyObject*)&py_rangerr_type);

}

但是当我尝试调用set / get方法时会抛出以下错误:

>>> import rangerr as r
>>> r.rangerr.min
<attribute 'min' of 'rangerr' objects>
>>> r.rangerr.min=2
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: can't set attributes of built-in/extension type 'rangerr'
>>> type(r.rangerr.min)
<type 'getset_descriptor'>

我是否需要添加类似&#34; PyMemberDef&#34; ?感谢您的任何指示或帮助。

1 个答案:

答案 0 :(得分:1)

不能直接为参数设置默认值。您可以向模块添加一个函数,将全局默认值设置为静态变量。

如果您只想初始化您的实例,则应定义Service,如

PyTypeObject.tp_init

如果您没有为static int py_rangerr_init(struct py_rangerr *self, PyObject *args, PyObject *kwds) { static char *kwlist[] = {"min", "max", NULL}; int min = 0, max = 1; if (!PyArg_ParseTupleAndKeywords(args, kwds, "|ii", kwlist, &min, &max)) return -1; self->range.min = min; self->range.max = max; return 0; } static PyTypeObject py_rangerr_type = { PyObject_HEAD_INIT(NULL) [...] .tp_init = (initproc) py_rangerr_init, }; min提供值,则会使用第max行中的默认值。

关于您对int min = 0, max = 1;的使用情况的问题,对于类似PyMemberDef的类型,它可能更少,重复性更低,更不容易使用int代替{ {1}}。

除了setter的typedef:

PyTypeObject.tp_members