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

小型双轮差速底盘寻迹功能的实现

1. 功能说明

寻迹机器人是一种能够跟踪特定物体或线路的机器人。它们通常具有以下功能和特点:

      ① 传感器:寻迹机器人配备了用于感知环境的传感器,如摄像头、灰度传感器等。这些传感器可以探测地面上的标记、颜色、纹理或其他特定特征,以确定要跟踪的目标。

      ② 自主导航:寻迹机器人通常具备自主导航能力,可以根据目标物体的位置和运动轨迹进行移动和调整。它们可能使用轮式、履带或其他移动机构来在地面上移动。

      ③ 跟踪精度:寻迹机器人通常被设计为能够实时跟踪目标物体,并尽可能准确地保持距离和方向。一些高级寻迹机器人还可以通过预测目标物体的运动来提高跟踪的精度。

      ④ 应用场景:寻迹机器人可以应用于多种场景,如工业生产线上的零部件跟踪、物流仓库中的货物识别与追踪、安防领域中的行人监控等。它们在自动化、智能化和效率提升方面具有广泛的应用前景。

本文示例将实现R023样机小型双轮差速底盘机器人沿直线寻迹行走的一个功能。

2. 结构装配

按照下图所示方式进行安装:

3. 电子硬件

在这个示例中,我们采用了以下硬件,请大家参考:

主控板

Basra主控板(兼容Arduino Uno)‍

扩展板

Bigfish2.1扩展板‍

传感器灰度传感器
电池7.4V锂电池

按照下图所示方式进行电路连接:

4. 功能实现

编程环境:Arduino 1.8.19

① 下面提供一个控制轮子转动方向和速度的参考例程(Test1.ino):

/*------------------------------------------------------------------------------------版权说明:Copyright 2023 Robottime(Beijing) Technology Co., Ltd. All Rights Reserved.Distributed under MIT license.See file LICENSE for detail or copy athttps://opensource.org/licenses/MITby 机器谱 2023-07-14 https://www.robotway.com/------------------------------*/#include <Servo.h>//调用舵机库Servo myservo;//声明一个舵机类void setup()//Arduino的设置函数{myservo.attach(4);//绑定控制舵机的引脚}void loop()//Arduino的循环函数{for(int i=0; i<180; i++){//通过调节i值控制舵机的运行参数myservo.write(i);//输出控制舵机的运行参数delay(500);//延时}}

② 下面提供一个控制轮子前进、停止、左转、右转、左微调、右微调的参考例程(Test2.ino):

/*------------------------------------------------------------------------------------版权说明:Copyright 2023 Robottime(Beijing) Technology Co., Ltd. All Rights Reserved.Distributed under MIT license.See file LICENSE for detail or copy athttps://opensource.org/licenses/MITby 机器谱 2023-07-14 https://www.robotway.com/------------------------------*/#include <Servo.h>#define middle1 88//定义对应舵机的中间值,即停止转动的值#define middle2 88//此值需要测量,各个舵机不一定相同Servo myservo[2];//定义一个舵机类数组void setup(){myservo[0].attach(4);myservo[1].attach(3);}void loop(){Left();//调用左转函数delay(1000);Right();delay(1000);Forwards();delay(1000);stop();delay(1000);}void Left()//左转函数{myservo[0].write(middle1);myservo[1].write(middle2 + 20);}void Right()//右转函数{myservo[0].write(middle1 - 20);myservo[1].write(middle2);}void Forwards()//前进函数{myservo[0].write(middle1 - 20);myservo[1].write(middle2 + 20);}void stop()//后退函数{myservo[0].write(middle1);myservo[1].write(middle2);}

③ 下面提供一个将灰度传感器数据显示到LED点阵上的参考例程(Test3.ino):

