商城订单模块实战 - 数据库设计、ABA问题处理、读写分离分库分表

news2025/1/15 13:40:00

引言

订单系统可以说是整个电商系统中最重要的一个子系统,因此订单数据可以算作电商企业最重要的数据资产。这篇文章我们来看看在我们的商城系统中订单服务是如何实现的,特别是在设计和实现一个订单系统的过程中有哪些问题是需要特别考虑的。

业务分析

订单系统业务分析

对于一个合格的订单系统,最基本的要求是什么?数据不能出错。用户的每一次购物,从下单开始到支付、发货,再到收货,流程中的每个环节,都需要同步更新订单数据,每次更新操作可能都需要同时更新好几张表。这些操作可能会随机分发到不同的服务器节点上执行,服务器或网络都有可能会出问题,在这么复杂的情况下,如何保证订单数据不出错呢?

  • 第一,代码必须是正确的没有Bug,当然这个要求很简单也很复杂,全是bug系统无法正常运行,但是也没有什么系统能保证没有一个bug。当然要确保不能因为代码Bug而导致数据错误。
  • 第二,要能够正确地使用事务。比如,在创建订单的时候,如果需要同时在订单表和订单商品表中插入数据,那么我们必须在一个数据库事务中执行这些插入数据的INSERT 语句,数据库事务可以确保:执行需要同时进行的操作语句时,要么一起成功,要么一起失败。而实际上,在微服务下,仅仅使用数据库事务是不够的,很多时候还需要分布式事务。后面课程会专门的讲解分布式事务在项目中的实现。

即使满足了上面列举的这两个基本要求,某些特殊情况也仍然可能会引发数据错误,是什么样的数据错误问题?如何解决呢?都都会在后续讲述

在此之前,我们需要首先了解对于一个订单系统而言,它的核心功能和数据结构是怎样的。其实任何一个公司的电商系统,其订单系统的功能都是独一无二的,因为订单系统会基于其业务配置了很多的功能,并且都很复杂。因此我们的电商系统只能化繁为简,聚焦那些最核心的、共通的业务功能和数据模型,并且以此为基础讨论其中的实现技术。

订单系统的核心功能和数据表

为了支撑订单模块的必备功能,一般订单数据库中至少需要具备如下4张表。

  • 订单主表:也称订单表,用于保存订单的基本信息,也就是我们的order表。
  • 订单商品表:用于保存订单中的商品信息,也就是我们的order_item表。
  • 订单支付表:用于保存订单的支付和退款信息。
  • 订单优惠表:用于保存订单使用的所有优惠信息。

这4张表之间的关系是订单主表与后面的几个子表都是一对多的关系,关联的外键就是订单主表的主键,即订单ID。

在我们的商城系统中,做了适度的改造和简化,表现在:
因为取消了一般的促销优惠,自然没有专门的订单优惠表;
订单的是否支付和是否退款直接保存在订单主表中,没有设计单独订单支付表用以保存支付和退款相关信息;
我们系统中没有单独的库存系统,所以库存扣减由订单系统发起,由产品服务执行实际的扣减库存操作。
对应到存储上,则是oms_order作为订单主表,其中的status字段表示了订单状态,包括

0->待付款;1->待发货;2->已发货;3->已完成;4->已关闭;5->无效订单

产品的库存则是保存在产品服务product对应数据库tl_goods的sku_stock表中。

从上面我们对订单的流程描述来看,订单系统的实现其实并不复杂,就是标准的CRUD。

但是前面我们也说过,某些情况也仍然可能会引发数据错误,是哪些情况呢?我们来一一分析下。

订单重复下单问题

仔细分析一下订单创建的场景:订单系统为用户提供创建订单的HTTP接口,用户在浏览器页面上点击“提交订单”按钮,浏览器向订单系统发送一条创建订单的请求,订单系统的后端服务收到请求,向数据库的订单表中插入一条订单数据,至此,订单创建成功。

那么我们设想一下,用户在点击“提交订单”的按钮时,不小心点了两下,那么浏览器就会向服务端连续发送两条创建订单的请求,最终的结果将会是什么?很自然会创建两条一模一样的订单。这样肯定是不行的,因此我们还需要做好防重工作,怎么做呢?

可能有人会想到,前端页面上应该防止用户重复提交表单,当用户“提交订单”的按钮后,将该按钮置灰不可用。但是仔细想想,即使前端控制了用户不重复提交,网络错误也有可能会导致重传,很多RPC框架和网关都拥有自动重试机制,所以对于订单服务来说,重复请求的问题是客观存在的。

