极客工坊

 找回密码
 注册

QQ登录

只需一步,快速开始

查看: 1413|回复: 0

机器人制作开源方案 | 森林管理员

[复制链接]
发表于 2023-12-29 09:36:26 | 显示全部楼层 |阅读模式
本帖最后由 机器谱 于 2023-12-29 09:36 编辑

作者:李佳骏、常睿康、张智斌、李世斌、高华耸
单位:山西能源学院
指导老师:赵浩成、郜敏
1. 研究背景
      森林作为地球上可再生自然资源及陆地生态的主体,在人类生存和发展的历史中起着不可代替的作用,它不仅能提供国家建设和人民生活所需的木材及林副产品,而且还肩负着释放氧气调节气候环境,涵养水源、保持水土、防风固沙、美化环境、净化空气、减少噪音及旅游保健等多种使命,同时森林还是农牧业稳产高产的重要条件。然而,森林火灾是森林最危险的敌人,也是林业最可怕的灾害,也会给森林带来最有害、具有毁灭性的后果。森林火灾不只是烧毀片片森林,伤害林内动物,而旦还降低森林的更新能力,引起土壤的贫瘠和破坏森林涵养水源的作用,甚至号致生态环境失去平衡。2019年3月30日,四川省凉山州木里县就曾经发生过一次非常严重的森林火灾。2019年3月30日是一个雨天,木里县喇嘛寺沟附近下午5点左右出现了雷雨天气,虽然只是一次持续了30秒左右的阵雨,但是隆隆雷声却一直回荡在天空,不绝于耳。下午6点时一场森林火灾在山上悄然而至,起火点位于海拔3800余米的山上,山中地形复杂,遍布沟壑,无论是交通还是通讯情况都非常不好,消防官兵难以迅速赶往现场进行救援。经过相关部门得知消息后初步判断本次火灾的原因,可能是由于之前降兩时的雷击导致,多个起火点都位于山崖上,火场平均海拔高达4千米。31日下午,689位救火人员带着各种设备登山,前往海拔4000余米的原始森林开展救火工作,由于当日风向不定,风力风向突变导致山火爆燃形成一个巨大的火球。救火人员兵分两路进行避险,其中一路安全撤离,另一路的27位森林消防指战员和三名地方救火人员就此和其他人失去了联系。4月2日早晨六点30分,田火山山上的明火终于被扑灭。而制作山林隔离带是有效制止森林大火蔓延的的有效方式,且该方面大多是通过人工下地来完成的,所以我们设计了一款可通过远程遥控来实现砍伐、运输一体化的全地形机器人,减少人工的需求量和危险性。同时,目前生活中的伐木大多由人工实地操作,我们的作品也可以代替其完成系统化的作业。

2. 主要功能
       本小组作品的主要功能有在大火发生时,及时进入火场中,对还未涉及到的地域进行砍伐树木,降低火灾面积。本小车为全地形小车,可以适应森林中的各种地形,如山坡、地坑等等。小车一侧设有抓夹,在另一侧砍伐完树木后抓夹可以及时将树枝抓起放到车后部,最终由小车将其带出森林。在救援的同时小车前部设有摄像头以及湿度传感器,实时监控并记录火灾情况。本作品不仅可以用作在森林防护中制作隔离带的一体化工作,也可以用作日常生活中做伐木一体机器人使用。

3. 场景调研
       森林火灾的起因主要有两大类:人为火和自然火。人为火包括:生产性火源(农、林、牧业生产用火,林副业生产用火,工矿运输生产用火等)、非生产性火源(野外吸烟、做饭、烧纸、取暖等)、故意纵火。自然火包括:雷电火、自燃等。由于我国的森林火灾中,由吸烟、烧荒和上坟烧纸引起的火灾占了绝大多数,所以我们小组场地模拟森林火灾的起因为烧荒。在实际的地形中会存在山坡和小断层,我们小组在模拟时,模拟机器人爬小山坡和爬小台阶(断层)。同时,模拟了砍伐树木的过程,再砍伐完后,车另一侧的抓夹会将树木夹起搬运出森林。在进行就救援以及砍伐树木时,车前的摄像头可进行实时监控,可记录下在砍伐树木时周遭的实时动态,同时还有湿度传感器的实时监测。

4. 机器人本体技术路线说明
       ① 机械结构:整体车身分为两部分,前端车体运用柔性结构,并采用履带式车轮,后部采用传统轮胎,有着很好的抓地力。车体大致结合军用坦克的履带的优越性并结合实际,改装了后置轮胎,使得前后轮可以通过配合,从而顺利翻越一些工作途中所遇到的障碍。
      ② 驱动系统:该车加装了6个双直流电机共同驱动,六个电机同时驱动,为小车提供了足够的动力,减少了动力的损失,为通过障碍物提供了足够的动力条件,电锯有单舵机制作的机械臂和电机控制的电锯体实现横向自由伸展,砍伐树木。机械爪由两个舵机控制的机械臂和单舵机控制的爪体构成,可实现横向伸展并调整爪子的摆动,实现抓取。
       ③ 控制系统:主板采用了Basra主控板,通过ESP8266Wi-Fi模块与电脑连接实现远程实时无限操作,通过摄像头、温湿度传感器来实时反馈现场情况,为人们的勘测和救援提供信息帮助。
       ④ 感知系统:车体全程操作由人工在远程通过WiFi模块实时操作,在车体的前方和中段均装有摄像头,前部摄像头可帮我我们锁定目标树木的方向,也可以规避障碍物,在侧端的摄像头可对目标数目进行定位,当其进入合适位置后进行砍伐和搬运。在车的顶部同时也装配了温湿度传感器,对周围环境进行监控,并及时反馈,为人工救援行动提供保障。

5. 示例程序
  1. #include <Servo.h>

  2. int _ABVAR_1_data = 0 ;

  3. int in1 = 5;

  4. int in2 = 6;

  5. int in3 = 9;

  6. int in4 = 10;

  7. int in5 = 12;

  8. int SERVO_SPEED=20;                                        //定义舵机转动快慢的时间

  9. int ACTION_DELAY=200;                                      //定义所有舵机每个状态时间间隔

  10. Servo myServo[4];

  11. int f = 50;                                                     //定义舵机每个状态间转动的次数,以此来确定每个舵机每次转动的角度

  12. int servo_port[4] = {3,4,7,8};                            //定义舵机引脚

  13. int servo_num = sizeof(servo_port) / sizeof(servo_port[0]);     //定义舵机数量

  14. float value_init[4] = {10, 40, 20, 40};                //定义舵机初始角度

  15. //10, 110, 20, 40chu    100, 110, 60, 30   kaishi     100, 40, 60, 30zhua     100, 40, 40, 30qi    10, 40, 60, 30weizhi   

  16. //float value_init[6] = {10, 130, 60, 30};                //定义舵机初始角度

  17. //float value_init[6] = {0, 40, 40, 40};

  18. void setup()

  19. {

  20. Serial.begin(9600);

  21. pinMode(in1, OUTPUT);

  22.   pinMode(in2, OUTPUT);

  23.   pinMode(in3, OUTPUT);

  24.   pinMode(in4, OUTPUT);

  25.     pinMode(in5, OUTPUT);



  26.        for(int i=0;i<servo_num;i++){

  27.     ServoGo(i,value_init[i]);

  28.   }

  29. }

  30. void ServoStart(int which)

  31. {

  32.   if(!myServo[which].attached())myServo[which].attach(servo_port[which]);

  33.   pinMode(servo_port[which], OUTPUT);

  34. }

  35. void ServoStop(int which)

  36. {

  37.   myServo[which].detach();

  38.   digitalWrite(servo_port[which],LOW);

  39. }

  40. void ServoGo(int which , int where)

  41. {

  42.   if(where!=200)

  43.   {

  44.     if(where==201) ServoStop(which);

  45.     else

  46.     {

  47.       ServoStart(which);

  48.       myServo[which].write(where);

  49.     }

  50.   }

  51. }

  52. void servo_move(float value0, float value1, float value2, float value3)

  53. {

  54.   float value_arguments[] = {value0, value1, value2, value3};

  55.   float value_delta[servo_num];

  56.   for(int i=0;i<servo_num;i++)

  57.   {

  58.     value_delta[i] = (value_arguments[i] - value_init[i]) / f;

  59.   }

  60.   for(int i=0;i<f;i++)

  61.   {

  62.     for(int k=0;k<servo_num;k++)

  63.     {

  64.       value_init[k] = value_delta[k] == 0 ? value_arguments[k] : value_init[k] + value_delta[k];

  65.       /**************************串口查看输出*****************************/

  66. //      Serial.print(value_init[k]);

  67. //      Serial.print(" ");

  68.     }

  69. //    Serial.println();

  70.       /**************************串口查看输出*****************************/



  71.     for(int j=0;j<servo_num;j++)

  72.     {

  73.       ServoGo(j,value_init[j]);

  74.     }

  75.     delay(SERVO_SPEED);

  76.   }

  77.   delay(ACTION_DELAY);

  78. }

  79. void loop()

  80. {

  81. /*   servo_move(10, 40, 20, 40);

  82.      delay(500);

  83.        servo_move(100, 20, 20, 40);

  84.      delay(500);

  85.      servo_move(100, 110, 20, 40);

  86.      delay(500);

  87.      servo_move(100, 110, 60, 40);

  88.      delay(500);   

  89.     servo_move(100, 40, 60, 40);

  90.      delay(500);

  91.   servo_move(10, 40, 60, 40);

  92.    delay(500);

  93.   servo_move(10, 40, 20, 40);*/

  94. _ABVAR_1_data = Serial.read() ;

  95. if (( ( _ABVAR_1_data ) == ( 1 ) ))

  96. {

  97.        analogWrite(in1, 0);                        

  98.       analogWrite(in2, 0);

  99.       analogWrite(in3, 0);

  100.       analogWrite(in4, 160);

  101.       delay (10);

  102. }

  103. if (( ( _ABVAR_1_data ) == ( 2 ) ))

  104. {

  105.   for(int i=0;i<10;i++)

  106. {

  107.     analogWrite(in1, 160);

  108.       analogWrite(in2, 0);

  109.       analogWrite(in3, 0);

  110.       analogWrite(in4, 0);

  111.       delay (10);

  112. }

  113. }

  114. if (( ( _ABVAR_1_data ) == ( 12 ) ))

  115. {     

  116.    for(int i=0;i<10;i++)

  117. {

  118.   servo_move(190, 40, 20, 130);

  119.   analogWrite(in1,150);

  120.       analogWrite(in2,0);     

  121.         analogWrite(in4,150);

  122.       analogWrite(in3,0);

  123.     delay (1000);

  124. }

  125. }

  126. if (( ( _ABVAR_1_data ) == ( 3 ) ))

  127. {     

  128.    

  129.   analogWrite(in1,150);

  130.       analogWrite(in2,0);     

  131.         analogWrite(in4,150);

  132.       analogWrite(in3,0);

  133.     delay (1000);

  134. }

  135. if (( ( _ABVAR_1_data ) == ( 0 ) ))

  136. {       analogWrite(in1,0);

  137.       analogWrite(in2,0);     

  138.         analogWrite(in4,0);

  139.       analogWrite(in3,0);

  140. }

  141. if (( ( _ABVAR_1_data ) == ( 4) ))   //jixiebi1

  142. {     

  143. servo_move(10, 40, 40, 130);

  144. delay(500);

  145.   delay(500);

  146.    delay(500);

  147.   analogWrite(in5, 250);

  148.    delay(500);

  149. }

  150. if (( ( _ABVAR_1_data ) == ( 5) ))   //jixiebi1

  151. {     

  152. servo_move(10, 40, 20, 60);

  153.      delay(500);

  154.   analogWrite(in5, 0);

  155. }

  156. if (( ( _ABVAR_1_data ) == ( 7) ))   //10, 110, 20, 40chu    100, 110, 60, 30   kaishi     100, 40, 60, 30zhua     100, 40, 40, 30qi    10, 40, 60, 30weizhi   

  157. {     

  158. servo_move(100, 110, 60, 30);

  159.      delay(500);



  160. }

  161. if (( ( _ABVAR_1_data ) == ( 8) ))   //qi

  162. {     

  163. servo_move(100, 40, 60, 30);

  164.      delay(500);



  165. }

  166. if (( ( _ABVAR_1_data ) == ( 9) ))   //fan

  167. {     

  168. servo_move( 100, 40, 40, 30);

  169.   delay(1000);  



  170. }

  171. if (( ( _ABVAR_1_data ) == ( 10) ))   //fan

  172. {     

  173. servo_move( 10, 40, 60, 30);

  174.   delay(1000);  



  175. }

  176. if (( ( _ABVAR_1_data ) == ( 11) ))   //fan

  177. {   

  178.     servo_move(10, 40, 20, 40);

  179.      delay(500);

  180.        servo_move(100, 20, 20, 40);

  181.      delay(500);

  182.      servo_move(100, 130, 20, 40);

  183.      delay(500);

  184.      servo_move(100, 130, 50, 40);

  185.      delay(500);   

  186.     servo_move(100, 40, 50, 40);

  187.      delay(500);

  188.   servo_move(10, 40, 50, 40);

  189.    delay(500);

  190.   servo_move(10, 40, 20, 40);

  191. }

  192. }
复制代码

6. 创新点
      小车行进结构利用“太极”式刚柔结合进行翻越障碍。当小车遇到有一定角度的倾斜坡的时候,此时小车的刚性材料组成的车轮即可派上用场,通过调整履带的旋转自由度,使得小车在上坡的过程中不会发射管前倾和后倾。在小车遇到有断崖(台阶)的地面环境时,其柔性材料做成的车轮,经过调整前轮的自由度和前轮和后轮之间的距离,可以做出俯仰抬头的动作,从而轻松爬上断崖(台阶}。与此同时,俩种不同材料组成的车轮,也解决了在移动过程中出现的打滑问题,使小车的稳定性更高。其次,小车的机械臂可以180°进行旋转,不仅使小车在工作时候可接触的范围更广,还减小了车的体积,从而加快了小车的速度,车身在此期间也可以保持到最佳的位置。在此期间,同时可以利用小车体积小的优势,可以顺利通过小的隧道进行工作。最后,我们设计的小车使其重心集中在了整个小车的最中心,减少了了在通过障碍工作途中造成小车前倾或者后倾,从而翻车的情况,加强了小车的稳定性。

7. 作品难点
       ① 本作品在左右两侧分别装有不同的机械结构,应对其进行有效合理的位置和结构设计,保证其相对应与其他传感器的配合。
       ② 小车在运行过程中需通过不同的障碍结构,而由于车身较小,故掌握小车在登台阶和上坡时的平衡很是关键,防止侧翻。故我们在进行上(下)坡(台阶)会将左右机械臂摆至前(后)进行平衡。
       ③ 在小车砍到设计的隔离带树木后,放在车后方拉出时,怎么样去寻找最合适的路线去拉走树木,从而防止其他树木在路途中阻挡砍到的树木撤离火源。

更多详情请见 :【S043】森林管理员 https://www.robotway.com/h-col-309.html
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 注册

本版积分规则 需要先绑定手机号

Archiver|联系我们|极客工坊

GMT+8, 2024-4-28 14:48 , Processed in 0.048793 second(s), 17 queries .

Powered by Discuz! X3.4 Licensed

Copyright © 2001-2021, Tencent Cloud.

快速回复 返回顶部 返回列表