eShopOnContainers 知多少[8]:Ordering microservice

2023-11-06 13:48

本文主要是介绍eShopOnContainers 知多少[8]:Ordering microservice,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 引言

Ordering microservice(订单微服务)就是处理订单的了,它与前面讲到的几个微服务相比要复杂的多。主要涉及以下业务逻辑:

  1. 订单的创建、取消、支付、发货

  2. 库存的扣减

2. 架构模式

640?wx_fmt=png

如上图所示,该服务基于CQRS 和DDD来实现。

640?wx_fmt=png

从项目结构来看,主要包括7个项目:

  1. Ordering.API:应用层

  2. Ordering.Domain:领域层

  3. Ordering.Infrastructure:基础设施层

  4. Ordering.BackgroundTasks:后台任务

  5. Ordering.SignalrHub:基于Signalr的消息推送和实时通信

  6. Ordering.FunctionalTests:功能测试项目

  7. Ordering.UnitTests:单元测试项目

从以上的项目定义来看,该微服务的设计并符合DDD经典的四层架构。

640?wx_fmt=png

核心技术选型:

  1. ASP.NET Core Web API

  2. Entity Framework Core

  3. SQL Server

  4. Swashbuckle(可选)

  5. Autofac

  6. Eventbus

  7. MediatR

  8. SignalR

  9. Dapper

  10. Polly

  11. FluentValidator

3. 简明DDD

领域驱动设计是一种方法论,用于解决软件复杂度问题。它强调以领域为核心驱动设计。主要包括战略和战术设计两大部分,其中战略设计指导我们在宏观层面对问题域进行识别和划分,从而将大问题划分为多个小问题,分而治之。而战术设计从微观层面指导我们如何对领域进行建模。640?wx_fmt=png其中战术设计了引入了很多核心要素,指导我们建模:

  1. 值对象(Value Object)

  2. 实体(Entity)

  3. 领域服务(Domain Service)

  4. 领域事件(Domain Event)

  5. 资源库(Repository)

  6. 工厂(Factory)

  7. 聚合(Aggregate)

  8. 应用服务(Application Service) 640?wx_fmt=png

其中实体、值对象和领域服务用于表示领域模型,来实现领域逻辑。 聚合用于封装一到多个实体和值对象,确保业务完整性。 领域事件来丰富领域对象之间的交互。 工厂、资源库用于管理领域对象的生命周期。 应用服务是用来表达用例和用户故事。

有了以上的战术设计要素还不够,如果它们糅合在一起,还是会很混乱,因此DDD再通过分层架构来确保关注点分离,即将领域模型相关(实体、值对象、聚合、领域服务、领域事件)放到领域层,将资源库、工厂放到基础设施层,将应用服务放到应用层。以下就是DDD经典的四层架构:640?wx_fmt=png

以上相关图片来源于:张逸 · 领域驱动战略设计实践

4. Ordering.Domain:领域层

640?wx_fmt=png

如果对订单微服务应用DDD,那么要摒弃传统的面向数据库建模的思想,转向领域建模。该项目中主要定义了以下领域对象:

  • Order:订单

  • OrderItem:订单项

  • OrderStatus:订单状态

  • Buyer:买家

  • Address:地址

  • PaymentMethod:支付方式

  • CardType:银行卡片类型

在该示例项目中,定义了两个聚合:订单聚合和买家聚合,其中Order和Buyer分属两个聚合根,其中订单聚合通过持有买家聚合的唯一ID进行关联。如下图所示:640?wx_fmt=png

我们依次来看其对实体、值对象、聚合、资源库、领域事件的实现方式。

4.1. 实体、值对象与聚合

640?wx_fmt=png

实体与值对象最大的区别在于,实体有标识符可变,值对象不可变。为了保证领域的不变性,也就是更好的封装,所有的属性字段都设置为 privateset,集合都设置为只读的,通过构造函数进行初始化,通过暴露方法供外部调用修改。 从类图中我们可以看出,其主要定义了一个 Entity抽象基类,所有的实体通过继承 Entity来实现命名约定。这里面有两点需要说明:

  1. 通过 Id属性确保唯一标识符

  2. 重写 Equals和 GetHashCode方法(hash值计算: this.Id.GetHashCode()^31)

  3. 定义 DomainEvents来存储实体关联的领域事件(领域事件的发生归根结底是由于领域对象的状态变化引起的,而领域对象[实体、值对象和聚合])中值对象是不可变的,而聚合往往包含多个实体,所以将领域事件关联在实体上最合适不过。)