解决办法是,让订单服务具备幂等性。什么是幂等性?幂等操作的特点是,操作任意多次执行所产生的影响,均与一次执行所产生的影响相同。也就是说,对于幂等方法,使用同样的参数,对它进行多次调用和一次调用,其对系统产生的影响是一样的。

例如:

update tableA set count = 10 where id = 1

这个操作多次执行,id 等于1的记录中的 count字段的值都为10,这个操作就是幂等的,我们不用担心这个操作被重复。

update tableA set count = count + 1 where id = 1;

这样的SQL操作就不是幂等的,一旦重复,结果就会产生变化。

所以,不用担心幂等方法的重复执行会对系统造成任何改变。如果创建订单的服务具备幂等性,那么无论创建订单的请求发送了多少次,正确的结果都是数据库只有一条新创建的订单记录。

这里又会涉及一个不太好解决的问题:对于订单服务来说,如何判断收到的创建订单的请求是不是重复请求呢?

在插入订单数据之前,先查询一下订单表里面有没有重复的订单,是不是就可以做出判斯了呢?这个方法看起来容易。实际上却很难实现。原因是我们很难通过SQL的WHERE语句来定义“重复的订单”,如果订单的用户、商品、数量和价格都一样,是否就能认为它们是重复订单呢?这个其实是无法确定的,因为有可能用户就是连续下了两个一模一样的订单。

这个问题的思路是利用数据库的唯一约束来判断数据是否重复。在数据库的最佳实践中,其中一条是要求数据库的每个表都有主键。在非分库分表的情况下,我们在向数据库的表中插入一条记录的时候,无需提供主键,插入的同时由数据库自动生成一个主键。这样,重复的请求就会导致插入重复的数据。

表的主键是自带唯一约束的,如果我们在一条INSERT语句中提供了主键,并且这个主键的值已经存在于表中,那么这条INSERT语句就会执行失败,数据也不会成功插入表中。我们可以利用数据库的这种“主键唯一约束”特性,在插入数据的时候带上主键,来解决创建订单服务的幂等性问题。

具体做法如下:首先,为订单系统增加一个“生成订单号”的服务,这个服务没有参数,返回值就是一个新的、全局唯一的订单号。在用户进入创建订单的页面时,前端页面会先调用这个生成订单号的服务得到一个订单号,在用户提交订单的时候,在创建订单的请求中带着这个订单号。

这个订单号就是订单表的主键,这样,无论是用户原因,还是网络原因等各种情况导致的重试,这些重复请求中的订单号都是相同的。订单服务在订单表中插入数据的时候,这些重复的INSERT语句中的主键,都是同一个订单号。数据库的主键唯一约束特性就可以保证,只有一次INSERT语句的执行是成功的,这样就实现了创建订单服务的幂等性。

时序图如下:
在这里插入图片描述

所以,可以看到,在PortalOrderController中专门提供了generateOrderId方法供外部系统获得订单ID。

而秒杀系统相关的微服务中虽然没有提供类似的generateOrderId方法,但依然注意了避免重复下单问题,在生成订单确认信息时,将预先生成的订单ID传递给了前端订单确认页。

还有一点需要注意的是,在具体实现时,如果是因为重复订单导致插入订单表的语句失败,那么订单服务就不要再把这个错误返回给前端页面了。否则,就有可能会出现用户点击创建订单按钮后,页面提示创建订单失败,而实际上订单已经创建成功了。正确的做法是,遇到这种情况,订单服务直接返回“订单创建成功”的响应即可。
要做到这一点,可以捕获java.sql.SQLIntegrityConstraintViolationException或者org.springframework.dao.DuplicateKeyException来实现。

订单ABA问题和解决

订单系统中,各种更新订单的服务同样也需要具备幂等性。

更新订单的服务,比如支付、发货等这些步骤中的更新订单操作,最终都会落到订单库上,都是对订单主表进行更新操作。

比如对支付操作的数据库的更新操作、无论是执行一次还是重复执行多次,订单状态都是已支付,不用我们额外设置任何逻辑,这就是天然幂等性。

在实现这些更新订单的服务时,还有哪些问题需要特别注意呢?在并发环境下,我们需要特别注意ABA问题。

什么是更新下的ABA问题呢?我们知道并发编程下的CAS有ABA问题,这个ABA问题和并发的ABA问题有相似之处。我们来看这么一个例子:
订单支付完成,填入物流单号666提交后,发现填错了,修改成正确的单号888,对于订单服务来说,这里就产生了两个更新订单的请求。
按照我们的设想,正常情况下,订单中的快递单号会先更新成666,再更新成888,这是没有问题的。但是现实生活有很多不正常的情况,比如,更新成666的请求到了,快递单号更新成666,然后更新成888的请求到了,快递单号又更新成888。但是订单服务在向调用方返回666更新成功的响应时,这个响应在网络传输过程中丢失了。如果调用方没有收到成功响应,触发自动重试逻辑,再次发起更新成666的请求,快递单号将会再次更新成666,这种情况下数据显然就会出错了。这就是ABA问题。

那么ABA问题应该怎么解决呢?仔细想想并发编程里怎么解决ABA问题的?版本戳。所以这里同样可以使用版本戳。

为订单主表增加一列,列名可以叫 version、也就是“版本号”的意思。每次查询订单的时候,版本号需要随着订单数据返回给页面。页面在更新数据的请求时,需要把该版本号作为更新请求的参数再带回给订单更新服务。

订单服务在更新数据的时候需要比较订单当前数据的版本号与消息中的版本号是否一致,如果不一致就拒绝更新数据。如果版本号一致,则还需要在更新数据的同时,把版本号加1。当然需要特别注意的是,“比较版本号、更新数据和把版本号加1”这个过程必须在同一个事务里面执行,只有这一系列操作具备原子性,才能真正保证并发操作的安全性。

具体的SQL语句参考如下:

UPDATE orders set tracking_number = 666,version = version + 1 WHERE version = ?;

版本号的机制可用于保证,从打开某条订单记录开始,一直到这条订单记录更新成功,这期间不会存在有其他人修改过这条订单数据的情况。因为如果被其他人修改过,数据库中的版本号就会发生改变,那么更新订单的操作就不会执行成功,而只能重新查询新版本的订单数据,然后再尝试更新。

所以可以看到,在order中专门设计了version字段:
在这里插入图片描述

但是因为牵涉到更新订单的操作未执行成功(表现为update语句返回行数为0)时的重试机制,代码修改较大,所以在OmsOrderMapper.xml和相关订单业务方法中没有实现上述的ABA解决方案,感兴趣的同学可以自行调整。

总的来说,因为网络、服务器等导致的不确定因素,重试请求是普遍存在且不可避免的问题。具有幂等性的服务可以克服由于重试问题而导致的数据错误。

所以,总的来说,对于创建订单的服务,可以通过预先生成订单号作为主键,然后利用数据库中“主键唯一约束”的特性,避免重复写入订单,实现创建订单服务的幂等性。对于更新订单的服务,可以通过一个版本号机制,即在每次更新数据之前校验版本号,以及在更新数据的同时自增版本号这样的方式来解决ABA问题,以确保更新订单服务的幂等性。

通过这样两种具备幂等性的实现方法,我们可以保证,无论是不是重复请求,订单表中的数据都是正确的。

当然这里讲到的实现订单幂等性的方法,在其他需要实现幂等性的服务中也完全可以套用,只需要这个服务操作的数据保存在数据库中,并且数据表带有主键即可。

实现服务幂等性的方法,远不止本章介绍的这两种,其实,实现幂等性的方法可分为两大类,一类是通过一些精巧的设计让更新本身就是幂等的,这种方法并不能适用于所有的业务。另一类是利用外部的具备一致性的存储(比如 MySQL)来做冲突检测,在设计幂等方法的时候,通常可以顺着这两个思路来展开。

读写分离与分库分表

使用Redis 作为MySQL的前置缓存,可以帮助MySQL挡住绝大部分的查询请求。这种方法对于像电商中的商品系统、搜索系统这类与用户关联不大的系统、效果特别好。因为在这些系统中、任何人看到的内容都是一样的,也就是说,对后端服来说,任何人的查询请求和返回的数据都是一样的。在这种情况下,Redis 缓存的命中率非常高,几乎所有的请求都可以命中缓存。

但是与用户相关的系统(不是用户系统本身,用户信息等相关数据在用户登录时进行缓存,就价值很高),使用缓存的效果就没有那么好了,比如,订单系统、账户系统、购物车系统、订单系统等等。对于这些系统而言,各个用户查询的信息与用户自身相关,即使同一个功能界面,用户看到的数据也是不一样的。

