机器人制作开源方案 | 森林管理员
?作者:李佳骏、常睿康、张智斌、李世斌、高华耸
单位:山西能源学院
指导老师:赵浩成、郜敏
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. 示例程序
#include <Servo.h>
int _ABVAR_1_data = 0 ;
int in1 = 5;
int in2 = 6;
int in3 = 9;
int in4 = 10;
int in5 = 12;
int SERVO_SPEED=20; ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?//定义舵机转动快慢的时间
int ACTION_DELAY=200; ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?//定义所有舵机每个状态时间间隔
Servo myServo[4];
int f = 50; ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? //定义舵机每个状态间转动的次数,以此来确定每个舵机每次转动的角度
int servo_port[4] = {3,4,7,8}; ? ? ? ? ? ? ? ? ? ? ? ? ? ?//定义舵机引脚
int servo_num = sizeof(servo_port) / sizeof(servo_port[0]); ? ? //定义舵机数量
float value_init[4] = {10, 40, 20, 40}; ? ? ? ? ? ? ? ?//定义舵机初始角度
//10, 110, 20, 40chu ? ?100, 110, 60, 30 ? kaishi ? ? 100, 40, 60, 30zhua ? ? 100, 40, 40, 30qi ? ?10, 40, 60, 30weizhi ?
//float value_init[6] = {10, 130, 60, 30}; ? ? ? ? ? ? ? ?//定义舵机初始角度
//float value_init[6] = {0, 40, 40, 40};
void setup()
{
Serial.begin(9600);
pinMode(in1, OUTPUT);
?pinMode(in2, OUTPUT);
?pinMode(in3, OUTPUT);
?pinMode(in4, OUTPUT);
? ?pinMode(in5, OUTPUT);
? ? ? for(int i=0;i<servo_num;i++){
? ?ServoGo(i,value_init[i]);
?}
}
void ServoStart(int which)
{
?if(!myServo[which].attached())myServo[which].attach(servo_port[which]);
?pinMode(servo_port[which], OUTPUT);
}
void ServoStop(int which)
{
?myServo[which].detach();
?digitalWrite(servo_port[which],LOW);
}
void ServoGo(int which , int where)
{
?if(where!=200)
?{
? ?if(where==201) ServoStop(which);
? ?else
? ?{
? ? ?ServoStart(which);
? ? ?myServo[which].write(where);
? ?}
?}
}
void servo_move(float value0, float value1, float value2, float value3)
{
?float value_arguments[] = {value0, value1, value2, value3};
?float value_delta[servo_num];
?for(int i=0;i<servo_num;i++)
?{
? ?value_delta[i] = (value_arguments[i] - value_init[i]) / f;
?}
?for(int i=0;i<f;i++)
?{
? ?for(int k=0;k<servo_num;k++)
? ?{
? ? ?value_init[k] = value_delta[k] == 0 ? value_arguments[k] : value_init[k] + value_delta[k];
? ? ?/**************************串口查看输出*****************************/
// ? ? ?Serial.print(value_init[k]);
// ? ? ?Serial.print(" ");
? ?}
// ? ?Serial.println();
? ? ?/**************************串口查看输出*****************************/
? ?for(int j=0;j<servo_num;j++)
? ?{
? ? ?ServoGo(j,value_init[j]);
? ?}
? ?delay(SERVO_SPEED);
?}
?delay(ACTION_DELAY);
}
void loop()
{
/* ? servo_move(10, 40, 20, 40);
? ? delay(500);
? ? ? servo_move(100, 20, 20, 40);
? ? delay(500);
? ? servo_move(100, 110, 20, 40);
? ? delay(500);
? ? servo_move(100, 110, 60, 40);
? ? delay(500); ?
? ?servo_move(100, 40, 60, 40);
? ? delay(500);
?servo_move(10, 40, 60, 40);
? delay(500);
?servo_move(10, 40, 20, 40);*/
_ABVAR_1_data = Serial.read() ;
if (( ( _ABVAR_1_data ) == ( 1 ) ))
{
? ? ? analogWrite(in1, 0); ? ? ? ? ? ? ? ? ? ? ? ?
? ? ?analogWrite(in2, 0);
? ? ?analogWrite(in3, 0);
? ? ?analogWrite(in4, 160);
? ? ?delay (10);
}
if (( ( _ABVAR_1_data ) == ( 2 ) ))
{
?for(int i=0;i<10;i++)
{
? ?analogWrite(in1, 160);
? ? ?analogWrite(in2, 0);
? ? ?analogWrite(in3, 0);
? ? ?analogWrite(in4, 0);
? ? ?delay (10);
}
}
if (( ( _ABVAR_1_data ) == ( 12 ) ))
{ ? ?
? for(int i=0;i<10;i++)
{
?servo_move(190, 40, 20, 130);
?analogWrite(in1,150);
? ? ?analogWrite(in2,0); ? ?
? ? ? ?analogWrite(in4,150);
? ? ?analogWrite(in3,0);
? ?delay (1000);
}
}
if (( ( _ABVAR_1_data ) == ( 3 ) ))
{ ? ?
?
?analogWrite(in1,150);
? ? ?analogWrite(in2,0); ? ?
? ? ? ?analogWrite(in4,150);
? ? ?analogWrite(in3,0);
? ?delay (1000);
}
if (( ( _ABVAR_1_data ) == ( 0 ) ))
{ ? ? ? analogWrite(in1,0);
? ? ?analogWrite(in2,0); ? ?
? ? ? ?analogWrite(in4,0);
? ? ?analogWrite(in3,0);
}
if (( ( _ABVAR_1_data ) == ( 4) )) ? //jixiebi1
{ ? ?
servo_move(10, 40, 40, 130);
delay(500);
?delay(500);
? delay(500);
?analogWrite(in5, 250);
? delay(500);
}
if (( ( _ABVAR_1_data ) == ( 5) )) ? //jixiebi1
{ ? ?
servo_move(10, 40, 20, 60);
? ? delay(500);
?analogWrite(in5, 0);
}
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 ?
{ ? ?
servo_move(100, 110, 60, 30);
? ? delay(500);
}
if (( ( _ABVAR_1_data ) == ( 8) )) ? //qi
{ ? ?
servo_move(100, 40, 60, 30);
? ? delay(500);
}
if (( ( _ABVAR_1_data ) == ( 9) )) ? //fan
{ ? ?
servo_move( 100, 40, 40, 30);
?delay(1000); ?
}
if (( ( _ABVAR_1_data ) == ( 10) )) ? //fan
{ ? ?
servo_move( 10, 40, 60, 30);
?delay(1000); ?
}
if (( ( _ABVAR_1_data ) == ( 11) )) ? //fan
{ ?
? ?servo_move(10, 40, 20, 40);
? ? delay(500);
? ? ? servo_move(100, 20, 20, 40);
? ? delay(500);
? ? servo_move(100, 130, 20, 40);
? ? delay(500);
? ? servo_move(100, 130, 50, 40);
? ? delay(500); ?
? ?servo_move(100, 40, 50, 40);
? ? delay(500);
?servo_move(10, 40, 50, 40);
? delay(500);
?servo_move(10, 40, 20, 40);
}
}
6. 创新点
? ? ? 小车行进结构利用“太极”式刚柔结合进行翻越障碍。当小车遇到有一定角度的倾斜坡的时候,此时小车的刚性材料组成的车轮即可派上用场,通过调整履带的旋转自由度,使得小车在上坡的过程中不会发射管前倾和后倾。在小车遇到有断崖(台阶)的地面环境时,其柔性材料做成的车轮,经过调整前轮的自由度和前轮和后轮之间的距离,可以做出俯仰抬头的动作,从而轻松爬上断崖(台阶}。与此同时,俩种不同材料组成的车轮,也解决了在移动过程中出现的打滑问题,使小车的稳定性更高。其次,小车的机械臂可以180°进行旋转,不仅使小车在工作时候可接触的范围更广,还减小了车的体积,从而加快了小车的速度,车身在此期间也可以保持到最佳的位置。在此期间,同时可以利用小车体积小的优势,可以顺利通过小的隧道进行工作。最后,我们设计的小车使其重心集中在了整个小车的最中心,减少了了在通过障碍工作途中造成小车前倾或者后倾,从而翻车的情况,加强了小车的稳定性。
7. 作品难点
? ? ? ① 本作品在左右两侧分别装有不同的机械结构,应对其进行有效合理的位置和结构设计,保证其相对应与其他传感器的配合。
? ? ? ② 小车在运行过程中需通过不同的障碍结构,而由于车身较小,故掌握小车在登台阶和上坡时的平衡很是关键,防止侧翻。故我们在进行上(下)坡(台阶)会将左右机械臂摆至前(后)进行平衡。
? ? ? ③ 在小车砍到设计的隔离带树木后,放在车后方拉出时,怎么样去寻找最合适的路线去拉走树木,从而防止其他树木在路途中阻挡砍到的树木撤离火源。
更多详情请见 :【S043】森林管理员
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。 如若内容造成侵权/违法违规/事实不符,请联系我的编程经验分享网邮箱:veading@qq.com进行投诉反馈,一经查实,立即删除!