640?wx_fmt=png

同样,值对象也是通过继承抽象基类 ValueObject来进行约定。其主要也是重载了 EqualsGetHashCode和方法。这里面有必要学习其 GetHashCode的实现技巧:

 
  1. // ValueObject.cs

  2. protected abstract IEnumerable<object> GetAtomicValues();

  3. public override int GetHashCode()

  4. {

  5.    return GetAtomicValues()

  6.     .Select(x => x != null ? x.GetHashCode() : 0)

  7.     .Aggregate((x, y) => x ^ y);

  8. }


  9. //Address.cs

  10. protected override IEnumerable<object> GetAtomicValues()

  11. {

  12.    // Using a yield return statement to return each ele

  13.    yield return Street;

  14.    yield return City;

  15.    yield return State;

  16.    yield return Country;

  17.    yield return ZipCode;

  18. }

可以看到,通过在基类定义 GetAtomicValues方法,用来要求子类指定需要hash的字段,然后将每个字段取hash值,然后通过异或运算再行聚合得到唯一hash值。

所有对聚合中领域对象的操作都是通过聚合根来维护的。因此我们可以看到聚合根中定义了许多方法来处理领域逻辑。

4.2. 仓储

640?wx_fmt=png聚合中的领域对象的持久化借助仓储来完成的。其提供统一的入口来进行聚合内相关领域对象的CRUD,从而完成透明持久化。从图中看出, IRepository定义了一个 IUnitOfWork属性,其代表工作单元,主要定义了两个方法 SaveChangesAsyncSaveEntitiesAsync,借助事务一次性提交所有更改,以确保数据的完整性和有效性。

4.3. 领域事件

640?wx_fmt=png

从类图中可以看出一个共同特征,都实现了 INotification接口。对MediatR熟悉的肯定一眼就明白了。是的,这个是 MediatR中定义的接口。借助MediatR,来实现事件处理管道。通过进程内事件处理管道来驱动命令接收,并将它们(在内存中)路由到正确的事件处理器。 关于MeidatR可以参考我的这篇博文:MediatR 知多少

而关于领域事件的处理,是通过继承 INotificationHanlder接口来实现,这样 INotificationINotificationHandler通过Ioc容器的服务注册,自动完成事件的订阅。而领域事件的处理其下放到了 Ordering.Api中处理了。这里大家可能会有疑惑,既然叫领域事件,那为什么领域事件的处理不放到领域层呢?我们可以这样理解,事件是领域内触发,但对事件的处理,其并非都是业务逻辑的相关处理,比如订单创建成功后发送短信、邮件等就不属于业务逻辑。

eShopOnContainers中领域事件的触发时机并非是即时触发,选择的是延迟触发模式。具体的实现,后面会讲到。

5. Ordering.Infrastructure:基础设施层

基础设施层主要用于提供基础服务,主要是用来实体映射和持久化。

640?wx_fmt=png

从图中可以看到,主要包含以下业务处理:

  1. 实体类型映射

  2. 幂等性控制器的实现

  3. 仓储的具体实现

  4. 数据库上下文的实现(UnitOfWork的实现)

  5. 领域事件的批量派发

这里着重下第2、4、5点的介绍。

5.1. 幂等性控制器

幂等性是指某个操作多次执行但结果相同,换句话说,多次执行操作而不改变结果。举例来说:我们在写预插脚本时,会添加条件判断,当表中不存在数据时才将数据插入到表中。无论重复运行多少次 SQL 语句,结果一定是相同的,并且结果数据会包含在表中。

那怎样确保幂等性呢?一种方式就是确保操作本身的幂等性,比如可以创建一个表示“将产品价格设置为¥25”而不是“将产品价格增加¥5”的事件。此时可以安全地处理第一条消息,无论处理多少次结果都一样,而第二个消息则完全不同。 但是假设价格是一个时刻在变的,而你当前的操作就是要将产品价格增加¥5怎么办呢?显然这个操作是不能重复执行的。那我如何确保当前的操作只执行一次呢? 一种简便的方法就是记录每次执行的操作。该项目中的 Idempotency文件夹就是来做这件事的。