比如,“我的订单”这个功能,用户看到的都是自己的订单数据。在这种情况下,缓存的命中率就比较低了,会有相当一部分查询请求因为命中不了缓存,穿透到 MySQL 数据库中。

随着系统的用户数量越来越多,穿透到MySQL 数据库中的读写请求也会越来越多,当单个MySQL支撑不了这么多的并发请求时,该怎么办?

读写分离

读写分离是提升 MySQL 并发能力的首选方案,当单个MySQL无法满足要求的时候,只能用多个MySQL实例来承担大量的读写请求。MySQL与大部分常用的关系型数据库一样,都是典型的单机数据库,不支持分布式部署。用一个单机数据库的多个实例组成一个集群,提供分布式数据库服务,是一件非常困难的事情。

一个简单且非常有效的是用多个具有相同数据的MySOL实例来分担大量查询请求,也就是“读写分离”。很多系统,特别是互联网系统,数据的读写比例严重不均衡,读写比例一般在9:1到几十比1,即平均每发生几十次查询请求,才会有一次更新请求,那就是说数据库需要应对的绝大部分请求都是只读查询请求。

分布式存储系统支持分布式写是非常困难的,因为很难解决好数据一致性的问题。但分布式读相对来说就简单得多,能够把数据尽可能实时同步到只读实例上,它们就可以分担大量的查询请求了。

读写分离的另一个好处是,实施起来相对比较简单。把使用单机MySQL的系统升级为读写分离的多实例架构非常容易,一般不需要修改系统的业务逻辑,只需要简单修改DAO (Data Access Object,一般指应用程序中负责访问数据库的抽象层)层的代码,把对数据库的读写请求分开,请求不同的MySQL实例就可以了。通过读写分离这样一个简单的存储架构升级,数据库支持的并发数量就可以增加几倍到十几倍。所以,当系统的用户数越来越多时,读写分离应该是首要考虑的扩容方案。

在这里插入图片描述

主库负责执行应用程序发来的数据更新请求,然后将数据变更同步到所有的从库中。这样,主库和所有从库中的数据一致,多个从库可以共同分担应用的查询请求。

读写分离的数据不一致问题

读写分离的一个副作用是,可能会存在数据不一致的问题。原因是数据库中的数据在主库完成更新后,是异步同步到每个从库上的,这个过程会有一个微小的时间差。正常情况下,主从延迟非常小,以几毫秒计。但即使是这样小的延迟,也会导致在某个时刻主库和从库上数据不一致的问题。

应用程序需要能够接受并克服这种主从不一致的情况,否则就会引发一些由于主从延迟而导致的数据错误。

回顾我们的订单系统业务,用户对购物车发起商品结算创建订单,进入订单页,打开支付页面进行支付,支付完成后,按道理应该再返回到支付之前的订单页。但如果这时马上自动返回到订单页,就很有可能会出现订单状态还是显示“未支付”的问题。因为支付完成后,订单库的主库中订单状态已经更新了,但订单页查询的从库中这条订单记录的状态可能还未更新,如何解决这种问题呢?

其实这个问题并没有特别好的技术手段来解决,所以可以看到,稍微上点规模的电商网站并不会支付完成后自动跳到到订单页,而是增加了一个支付完成页面,这个页面其实没有任何新的有效信息,就是告诉你支付成功的信息。如果想再查看一下刚刚支付完成的订单,需要手动选择,这样就能很好地规避主从同步延迟的问题。

如果是那些数据更新后需要立刻查询的业务,这两个步骤可以放到一个数据库事务中,同一个事务中的查询操作也会被路由到主库,这样就可以规避主从不一致的问题了,还有一种解决方式则是对查询部分单独指定进行主库查询。

总的来说,对于这种因为主从延迟而带来的数据不一致问题,并没有一种简单方便且通用的技术方案可以解决,对此,我们需要重新设计业务逻辑,尽量规避更新数据后立即去从库查询刚刚更新的数据。

分库分表

除了访问MySQL的并发问题,还要解决海量数据的问题,很多的时候,我们会使用分布式的存储集群,因为MySQI本质上是一个单机数据库,所以很多场景下,其并不适合存储TB级别以上的数据。

但是绝大部分电商企业的在线交易类业务,比如订单、支付相关的系统,还是无法离开MySQL的。原因是只有MySOL 之类的关系型数据库,才能提供金融级的事务保证。目前的分布式事务的各种解法方案多少都有些不够完善。