/*------------------------------------------------------------------------------------版权说明:Copyright 2023 Robottime(Beijing) Technology Co., Ltd. All Rights Reserved.Distributed under MIT license.See file LICENSE for detail or copy athttps://opensource.org/licenses/MITby 机器谱 2023-07-14 https://www.robotway.com/------------------------------*/#include <LedControl.h>//调用点阵库函数#include <Servo.h>LedControl lc=LedControl(12,11,13,1);//声明点阵类,并设置对应的引脚int pin[3] = {A0, A4, A3};//设置传感器的对应的三个引脚byte value;//声明传感器值变量void setup(){LedInit();//初始化点阵}/************************************************************************************此程序用到了for与switch的配合框架,可用于多传感器的实时处理,请细细体会!具体解析:for循环中使用了位处理,这样的结果就是value的一个数据位对应一个传感器的状态,此程序value的类型为byte,则可支持8个传感器,如果要使用更多传感器可定义int等。传感器触发时返回值为0,因此value值与传感器触发的状态对应关系以A0触发为例:A0传感器触发-->二进制:00000110-->十六进制:0x06-->对应case 0x06;所以这样做的好处就是当传感器的状态发生改变时程序可以快速的到达指定的处理方式**************************************************************************************/void loop(){value = 0;for(int i=0; i<3; i++){//通过循环检测,读取传感器的状态值value |= (digitalRead(pin[i]) << i);//通过位处理得到结果值,digitalRead()用于读取数字值}switch (value) {//根据结果值进行相应的事件处理case 0x00://全部触发LedOn(0);//点亮相应的点阵LedOn(1);LedOn(2);break;case 0x01://触发右边两个LedOn(1);LedOn(2);break;case 0x03://触发右边一个LedOn(2);break;case 0x04://触发左边两个LedOn(0);LedOn(1);break;case 0x05://触发中间一个LedOn(1);break;case 0x06://触发左边一个LedOn(0);break;default:;}}void LedOn(int key)//根据参数点亮相应的点阵LED{lc.clearDisplay(0);for(int i=0; i<2; i++){for(int j=3*key; j<3*key+2; j++){lc.setLed(0, i, j, true);}}}void LedInit() //点阵初始化函数{lc.shutdown(0,false);lc.setIntensity(0,8);lc.clearDisplay(0);}

④ 下面提供一个小车行走直线的参考例程(Test4.ino):

/*------------------------------------------------------------------------------------版权说明:Copyright 2023 Robottime(Beijing) Technology Co., Ltd. All Rights Reserved.Distributed under MIT license.See file LICENSE for detail or copy athttps://opensource.org/licenses/MITby 机器谱 2023-07-14 https://www.robotway.com/------------------------------*/#include <Servo.h>#define middle1 88#define middle2 88Servo myservo[2];int pin[3] = {A0, A4, A3};byte value;byte value_his = 0;//记录上一次的传感器值void setup(){myservo[0].attach(4);myservo[1].attach(3);}void loop(){value = 0;for(int i=0; i<3; i++){value |= (digitalRead(pin[i]) << i);}if(value == 0x07){//当传感器都没有触发时默认为上一次的值value = value_his;}switch (value) {case 0x00://全部触发Forwards();break;case 0x01://触发右边两个while(digitalRead(pin[1])){//通过while循环使小车回到跑道中间Right();}break;case 0x03://触发右边一个while(digitalRead(pin[1])){Right();}break;case 0x04://触发左边两个while(digitalRead(pin[1])){Left();}break;case 0x05://触发中间一个Forwards();break;case 0x06://触发左边一个while(digitalRead(pin[1])){Left();}break;default:stop();}value_his = value;}void Left(){myservo[0].write(middle1);myservo[1].write(middle2 + 20);}void Right(){myservo[0].write(middle1 - 20);myservo[1].write(middle2);}void Forwards(){myservo[0].write(middle1 - 20);myservo[1].write(middle2 + 20);}void stop(){myservo[0].write(middle1);myservo[1].write(middle2);}

⑤ 下面提供一个小车识别十字路口的参考例程(Test5.ino):

