Odoo开发之记录集 – 使用模型数据

KaiZhaoKZ 2020-05-27

Odoo 12开发之记录集 – 使用模型数据

一·启动服务链接数据库

# 1.启动服务,链接数据库
 	./odoo-bin shell -c debian/odoo.conf  -d library_db
    
# 2. self   当前操作的对象
	res.users(1,)
    
# 3. self._name  # 当前模块名称 ,获得记录集模型名
	‘res.users‘
    
# 4. self.login  # 记录值,至于怎么用.未知
	‘__system__‘
    
# 5. self.name     #  name 值为OdooBot
	OdooBot

二·执行环境

环境属性

# 1. self.env , 获得当前环境
	<odoo.api.Environment object at 0x7f78a26026a0>
    
# 2. self.env中的执行环境属性
	# env.cr  正在使用的数据库游标(cursor)
    	<odoo.sql_db.Cursor object at 0x7f9df38efc50>
    # env.user 是当前用户的记录
    	res.users(1,)
    # env.uid 是会话用户的id
    	1 
    # env.context 是会话上下文的不可变字典    
    	{‘lang‘: ‘en_US‘, ‘tz‘: ‘Europe/Brussels‘}
        # 由于是 frozendict 由odoo 自行封装的字典,不可变
           env.context[‘test‘]=‘test‘
           错误: NotImplementedError: ‘__setitem__‘ not supported on frozendic
                
     # search([(‘name‘,‘like‘,‘Ad‘)]) 搜索查询,search和domain表达式结合使用
    	self.env[‘res.partner‘].search([(‘name‘,‘like‘,‘Ad‘)])
     
     # browse() 也是搜索查询,browse放id列表如[1,2,3,4]
        self.env[‘res.partner‘].browse([1,2,3,4])

环境上下文

### 环境上下是带有会话数据的字典. 
### 作用:用于客户端和服务端ORM和业务逻辑中. 可以将信息从一个视图中传递到另一个视图中.
	# 比如:上一个视图中活跃的id,通过点击链接或者按钮.将默认值带入到下一个视图中.
    # 形式:	
		{‘lang‘: ‘en_US‘, ‘tz‘: ‘Europe/Brussels‘, ‘uid‘: 2}
# 1. 查看上下文命令
	self.context_get()
	self.env.context
	# lang :用户语言 tz:时区信息 uid:当前用户的id

修改记录集执行环境

### 记录集执行环境是不可变的.不可被修改,因此需要创建一个新的环境来修改
	# 1. env.with_context({diciionary}) # 替换原上下文为新的上下文
    # 2. env.with_context(key=value,...) # 修改当前上下文,为一些键设置值
    # 3. env.sudo(user) # 传入一条用户记录并返回该用户的环境.如果未传用户,则使用	__system__超级用户. 可绕过安全规则执行指定操作
    
# env.ref()函数 , 传入一个外部标识符字符串并返回它的记录
	self.env.ref(‘base.user_root‘) 
    res.users(1,)

三· 记录集和作用域domain查询数据

创建记录

### search方法 接收一个域表达式并返回符合条件的记录集. 空域[]将返回所有的记录

    # 关键字参数:
        # order是一个数据库查询语句种的 order by使用的字符串.通常是一个逗号分隔的字段名列表。每个字段都可接DESC关键字,用于表示倒序排列。
        # limit 设置获取记录的最大条数
        # offset 忽略钱n条记录,配合limit使用
    # 用法:
    	self.env[‘res.partner‘].search([(‘name‘, ‘like‘, ‘Pac‘)])
        # 结果
        res.partner(42, 62)
 	   
    
### search_count()方法 返回记录条数. 先知道条数,节约内存
	# 用法
    	self.env[‘res.partner‘].search_count([])
        #结果
        681
### browse()方法接收一个ID列表或单个ID并返回这些记录的记录集
	# 用法
    	self.env[‘res.partner‘].browse([42, 62])
        # 结果
        res.partner(42, 62)

域表达式