虽然 MySQL 无法支持这么大的数据量,以及这么高的并发需求,但是交易类系统必须用它来保证数据一致性,那么,如何才能解决这个问题呢?这个时候我们就要考虑分片,也就是拆分数据。

如果一个数据库无法支撑1TB的数据,那就把它拆分成100个库,每个库就只有10GB的数据了。这种拆分操作就是MySOL的分库分表操作。

如何规划分库分表

以订单表为例,首先,我们需要思考的问题是,选择分库还是分表,或者两者都有,分库就是把数据拆分到不同的MySQL 数据库实例中,分表就是把数据拆分到一个数据库的多张表里面。

在考虑到底是选择分厍还是分表之前,我们需要首先明确一个原则,那就是能小拆就小非,能少抖就小多拆。原因很简单,数据拆得越分散,并发和维护就越麻烦,系统出问题的概率也就越大。

遵循上面这个原则,还需要进一步了解,哪种情况适合分表,哪种情况适合分库。选择分厍或是分表的目的是解决如下两个问题。

第一,是为了解决因数据量太大而导致查询慢的问题。这里所说的“查询”,其实主要是事务中的查询和更新操作,因为只读的查询可以通过缓存和主从分离来解决。分表主要用于解决因数据量大而导致的查询慢的问题。

第二,是为了应对高并发的问题。如果一个数据库实例撑不住,就把并发请求分散到多个实例中,所以分库可用于解决高并发的问题。
简单地说,如果数据量太大,就分表;如果并发请求量高,就分库。一般情况下,我们的解决方案大都需要同时做分库分表,我们可以根据预估的并发量和数据量,分别计算应该拆分成多少个库以及多少张表。

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

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

相关文章

逾 200 家港企参与! GoGBA大湾区发展日(广州)圆满举行

2023年4月26日 – 由香港特别行政区政府政制及内地事务局粤港澳大湾区发展办公室、香港特别行政区政府驻粤经济贸易办事处(驻粤办)、香港贸易发展局(香港贸发局)广州办事处,以及香港贸发局GoGBA商贸支援合办的GoGBA大湾…

BSN-DDC基础网络详解(十):官方DDC应用SDK

官方 SDK 是 BSN 联盟为平台方推出的可快速接入 DDC 网络的工具包,目前 DID 和各个开放联盟链的官方 DDC SDK 都使用 Java 语言开发,其它主流语言的 SDK 根据市场反馈我们将陆续增加。如果算力中心方和平台方的业务系统的开发语言与 SDK 不匹配&#xff…

基于DSP+FPGA+ADS1282支持31Bit高精度数据采集方案(一)

3.1 系统需求分析 3.1.1 系统功能设计要求 本硬件处理平台的主要任务有三类,一是数据采集,包括采集惯性测量元件 的输出信号,接收外部系统校正信息,如 GPS 信息等;二是数据处理与计算,包 括惯性测量…

如何实现自动化按图片搜索淘宝商品(拍立淘)功能?拍立淘API接口item_search_img

我们都知道淘宝平台推出了拍立淘功能,如果大家遇到了自己喜欢的商品,就可以拍一张照片,在淘宝用拍立淘搜索就能够出现相似的同款,这样就不用再去找别人要链接了。淘宝拍立淘主要是通过图片识别来找相似主图的宝贝,那么…

基于JavaSpringmvc+myabtis+html的鲜花商城系统设计和实现

基于JavaSpringmvcmyabtishtml的鲜花商城系统设计和实现 博主介绍:5年java开发经验,专注Java开发、定制、远程、指导等,csdn特邀作者、专注于Java技术领域 作者主页 超级帅帅吴 Java项目精品实战案例《500套》 欢迎点赞 收藏 ⭐留言 文末获取源码联系方式…

分布式的流处理平台Kafka

目录: 一、简介二、基本概念三、生产者使用详解四、发送消息五、消费者代码示例 一、简介 ApacheKafka 是一个分布式的流处理平台。它具有以下特点: 支持消息的发布和订阅,类似于 RabbtMQ、ActiveMQ 等消息队列;支持数据实时处理…

从零开始实现VAE和CVAE

扩散模型可以看作是一个层次很深的VAE(变分自编码器),前向(forward,或者译为正向)的过程,通过在多个尺度上添加噪声来逐步扰乱数据分布;然后是反向的过程,去学习如何恢复数据结构,上…

