当前位置: 首页 > article >正文

【架构实战】六边形架构与整洁架构实战

一、架构演进概述随着业务复杂度增加软件架构也在不断演进架构演进历程分层架构 → 六边形架构 → 整洁架构 → 微服务架构核心目标实现高内聚、低耦合架构质量评估独立性框架、数据库、UI的可替换性可测试性业务逻辑可独立测试可维护性代码易于理解和修改二、六边形架构Hexagonal Architecture1. 核心理念六边形架构又称端口与适配器Ports and Adapters由Alistair Cockburn提出。 核心思想 - 业务逻辑处于核心位置 - 通过端口与外部世界交互 - 适配器负责协议转换2. 架构图┌─────────────────────────────────┐ │ 外部世界 │ │ (数据库、Web UI、外部服务) │ └─────────────┬───────────────────┘ │ ┌─────────────▼───────────────────┐ │ 适配器层 │ │ ┌─────────────┬─────────────┐ │ │ │ 驱动适配器 │ 驱动适配器 │ │ │ │ (Primary) │ (Primary) │ │ │ └──────┬──────┴──────┬──────┘ │ └─────────┼─────────────┼────────┘ │ │ ┌─────────▼─────────────▼─────────┐ │ 端口层 │ │ (输入端口) (输出端口) │ │ ┌─────────────┬─────────────┐ │ │ │ │ │ │ │ └──────┬──────┴──────┬──────┘ │ └─────────┼─────────────┼────────┘ │ │ ┌─────────▼─────────────▼─────────┐ │ 应用层 │ │ (用例/服务编排) │ └─────────┬─────────────────────┘ │ ┌─────────▼─────────────────────┐ │ 领域层 │ │ (核心业务逻辑、实体、值对象) │ └─────────────────────────────────┘3. 核心概念端口Port定义了系统与外部交互的方式分为输入端口驱动端口和输出端口被驱动端口端口是接口不包含实现适配器Adapter负责实现端口定义的接口处理外部协议和格式转换适配器可以随时替换4. 代码实现领域层核心// 领域实体publicclassOrder{privateOrderIdid;privateCustomerIdcustomerId;privateOrderStatusstatus;privateListOrderItemitems;publicvoidaddItem(Productproduct,intquantity){if(status!OrderStatus.DRAFT){thrownewBusinessException(只有草稿状态可以添加商品);}items.add(newOrderItem(product,quantity));}publicvoidsubmit(){if(items.isEmpty()){thrownewBusinessException(订单不能为空);}this.statusOrderStatus.SUBMITTED;}}// 领域服务publicclassOrderDomainService{publicvoidvalidateOrder(Orderorder){// 业务规则验证}}输入端口Driving Port// 输入端口接口由应用层或领域层定义publicinterfaceOrderInputPort{OrderDTOcreateOrder(CreateOrderCommandcommand);OrderDTOupdateOrder(LongorderId,UpdateOrderCommandcommand);voidcancelOrder(LongorderId);OrderDTOgetOrder(LongorderId);PageOrderDTOlistOrders(OrderQueryquery);}输出端口Driven Port// 输出端口接口由应用层定义由基础设施实现publicinterfaceOrderRepository{OrderfindById(OrderIdid);Ordersave(Orderorder);voiddelete(OrderIdid);}publicinterfaceProductRepository{ProductfindById(ProductIdid);ListProductfindByIds(ListProductIdids);}publicinterfaceEventPublisher{voidpublish(DomainEventevent);}publicinterfaceIdGenerator{OrderIdgenerateId();}应用层用例// 应用服务实现输入端口ServiceTransactionalpublicclassOrderApplicationServiceimplementsOrderInputPort{privatefinalOrderRepositoryorderRepository;privatefinalProductRepositoryproductRepository;privatefinalIdGeneratoridGenerator;privatefinalEventPublishereventPublisher;publicOrderApplicationService(OrderRepositoryorderRepository,ProductRepositoryproductRepository,IdGeneratoridGenerator,EventPublishereventPublisher){this.orderRepositoryorderRepository;this.productRepositoryproductRepository;this.idGeneratoridGenerator;this.eventPublishereventPublisher;}OverridepublicOrderDTOcreateOrder(CreateOrderCommandcommand){// 创建订单OrderorderOrder.create(idGenerator.generateId(),CustomerId.of(command.getCustomerId()));// 添加商品for(OrderItemDTOitemDTO:command.getItems()){ProductproductproductRepository.findById(ProductId.of(itemDTO.getProductId()));order.addItem(product,itemDTO.getQuantity());}// 提交订单order.submit();// 保存orderRepository.save(order);// 发布事件eventPublisher.publish(newOrderCreatedEvent(order));returntoDTO(order);}}适配器层// Web适配器Driving AdapterRestControllerpublicclassOrderControllerimplementsOrderInputPort{privatefinalOrderApplicationServiceorderService;PostMapping(/orders)publicResponseEntityOrderDTOcreateOrder(RequestBodyCreateOrderRequestrequest){CreateOrderCommandcommandtoCommand(request);OrderDTOresultorderService.createOrder(command);returnResponseEntity.ok(result);}// ... 其他接口方法}// JPA适配器Driven AdapterRepositorypublicclassJpaOrderRepositoryimplementsOrderRepository{AutowiredprivateOrderJpaRepositoryjpaRepository;OverridepublicOrderfindById(OrderIdid){returnjpaRepository.findById(id.getValue()).map(this::toDomain).orElseThrow(()-newEntityNotFoundException(Order not found));}OverridepublicOrdersave(Orderorder){OrderEntityentitytoEntity(order);jpaRepository.save(entity);returnorder;}}// 消息适配器Driven AdapterComponentpublicclassKafkaEventPublisherimplementsEventPublisher{AutowiredprivateKafkaTemplateString,ObjectkafkaTemplate;Overridepublicvoidpublish(DomainEventevent){kafkaTemplate.send(domain-events,event.getClass().getSimpleName(),event);}}// UUID适配器Driven AdapterComponentpublicclassUuidIdGeneratorimplementsIdGenerator{OverridepublicOrderIdgenerateId(){returnOrderId.of(UUID.randomUUID().toString());}}三、整洁架构Clean Architecture1. 核心理念整洁架构由Robert C. Martin提出强调 - 依赖规则外层依赖内层内层不知道外层 - 层次分离明确的职责边界 - 独立性强业务逻辑不依赖任何外部框架2. 架构层次┌─────────────────────────────────────────────────────────────────┐ │ 外层框架 驱动 │ │ (Web框架、数据库、UI、外部接口、测试框架) │ ├─────────────────────────────────────────────────────────────────┤ │ 接口适配器层 │ │ (Controllers、Gateways、Presenters) │ ├─────────────────────────────────────────────────────────────────┤ │ 应用层 │ │ (用例、命令、查询、DTO) │ ├─────────────────────────────────────────────────────────────────┤ │ 领域层 │ │ (实体、值对象、领域服务、领域事件、接口) │ ├─────────────────────────────────────────────────────────────────┤ │ 内层企业业务规则 │ └─────────────────────────────────────────────────────────────────┘ ▲ │ 依赖方向 │ ▼3. 依赖规则依赖规则Dependency Rule - 源码依赖只能从外向内 - 内层不知道外层的存在 - 外层通过接口依赖内层 - 具体的实现细节放在外层4. 代码结构com.example.cleanarchitecture ├── domain # 领域层最内层 │ ├── model │ │ ├── Order.java │ │ ├── Product.java │ │ └── Customer.java │ ├── service │ │ └── OrderDomainService.java │ ├── event │ │ └── DomainEvent.java │ └── port # 端口接口 │ ├── inbound │ │ ├── CreateOrderUseCase.java │ │ └── OrderQueryService.java │ └── outbound │ ├── OrderRepository.java │ ├── ProductRepository.java │ └── EventPublisher.java ├── application # 应用层 │ ├── usecase │ │ ├── CreateOrderUseCaseImpl.java │ │ └── GetOrderUseCaseImpl.java │ ├── dto │ │ ├── CreateOrderCommand.java │ │ └── OrderDTO.java │ └── mapper │ └── OrderMapper.java ├── adapter # 适配器层 │ ├── inbound │ │ ├── web │ │ │ └── OrderController.java │ │ └── grpc │ │ └── OrderGrpcAdapter.java │ └── outbound │ ├── persistence │ │ ├── JpaOrderRepository.java │ │ └── JpaProductRepository.java │ ├── messaging │ │ └── KafkaEventPublisher.java │ └── external │ └── PaymentGatewayAdapter.java └── configuration # 配置层 └── DependencyInjectionConfig.java5. 整洁架构实现领域层// 领域实体publicclassOrder{privateOrderIdid;privateCustomerIdcustomerId;privateOrderStatusstatus;privateListOrderItemitems;publicstaticOrdercreate(OrderIdid,CustomerIdcustomerId){returnnewOrder(id,customerId,OrderStatus.DRAFT,newArrayList());}publicvoidaddItem(Productproduct,intquantity){// 业务规则if(status!OrderStatus.DRAFT){thrownewDomainException(只有草稿订单可以添加商品);}items.add(newOrderItem(product,quantity));}}// 输入端口用例接口publicinterfaceCreateOrderUseCase{OrderDTOexecute(CreateOrderCommandcommand);}// 输出端口仓储接口publicinterfaceOrderRepository{OptionalOrderfindById(OrderIdid);Ordersave(Orderorder);voiddelete(OrderIdid);ListOrderfindByCustomerId(CustomerIdcustomerId);}应用层// 用例实现ServicepublicclassCreateOrderUseCaseImplimplementsCreateOrderUseCase{privatefinalOrderRepositoryorderRepository;privatefinalProductRepositoryproductRepository;privatefinalIdGeneratoridGenerator;publicCreateOrderUseCaseImpl(OrderRepositoryorderRepository,ProductRepositoryproductRepository,IdGeneratoridGenerator){this.orderRepositoryorderRepository;this.productRepositoryproductRepository;this.idGeneratoridGenerator;}OverridepublicOrderDTOexecute(CreateOrderCommandcommand){// 1. 验证客户CustomerIdcustomerIdCustomerId.of(command.getCustomerId());// 2. 创建订单OrderIdorderIdidGenerator.generateId();OrderorderOrder.create(orderId,customerId);// 3. 添加商品for(OrderItemCommanditemCmd:command.getItems()){ProductproductproductRepository.findById(ProductId.of(itemCmd.getProductId())).orElseThrow(()-newDomainException(商品不存在));order.addItem(product,itemCmd.getQuantity());}// 4. 提交订单order.submit();// 5. 保存orderRepository.save(order);// 6. 返回DTOreturnOrderDTO.from(order);}}适配器层// Web控制器入站适配器RestControllerRequestMapping(/api/orders)publicclassOrderController{privatefinalCreateOrderUseCasecreateOrderUseCase;privatefinalGetOrderUseCasegetOrderUseCase;PostMappingpublicResponseEntityOrderDTOcreateOrder(RequestBodyValidCreateOrderRequestrequest){CreateOrderCommandcommandCreateOrderCommand.from(request);OrderDTOresultcreateOrderUseCase.execute(command);returnResponseEntity.status(HttpStatus.CREATED).body(result);}GetMapping(/{id})publicResponseEntityOrderDTOgetOrder(PathVariableLongid){OrderDTOresultgetOrderUseCase.execute(id);returnResponseEntity.ok(result);}}// JPA仓储出站适配器RepositorypublicclassJpaOrderRepositoryimplementsOrderRepository{AutowiredprivateOrderJpaRepositoryjpaRepository;OverridepublicOptionalOrderfindById(OrderIdid){returnjpaRepository.findById(id.getValue()).map(entity-/* 转换为领域对象 */);}OverridepublicOrdersave(Orderorder){OrderEntityentitytoEntity(order);jpaRepository.save(entity);returnorder;}}四、架构对比1. 六边形 vs 整洁架构维度六边形架构整洁架构核心理念端口与适配器依赖规则层次划分端口/适配器/应用/领域实体/用例/接口适配器/框架适用场景轻量级应用复杂企业应用学习曲线较平缓较陡峭灵活性高非常高2. 分层架构对比维度传统三层六边形整洁架构可测试性差好非常好可维护性一般好非常好耦合度高低最低复杂度低中高五、实战选择1. 何时使用六边形适合场景 - 微服务内部设计 - 需要与多个外部系统交互 - 希望保持业务逻辑的纯粹性 - 团队对DDD有一定了解2. 何时使用整洁架构适合场景 - 复杂的企业级应用 - 对可测试性要求极高 - 需要长期维护的项目 - 团队有较强的架构能力3. 实际项目建议// 项目结构建议根据复杂度选择// 简单项目六边形架构com.example ├── domain │ ├── model │ └── port # 端口定义 ├── application # 用例 └── adapter # 适配器// 复杂项目整洁架构com.example ├── domain │ ├── model │ ├── service │ └── port ├── application │ ├── usecase │ └── dto └── adapter ├── inbound └── outbound六、测试策略1. 领域层测试// 领域实体测试无需Spring纯单元测试classOrderTest{TestvoidshouldAddItemToDraftOrder(){// GivenOrderorderOrder.create(OrderId.of(123),CustomerId.of(456));// Whenorder.addItem(Product.of(P1,Money.of(100)),2);// ThenassertEquals(1,order.getItems().size());assertEquals(Money.of(200),order.getTotalAmount());}TestvoidshouldRejectAddItemToSubmittedOrder(){// GivenOrderorderOrder.create(OrderId.of(123),CustomerId.of(456));order.submit();// When/ThenassertThrows(BusinessException.class,()-order.addItem(Product.of(P1,Money.of(100)),2));}}2. 用例层测试// 用例测试使用MockExtendWith(MockitoExtension.class)classCreateOrderUseCaseTest{MockprivateOrderRepositoryorderRepository;MockprivateProductRepositoryproductRepository;MockprivateIdGeneratoridGenerator;InjectMocksprivateCreateOrderUseCaseImpluseCase;TestvoidshouldCreateOrderSuccessfully(){// GivenCreateOrderCommandcommandCreateOrderCommand.builder().customerId(C123).items(List.of(newOrderItemCommand(P1,2))).build();when(productRepository.findById(any())).thenReturn(Optional.of(Product.of(P1,Money.of(100))));when(idGenerator.generateId()).thenReturn(OrderId.of(O123));// WhenOrderDTOresultuseCase.execute(command);// ThenassertNotNull(result);verify(orderRepository).save(any());}}七、总结架构设计的选择建议六边形架构适合微服务、对接多外部系统整洁架构适合复杂业务、需要高可测试性核心原则业务逻辑与外部依赖解耦实施建议从六边形架构开始随着业务复杂度增加演进到整洁架构重视领域模型的设计保持架构的持续演进个人观点仅供参考

相关文章:

【架构实战】六边形架构与整洁架构实战

一、架构演进概述 随着业务复杂度增加,软件架构也在不断演进: 架构演进历程: 分层架构 → 六边形架构 → 整洁架构 → 微服务架构核心目标:实现高内聚、低耦合 架构质量评估: 独立性:框架、数据库、UI的可替…...

神经机器翻译数据集构建:Europarl语料处理与优化

1. 神经机器翻译数据集构建实战:从Europarl语料到模型训练在自然语言处理领域,机器翻译一直是最具挑战性的任务之一。2014年,随着神经机器翻译(Neural Machine Translation, NMT)技术的突破,基于编码器-解码…...

电话号码精确定位:免费开源工具的实用指南与深度解析

电话号码精确定位:免费开源工具的实用指南与深度解析 【免费下载链接】location-to-phone-number This a project to search a location of a specified phone number, and locate the map to the phone number location. 项目地址: https://gitcode.com/gh_mirro…...

AI 英语教学智能体开发

AI 英语教学智能体开发已进入“认知模拟”与“超低延迟交互”的深水区。技术栈的构建不再仅仅是调用 API,而是涉及从底层推理到上层教学法编排的整套国产化生态。以下是国内开发 AI 英语教学智能体的核心技术维度:1. 基座模型与国产化适配国内开发者目前…...

Elasticsearch架构核心:Node节点详解与角色功能全解析

Elasticsearch架构核心:Node节点详解与角色功能全解析一、前言二、什么是 Elasticsearch Node(节点)?1. 官方定义2. 通俗理解3. 节点核心特点三、节点角色与功能流程图四、Elasticsearch 节点的 5 种核心角色与功能1. 主节点&…...

保姆级教程:在Gazebo 11中为WAM-V无人艇模型添加AprilTag(Ubuntu 20.04环境)

深度解析:在Gazebo 11中为WAM-V无人艇集成AprilTag的完整实践指南(Ubuntu 20.04环境) 当你在ROS/Gazebo仿真环境中需要对现有机器人模型进行功能扩展时,往往会遇到模型文件嵌套复杂、修改位置不明确的困扰。本文将以WAM-V无人艇为…...

批量给文件改名的方法有哪些?这5个实用技巧新手也能秒会

在日常办公和学习中,我们常常需要处理大量文件,杂乱的文件名不仅影响工作效率,还可能导致文件查找困难。无论是学生整理作业、职场人士归档资料,还是摄影爱好者管理照片,批量给文件改名都是必备技能。本文将详细介绍5种…...

别再只跑Demo了!手把手教你用DINOv2的Patch特征做简单的图像前景分割