/*------------------------------------------------------------------------------------版权说明:Copyright 2023 Robottime(Beijing) Technology Co., Ltd. All Rights Reserved.Distributed under MIT license.See file LICENSE for detail or copy athttps://opensource.org/licenses/MITby 机器谱 2023-07-14 https://www.robotway.com/------------------------------*/#include <Servo.h>#include <LedControl.h>#define middle1 88#define middle2 88Servo myservo[2];LedControl lc=LedControl(12,11,13,1);int pin[3] = {A0, A4, A3};byte value;byte value_his = 0;int time[3];//用于记录传感器的触发时间void setup(){LedInit();myservo[0].attach(4);myservo[1].attach(3);}void loop(){value = 0;for(int i=0; i<3; i++){value |= (digitalRead(pin[i]) << i);if(!digitalRead(pin[i])){time[i] = millis();//调用mills函数可以得到此时单片机的运行时间}}if(TimeDeal()){if(millis() > 1000){//用于排除刚开机时的误判LedDis();//十字路口显示}}if(value == 0x07){//当传感器都没有触发时默认为上一次的值value = value_his;}switch (value) {case 0x00://全部触发Forwards();break;case 0x01://触发右边两个while(digitalRead(pin[1])){Right();}break;case 0x03://触发右边一个while(digitalRead(pin[1])){Right();}break;case 0x04://触发左边两个while(digitalRead(pin[1])){Left();}break;case 0x05://触发中间一个Forwards();break;case 0x06://触发左边一个while(digitalRead(pin[1])){Left();}break;default:stop();}value_his = value;lc.clearDisplay(0);}void Left(){myservo[0].write(middle1);myservo[1].write(middle2 + 20);}void Right(){myservo[0].write(middle1 - 20);myservo[1].write(middle2);}void Forwards(){myservo[0].write(middle1 - 20);myservo[1].write(middle2 + 20);}void stop(){myservo[0].write(middle1);myservo[1].write(middle2);}bool TimeDeal()//十字路口识别函数{if(millis() > 500){if((abs(time[1] - time[0]) < 100) && (abs(time[1] - time[2]) < 100)){//当中间传感器与另外两个传感器触发的时间小于100毫秒时判定为十字路口return true;}elsereturn false;   }}void LedDis()//十字路口显示函数{for(int i=3; i<5; i++){for(int j=0; j<8; j++){lc.setLed(0, i, j, true);}}for(int i=3; i<5; i++){for(int j=0; j<8; j++){lc.setLed(0, j, i, true);}}}void LedInit(){lc.shutdown(0,false); //start the 8*8 ledlc.setIntensity(0,8);lc.clearDisplay(0);}

 ⑥ 下面提供一个小车实现寻迹的完整程序(TrackingCar.ino):

/*------------------------------------------------------------------------------------版权说明:Copyright 2023 Robottime(Beijing) Technology Co., Ltd. All Rights Reserved.Distributed under MIT license.See file LICENSE for detail or copy athttps://opensource.org/licenses/MITby 机器谱 2023-07-14 https://www.robotway.com/------------------------------*/#include <LedControl.h>#include <Servo.h>#define middle1 88#define middle2 88LedControl lc=LedControl(12,11,13,1);Servo myservo[2];int pin[3] = {A0, A4, A3};int time[3];byte value;byte value_his = 0;int flag = 0;int times = 0;void setup(){LedInit();Serial.begin(9600);//串口,用于调试myservo[0].attach(4);myservo[1].attach(3);}void loop(){value = 0;for(int i=0; i<3; i++){value |= (digitalRead(pin[i]) << i);if(!digitalRead(pin[i])){time[i] = millis();}}if(TimeDeal()){times++;Serial.print(times);}else{if(times > 1){Serial.println();flag += 1;Serial.println(flag);}times = 0;}if(flag == 3){while(1){stop();}}if(value == 0x07){value = value_his;}switch (value) {case 0x00://全部触发LedOn(0);LedOn(1);LedOn(2);Forwards();//delay(500);break;case 0x01://触发右边两个LedOn(1);LedOn(2);while(digitalRead(pin[1])){Right();}break;case 0x03://触发右边一个LedOn(2);while(digitalRead(pin[1])){Right();}break;case 0x04://触发左边两个LedOn(0);LedOn(1);while(digitalRead(pin[1])){Left();}break;case 0x05://触发中间一个LedOn(1);Forwards();break;case 0x06://触发左边一个LedOn(0);while(digitalRead(pin[1])){Left();}break;default:stop();}value_his = value;lc.clearDisplay(0);}bool TimeDeal(){if(millis() > 500){if((abs(time[1] - time[0]) < 100) && (abs(time[1] - time[2]) < 100)){return true;}elsereturn false;   }}void Left(){myservo[0].write(middle1);myservo[1].write(middle2 + 20);}void Right(){myservo[0].write(middle1 - 20);myservo[1].write(middle2);}void Forwards(){myservo[0].write(middle1 - 20);myservo[1].write(middle2 + 20);}void stop(){myservo[0].write(middle1);myservo[1].write(middle2);}void LedOn(int key){for(int i=0; i<2; i++){for(int j=3*key; j<3*key+2; j++){lc.setLed(0, i, j, true);}}}void LedInit(){lc.shutdown(0,false); //start the 8*8 ledlc.setIntensity(0,8);lc.clearDisplay(0);}

 程序源代码资料内容详见 小型双轮差速底盘-寻迹与路口