### domain用于过滤数据记录. 使用的是特殊语法来供Odoo ORM解析	
	# 写法:
		(‘字段名’, ‘运算符’, ‘值’) 组成的元组
    # 字段名:是一个有待过滤的字段,可用点号标记,来表示关联模型种的字段
    # 值: 分为两种运行上下文: 
    	 ① 窗口操作或字段属性等客户端,使用原生字段值,不能使用点标记符
         ② 服务端可以是一个对象
    # 运算符
    	# 比较: <,>,<=,>=,=,!=
        # =like 和 =ilike 
        # like匹配 ‘%value%‘模式,ilike与其相似.
        # not like 和 not ilike
        # child of 支持层级关联的模型中 查找层级关系种的子级值
        # in 和 not in 在和不在
        
	 # 多条件,即包含多个元祖  默认是 AND 运算符
    	# 例子: 此例子为编造
        self.env[‘res.partner‘].search([(‘message_follower_ids‘, ‘in‘, [user.partner_id.id]),(‘user_id‘, ‘=‘, user.id)])
     
     # 显式逻辑运算符 ‘&‘符号表示 AND 运算符(默认值),管道运算符’|‘表示OR运算符
    	# 例子:
            [‘|‘,(‘message_follower_ids‘, ‘in‘, [user.partner_id.id]),‘|‘,(‘user_id‘, ‘=‘, user.id),(‘user_id‘, ‘=‘, False)]

四·记录集中访问数据

访问记录中数据

# 1.记录集仅有一条记录时,称为单例
	self.name  --> OdooBot
### 注意:
	尝试访问有多条记录的记录集字段值会产生错误,所以在不确定操作的是否为单例数据集时就会产生问题。对于设计仅操作单例的方法,
    可在开头处使用 self.ensure_one(),如果 self 不是单例时将抛出错误。

访问关联字段

### 包含:many-to-one, one-to-many和many-to-many
    # 对于many-to-one,其值可以是单例或空记录集
        self.company_id ---> res.company(1,)
        self.company_id.name  ---> ‘YourCompany‘
        self.company_id.currency_id   ---> res.currency(1,)
        self.company_id.currency_id.name   ---> ‘EUR‘

    # 空记录可像单例一样操作,访问其字段值不会返回错误而是返回 False 
    	self.company_id.parent_id --->  res.company()
        self.company_id.parent_id.name ---> False

访问时间和日期值

# 1.查询上次admin用户登录日期
	self.browse(2).login_date ---> datetime.datetime(2019, 1, 8, 9, 2, 54, 45546)

#  2. 可以利用python的datetime模块进行时间计算
Odoo 还在odoo.tools.date_utils模块中提供额外的便利函数
# start_of(value, granularity)是某个特定刻度时间区间的开始时间,这些刻度有year, quarter, month, week, day或hour
    #  end_of(value, granularity)是某个特定刻度时间区间的结束时间
	#  add(value, **kwargs)为指定值加上一个时间间隔。**kwargs参数由一个relativedelta对象来定义时间间隔。这些参数可以是years, months, weeks, days, hours, minutes等等
    #  subtract(value, **kwargs)为指定值减去一个时间间隔 
    
from odoo.tools import date_utils
from datetime import datetime

# 范围时间当前时间和指定时间参数:week,本周周一那天.如今天是2020年5月27,一周前是2020年5月25	
# 输入: date_utils.start_of(datetime.now(), ‘week‘)
	    datetime.datetime(2020, 5, 25, 0, 0)
    
# end_of 同理 末尾时间
# 输入:date_utils.end_of(datetime.now(), ‘week‘)
	  datetime.datetime(2020, 5, 31, 23, 59, 59, 999999)
    
# add 指定添加多长时间
# 输入: date_utils.add(datetime.today(), months=2)
		datetime.datetime(2020, 7, 27, 2, 40, 55, 424214)
    
# subtract 指定减去多长时间
# 输入: date_utils.subtract(datetime.today(), months=2)
		datetime.datetime(2020, 3, 27, 2, 42, 21, 390561)
odoo.fields.Date和the odoo.fields.Datetime
# 1. fields.Date.today()   # 返回当前日期,类型是datetime
	 datetime.date(2020, 5, 27)
# 2. fields.Datetime.now() # 返回当前日期+时间,类型是datetime
	 datetime.datetime(2020, 5, 27, 2, 46, 10)
