繁体   English   中英

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

[英]Adding new python type : TypeError: can't set attributes of built-in/extension type

下面的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'>

我是否需要添加类似“ PyMemberDef”的内容? 感谢您的指导或帮助。

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

如果只想初始化实例,则应定义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,
};

如果您没有给出minmax的值,则int min = 0, max = 1;行的默认值int min = 0, max = 1; 将会被使用。

关于你对的使用问题PyMemberDef ,它是,对于类型,如int它可能是少重复不易出错使用PyTypeObject.tp_members而不是PyTypeObject.tp_getset

除了setter的typedef是:

typedef int (*setter)(PyObject *, PyObject *, void *);

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM