Python 框架学习 Django篇 (六) ORM关联

像是上一章我们很少会通过页面点击去添加和绑定关系表,更多的时候都是通过django的语法实现,接下来我们做一个案例 django rom是怎么操作外键关系的

创建mode模型表

Django_demo/mgr/models.py

# 国家表
class Country(models.Model):name = models.CharField(max_length=100)# 学生表, country 字段是国家表的外键,形成一对多的关系
class Student(models.Model):name    = models.CharField(max_length=100)grade   = models.PositiveSmallIntegerField()country = models.ForeignKey(Country,on_delete=models.PROTECT)
python manage.py makemigrations 
python manage.py migrate

添加测试数据

python manage.py shell
#注意应用名称
from paas.models import *
c1 = Country.objects.create(name='中国')
c2 = Country.objects.create(name='美国')
c3 = Country.objects.create(name='法国')
Student.objects.create(name='白月', grade=1, country=c1)
Student.objects.create(name='黑羽', grade=2, country=c1)
Student.objects.create(name='大罗', grade=1, country=c1)
Student.objects.create(name='真佛', grade=2, country=c1)
Student.objects.create(name='Mike', grade=1, country=c2)
Student.objects.create(name='Gus',  grade=1, country=c2)
Student.objects.create(name='White', grade=2, country=c2)
Student.objects.create(name='Napolen', grade=2, country=c3)

一、外键表使用

1、字段访问

#获取student表中 name='白月' 的数据
s1 = Student.objects.get(name='白月')#将拿到的数据通过外键直接获取到对应外键表的name字段数据并输出 
s1.country.name 

案例

>>> s1 = Student.objects.get(name='白月') 
>>> s1.country.name 
'中国' 

2、单个字段过滤

如果我们想要查询学生表中所有关于1年纪的学生,对应表字段是grade 年纪

Student.objects.filter(grade=1).values()

返回

<QuerySet [{'id': 1, 'name': '白月', 'grade': 1, 'country_id': 1}, {'id': 3, 'name': '大罗', 'grade': 1, 'country_id': 1}, {'id': 5, 'name': 'Mike', 'grade': 1, 'country_id': 2}, {
'id': 6, 'name': 'Gus', 'grade': 1, 'country_id': 2}]>

3、多字段过滤

如果我要同时实现多个条件呢,比如同时满足一年级, 并且国籍是中国的学生

我们不能直接添加多个选项  Student.objects.filter(grade=1,country='中国')

因为,Student表中 country 并不是国家名称字符串字段,而是一个外键字段,对应 Country 表中 id 字段,或许我们可以和上一张用sql语句一样,先查id,然后在查其他信息

cn = Country.objects.get(name='中国')
Student.objects.filter(grade=1,country_id=cn.id).values()#或者cn = Country.objects.get(name='中国')
Student.objects.filter(grade=1,country=cn).values()

 返回

>>> cn = Country.objects.get(name='中国') 
>>> Student.objects.filter(grade=1,country_id=cn.id).values() 
<QuerySet [{'id': 1, 'name': '白月', 'grade': 1, 'country_id': 1}, {'id': 3, 'name': '大罗', 'grade': 1, 'country_id': 1}]> >>> cn = Country.objects.get(name='中国') 
>>> Student.objects.filter(grade=1,country=cn).values() 
<QuerySet [{'id': 1, 'name': '白月', 'grade': 1, 'country_id': 1}, {'id': 3, 'name': '大罗', 'grade': 1, 'country_id': 1}]> 

4、多字段过滤优化

但是上面的方法其实相对来说过于繁琐,并且需要查询两次,性能不高,

Django ORM 中,对外键关联,有更方便的语法,使用外键+ 双下划线 + 字段名称

Student.objects.filter(grade=1,country__name='中国').values()

 返回

<QuerySet [{'id': 1, 'name': '白月', 'grade': 1, 'country_id': 1}, {'id': 3, 'name': '大罗', 'grade': 1, 'country_id': 1}]> 

5、过滤优化--指定显示字段