相关文章:

小型双轮差速底盘寻迹功能的实现

1. 功能说明 寻迹机器人是一种能够跟踪特定物体或线路的机器人。它们通常具有以下功能和特点&#xff1a; ① 传感器&#xff1a;寻迹机器人配备了用于感知环境的传感器&#xff0c;如摄像头、灰度传感器等。这些传感器可以探测地面上的标记、颜色、纹理或其他特定特征&#xf…...

第七篇:k8s集群使用helm3安装Prometheus Operator

安装Prometheus Operator 目前网上主要有两种安装方式&#xff0c;分别为&#xff1a;1. 使用kubectl基于manifest进行安装 2. 基于helm3进行安装。第一种方式比较繁琐&#xff0c;需要手动配置yaml文件&#xff0c;特别是需要配置pvc相关内容时&#xff0c;涉及到的yaml文件太…...

Chrome 75不支持保存成mhtml的解决方法

在Chrome 75之前&#xff0c;可以设置chrome://flags -> save as mhtml来保存网页为mhtml。 升级新版&#xff0c;发现无法另存为/保存网页为MHTML了。 在网上搜索无果后&#xff0c;只得从chromium项目的commits中查找&#xff0c;原来chrome搞了个"Chrome Flag Owner…...

工程监测振弦采集仪应用于岩土工程监测案例

振弦采集仪是一种用于测量地面或岩土中振动参数的仪器&#xff0c;可以对地基、土壤和岩体的性质及其变化进行监测。在岩土工程监测中&#xff0c;振弦传感器被广泛应用于测量土体或岩体的振动情况&#xff0c;以了解地震或其他自然灾害的影响。 以下是一个振弦采集仪应用岩土工…...

配置HDFS单机版,打造数据存储的强大解决方案

目录 简介&#xff1a;步骤&#xff1a;安装java下载安装hadoop配置hadoop-env.sh配置 core-site.xml配置hdfs-site.xml初始化hdfs文件系统启动hdfs服务验证hdfs 结论&#xff1a; 简介&#xff1a; Hadoop分布式文件系统&#xff08;HDFS&#xff09;是Hadoop生态系统中的一个…...

U盘删除的文件怎么找回?4个简单方法分享!

“在u盘里不小心删除的文件到底还能不能找回来呀&#xff1f;真的好着急啊&#xff01;这个u盘对我来说真的很重要&#xff0c;怎么恢复里面的数据呢&#xff1f;请各位大佬帮帮我吧&#xff01;” 作为一个便捷的存储工具&#xff0c;u盘逐渐获得大众的青睐。在互联网时代&…...

【雕爷学编程】MicroPython动手做(27)——物联网之掌控板小程序2

知识点&#xff1a;什么是掌控板&#xff1f; 掌控板是一块普及STEAM创客教育、人工智能教育、机器人编程教育的开源智能硬件。它集成ESP-32高性能双核芯片&#xff0c;支持WiFi和蓝牙双模通信&#xff0c;可作为物联网节点&#xff0c;实现物联网应用。同时掌控板上集成了OLED…...

形参动态内存开辟和柔性数组

//柔性数组 //定义&#xff1a;结构体最后一个成员允许是未知大小的数组 // 优点;在开辟空间时&#xff0c;连续开辟&#xff0c;便于释放空间&#xff0c;不会因多次开辟&#xff0c;导致释放空间出错 // 开辟空间时&#xff0c;节省动态开辟次数&#xff0c;节省空间&am…...

【LLM系列之指令微调】长话短说大模型指令微调的“Prompt”

1 指令微调数据集形式“花样”太多 大家有没有分析过 prompt对模型训练或者推理的影响&#xff1f;之前推理的时候&#xff0c;发现不加训练的时候prompt&#xff0c;直接输入模型性能会变差的&#xff0c;这个倒是可以理解。假如不加prompt直接训练&#xff0c;是不是测试的时…...

MacOS使用brew如何下载Nginx