喜报 | 国家发明专利证书! 再添2项!

​近日,擎创科技自主研发的《一种基于倒序表的实时日志聚类分析方法》以及《一种基于社区检测的运维告警场景生成方法》正式获得国家颁发的发明专利证书!擎创的专业性、自主性、创新能力、技术水平以及研发实力在得到了确切的肯定。 作为智能运维领域领先…

DJ4-5 路由算法:LS 和 DV

目录 一、迪杰斯特拉算法 1. 术语定义 2. 算法描述 3. 举例说明 4. 构建从源节点到目的节点的路径 5. 构建最低费用路径树 6. 构建转发表 二、距离向量路由算法 1. 术语定义 2. 举例说明 3. 距离向量表 4. 更新距离向量表 5. 举例说明 三、距离向量路由算法 PLUS…

多维评测指标解读2022MSU世界编码器大赛结果

是极致性能,更是最佳商用。 19项第一之上,是63%的极致带宽降低 近日,2022 MSU世界视频编码器大赛成绩正式揭晓。报告显示,阿里媒体处理服务MPS(Alibaba Media Processing Service)s264及s265编码器共计斩获…

【黑马旅游案例记录(结合ES)】

黑马旅游案例记录 11.9.黑马旅游案例11.9.1.酒店搜索和分页11.9.1.1.需求分析11.9.1.2.定义实体类11.9.1.3.定义controller11.9.1.4.实现搜索业务 11.9.2.酒店结果过滤11.9.2.1.需求分析11.9.2.2.修改实体类11.9.2.3.修改搜索业务 11.9.3.我周边的酒店11.9.3.1.需求分析11.9.3.…

10 【Sass语法介绍-继承】

1.前言 在我们编写样式的时候,很多情况下我们几个不同的类会有相同的样式代码,同时这几个类又有其自己的样式代码,这使我们就可以通过 Sass 提供的继承 extend 来实现。本节内容我们将讲解 Sass 继承的语法以及继承的多重延伸等等&#xff0…

【无功功率控制】连接到无限电网的小型风电场的无功功率控制(Simulink)

💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…

MongoDB【常用命令】

目录 1:基本常用命令 1.1:演示案例 1.2:数据库操作 1.2.1:选择和创建数据库,查看当前正在使用的数据库命令 1.2.2:数据库的删除 1.3:集合操作 1.3.1:集合的显式创建&#xff0…

安全意识培训:如何提高员工网络安全意识?

随着网络技术的不断发展和应用,网络安全已经成为企业必须关注和重视的问题。尤其是在今天,企业数字化转型的大背景下,网络安全问题日益凸显。对于企业而言,员工是企业安全的第一道防线,提高员工的网络安全意识已经成为…

制作自己的镜像并且推送到docker hub上去。

1、在docker hub(Docker)注册账号:比如我的账号是:zhangyi0833 2、在本机上制作自己已经安装了自己想要的工具的镜像,比如我这里安装了cgdb在centos8上面。通过命令制作自己的镜像: docker commit -m"提交的描述信息" -a"镜像…

如何复刻Midjourney的成功?

AI绘画的大模型和应用非常多,但最有名的非Stable Diffusion和Midjourney莫属,其中,尤其是Midjourney(以下简称MJ),仅11位成员,8个研发人员中的一半都是尚未毕业的本科生,从未融资,成立3年,千万用…

(原创)Flutter基础入门:手把手教你搭建Flutter混合项目:模块代码依赖方式集成

前言 Flutter是Google开源的构建用户界面(UI)工具包 支持在不同平台构建一致的ui效果 但在实际业务中,一般不会整个APP都用纯Flutter开发 尤其一些老的项目,会采用接入Flutter的方式来混合开发 那么今天就主要讲一下如何搭建一个…

外卖app开发流程全解析

外卖app开发是现代餐饮业的一个必备部分。在这个数字化时代,人们更愿意使用手机应用程序来订购食品。因此,为了满足客户需求,餐饮企业需要开发自己的外卖app。 第一步:确定目标受众 在开始外卖app的开发之前,需要确定…

Shiro-721---漏洞复现

漏洞原理 Shiro rememberMe 反序列化远程代码执行漏洞 由于 Apache Shiro cookie 中通过 AES-128-CBC 模式加密的 rememberMe 字段存 在问题,用户可通过 Padding Oracle 加密生成的攻击代码来构造恶意的 rememberMe 字段,并重新请求网站,进…