- 追加された行はこの色です。
- 削除された行はこの色です。
#contents
*課題2[#p5eb685a]
指定の道をあらかじめ模造紙に描き、第1コース・第2コースそれぞれの道順に沿って製作したロボットを動かす。また、その途中に、350mlの空き缶を指定の位置から、指定の位置に運ぶ。
*コース[#l49cfc9f]
第2コース
#ref(2018a-mission2.png)
1Aをスタート
2Bを右折
3Jで一時停止の後、Yの空き缶をキャッチしてJに戻る
4JからKに向かい、Kを直進
5Lを直進
6Iを左折
7Hで一時停止の後、左折
8Gで一時停止の後、右折
9Fで一時停止の後、右折
10Eを直進
11Dで一時停止の後、空き缶をXに置いてDに戻り、Cに向かう
12Cで一時停止の後、直進
13Bを直進
14Aで停止
(一時停止の指定がある場所は、1秒間停止すること)
*ロボットの説明[#fbbc27ea]
**車輪部分[#k29f6177]
課題1と同様、シンプルな3輪構造で、前輪が2輪、後輪が1輪となっている。前輪の2輪が駆動し、後輪の2輪が自由に回転する。
**センサーライト部分 [#zcdc4beb]
2つの車輪ほぼ真ん中前方に、確実に明るさを読み取れるような高さに設置した。
#ref(IMG_1298.JPG)
**アーム[#y0f7317f]
#ref(IMG_1301.JPG)
#ref(アーム.png)
モータBに歯車を直接取り付け、
アームの動作は缶をつかむ動作、缶の方向を変える動作、缶を離す動作の3段階に分かれている。
***缶をつかむ動作 [#u0daf4ae]
モータBを稼働させるだけでは先にアームの方向が変わってしまうので、図のようにある程度固定し、缶をつかむ動作のみが稼働するようにした。
#ref(IMG_1300neo.png)
***缶の方向を変える動作 [#ce6f913e]
缶をつかむ動作を終えると缶をつかむ動きが固定され、缶の方向をかえる力が大きくなる。すると、上の図の部品が外れ、缶の方向が左に変わる。
***缶を離す動作 [#kefdb2d7]
缶の方向が変わると赤で囲まれた部分が回転し、それとともに黄で囲まれた部分が上昇し、回転部分にひっかかる。すると、缶の方向を変える部分が固定され、モータBを回すと、缶を離す。
#ref(IMG_1299ex.png)
*交差点の感知 [#vb1866bb]
今回、私は関数DAI(DAIに深い意味はない)を使って交差点を感知する方法として採用した。関数DAIは4つの範囲で図に示した数値を足されることになっている。
#ref(白黒数値.png)
ライントレース中は4つの範囲を行き来するのでほぼ±0に保たれ、交差点にさしかかるとDEEPBLACKの範囲上に比較的長時間存在するので関数DAIが増加し、ある一定値を超えると次の動作へうつる。
また、関数DAIを正確に測るために交差点前には"DAI *= 0"(関数DAIに0をかけて)にする。
#ref(白黒数値2.png)
#define LEFT_LINE if(DEEPWHITE){RIGHT DAI -= 3;} else if(WHITE){BIT_RIGHT DAI -= 1;} else if(DEEPBLACK){LEFT DAI += 3;} else if(BLACK){BIT_LEFT DAI += 1;} else {STRAGHT DAI *= 0;}
#define RIGHT_LINE if(DEEPWHITE){LEFT DAI -= 3;} else if(WHITE){BIT_LEFT DAI -= 1;} else if(DEEPBACK){RIGHT DAI += 3;} else if(BLACK){BIT_RIGHT DAI += 1;} else {STRAGHT DAI *= 0;}
プログラムのようにその時位置している範囲に従ってそれ相応の数値を足され、指定の方向へ移動する。どの範囲にも属さない時(丁度白と黒の境界にある時)は直進する。
右[#gabb9e62]
#define RIGHT OnFwd(OUT_A,30);OnRev(OUT_C,30);Wait(1);Off(OUT_AC);
少し右[#f2a3eb93]
#define BIT_RIGHT OnFwd(OUT_A,30);OnRev(OUT_C,10);Wait(1);Off(OUT_AC);
左[#yf200910]
#define LEFT OnFwd(OUT_C,30);OnRev(OUT_A,30);Wait(1);Off(OUT_AC);
少し左[#m9a0ad8a]
#define BIT_LEFT OnFwd(OUT_C,30);OnRev(OUT_A,10);Wait(1);Off(OUT_AC);
直線方向[#d825c59f]
#define STRAGHT OnFwd(OUT_AC,30);Wait(1);Off(OUT_AC);
各方向に1000分の1秒進み、その時間毎に今存在している範囲を感知することで正確にライントレースできるようにした。
#define KANSU DAI < 150
ライントレース中は4つの範囲を行き来するのでほぼ±0に保たれ、交差点にさしかかるとDEEPBLACKの範囲上に比較的長時間存在するので関数DAIが増加し、関数DAIが150を超えると次の動作へうつる。
また、関数DAIを正確に測るためにライントレース前には"DAI *= 0"(関数DAIに0をかけて0にする)をする。
*交差点からの右折等 [#nb589184]
*プログラミングにあたっての準備[#heeafd24]
**色分けし、それに伴う方向に動くプログラム [#u32bece1]
白
#define DEEPWHITE SENSOR_1 > 48
少し白 [#g54731d9]
#define WHITE SENSOR_1 > 44
少し黒ece9d1]
#define BLACK SENSOR_1 < 40
黒 [#hdece4ff]
#define DEEPBLACK SENSOR_1 < 36
右[#gabb9e62]
#define RIGHT OnFwd(OUT_A,30);OnRev(OUT_C,30);Wait(1);Off(OUT_AC);
少し右[#f2a3eb93]
#define BIT_RIGHT OnFwd(OUT_A,30);OnRev(OUT_C,10);Wait(1);Off(OUT_AC);
左[#yf200910]
#define LEFT OnFwd(OUT_C,30);OnRev(OUT_A,30);Wait(1);Off(OUT_AC);
少し左[#m9a0ad8a]
#define BIT_LEFT OnFwd(OUT_C,30);OnRev(OUT_A,10);Wait(1);Off(OUT_AC);
直線方向[#d825c59f]
#define STRAGHT OnFwd(OUT_AC,30);Wait(1);Off(OUT_AC);
#ref(白黒判別.png)
※上のプログラムを組み合わせて図のように動作するようにした。
#define LEFT_LINE if(DEEPWHITE){RIGHT DAI -= 3;} else if(WHITE){BIT_RIGHT DAI -= 1;} else if(DEEPBLACK){LEFT DAI += 3;} else if(BLACK){BIT_LEFT DAI += 1;} else {STRAGHT DAI *= 0;}
#define RIGHT_LINE if(DEEPWHITE){LEFT DAI -= 3;} else if(WHITE){BIT_LEFT DAI -= 1;} else if(DEEPBACK){RIGHT DAI += 3;} else if(BLACK){BIT_RIGHT DAI += 1;} else {STRAGHT DAI *= 0;}
#define KANSU DAI < 150
#define KANSU DAI < 150
それぞれ左・右のライントレース時に使うプログラム
#br
#define OFF Off(OUT_AC);
#br
#define WAIT Wait(1000);
車輪を止めるプログラムと停止するプログラム
#br
#define CATCH OnFwd(OUT_AC,20);Wait(300);OFF OnFwd(OUT_B,30);Wait(300);Off(OUT_B);OnRev(OUT_AC,20);Wait(300);OFF
#br
#define RELEASE OnFwd(OUT_B,30);Wait(1000);Off(OUT_B);Wait(500);OnRev(OUT_B,30);Wait(500);Off(OUT_B);
それぞれアームを閉開するプログラム
#br
#define SIMPLE_RIGHT OnFwd(OUT_A,40);OnRev(OUT_C,40);
#br
#define SIMPLE_LEFT OnFwd(OUT_C,40);OnRev(OUT_A,40);
それぞれ右・左にその場で旋回するプログラム
#br
#define GO_STRAIGHT OnFwd(OUT_A,40);OnFwd(OUT_C,40);Wait(190);OFF
#define MOVE_RIGHT SIMPLE_RIGHT Wait(750); OFF
#br
#define GO_STRAIGHT OnFwd(OUT_A,40);OnFwd(OUT_C,40);Wait(190);OFF
#define MOVE_LEFT SIMPLE_LEFT Wait(750); OFF
それぞれ交差点で右・左に曲がるプログラム
*プログラム本編 [#d73f6990]
task main()
{
SetSensorLight(S1);
int DAI=0;
while(KANSU){
RIGHT_LINE
}
OFF
DAI *= 0;
PlaySound(SOUND_DOWN);
GO_STRAIGHT
MOVE_RIGHT
while(KANSU){
RIGHT_LINE
}
OFF
DAI *= 0;
PlaySound(SOUND_UP);
WAIT
CATCH
OnFwd(OUT_B,10);
GO_STRAIGHT
MOVE_LEFT
while(BLACK){
OnFwd(OUT_AC,20);
}
OFF
repeat(2){
while(KANSU){
LEFT_LINE
}
OFF
DAI *= 0;
SIMPLE_RIGHT
Wait(50);
OFF
while(BLACK){
OnFwd(OUT_AC,30);
}
OFF
}
while(KANSU){
LEFT_LINE
}
OFF
PlaySound(SOUND_UP);
WAIT
DAI *= 0;
OnFwd(OUT_AC,30);
Wait(50);
OFF
SIMPLE_LEFT
Wait(450);
OFF
while(KANSU){
LEFT_LINE
}
OFF
DAI *= 0;
while(BLACK){
OnFwd(OUT_A,30);OnFwd(OUT_C,30);
}
OFF
while(KANSU){
RIGHT_LINE
}
OFF
DAI *= 0;
while(BLACK){
OnFwd(OUT_A,30);OnFwd(OUT_C,30);
}
OFF
while(KANSU){
LEFT_LINE
}
OFF
DAI *= 0;
GO_STRAIGHT
MOVE_LEFT
while(KANSU){
LEFT_LINE
}
OFF
WAIT
DAI *= 0;
GO_STRAIGHT
MOVE_RIGHT
while(BLACK){
OnFwd(OUT_A,30);OnFwd(OUT_C,30);
}
OFF
while(KANSU){
RIGHT_LINE
}
OFF
WAIT
DAI *= 0;
PlaySound(SOUND_UP);
GO_STRAIGHT
MOVE_RIGHT
while(KANSU){
RIGHT_LINE
}
OFF
SIMPLE_RIGHT
Wait(50);
OFF
while(BLACK){
OnFwd(OUT_AC,30);
}
OFF
while(KANSU){
RIGHT_LINE
}
OFF
Off(OUT_B);
WAIT
RELEASE
OnFwd(OUT_AC,30);
Wait(80);
OFF
SIMPLE_RIGHT
Wait(500);
OFF
while(KANSU){
RIGHT_LINE
}
OFF
DAI *= 0;
GO_STRAIGHT
MOVE_RIGHT
while(KANSU){
RIGHT_LINE
}
OFF
WAIT
DAI *= 0;
PlaySound(SOUND_UP);
while(BLACK){
OnFwd(OUT_A,30);OnFwd(OUT_C,29);
}
OFF
while(KANSU){
RIGHT_LINE
}
OFF
DAI *= 0;
PlaySound(SOUND_FAST_UP);
}
*課題2を終えて [#oe801b50]
**苦労した点 [#p3e849d8]
作成の過程では、センサーライトやアームを所定の位置に設置しつつ、全体のバランスを保つことが難しかった。また、アーム部分を1つのモータで2段階動かすことにこだわり過ぎて、時間と労力を使い過ぎた。プログラミングの過程では、交差点などを正常に通過させる確率を上げることが大変だった。
**工夫した点 [#u95b7b61]
上では苦労したと書いたが、やはり、あらゆる工夫を取り込んで、1つのモータで2段階動かすことができたことがよかった。また、ライントレースに関数を使用することで安定して交差点を認識できるようにした。