640?wx_fmt=png

从类图来看很简单,就是每次发送事件时生成一个唯一的Guid,然后构造一个 ClientRequest对象实例持久化到数据库中,每次借助MediatR发送消息时都去检测消息是否已经发送。

640?wx_fmt=png

5.2. UnitOfWork(工作单元的实现)

640?wx_fmt=png

从代码来看,主要干了两件事:

  1. 在提交变更之前,触发所有的领域事件

  2. 批量提交变更

这里需要解释的一点是,为什么要在持久化之前而不是之后进行领域事件的触发呢? 这种触发就是延迟触发,将领域事件的发布与领域实体的持久化放到一个事务中来达到一致性。 当然这有利有弊,弊端就是当领域事件的处理非常耗时,很有可能会导致事务超时,最终导致提交失败。而避免这一问题,也只有做事务拆分,这时就要考虑最终一致性和相应的补偿措施,显然更复杂。

至此,我们可以总结下聚合、仓储与数据库之间的关系,如下图所示。640?wx_fmt=png

6. Ordering.Api:应用层

应用层通过应用服务接口来暴露系统的全部功能。在这里主要涉及到:

  1. 领域事件的处理

  2. 集成事件的处理

  3. CQRS的实现

  4. 服务注册

  5. 认证授权

  6. 集成事件的订阅

640?wx_fmt=png

6.1. 领域事件和集成事件

对于领域事件和集成事件的处理,我们需要先明白二者的区别。领域事件是发生在领域内的通信(同步或异步均可),而集成事件是基于多个微服务(其他限界上下文)甚至外部系统或应用间的异步通信。 领域事件是借助于MediatR的 INotificationINotificationHandler的接口来实现。

其中 Application/Behaviors文件夹中是实现MediatR中的 IPipelineBehavior接口而定义的请求处理管道。

640?wx_fmt=png

集成事件的发布订阅是借助事件总线来完成的,关于事件总线之前有文章详述,这里不再赘述。在此,仅代码举例其订阅方式。

 
  1. private void ConfigureEventBus(IApplicationBuilder app)

  2. {

  3.    var eventBus = app.ApplicationServices.GetRequiredService<BuildingBlocks.EventBus.Abstractions.IEventBus>();


  4.    eventBus.Subscribe<UserCheckoutAcceptedIntegrationEvent, IIntegrationEventHandler<UserCheckoutAcceptedIntegrationEvent>>();

  5. // some other code

  6. }

6.2. 基于MediatR实现的CQRS

CQRS(Command Query Responsibility Separation):命令查询职责分离。是一种用来实现数据模型读写分离的架构模式。顾名思义,分为两大职责:

  1. 命令职责

  2. 查询职责

其核心思想是:在客户端就将数据的新增修改删除等动作和查询进行分离,前者称为Command,通过Command Bus对领域模型进行操作,而查询则从另外一条路径直接对数据进行操作,比如报表输出等。

640?wx_fmt=png

对于命令职责,其是借助于MediatR充当的CommandBus,使用 IRequest来定义命令,使用 IRequestHandler来定义命令处理程序。我们可以看下 CancelOrderCommandCancelOrderCommandHandler的实现。

 
  1. public class CancelOrderCommand : IRequest<bool>

  2. {


  3.    [DataMember]

  4.    public int OrderNumber { get; private set; }


  5.    public CancelOrderCommand(int orderNumber)

  6.    {

  7.        OrderNumber = orderNumber;

  8.    }

  9. }


  10. public class CancelOrderCommandHandler : IRequestHandler<CancelOrderCommand, bool>

  11. {

  12.    private readonly IOrderRepository _orderRepository;


  13.    public CancelOrderCommandHandler(IOrderRepository orderRepository)

  14.    {

  15.        _orderRepository = orderRepository;

  16.    }


  17.    public async Task<bool> Handle(CancelOrderCommand command, CancellationToken cancellationToken)

  18.    {

  19.        var orderToUpdate = await _orderRepository.GetAsync(command.OrderNumber);

  20.        if(orderToUpdate == null)

  21.        {

  22.            return false;

  23.        }


  24.        orderToUpdate.SetCancelledStatus();

  25.        return await _orderRepository.UnitOfWork.SaveEntitiesAsync();

  26.    }

  27. }