如果返回结果只需要 学生姓名 和 国家名两个字段,可以这样指定values内容

Student.objects.filter(grade=1,country__name='中国').values('name','country__name')

 返回

<QuerySet [{'name': '白月', 'country__name': '中国'}, {'name': '大罗', 'country__name': '中国'}]>

6、字段显示重命名

有个问题,我们返回数据的字段是country__name 这样的,双下划线很奇怪,有时候前后端定义好了接口格式,必须让用countryname的话就需要去做字段的重命名

from django.db.models import F# annotate 可以将表字段进行别名处理
Student.objects.annotate(countryname=F('country__name'),studentname=F('name'))\.filter(grade=1,countryname='中国').values('studentname','countryname')

7、反向访问

 Django ORM中,关联表正向关系是通过表外键字段(或者多对多)表示

而反向访问,则是通过将model模板名称转换成小写表示的,比如说你已经获取到了某个国家信息,需要通过国家信息反向查看属于这个国家的学生数据

 案例

#声明获取国家表中name字段为中国的数据
cn = Country.objects.get(name='中国')#先将要查看的表改为全小写,并且声明_set来获取所有的反向外键关联对象
cn.student_set.all()

返回

>>> cn = Country.objects.get(name='中国') 
>>> cn.student_set.all() 
<QuerySet [<Student: Student object (1)>, <Student: Student object (2)>, <Student: Student object (3)>, <Student: Student object (4)>]> 
>>>  

小知识

Django还给出了一个方法,是在定义Model的时候,外键字段使用 related_name 参数