# 3. fields.Date.context_today(record, timestamp=None)
   fields.Date.context_today(self, timestamp=None)
     datetime.date(2020, 5, 27)

	 # 在会话上下文中返回带有当前日期的字符串。时间从记录上下文中获取。可选项timestamp参数是一个datetime对象,如果传入将不使用当前时间,而使用传入值
 
# 4. fields.Datetime.context_timestamp(record, timestamp)
	 # 将原生的datetime值(无时区)转换为具体时区的datetime。时区从记录上下文中提取,因此使了前述函数名。

转换文本形式的日期和时间

# fields.Date和fields.Datetime都提供了如下函数
# 1. to_date将字符串转换为date对象
# 2. to_datetime(value)将字符串转换为datetime对象
# 3. to_string(value)将 date和datetime对象转换成字符串格式

# 4. Odoo 预设文本格式默认值:
    odoo.tools.DEFAULT_SERVER_DATE_FORMAT ---> %Y-%m-%d
    odoo.tools.DEFAULT_SERVER_DATETIME_FORMAT ---> Y-%m-%d %H:%M:%S
            
from odoo import fields
# 输入: fields.Datetime.to_datetime(‘2019-01-12 13:48:50‘)
		datetime.datetime(2019, 1, 12, 13, 48, 50)  
# 注意:
	其他格式时间字符串转换成日期类型需要使用 datetime.strptime单独转换
    from datetime import datetime
	# 输入: datetime.strptime(‘1/1/2019‘, ‘%d/%m/%Y‘)
	datetime.datetime(2019, 1, 1, 0, 0)

五·记录中写入

# odoo写入分为两种模式:
	# 1. 使用对象形式直接分配 . 简单但一次只能操作一条记录,效率较低
	# 2. 使用write() 方法 .  写入关联字段时使用特殊语法,但每条命令可写入多个字段和记录,记录计算更为高效

对象形式分配值写入

# 查询一条数据
	root = self.env[‘res.users‘].browse(1)
# 更改root对象name值
	root.name = ‘Superuser‘
# root.name
	Superuser
 
###

通过 write()方法写入

# 用write()方法来同时更新多条记录中的多个字段,仅需一条数据库命令
# write() 接收一个字典来进行字段和值的映射

# 得到一个空对象
	Partner = self.env[‘res.partner‘]
# 查询值
	recs = Partner.search( [(‘name‘, ‘ilike‘, ‘Azure‘)] )
# 修改值, 修改成功为True
	recs.write({‘comment‘: ‘Hello!‘})

    
### 在写入many-to-one字段时,写入的值必须是关联记录的ID。
	例如,我们不用self.write({‘user_id’: self.env.user}),而应使用self.write({‘user_id’: self.env.user.id})   
    
### 写入to-many字段时,写入的值必须使用和 XML 数据文件相同的特殊语法
	比如,我们设置图书作者列表为author1和author2,这是两条 Partner 记录。| 管道运算符可拼接记录来创建一个记录集,因此使用对象形式的分配可以这么写
    publisher.child_ids = author1 | author2
    # 使用write()方法
    book.write( { ‘child_ids‘: [(6, 0, [author1.id, author2.id])] } )
    ### (4, id, _)添加一条记录
    ### (6, _, [ids])替换关联记录列表为所传入的列表

写入日期和时间值

### 可以使用文本形式值写入日期和时间:
# 如:
	# 搜索到一个对象
	demo = self.search([(‘login‘, ‘=‘, ‘demo‘)])
    # 更改demo.login_date值
	demo.login_date = ‘2019-01-01 09:00:00‘
    # 查看
    demo.login_date ---> datetime.datetime(2019, 1, 1, 9, 0)

创建和删除记录

#创建和删除记录 通过 create()和unlink()模型方法实现

### create方法
    # 得到一个空的对象 ,这个对象仅被加载到了内存种,没有实际保存到数据库种
        Partner = self.env[‘res.partner‘]
    # 创建一个对象,此时就被保存到数据库种
        new = Partner.create({‘name‘: ‘ACME‘, ‘is_company‘: True})
        res.partner(64,)
    # 查看cutomer属性值
        print(new.customer) # customer标记默认为 True