首先&#xff0c;第一步切换源&#xff1a; 切换 brew.git 仓库地址&#xff1a; cd "$(brew --repo)" git remote set-url origin https://mirrors.aliyun.com/homebrew/brew.git 替换 homebrew-core.git 仓库地址: cd "$(brew --repo)/Library/Taps/home…...

linux ftp

使用ftp连接本机进行文件传输 1、下载vsftpd服务器程序 apt install vsftpd 2、使用tcp抓包 tcpdump -nt -i lo port 20 在FTP连接到本地主机&#xff08;127.0.0.1&#xff09;时&#xff0c;数据可能通过本地回环接口&#xff08;loopback interface&#xff09;传输&…...

你知道HTTP与HTTPS有什么区别吗?

作者&#xff1a;Insist-- 个人主页&#xff1a;insist--个人主页 作者会持续更新网络知识和python基础知识&#xff0c;期待你的关注 目录 一、什么是HTTP&#xff1f; 二、什么是HTTPS&#xff1f; 三、HTTPS 的工作原理 1、客户端发起 HTTPS 请求 2、服务端的配置 3、…...

keil使用printf函数重定串口输出,程序卡在Reset_Handler

最近在做国产芯片GD32F103项目&#xff0c;使用printf()函数重定向USART0串口输出&#xff0c;发现程序没有运行&#xff0c;单步调试发现&#xff0c;程序卡在startup_gd32f10x.s文件的Reset_Handler处&#xff0c;记录一下解决方法。 解决办法&#xff1a; 1、引用头文件#in…...

Redis预热 雪崩 击穿 穿透

redis预热 在Redis中&#xff0c;预热是指在实际的负载之前&#xff0c;提前将数据加载到内存中&#xff0c;以便在请求到来时能够快速响应。预热可以减少冷启动时的延迟&#xff0c;并提高系统的性能。 有几种方法可以进行Redis的预热&#xff1a; 使用持久化机制&#xff1…...

Shell脚本学习-MySQL单实例和多实例启动脚本

已知MySQL多实例启动命令为&#xff1a; mysqld_safe --defaults-file/data/3306/my.cnf & 停止命令为&#xff1a; mysqladmin -uroot -pchang123 -S /data/3306/mysql.sock shutdown 请完成mysql多实例的启动脚本的编写&#xff1a; 问题分析&#xff1a; 要想写出脚…...

vue3搭建(vite+create-vue)

目录 前提条件 输入命令 对于Add an End-to-End Testing Solution nightwatch和Cypress 和 Playwright 运行 前提条件 熟悉命令行已安装 16.0 或更高版本的 Node.js &#xff08;node -v查看版本&#xff09; 输入命令 npm init vuelatest 这一指令将会安装并执行 create-…...

服务器中了360后缀勒索病毒怎么解决,360后缀勒索病毒解密数据恢复

某医药公司是一家小型企业&#xff0c;拥有自己的服务器存储重要数据和文件。某天早上&#xff0c;IT管理员发现企业服务器中了360后缀的勒索病毒&#xff0c;所有数据文件都被加密了。这个病毒的入侵让公司业务受到严重影响&#xff0c;企业立即启动了勒索病毒解密数据恢复的措…...

3000字详解:风控核心岗位及核心价值

01、信贷场景中所谓风控是什么&#xff1f; 从一个小故事说起&#xff1a; “风控是什么&#xff1f;” “你走过大桥么&#xff1f;” “桥上有栏杆么&#xff1f;” “有” “你过桥时会扶栏杆么” “一般不扶” “那栏杆是不是没必要有呢” “那还是得有啊&#xf…...

fiddler 手机抓包(含https) 完整流程

第一部分&#xff1a;下载并安装fiddler 一.使用任一浏览器搜索【fiddler下载安装】&#xff0c;并下载fiddler 安装包。 二.fiddler安装包下载成功后&#xff0c;将下载的fiddler压缩包解压到自定义文件夹【fiddler】或者解压到当前文件夹下&#xff0c;双击文件夹中的【fidd…...

ChatGPT学python——制作自己的AI模型(一)初步了解

前言 「作者主页」&#xff1a;雪碧有白泡泡 「个人网站」&#xff1a;雪碧的个人网站 「推荐专栏」&#xff1a; ★java一站式服务 ★ ★前端炫酷代码分享 ★ ★ uniapp-从构建到提升★ ★ 从0到英雄&#xff0c;vue成神之路★ ★ 解决算法&#xff0c;一个专栏就够了★ ★ 架…...

python/java环境配置

环境变量放一起 python&#xff1a; 1.首先下载Python Python下载地址&#xff1a;Download Python | Python.org downloads ---windows -- 64 2.安装Python 下面两个&#xff0c;然后自定义&#xff0c;全选 可以把前4个选上 3.环境配置 1&#xff09;搜高级系统设置 2…...

基于uniapp+WebSocket实现聊天对话、消息监听、消息推送、聊天室等功能,多端兼容

基于 ​UniApp + WebSocket​实现多端兼容的实时通讯系统,涵盖WebSocket连接建立、消息收发机制、多端兼容性配置、消息实时监听等功能,适配​微信小程序、H5、Android、iOS等终端 目录 技术选型分析WebSocket协议优势UniApp跨平台特性WebSocket 基础实现连接管理消息收发连接…...

基于服务器使用 apt 安装、配置 Nginx

&#x1f9fe; 一、查看可安装的 Nginx 版本 首先&#xff0c;你可以运行以下命令查看可用版本&#xff1a; apt-cache madison nginx-core输出示例&#xff1a; nginx-core | 1.18.0-6ubuntu14.6 | http://archive.ubuntu.com/ubuntu focal-updates/main amd64 Packages ng…...

3-11单元格区域边界定位(End属性)学习笔记

返回一个Range 对象&#xff0c;只读。该对象代表包含源区域的区域上端下端左端右端的最后一个单元格。等同于按键 End 向上键(End(xlUp))、End向下键(End(xlDown))、End向左键(End(xlToLeft)End向右键(End(xlToRight)) 注意&#xff1a;它移动的位置必须是相连的有内容的单元格…...

鸿蒙DevEco Studio HarmonyOS 5跑酷小游戏实现指南

1. 项目概述 本跑酷小游戏基于鸿蒙HarmonyOS 5开发&#xff0c;使用DevEco Studio作为开发工具&#xff0c;采用Java语言实现&#xff0c;包含角色控制、障碍物生成和分数计算系统。 2. 项目结构 /src/main/java/com/example/runner/├── MainAbilitySlice.java // 主界…...

HashMap中的put方法执行流程(流程图)

1 put操作整体流程 HashMap 的 put 操作是其最核心的功能之一。在 JDK 1.8 及以后版本中&#xff0c;其主要逻辑封装在 putVal 这个内部方法中。整个过程大致如下&#xff1a; 初始判断与哈希计算&#xff1a; 首先&#xff0c;putVal 方法会检查当前的 table&#xff08;也就…...

论文笔记——相干体技术在裂缝预测中的应用研究

目录 相关地震知识补充地震数据的认识地震几何属性 相干体算法定义基本原理第一代相干体技术&#xff1a;基于互相关的相干体技术&#xff08;Correlation&#xff09;第二代相干体技术&#xff1a;基于相似的相干体技术&#xff08;Semblance&#xff09;基于多道相似的相干体…...

R 语言科研绘图第 55 期 --- 网络图-聚类

在发表科研论文的过程中&#xff0c;科研绘图是必不可少的&#xff0c;一张好看的图形会是文章很大的加分项。 为了便于使用&#xff0c;本系列文章介绍的所有绘图都已收录到了 sciRplot 项目中&#xff0c;获取方式&#xff1a; R 语言科研绘图模板 --- sciRplothttps://mp.…...

【从零开始学习JVM | 第四篇】类加载器和双亲委派机制(高频面试题)

前言&#xff1a; 双亲委派机制对于面试这块来说非常重要&#xff0c;在实际开发中也是经常遇见需要打破双亲委派的需求&#xff0c;今天我们一起来探索一下什么是双亲委派机制&#xff0c;在此之前我们先介绍一下类的加载器。 目录 ​编辑 前言&#xff1a; 类加载器 1. …...

基于鸿蒙(HarmonyOS5)的打车小程序

1. 开发环境准备 安装DevEco Studio (鸿蒙官方IDE)配置HarmonyOS SDK申请开发者账号和必要的API密钥 2. 项目结构设计 ├── entry │ ├── src │ │ ├── main │ │ │ ├── ets │ │ │ │ ├── pages │ │ │ │ │ ├── H…...