解锁DINOv2的实战潜力:用Patch特征实现零样本图像分割 在计算机视觉领域,我们常常陷入一个怪圈:花费大量时间跑通各种模型的Demo,却很少思考这些预训练特征在实际任务中的真正价值。DINOv2作为Meta开源的视觉大模型,其…...

从市场调研到用户画像:因子分析如何帮你发现隐藏的‘消费者因子’?

解码消费者心智:如何用因子分析从海量问卷中提炼黄金洞察 当市场部同事将一份包含87个问题的用户满意度问卷扔到你桌上时,那些密密麻麻的评分数据就像未经处理的矿石——价值连城却难以直接利用。这正是因子分析大显身手的时刻。想象一下,你不…...

从编码原理到实战:彻底搞懂QT中文乱码,让你的应用告别“火星文”(UTF-8/GBK转换详解)

从编码原理到实战:彻底搞懂QT中文乱码,让你的应用告别“火星文”(UTF-8/GBK转换详解) 在QT开发中,中文乱码问题就像一位不请自来的“老朋友”,总会在你最意想不到的时候出现。无论是控件显示、文件读写还是…...

2025年MLOps必备的10个Python库解析

1. 为什么2025年的MLOps需要这10个Python库?三年前部署一个机器学习模型还需要手动编写数百行部署脚本,现在MLOps工具链的成熟度已经让模型部署变得像调用API一样简单。作为经历过完整MLOps演进周期的从业者,我亲历了从手工运维到自动化管道的…...