以上代码中,有一点需要指出,就是所有Command中的属性都定义为 privateset,通过构造函数进行赋值,以确保Command的不变性。

对于查询职责,通过定义查询接口,借助Dapper直接写SQL语句来完成对数据库的直接读取。640?wx_fmt=png

而对于定义的命令,为了确保每个命令的合法性,通过引入第三方Nuget包 FluentValdiation来进行命令的合法性校验。其代码也很简单,参考下图。640?wx_fmt=png

6.3. 服务注册

整个订单微服务中所有服务的注册,都是放到应用层来做的,在 Ordering.Api\Infrastructure\AutofacModules文件夹下通过继承 Autofac.Module定义了两个Module来进行服务注册:

  • ApplicationModule:自定义接口相关服务的注册

  • MediatorModule:Mediator相关接口服务的注册

将所有的服务注册都放到高层模块来进行注册,有点违背关注点分离,各层应该关注本层的服务注册,所以这中实现方式是有待改进的。而具体如何改进,这里给大家提供一个线索,可参考ABP是如何实现进行服务注册的分离和整合的。

这里顺带提一下 Autofac这个Ioc容器的一个限制,就是所有的服务注册必须在程序启动时完成注册,不允许运行时动态注册。

7. Ordering.BackgroundTasks:后台任务

后台任务,顾名思义,后台静默运行的任务,也称计划任务。在.NET Core 中,我们将这些类型的任务称为托管服务,因为它们是在主机/应用程序/微服务中托管的服务/逻辑。请注意,这种情况下托管服务仅简单表示具有后台任务逻辑类。

那我们如何实现托管服务了,一种简单的方式就是使用.NET Core 2.0之后版本中提供了一个名为 IHostedService的新接口。当然也可以选择其他的一些后台任务框架,比如HangFire、Quartz。640?wx_fmt=png

该示例项目就是基于 BackgroundService定义的一个后台任务。该任务主要用于轮询订单表中处于已提交超过1分钟的订单,然后发布集成事件到事件总线,最终用来将订单状态更新为待核验(库存)状态。

 
  1. public abstract class BackgroundService : IHostedService, IDisposable

  2. {

  3.    protected BackgroundService();


  4.    public virtual void Dispose();

  5.    public virtual Task StartAsync(CancellationToken cancellationToken);

  6.    [AsyncStateMachine(typeof(<StopAsync>d__4))]

  7.    public virtual Task StopAsync(CancellationToken cancellationToken);

  8.    protected abstract Task ExecuteAsync(CancellationToken stoppingToken);

  9. }

BackgroundService的方法申明中我们可以看出仅需实现 ExecuteAsync方法即可。

完成后台任务的定义后,将服务注册到Ioc容器中即可。

 
  1. public IServiceProvider ConfigureServices(IServiceCollection services)

  2. {

  3. //Other DI registrations;

  4. // Register Hosted Services

  5. services.AddSingleton<IHostedService, GracePeriodManagerService>();

  6. services.AddSingleton<IHostedService, MyHostedServiceB>();

  7. services.AddSingleton<IHostedService, MyHostedServiceC>();

  8. //...

  9. }

640?wx_fmt=png

总之, IHostedService接口为 ASP.NET Core Web 应用程序启动后台任务提供了一种便捷的方法。它的优势主要在于:当主机本身关闭时,可以利用取消令牌来优雅的清理后台任务。

8. Ordering.SignalrHub:即时通信

在订单微服务中,当订单状态变更时,需要实时推送订单状态变更消息给客户端。而这就涉及到实时通信。实时 HTTP 通信意味着,当数据可用时,服务端代码会推送内容到已连接的客户端,而不是服务端等待客户端来请求新数据。

而对于实时通信,ASP.NET Core中SignalR可以满足我们的需求,其支持几种处理实时通信的技术以确保实时通信的可靠传输。

  • WebSockets

  • Server-Sent Events

  • Long Polling

640?wx_fmt=png

