6167
V2EX  ›  Django

如何设置 django 模型,使得其中的 b 字段的初始值是 a 字段的值,并且 b 字段可修改

  •  
  •   6167 · May 24, 2018 · 3591 views
    This topic created in 2921 days ago, the information mentioned may be changed or developed.

    代码如下 class Pool ( models.Model): amount=models.IntegerFields('数量') last=models.IntegerFields('剩余数量',default=0)

    目前用的是在模型下添加
    def save(self,*args,**kwargs):
    	self.last=self.amount
        super(Pool,self).save(*args,**kwargs)
    

    Pool 表是另外一个表的外键,再通过另一表的视图函数修改 last 字段时, 显示错误,要修改 last 必须修改 amount,求解。

    9 replies    2018-05-25 18:27:52 +08:00
    6167
        1
    6167  
    OP
       May 24, 2018
    说的好像有点复杂,简单点说就是如何设置 last 字段的默认值 default,使得 last 字段的值等于 amount 字段的值
    twor
        2
    twor  
       May 24, 2018
    last=models.IntegerFields('剩余数量',default=-1)
    if self.last == -1 :
    ----self.last = self.amount

    写完,我自己也感觉不好意思
    fcfangcc
        3
    fcfangcc  
       May 24, 2018
    flask-sqlalchemy 里面可以这么用,可以搜索下 django 能不能这么用

    <script src="https://gist.github.com/fcfangcc/23e42a52f3faa6d3926f615074064197.js"></script>
    siteshen
        4
    siteshen  
       May 25, 2018
    看了下 Django 的代码,直接设置 default 做不到,因为 default 作为 callable 调用时,不会传入任何参数。
    你的答案方案在于,没有区分 self.last 的值,是“初始值”还是“被设置的值”。
    和 @twor 答案类似,不过用了一个无意义的变量作为 sentinel value。

    class Pool():
    UNSET = object()

    last = models.IntegerFields(default=UNSET)

    def save(self):
    if self.last == self.UNSET:
    self.last = self.amount

    super(Pool, self).save()
    6167
        5
    6167  
    OP
       May 25, 2018
    @siteshen 基本理解了,不过代码里的 UNSET 变量在执行 python manage.py migrate 命令的时候无法序列化,显示错误 ValueError:Cannot Serialize:object,有没有解决办法
    jinhao7773
        6
    jinhao7773  
       May 25, 2018
    为什么一定要设置值,而且重载 save 也不是万全的,比如直接改动数据库或使用 update()修改 amount,并不会同步修改 last,其实最好就是直接让 last 为 null,写个 get_last 方法返回 self.last or self.amount 就行,不直接去读 last。
    Gothack
        7
    Gothack  
       May 25, 2018
    用 @ receiver 就可以吧
    vZexc0m
        8
    vZexc0m  
       May 25, 2018
    利用 objectManager 可以实现。
    siteshen
        9
    siteshen  
       May 25, 2018
    @6167 放里面是为了最小权限原则,不行的话放外面。下面这个是 drf_yasg.utils 里的代码,用的 is 比较,比我之前写的完善些。

    class unset(object):
    """Used as a sentinel value for function parameters not set by the caller where ``None`` would be a valid value."""
    pass


    def swagger_auto_schema(method=None, methods=None, auto_schema=unset):
    if auto_schema is not unset:
    data['auto_schema'] = auto_schema
    About   ·   Help   ·   Advertise   ·   Blog   ·   API   ·   FAQ   ·   Solana   ·   1693 Online   Highest 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 45ms · UTC 16:26 · PVG 00:26 · LAX 09:26 · JFK 12:26
    ♥ Do have faith in what you're doing.