用E4A中文编程,30分钟搞定一个能远程控制STM32的安卓APP(基于OneNET MQTT)

用E4A中文编程30分钟构建OneNET物联网控制APP全指南 当第一次听说能用中文编写安卓APP控制硬件设备时,我的反应和多数嵌入式开发者一样——这要么是夸大其词,要么功能极其有限。直到在某个智能家居项目截止日前48小时,因Java界面卡死而被迫尝…...

C++26反射让constexpr容器成为现实?揭秘编译期JSON Schema校验器的7层元编程架构(含完整Doxygen生成的反射依赖图)

更多请点击: https://intelliparadigm.com 第一章:C26反射核心机制与constexpr容器的范式突破 C26 将首次在标准中引入原生、零开销的编译期反射(std::reflect)设施,配合全面 constexpr 化的容器(如 std::…...

【收藏备用】2026年Java程序员必看:不用弃坑,靠大模型轻松涨薪(小白/在职通用)

说真的,2025到2026这一年,看着身边一群搞Java的兄弟纷纷转型大模型,心里挺有感触的。我们当初入门的时候,都是从写接口、搭Spring Boot、连MySQL、配Redis开始,一天天稳扎稳打,以为凭着这些硬技能就能安安稳…...

STC8H8K64U vs. 新唐MS51:硬件PWM库函数生态与开发效率深度对比

STC8H8K64U与新唐MS51硬件PWM开发体验全维度对比 1. 开发环境与生态支持差异 在嵌入式开发领域,芯片厂商提供的开发工具链和软件支持往往决定了实际开发效率。STC8H8K64U和新唐MS51在硬件PWM模块的支持上呈现出截然不同的开发体验。 STC8H8K64U的库函数现状&#xf…...

(十二)Scanner 输入校验、if 嵌套、逻辑运算符 (并且)、||(或者)综合练习