该示例项目的实现思路很简单:

  1. 订阅订单状态变更相关的集成事件

  2. 继承 SignalR.Hub定义一个 NotificationsHub

  3. 在集成事件处理程序中调用Hub进行消息的实时推送

 
  1. // 订阅集成事件

  2. private void ConfigureEventBus(IApplicationBuilder app)

  3. {

  4.    var eventBus = app.ApplicationServices.GetRequiredService<IEventBus>();  

  5.    eventBus.Subscribe<OrderStatusChangedToAwaitingValidationIntegrationEvent, OrderStatusChangedToAwaitingValidationIntegrationEventHandler>();

  6.    eventBus.Subscribe<OrderStatusChangedToPaidIntegrationEvent, OrderStatusChangedToPaidIntegrationEventHandler>();

  7.    eventBus.Subscribe<OrderStatusChangedToStockConfirmedIntegrationEvent, OrderStatusChangedToStockConfirmedIntegrationEventHandler>();

  8.    eventBus.Subscribe<OrderStatusChangedToShippedIntegrationEvent, OrderStatusChangedToShippedIntegrationEventHandler>();

  9.    eventBus.Subscribe<OrderStatusChangedToCancelledIntegrationEvent, OrderStatusChangedToCancelledIntegrationEventHandler>();

  10.    eventBus.Subscribe<OrderStatusChangedToSubmittedIntegrationEvent, OrderStatusChangedToSubmittedIntegrationEventHandler>();  

  11. }


  12. // 定义SignalR.Hub

  13. [Authorize]

  14. public class NotificationsHub : Hub

  15. {


  16.    public override async Task OnConnectedAsync()

  17.    {

  18.        await Groups.AddToGroupAsync(Context.ConnectionId, Context.User.Identity.Name);

  19.        await base.OnConnectedAsync();

  20.    }


  21.    public override async Task OnDisconnectedAsync(Exception ex)

  22.    {

  23.        await Groups.AddToGroupAsync(Context.ConnectionId, Context.User.Identity.Name);

  24.        await base.OnDisconnectedAsync(ex);

  25.    }

  26. }


  27. // 在集成事件处理器中调用Hub进行消息的实时推送

  28. public class OrderStatusChangedToPaidIntegrationEventHandler : IIntegrationEventHandler<OrderStatusChangedToPaidIntegrationEvent>

  29. {

  30.    private readonly IHubContext<NotificationsHub> _hubContext;


  31.    public OrderStatusChangedToPaidIntegrationEventHandler(IHubContext<NotificationsHub> hubContext)

  32.    {

  33.        _hubContext = hubContext ?? throw new ArgumentNullException(nameof(hubContext));

  34.    }


  35.    public async Task Handle(OrderStatusChangedToPaidIntegrationEvent @event)

  36.    {

  37.        await _hubContext.Clients

  38.            .Group(@event.BuyerName)

  39.            .SendAsync("UpdatedOrderState", new { OrderId = @event.OrderId, Status = @event.OrderStatus });

  40.    }

  41. }

8. 最后

订单微服务在整个eShopOnContainers中属于最复杂的一个微服务了。 通过对DDD的简要介绍,以及对每一层的技术选型以及实现的思路和逻辑的梳理,希望对你有所帮助。

原文链接:https://www.cnblogs.com/sheng-jie/p/10312605.html

 

.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com

640?wx_fmt=jpeg

这篇关于eShopOnContainers 知多少[8]:Ordering microservice的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/356994

相关文章

视觉语言模型(VLMs)知多少?

最近这几年,自然语言处理和计算机视觉这两大领域真是突飞猛进,让机器不仅能看懂文字,还能理解图片。这两个领域的结合,催生了视觉语言模型,也就是Vision language models (VLMs) ,它们能同时处理视觉信息和文字数据。 VLMs就像是AI界的新宠,能搞定那些既需要看图又需要读文的活儿,比如给图片配文字、回答有关图片的问题,或者根据文字描述生成图片。以前这些活儿都得靠不同

比亚迪汽车在用的音频芯片--AD2438专利知多少

