odoo 开发入门教程系列-模型和基本字段
模型和基本字段
在上一章的末尾,我们创建一个odoo模块。然而,此时它仍然是一个空壳,不允许我们存储任何数据。在我们的房地产模块中,我们希望将与房地产相关的信息(名称(name)、描述(description)、价格(price)、居住面积(living area)…)存储在数据库中。odoo框架提供了数据库交互的工具
开始练习前,请确保estate模块已被安装,也就是说必须以installed的状态出现在Apps列表中,如下

对象关系映射(Object-Relational Mapping)
参考: 和本主题关联文档可参考 Models API.
ORM 层是odoo的一个关键组件。该层避免了手动写入大部分SQL并提供可扩展性和安全服务.
业务对象被定义为继承于 Model的Python类。可以通过在定义中设置属性来配置model。最重要的属性为 _name,该属性定义了model在odoo系统中的属性。以下为model的最小化定义:
from odoo import models
class TestModel(models.Model):
_name = "test.model"
该定义足够ORM生成一张名为test_model的表。model _name中的 . 会被ORM自动化转为_ 。按约定所有的model位于一个名为 models 的目录,并且每个mode被定义为一个Python文件。
来看下 crm_recurring_plan 表是怎么定义的,以及对应Python文件是怎么导入的:
在
odoo/addons/crm/models/crm_recurring_plan.py中定义model(源码链接)# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details. from odoo import fields, models class RecurringPlan(models.Model):
_name = "crm.recurring.plan"
_description = "CRM Recurring revenue plans"
_order = "sequence" name = fields.Char('Plan Name', required=True, translate=True)
number_of_months = fields.Integer('# Months', required=True)
active = fields.Boolean('Active', default=True)
sequence = fields.Integer('Sequence', default=10) _sql_constraints = [
('check_number_of_months', 'CHECK(number_of_months >= 0)', 'The number of month can\'t be negative.'),
]
在
crm/models/__init__.py中导入crm_recurring_plan.py(源码链接)# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details. from . import res_users
from . import calendar
from . import crm_lead
from . import crm_lost_reason
from . import crm_stage
from . import crm_team
from . import ir_config_parameter
from . import res_config_settings
from . import res_partner
from . import digest
from . import crm_lead_scoring_frequency
from . import utm
from . import crm_recurring_plan
在
crm/__init__.py中导入models包 (源码链接)# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details. from . import controllers
from . import models
from . import report
from . import wizard from odoo import api, SUPERUSER_ID
练习
创建estate_property表的最小化模型
在
odoo14/custom/estate/models/estate_property.py中定义model#!/usr/bin/env python
# -*- coding: utf-8 -*- from odoo import model class EstateProperty(models.Model):
_name = 'estate.property'
estate_property.py在odoo14/custom/estate/models/__init__.py中导入#!/usr/bin/env python
# -*- coding:utf-8 -*- from . import estate_property
在
estate/__init__.py中导入models包#!/usr/bin/env python
# -*- coding:utf-8 -*- from . import models
重启odoo服务
python odoo-bin --addons-path=custom,odoo/addons -r myodoo -w test123 -d odoo -u estate
-u estate 表示更新 estate 模块,也就是说ORM将应用数据库模式变更。
启动过程中可以看到类似以下告警日志:
...
2022-12-14 06:46:02,771 23792 WARNING odoo odoo.models: The model estate.property has no _description
2022-12-14 06:46:02,920 23792 WARNING odoo odoo.models: The model estate.property has no _description
...
2022-12-14 06:46:03,498 23792 WARNING odoo odoo.modules.loading: The model estate.property has no access rules, consider adding one...
...
...
以防万一,可以看下到数据库看下表是否创建成功。pgAmin查看路径:Servers -> PostgreSQL 12 -> Databases (x) ->数据库名 -> Schemas -> public -> Tables
模型字段(Model Fields)
参考: 该主题相关文档可参考 Fields API
字段用于定义model可以存储啥及在哪里存储。 Fields被定义为model类的属性:
from odoo import fields, models
class TestModel(models.Model):
_name = "test.model"
_description = "Test Model"
name = fields.Char()
name 字段被定义为Char,代表Python unicode的 str 和SQL的 VARCHAR.
有两大类领域字段:‘简单’字段--直接存储在模型表中的原子值,形如Boolean, Float, Char, Text, Date 和Selection, ‘关系型’ 字段--连接相同或者不同模型的记录。
给模型表estate_property添加字段
添加以下字段到表中
| Field | Type |
|---|---|
| name | Char |
| description | Text |
| postcode | Char |
| date_availability | Date |
| expected_price | Float |
| selling_price | Float |
| bedrooms | Integer |
| living_area | Integer |
| facades | Integer |
| garage | Boolean |
| garden | Boolean |
| garden_area | Integer |
| garden_orientation | Selection |
The garden_orientation 字段必须有4种可选值:‘North’, ‘South’, ‘East’ 和‘West’。Selection(选择列表)定义为元组列表,查看示例
修改odoo14/custom/estate/models/estate_property.py文件
#!/usr/bin/env python
# -*- coding: utf-8 -*-
from odoo import models,fields
class EstateProperty(models.Model):
_name = 'estate.property'
_description = 'estate property table'
name = fields.Char(size=15)
description = fields.Text()
postcode = fields.Char(size=15)
date_availability = fields.Datetime('Availability Date')
expected_price = fields.Float('expected price', digits=(8, 2)) # 最大8位,小数占2位
selling_price = fields.Float('selling price', digits=(8, 2))
bedrooms = fields.Integer()
living_area = fields.Integer()
facades = fields.Integer()
garage = fields.Boolean('garage')
garden = fields.Boolean('garden')
garden_area = fields.Integer()
garden_orientation = fields.Selection(
string='Orientation',
selection=[('north', 'North'), ('south', 'South'), ('east', 'East'), ('West','West')],
help="garden orientation"
)
重启odoo服务
python odoo-bin --addons-path=custom,odoo/addons -r myodoo -w test123 -d odoo -u estate
数据库中验证