package dey5; import java.util.Scanner; public class Test06 {public static void main(String[] args) {Scanner scanner new Scanner(System.in);// 第一步:输入第一个运算数System.out.println("输入第一个运算数");if (scanner.hasNextInt()) {in…...

NodeMCU PyFlasher:零门槛ESP8266固件烧录完全指南

NodeMCU PyFlasher:零门槛ESP8266固件烧录完全指南 【免费下载链接】nodemcu-pyflasher Self-contained NodeMCU flasher with GUI based on esptool.py and wxPython. 项目地址: https://gitcode.com/gh_mirrors/no/nodemcu-pyflasher NodeMCU PyFlasher是一…...

3个场景告诉你:为什么Mac用户需要桌面歌词显示工具LyricsX

3个场景告诉你:为什么Mac用户需要桌面歌词显示工具LyricsX 【免费下载链接】Lyrics Swift-based iTunes plug-in to display lyrics on the desktop. 项目地址: https://gitcode.com/gh_mirrors/lyr/Lyrics 如果你正在使用Mac听音乐,是否曾有过这…...

PACS系统选型与部署避坑指南:医院影像科技术负责人必看的架构解析

PACS系统选型与部署避坑指南:医院影像科技术负责人必看的架构解析 在数字化医疗快速发展的今天,医学影像存储与传输系统(PACS)已成为医院信息化建设的核心支柱。作为连接影像设备、临床科室和放射科医生的"神经中枢",一套设计合理…...

抖音批量下载终极指南:如何免费高效获取无水印视频内容

抖音批量下载终极指南:如何免费高效获取无水印视频内容 【免费下载链接】douyin-downloader A practical Douyin downloader for both single-item and profile batch downloads, with progress display, retries, SQLite deduplication, and browser fallback supp…...

从一次内部攻防演练说起:我是如何利用CVE-2017-1000028漏洞“捡到”GlassFish管理员密码的

实战复盘:GlassFish任意文件读取漏洞的深度利用链 那是一个普通的周二下午,我正在为客户做内部网络的安全评估。按照惯例,我先用Nmap对目标网段进行扫描,突然在扫描结果中发现了一台开放4848端口的服务器——这个端口号让我瞬间警…...

你的MCP4725 DAC输出不准?可能是这3个硬件坑和2个软件误区(附STM32 F4实测排查指南)

MCP4725 DAC精度问题全解析:从硬件设计到软件优化的实战指南 在嵌入式系统开发中,数字模拟转换器(DAC)的精度问题常常让工程师们头疼不已。MCP4725作为一款性价比较高的12位DAC芯片,广泛应用于各种需要精确电压输出的场景。然而,很…...

Python实战:用PyCryptodome构建你的数据安全防线

1. PyCryptodome:Python开发者的加密利器 当你需要为Python应用添加加密功能时,PyCryptodome绝对是个绕不开的名字。这个库的前身是著名的PyCrypto,现在已经成为Python生态中最强大的密码学工具之一。我在多个实际项目中使用过它,…...

96个公共Tracker终极配置指南:让BT下载速度提升300%

96个公共Tracker终极配置指南:让BT下载速度提升300% 【免费下载链接】trackerslist Updated list of public BitTorrent trackers 项目地址: https://gitcode.com/GitHub_Trending/tr/trackerslist 还在为BT下载速度慢而烦恼吗?🤔 今天…...

3步解锁加密音乐:浏览器本地解密完全指南

3步解锁加密音乐:浏览器本地解密完全指南 【免费下载链接】unlock-music 在浏览器中解锁加密的音乐文件。原仓库: 1. https://github.com/unlock-music/unlock-music ;2. https://git.unlock-music.dev/um/web 项目地址: https://gitcode.c…...

用STM32G431玩超级玛丽!CubeMX+HAL库移植NES模拟器保姆级教程(附蓝桥杯板子适配)

用STM32G431玩超级玛丽!CubeMXHAL库移植NES模拟器保姆级教程(附蓝桥杯板子适配) 还记得小时候抱着红白机玩《超级玛丽》的快乐吗?现在,你可以亲手把这份童年回忆"塞进"一块STM32开发板里。本文将带你用STM32…...

RWKV-7 (1.5B World)镜像部署:腾讯云TI-ONE平台GPU容器配置

RWKV-7 (1.5B World)镜像部署:腾讯云TI-ONE平台GPU容器配置 1. 项目概述 RWKV-7 (1.5B World)是一款专为单卡GPU优化的轻量级对话模型,基于RWKV架构开发。这个1.5B参数的模型虽然体积小巧,却具备出色的多语言理解能力,特别适合在…...

协同过滤算法原理与商业化应用实践

1. 协同过滤的商业化应用全景当你在电商平台看到"猜你喜欢"的推荐商品,或者在视频网站发现首页推送的内容恰好符合你的口味,背后很可能就是协同过滤算法在发挥作用。这种技术已经成为现代商业中精准预测用户偏好的核心工具,它不需要…...

避坑指南:为什么ESP32的One-Wire驱动读不了AM2302?手把手教你用MicroPython中断搞定它

ESP32与AM2302的协议兼容性困境:用MicroPython中断方案破解温湿度读取难题 当你在ESP32上尝试用MicroPython的One-Wire驱动读取AM2302温湿度传感器时,大概率会遇到数据读取失败的情况。这不是代码写错了,而是两种协议之间存在微妙的时序差异。…...

如何用NVIDIA Profile Inspector解锁显卡隐藏性能:完整新手教程

如何用NVIDIA Profile Inspector解锁显卡隐藏性能:完整新手教程 【免费下载链接】nvidiaProfileInspector 项目地址: https://gitcode.com/gh_mirrors/nv/nvidiaProfileInspector 你是否觉得NVIDIA显卡的性能还有提升空间?是否想要获得比官方控制…...