### unlink()方法会删除记录
	# 搜索到某条记录
    rec = Partner.search([(‘name‘, ‘=‘, ‘ACME‘)])
    rec.unlink() # 删除 partner 关联字段的串联删除

拷贝模型记录值

### copy() 方法,会自动创建一条记录.这条记录会被保存到数据库中

# 查询出一条记录 
	demo = self.env.ref(‘base.user_demo‘)
# 拷贝
	new = demo.copy({‘name‘: ‘Daniel‘, ‘login‘: ‘daniel‘, ‘email‘: ‘‘})

六·重构记录集

# 1. recordset.ids 返回记录集元素的ID列表
# 2. recordset.ensure_one()检查是否为单条记录(单例);若不是,则抛出ValueError异常
# 3. recordset.filtered(func)返回一个过滤了的记录集,func可以是一个函数或一个点号分隔的表达式来表示字段路径,可参见下面的示例。
# 4. recordset.mapped(func)返回一个映射值列表。除函数外,还可使用文本字符串作为映射的字段名。
# 5. recordset.sorted(func)返回一个排好序的记录值。除函数外,文本字符串可用作排序的字段名。reverse=True是其可选参数。
>>> rs0 = self.env[‘res.partner‘].search([])
>>> len(rs0)
48

>>> starts_A = lambda r: r.name.startswith(‘A‘)
>>> rs1 = rs0.filtered(starts_A)
>>> print(rs1)
res.partner(63, 59, 14, 35)

>>> rs1.sorted(key=lambda r: r.id, reverse=True)
res.partner(63, 59, 35, 14)

>>> rs2 = rs1.filtered(‘is_company‘)
>>> print(rs2)
res.partner(14,)

>>> rs2.mapped(‘name‘)
[‘Azure Interior‘]

>>> rs2.mapped(lambda r: (r.id, r.name))
[(14, ‘Azure Interior‘)]
rs1 | rs2是一个集合的并运算,会生成一个包含两个记录集所有元素的记录集
rs1 + rs2是集合加法运算,会将两个记录集拼接为一个记录集,这可能会带来集合中有重复记录
rs1 & rs2是集合的交集运算,会生成一个仅在两个记录集中同时出现元素组成的数据集
rs1 – rs2是集合的差集运算,会生成在rs1中有但rs2中没有的元素组成的数据集

# 还可以使用分片标记,例如:
    rs[0]和rs[-1]分别返回第一个和最后一个元素
    rs[1:]返回除第一元素外的记录集拷贝。其结果和rs – rs[0]相同,但保留了排序
    
# 删除或添加元素来修改记录集
	self.author_ids |= author1:如果不存在author1,它会将author1加入记录集
	self.author_ids -= author1:如果author1存在于记录集中,会进行删除
	self.author_ids = self.author_ids[:-1]删除最后一条记录

七·底层 SQL 和数据库事务

# 数据库引入运算在一个数据库事务上下文中执行
# 通过数据库游标self.env.cr

# 执行事务缓冲的写运算 ,提交数据

	self.env.cr.commit()
    
# 取消上次 commit之后的写运算,如果尚未 commit,则回滚所有操作
	self.env.cr.rollback()
    
# execute() 方法 执行SQL语法 self.env.cr.execute()
	参数一: 运行的SQL 语句,
    可选参数:一个用作 SQL 参数值的元组或列表。这些值会用在%s占位符之处。
    
    # 用法: 
	self.env.cr.execute("SELECT id, login FROM res_users WHERE login=%s OR id=%s", (‘demo‘,1)
    self.env.cr.execute("SELECT id, login FROM res_users WHERE login=%s OR id=%s", (‘demo‘,1))

### select 查询记录
	# fetchall() 函数以元组列表的形式获取所有行
    	# 输入:self.env.cr.fetchall()
        [(1, ‘__system__‘), (6, ‘demo‘)]
    
    # dictfetchall()则以字典列表的形式获取
    	# 输入: self.env.cr.dictfetchall()
        [{‘id‘: 1, ‘login‘: ‘__system__‘}, {‘id‘: 6, ‘login‘: ‘demo‘}]

### 修改DML操纵语言
 self.env.cache.invalidate()

相关推荐