#国家表
class Country(models.Model):name = models.CharField(max_length=100)# country 字段是国家表的外键,形成一对多的关系
class Student(models.Model):name    = models.CharField(max_length=100)grade   = models.PositiveSmallIntegerField()country = models.ForeignKey(Country,on_delete = models.PROTECT,# 指定反向访问的名字related_name='students')
python manage.py makemigrations 
python manage.py migrate

查询

cn = Country.objects.get(name='中国')
students = cn.students.all()#拿到的值循环存储并且输出
student_names = [student.name for student in students]print(student_names)

返回

>>> print(student_names)
['白月', '黑羽', '大罗', '真佛'] 

8、反向过滤

如果我们想要获取所有一年纪学生的国家名称呢,同样可以依靠复合查询实现

# 先获取所有的一年级学生id列表
country_ids = Student.objects.filter(grade=1).values_list('country', flat=True)# 再通过id列表使用  id__in  过滤
Country.objects.filter(id__in=country_ids).values()

存在的问题就是重复请求,造成性能下降 ,用Django ORM 的方法

Country.objects.filter(students__grade=1).values()

 返回

>>> Country.objects.filter(students__grade=1).values() 
<QuerySet [{'id': 7, 'name': '中国'}, {'id': 7, 'name': '中国'}, {'id': 8, 'name': '美国'}, {'id': 8, 'name': '美国'}]> 

发现存在重复的数据,我们使用 .distinct() 去重

>>> Country.objects.filter(students__grade=1).values().distinct() 
<QuerySet [{'id': 7, 'name': '中国'}, {'id': 8, 'name': '美国'}]> 

 注意

我们前面在定义学生表时, 使用了related_name = "students" 去指定了反向关联students

所以这里通过国家表查询学生表数据的时候使用的字段名称是students__grade 的反向关联名称

如果定义时,没有指定related_name, 则应该使用 表名转化为小写 ,就是这样

Country.objects.filter(student__grade=1).values()

二、实现项目代码

我们在 mgr 目录下面新建 order.py 处理 客户端发过来的 列出订单、添加订单 的请求

1、添加增删改查主体程序

vi Django_demo/mgr/order.py

from django.http import JsonResponse
from django.db.models import F
from django.db import IntegrityError, transaction# 导入 Order 对象定义
from  paas.models import  Order,OrderMedicineimport jsondef dispatcherorder(request):# 根据session判断用户是否是登录的管理员用户if 'usertype' not in request.session:return JsonResponse({'ret': 302,'msg': '未登录','redirect': '/mgr/sign.html'},status=302)if request.session['usertype'] != 'mgr':return JsonResponse({'ret': 302,'msg': '用户非mgr类型','redirect': '/mgr/sign.html'},status=302)# 将请求参数统一放入request 的 params 属性中,方便后续处理# GET请求 参数 在 request 对象的 GET属性中if request.method == 'GET':request.params = request.GET# POST/PUT/DELETE 请求 参数 从 request 对象的 body 属性中获取elif request.method in ['POST','PUT','DELETE']:# 根据接口,POST/PUT/DELETE 请求的消息体都是 json格式request.params = json.loads(request.body)# 根据不同的action分派给不同的函数进行处理action = request.params['action']if action == 'list_order':return listorder(request)elif action == 'add_order':return addorder(request)# 订单 暂 不支持修改 和删除else:return JsonResponse({'ret': 1, 'msg': '不支持该类型http请求'})

2、添加路由

vi  Django_demo/mgr/urls.py


urlpatterns = [...path('orders', dispatcherorder), # 加上这行...
]

3、定义添加订单函数

接下来,我们添加函数 addorder,来处理添加订单请求,首先我们要了解的是

每次添加一个订单,都需要在2张表(Order 和 OrderMedicine )中添加记录 订单表和中间表

我们给两张表添加记录,就会往数据库写两次,如果中有一次写入失败了,就会形成脏数据

而对应解决这个问题的办法,就是利用数据库"事务"的机制

什么是事务

    把一批数据库操作放在 事务 中, 该事务中的任何一次数据库操作 失败了, 数据库系统就会让 整个事务就会发生回滚,撤销前面的操作, 数据库回滚到这事务操作之前的状态

在django中使用事务机制

直接通过关键字with transaction.atomic()  即可实现数据库批量操作

 Django_demo/mgr/order.py

def addorder(request):info  = request.params['data']# 从请求消息中 获取要添加订单的信息# 并且插入到数据库中#设置事务with transaction.atomic():new_order = Order.objects.create(name=info['name'] ,customer_id=info['customerid'])batch = [OrderMedicine(order_id=new_order.id,medicine_id=mid,amount=1)for mid in info['medicineids']]#  在多对多关系表中 添加了 多条关联记录OrderMedicine.objects.bulk_create(batch)return JsonResponse({'ret': 0,'id':new_order.id})

只要是在 with transaction.atomic() 下面 缩进部分的代码,对数据库相关的操作都视为在同一个事务中,如果其中有任何一步数据操作失败了, 前面的操作都会回滚,这样就解决了多次写入失败导致的脏数据问题

代码说明

batch = [OrderMedicine(order_id=new_order.id,medicine_id=mid,amount=1)for mid in info['medicineids']]#  在多对多关系表中 添加了 多条关联记录OrderMedicine.objects.bulk_create(batch)

正常来说,OrderMedicine 对应的是订单和药品的多对对记录关系表,要在多对多表中加上关联记录,就是添加一条记录,直接写入即可,如下

OrderMedicine.objects.create(order_id=new_order.id,medicine_id=mid,amount=1)

但你实际上去买药的时候,很少会单独买一样,多少要带点其他的 ,但是我们如果直接用循环把去写入上面的语句,循环几次插入几次,那么多次写入也会影响性能,我们可以使用django中的bulk_create 把多条数据的插入,放在一个SQL语句中完成

batch = [OrderMedicine(order_id=new_order.id,medicine_id=mid,amount=1)  for mid in info['medicineids']]#  在多对多关系表中 添加了 多条关联记录
OrderMedicine.objects.bulk_create(batch)

三、ORM 外键关联

接下来我们编写listorder 函数用来处理 列出订单请求,请求格式如下

[{id: 1, name: "华山医院订单001", create_date: "2018-12-26T14:10:15.419Z",customer_name: "华山医院",medicines_name: "青霉素"},{id: 2, name: "华山医院订单002", create_date: "2018-12-27T14:10:37.208Z",customer_name: "华山医院",medicines_name: "青霉素 | 红霉素 "}
] 

 1、基于接口案例返回值

 其中 ‘id’,’name’,‘create_date’ 这些字段的内容获取很简单,order表中就有这些字段

def listorder(request):# 返回一个 QuerySet 对象 ,包含所有的表记录qs = Order.objects.values('id','name','create_date')return JsonResponse({'ret': 0, 'retlist': newlist})

 但是customer_name 客户名称 medicines_name 药品名称是在订单表里面没有的,我们需要通过类似前面的方法从订单表的外键customer 获取到客户表的name字段,方法就是模型名称小写 + 双下划线 + 字段名称

def listorder(request):qs = Order.objects\.values('id','name','create_date',# 两个下划线,表示取customer外键关联的表中的name字段的值'customer__name')# 将 QuerySet 对象 转化为 list 类型retlist = list(qs)return JsonResponse({'ret': 0, 'retlist': retlist})

同样的道理 , 订单对应 的药品 名字段,是多对多关联, 也同样可以用 两个下划线 获取 关联字段的值

 Django_demo/mgr/order.py 

def listorder(request):qs = Order.objects\.values('id','name','create_date','customer__name',# 两个下划线,表示取medicines 关联的表中的name字段的值# 如果有多个,就会产生多条记录'medicines__name')# 将 QuerySet 对象 转化为 list 类型retlist = list(qs)return JsonResponse({'ret': 0, 'retlist': retlist})

2、重命名返回字段

 这个是能返回数据了,但存在的问题是双下划线和接口要求的单下划线不同,需要通过annotate做一下重命名

from django.db.models import Fdef listorder(request):# 返回一个 QuerySet 对象 ,包含所有的表记录qs = Order.objects\.annotate(customer_name=F('customer__name'),medicines_name=F('medicines__name'))\.values('id','name','create_date','customer_name','medicines_name')# 将 QuerySet 对象 转化为 list 类型retlist = list(qs)return JsonResponse({'ret': 0, 'retlist': retlist})

3、去除订单内多个不同药品

如果一个订单里面有多个药品,就会产生多条记录, 这不是我们要的。

根据接口,一个订单里面的多个药品, 用 竖线 隔开

def listorder(request):# 返回一个 QuerySet 对象 ,包含所有的表记录qs = Order.objects\.annotate(customer_name=F('customer__name'),medicines_name=F('medicines__name'))\.values('id','name','create_date','customer_name','medicines_name')# 将 QuerySet 对象 转化为 list 类型retlist = list(qs)# 可能有 ID相同,药品不同的订单记录, 需要合并newlist = []id2order = {}for one in retlist:orderid = one['id']if orderid not in id2order:newlist.append(one)id2order[orderid] = oneelse:id2order[orderid]['medicines_name'] += ' | ' + one['medicines_name']return JsonResponse({'ret': 0, 'retlist': newlist})

4、测试--获取订单信息

import  requests,pprint#添加认证
payload = {'username': 'root','password': '12345678'
}
#发送登录请求
response = requests.post('http://127.0.0.1:8000/api/mgr/signin',data=payload)
#拿到请求中的认证信息进行访问
set_cookie = response.headers.get('Set-Cookie')# 构建添加 客户信息的 消息体,是json格式
payload = {"action":"list_order",
}
url='http://127.0.0.1:8000/api/mgr/orders/'if set_cookie:# 将Set-Cookie字段的值添加到请求头中headers = {'Cookie': set_cookie}# 发送请求给web服务response = requests.post(url,json=payload,headers=headers)pprint.pprint(response.json())

返回

{'ret': 0,'retlist': [{'create_date': '2023-10-25T03:08:00Z','customer_name': 'zhangsan','id': 5,'medicines_name': 'gmkl','name': 'test'}]}

5、测试--添加订单信息

import  requests,pprint#添加认证
payload = {'username': 'root','password': '12345678'
}
#发送登录请求
response = requests.post('http://127.0.0.1:8000/api/mgr/signin',data=payload)
#拿到请求中的认证信息进行访问
set_cookie = response.headers.get('Set-Cookie')# 构建添加 客户信息的 消息体,是json格式
payload = {"action":"add_order",             #模式改为添加"data": {"name": "天山订单",                 #订单名称自定义"customerid": 1,              #客户表中的id值"medicineids": ["6"]          #这个是药品表中的id值}
}
url='http://127.0.0.1:8000/api/mgr/orders/'if set_cookie:# 将Set-Cookie字段的值添加到请求头中headers = {'Cookie': set_cookie}# 发送请求给web服务response = requests.post(url,json=payload,headers=headers)pprint.pprint(response.json())

四、特殊字段参数使用

1、唯一性约束处理

我在实际使用的是时候为数据的一致性,我给好几个字段,都添加了唯一性约束

ip_address = modes.CharField(max_length=200,unique=True)

但是在写入的时候如果某个字段发现因为已经有相同数据程序就会崩掉,我们要就不能直接使用create去添加数据,而是改用get_or_create

 案例

#处理数据逻辑
obj, created = clusterINFO.objects.get_or_create(唯一性字段=传入的值比如ip, 唯一性字段2=传入的值2, #表内其他非一致性字段defaults={'other_field': other_value}
)

上面这个方法当发现已经有字段时,则会给created变量返回一个值为false,如果发现可以正常写入就会返回一个true

2、传入值处理

一开始总是陷入一个误区,想着表内要啥数据就从客户端传啥数据

payload = {"action":"list_order","data": {key: valuekey: value}
}

 比方说,我要添加集群中的node数据,肯定要确保有这个集群名称吧,还要确定下集群的园区是什么,这些都不是node表里面有的,但也要跟随传输过去处理

#拿到请求后先去集群表里面去查环境和集群名对应的字段数据
cn = 集群表.objects.get(环境=  ,集群名=)#到到的cn如果没有报错就是存放着对应环境集群那一行的数据
cn.id  就能拿到id值了,剩下的就是写入#给node表插入数据
obj, created = clusterINFO.objects.get_or_create(...defaults={'node表关联的外键字段_id': cn.id  #给node表外键提供主键表的id值#这块外键的字段名要加_id表示外键表名}
)

3、前端代码访问api数据

把这个文件扔到templates模板目录下面,定义个路由从页面显示出来,在页面打开的时候自动去获取订单表信息循环打印出来

<html>
<head><title>Your Page Title</title>
</head>
<body>
<h1>Your Page Content</h1><div id="list-container"></div><!-- 将 JavaScript 代码放在 <script> 标签中 -->
<script>// 定义请求参数const params = {action: 'list_order'};fetch('/api/mgr/orders/',{method: 'POST',headers: {'Content-Type': 'application/json'},body: JSON.stringify(params)})//获取请求的结果数据json.then(response => response.json()).then(data => {// 打印返回的 JSON 数据console.log(data);// 解析JSON数据并提取retlist的值const retlistValues = data.retlist; // retlist的值是一个包含多个JSON对象的数组console.log(retlistValues)//获取页面div,一会把自定义的div加进去const existingDiv = document.getElementById("list-container")// 遍历retlist的值,并将字段显示在页面上retlistValues.forEach(obj => {// 在页面上创建新的元素来显示字段和值const div = document.createElement('div');for (let key in obj) {const span = document.createElement('span');span.innerText = `${key}: ${obj[key]}`;div.appendChild(span);}existingDiv.appendChild(div)});})</script>
</body>
</html>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/120389.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

手把手教你部署Jenkins教程,小白也能学会(多图预警)!

背景 公司的前端、后端构建及部署工作都是人工去做&#xff0c;随着业务扩大&#xff0c;项目迭代速度变快&#xff0c;人员增多&#xff0c;各种问题都暴露出来&#xff0c;将通过一个简单案例分享一下基于Jenkins的前后端自动化工作流搭建的过程&#xff0c;搭建完这套工作流…

如何编辑pdf?推荐福昕高级pdf编辑器

这里写目录标题 安装教程1.双击FoxitPhantomPDF941_L10N_Setup.exe安装2.打开FiX UZ1文件夹 复制plugins文件夹和FoxitPhantomPDF.exe到安装目录中替换3. 双击Express2BusinessFix New.reg导入注册表 如何复制页面如何修改pdf的内容福昕高级pdf编辑器安装包 【Note】学校要求加…

小型洗衣机哪个牌子质量好?家用小洗衣机推荐

随着人们的生活水平的提升&#xff0c;越来越多小伙伴来开始追求更高的生活水平&#xff0c;一些智能化的小家电就被发明出来&#xff0c;而且小型洗衣机是其中一个。现在通过内衣裤感染到细菌真的是越来越多&#xff0c;所以我们对内衣裤的清洗频次会高于普通衣服&#xff0c;…

2022年京东双十一手机数码全品类数据回顾

2023年双十一临近&#xff0c;特此带大家回顾一下去年双十一热门品类的一些战况数据。这一期是京东手机电脑数码。 整体表现来看&#xff0c;2022年双11大促京东手机、电脑、数码类产品并没有想象中的增长状态&#xff0c;无论是电脑中的笔记本、数码中的相机&#xff0c;或者是…

ICC2:分段长tree的流程

我正在「拾陆楼」和朋友们讨论有趣的话题,你⼀起来吧? 拾陆楼知识星球入口 分段长tree操作起来方法很多,这里提供两种ICC2分段长tree的方法。有需要的可以试试。 1.用原始sdc长一遍tree,找得到要做subtree部分,并预估latency值。 2.把sdc中添加subtree clock,subtree是…

国产 2443A 峰值功率分析仪

2443A 峰值功率分析仪 频率范围覆盖&#xff1a;9kHz至67GHz 产品综述 2443A峰值功率分析仪由峰值功率分析仪主机和系列化峰值功率探头组成&#xff0c;可用于测量和分析微波毫米波脉冲调制信号的多种幅度和时间参数&#xff0c;是表征脉冲调制信号特性的综合性测量与分析仪器。…

矢量图形编辑软件Illustrator 2023 mac中文版软件特点(ai2023) v27.9

illustrator 2023 mac是一款矢量图形编辑软件&#xff0c;用于创建和编辑排版、图标、标志、插图和其他类型的矢量图形。 illustrator 2023 mac软件特点 矢量图形&#xff1a;illustrator创建的图形是矢量图形&#xff0c;可以无限放大而不失真&#xff0c;这与像素图形编辑软…

gitlab查看、修改用户和邮箱,gitlab生成密钥

查看用户、邮箱 git config user.name git config user.email 修改用户、邮箱 git config --global user.name “xxx” git config --global user.email “xxxxxx.com” 生成ssh密钥 ssh-keygen -t rsa -C “xxxxxx.com” 查看SSH秘钥 cat ~/.ssh/id_rsa.pub 将秘钥复制&…

How to install the console system of i-search rpa on Centos 7

How to install the console system of i-search rpa on Centos 7 1、 准备1.1 、查看磁盘分区状态1.2、上传文件1.2.1、添加上传目录1.2.2、上传安装包1.2.3、解压安装包1.2.4、查看安装包结构 1.3、安装依赖包1.3.1、基础依赖包1.3.2 相关依赖 1.4、关闭防火墙1.5、解除SeLin…

【JAVA学习笔记】47 - 异常,try-catch处理,throw处理

项目代码 https://github.com/yinhai1114/Java_Learning_Code/tree/main/IDEA_Chapter12/scr/com/yinhai/exception_ 〇、异常处理的引入 程序出现一个小问题如int num1 10;int num2 0&#xff1b;num1 / num2 > 10 / 0 会抛出错误&#xff0c;但这样不算致命的小问题就…

Python 编写 Flink 应用程序经验记录(Flink1.17.1)

目录 官方API文档 提交作业到集群运行 官方示例 环境 编写一个 Flink Python Table API 程序 执行一个 Flink Python Table API 程序 实例处理Kafka后入库到Mysql 下载依赖 flink-kafka jar 读取kafka数据 写入mysql数据 flink-mysql jar 官方API文档 https://nigh…

汇编学习(1)

汇编、CPU架构、指令集、硬编码之间的关系 ● 汇编语言&#xff1a;这是一种低级语言&#xff0c;用于与硬件直接交互。它是由人类可读的机器码或指令组成的&#xff0c;这些指令告诉CPU如何执行特定的任务。每条汇编指令都有一个对应的机器码指令&#xff0c;CPU可以理解和执…

css属性clip-path的使用说明

前言 当ui设计上的图片、div等的形状不是长方形&#xff0c;而是多边形的时候&#xff0c;就可以借助clip-path这个css属性来实现。 clip-path CSS 属性使用裁剪方式创建元素的可显示区域。区域内的部分显示&#xff0c;区域外的隐藏。【from: MDN】 clip-path可以理解为一把剪…

Spring Cloud学习:二【详细】

目录 Nacos的配置 Nacos的单机启动 服务注册 Nacos服务分级存储模型 优先访问同集群的服务 根据权重负载均衡 环境隔离Namespace Nacos调用流程 Nacos与Eureka注册对比 Nacos与Eureka的共同点 Nacos与Eureka的区别 Nacos配置管理 统一配置 配置自动刷新 多环境配…

python自动化测试平台开发:自动化测试平台简介

一.测试平台简介 为什么需要测试平台 已有的开源测试平台不能满足需要&#xff0c;不要轻易造轮子 需要公司级别的定制 需要整合公司内部的多套平台 例子&#xff1a;DevOps平台、精准化测试平台、质量监控平台等等 常见的测试平台开发模式 大一统模式&#xff08;适合简单的…

从瀑布模式到水母模式:ChatGPT如何赋能软件研发全流程【文末送书五本】

从瀑布模式到水母模式&#xff1a;ChatGPT如何赋能软件研发全流程 前言内容简介购买链接作者简介专家推荐读者对象参与方式往期赠书回 &#x1f3d8;️&#x1f3d8;️个人简介&#xff1a;以山河作礼。 &#x1f396;️&#x1f396;️:Python领域新星创作者&#xff0c;CSDN实…

【C】关于动态内存的试题及解析

目录 第1题&#xff1a; 第2题&#xff1a; 第3题&#xff1a; 第4题&#xff1a; 第5题&#xff1a; 在学习了关于动态开辟内存的相关知识后&#xff0c;下面是一些涉及到动态开辟内存程序的试题及解析&#xff08;试题部分来自《高质量的C/C编程》、笔试题&#xff09;。 第1…

(1)(1.9) HC-SR04声纳

文章目录 前言 1 连接到自动驾驶仪 2 参数说明 前言 HC-SR04 声纳是一种价格低廉但量程很短&#xff08;最远只有 2m&#xff09;的测距仪&#xff0c;主要设计用于室内&#xff0c;但也成功地在室外的 Copter 上使用过。极短的测距范围使其用途有限。 &#xff01;Warning…

移远通信5G RedCap模组拿下首个中国移动5G物联网开放实验室5G及轻量化产品能力认证

10月21日&#xff0c;在2023世界物联网博览会期间&#xff0c;中国移动举办了以“智融万物 创见未来”为主题的物联网开发者大会暨物联网产业论坛。作为中国移动在物联网领域重要的合作伙伴&#xff0c;移远通信应邀参加论坛。 随着千行百业数智化进程的不断加速&#xff0c;5G…

酷开科技依托酷开系统推动家庭智能化加速发展

为什么越来越多的人会选择智能家居&#xff1f;因为智能家居的出现&#xff0c;大大方便了我们的生活&#xff0c;为生活提供便利舒适的体验&#xff1b;就如同洗衣机与洗碗机解放了我们的双手是一样的道理&#xff0c;智能家居是在生活的方方面面为我们提供更加便利化的可能性…