平台君上周分享AD2428后,有读者后台留言问有没有分析过AD2438?作为听人劝会发达的排头兵,平台君这就给大家带来AD2438芯片的一些分析信息。这可是现在全球新能源汽车销售冠军比亚迪也在用的汽车音频芯片。 平台君从AD2438芯片的管芯标识发现这款芯片应该是2022年设计的,而前一款AD2428是2017年的。 AD2438封装图及管芯标识图(图源:IPBrain集成电路

天童美语:全国测绘法宣传日|测绘知识知多少

每年8月29日是全国测绘法宣传日,这一特殊的日子,旨在提醒公众测绘科学在日常生活中的重要性和影响。8月29日是第19个全国测绘法宣传日。和南宁天童美语一起了解关于测绘的知识吧。      ·什么是测绘?      测绘是指对自然地理要素或者地表人工设施的形状、大小、空间位置及其属性等进行测定、采集、表述,以及对获取的数据、信息、成果进行处理和提供的活动。      测绘是测量与绘图的总称,它以地球

转格式软件知多少?格式转换方法盘点

随着手机和其他工具的普及,拍摄视频已成为我们日常生活中的一种娱乐形式。在享受录制视频乐趣的同时,我们需要灵活使用转格式软件,以满足不同播放设备和应用场景的要求。如何转换视频格式?许多人都关心视频格式转换问题。 视频格式转换可能看起来很简单,但在现实中,它涉及许多技术细节和操作步骤。许多人发现很难解决这个问题,不知道从哪里开始。你可以在线搜索不同的教程,尝试不同的软件和方法,但结果往往很差,甚至可

NEFU563 鸭子知多少?【递归】

题目链接: http://acm.nefu.edu.cn/JudgeOnline/problemshow.php?problem_id=563 题目大意: 有个人赶着鸭子去每个村庄卖,每经过一个村子卖去所赶鸭子的一半又2只。这样经过了N个村子 还剩下2只鸭子,问:他出发时所赶的鸭子共有多少只。 思路: 路过第i个村子剩的鸭子数 = 第i+1天刚开始赶的鸭子数 = (第i+1

日系车知多少

抵制日系车。 日系车品牌包括直接使用日系车品牌的车辆,原型车是日系车或关键部件使用日系产品的国产品牌,模仿日系车的国内品牌,和日系车有着亲缘关联的国际汽车品牌。 一、直接使用日系车品牌的:广州本田——本田雅阁、飞度、奥得赛; 天津丰田——丰田威驰; 长安玲木——玲木斯威夫特·羚羊、玲木奥拓(还有西安奥拓、江南奥拓); 郑州日产——日产派拉丁、日产D22皮卡; 东风风神——日产阳光、日产蓝鸟、蓝

RocketMQ相关知识知多少

一、RocketMQ的定义 官网网址:领域模型概述 | RocketMQ Apache RocketMQ 自诞生以来,因其架构简单、业务功能丰富、具备极强可扩展性等特点被众多企业开发者以及云厂商广泛采用。历经十余年的大规模场景打磨,RocketMQ 已经成为业内共识的金融级可靠业务消息首选方案,被广泛应用于互联网、大数据、移动互联网、物联网等领域的业务场景。【RocketMQ是一个队列模型的消

Marin说PCB之Max parallel知多少?

今天是个阳光明媚,万里乌云的好日子。小编我一如既往地到家打开电脑准备看腾讯视频的五十公里桃花坞的第四季,在看到汪苏泷汪台说650电台要解散的时候小编我差点也哭了。650电台之于桃花坞就像乐队的鼓手一样,都是一个团队的灵感啊,而且650电台已经成立四年了,现在说解散就解散是有点太可惜了。 好了,咱们不能感慨太多了,不然就有黑粉说你老是说这些废话,不能直接进入每期文章的主题吗?更有一些性质恶劣的

JAVA数组知多少

在我讲数组的时候,大家一定对数组有一定的了解或者学过一点,今天我会从一维和二维两方面来谈谈我的理解,多维的有兴趣也可以自己推一下。               下面我列一张表来比较一下一二维数组的异同:  一维数组数组的属性和方法唯一的属性length 数组的长度数组名.length对应下标的元素值数组名[下标]数组的定义方式数据类型 [] 数组名 = new 数据

图片知多少

欲观原文,请君移步微信 图片是由图形、图像等构成的平面媒体。在我们生活中随处可见,下面小编与大家一起聊一下关于图片的那些事儿。 #基本概念 ##像素 谈到图片,就离不开像素这个概念,像素是指由图片的小方格组成的,这些小方块都有一个明确的位置和被分配的色彩数值,小方格颜色和位置就决定该图像所呈现出来的样子。 当图片尺寸以像素为单位时,比如一个1920x1080的图片,表示的就是这张图片水