常见属性
现在假设要求 name 和expected_price字段值不为null,所以需要对其修改,如下,添加字段属性配置required=True
name = fields.Char(required=True)
expected_price = fields.Float('expected price', digits=(8, 2), required=True) # 最大8位,小数占2位
修改后重启odoo服务。
有些属性是所有字段都拥有的,最常见的几个属性如下:
string(str, default: 字段名称)UI上显示为字段的label (用户可见).
required(bool, default:False)如果为
True, 表示该字段值不能为空。创建记录时必须拥有默认值或给定的值。help(str, default:'')UI上为用户提供long-form 帮助提示
index(bool, default:False)要求odoo在该列上创建数据库索引
自动创建的字段(Automatic Fields)
参考: 该话题相关文档可参考 Automatic fields.
odoo会在所有model(当然,也可以配置禁止自动创建某些字段)中创建少数字段。这些字段有系统管理并且不能写,但是可以读取,如果必要的话:
id(Id)model记录的唯一标识
create_date(Datetime)记录创建日期
create_uid(Many2one)记录创建人
write_date(Datetime)记录最后修改时间
write_uid(Many2one)记录最后修改人
odoo 开发入门教程系列-模型和基本字段的更多相关文章
- 移动H5开发入门教程:12点webAPP前端开发经验
如果你是一名移动H5前端开发人员,25学堂的小编认为下面的分享的12点webAPP前端开发经验是你必须掌握的基础知识点.算是一篇移动H5开发入门教程吧! 1. viewport:也就是可视区域.对于桌 ...
- ActiveMQ详细入门教程系列(一)
一.什么是消息中间件 两个系统或两个客户端之间进行消息传送,利用高效可靠的消息传递机制进行平台无关的数据交流,并基于数据通信来进行分布式系统的集成.通过提供消息传递和消息排队模型,它可以在分布式环境下 ...
- WPF入门教程系列(一) 创建你的第一个WPF项目
WPF入门教程系列(一) 创建你的第一个WPF项目 WPF基础知识 快速学习绝不是从零学起的,良好的基础是快速入手的关键,下面先为大家摞列以下自己总结的学习WPF的几点基础知识: 1) C#基础语法知 ...
- Android Studio JNI开发入门教程
Android Studio JNI开发入门教程 2016-08-29 14:38 3269人阅读 评论(0) 收藏 举报 分类: JNI(3) 目录(?)[+] 概述 在Andorid ...
- SeaJS入门教程系列之使用SeaJS(二)
SeaJS入门教程系列之使用SeaJS(二) 作者: 字体:[增加 减小] 类型:转载 时间:2014-03-03我要评论 这篇文章主要介绍了SeaJS入门教程系列之使用SeaJS,着重介绍了SeaJ ...
- Silverlight,Windows 8应用开发实例教程系列汇总
Kevin Fan分享开发经验,记录开发点滴 Silverlight,Windows 8应用开发实例教程系列汇总 2012-06-18 01:05 by jv9, 2145 阅读, 3 评论, 收藏, ...
- WPF入门教程系列二十三——DataGrid示例(三)
DataGrid的选择模式 默认情况下,DataGrid 的选择模式为“全行选择”,并且可以同时选择多行(如下图所示),我们可以通过SelectionMode 和SelectionUnit 属性来修改 ...
- WPF入门教程系列三——Application介绍(续)
接上文WPF入门教程系列二——Application介绍,我们继续来学习Application 三.WPF应用程序的关闭 WPF应用程序的关闭只有在应用程序的 Shutdown 方法被调用时,应用程序 ...
- WPF入门教程系列二——Application介绍
一.Application介绍 WPF和WinForm 很相似, WPF与WinForm一样有一个 Application对象来进行一些全局的行为和操作,并且每个 Domain (应用程序域)中仅且只 ...
- 基于Nodejs生态圈的TypeScript+React开发入门教程
基于Nodejs生态圈的TypeScript+React开发入门教程 概述 本教程旨在为基于Nodejs npm生态圈的前端程序开发提供入门讲解. Nodejs是什么 Nodejs是一个高性能Ja ...
随机推荐
- ethcat开发记录 三
一.关于controlword的控制 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0 保留 待定 暂停 故障复位 模式有关 伺服使能 快停 上电 伺服准备好
- win10下 pytorch 跑模型 gpu利用率低
查阅资料后发现 Dataloader中的num_workers参数(线程数)设置为0,该为4后,nvidia-smi查看GPU占用率变为高(不要用任务管理器查看)
- FastReport和RDLC报表
最近在做报表的时候第一次接触到RDLC报表,对比于之前使用的FastReport报表来说,在使用体验上个人目前感觉RDLC灵活性相对较差,尤其是表格的格式多样的时候,不易修改.RDLC应用于格式简单的 ...
- 4组-Alpha冲刺-6/6
一.基本情况 队名:摸鲨鱼小队 组长博客:https://www.cnblogs.com/smallgrape/p/15574385.html 小组人数:8人 二.冲刺概况汇报 组长:许雅萍 过去两天 ...
- Shell脚本基本命令4
使用join连接字段 1.$ cat >sales 创建salse文件 #业务员数据 注释说明 #业务员量 joe 100 jane 200 herman 150 chris 300 2.$ ...
- MyBatis面试题汇总
1.什么是Mybatis? Mybatis是对象关系映射一个框架,它内部封装了JDBC,开发的时候只要关注SQL语句本身,可以严格控制sql的执行性能,灵活,其二可以通过XML或者注解来配置映射信息 ...
- 面向对象的练习总结(java)
三次作业总结博客 l 前言 第一次题目集是我刚刚接触java所做的第一套习题,本次题目难度不大,题量较多,涉及的知识点主要是基础的语法知识,出题人的意图是让我们尽快熟悉java的语法,由于事先有c语 ...
- 20193314白晨阳《Python程序设计》实验四 Python综合实践
课程:<Python程序设计> 班级: 1933 姓名: 白晨阳 学号:20193314 实验教师:王志强老师 实验日期:2021年6月13日 必修/选修: 公选课 实验内容: Pytho ...
- nodejs 反单引号用法(·)
这个反单引号就是数字1旁边(~)下面的那个符号,平时用得很少,虽然单引号和双引号是使用较多的,但我们还有第三个方案,就是ES6中的模板字符串(反引号). 在nodejs中用反单引号(·)主要基于以下作 ...
- bitmap_find_next_zero_area_off函数
备注: