本帖最后由 hello怪 于 2017-7-25 22:24 编辑 3 {4 H+ H9 Q' H( P; |5 c( W
! z8 Q" w5 ?* y; u(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html
& ^/ @" D( M! C9 X+ Q
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面: , j! W7 p4 v# h9 H2 G4 M+ ]
场地6 C& R4 V" n' m8 u
水平度和光滑度 在家里当然瓷砖的效果是最好的4 U, v- Y, @* K$ m2 g. t, C8 T
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直
$ O9 @3 ]7 e! U. C) m1 s
3 [( N8 X6 I% {6 ^9 H车子结构1 K8 { c8 R7 R0 g2 Q2 \
轮子间距) C, O7 Q# q; l7 c6 j \' m
轮子间距越大越好,稍微想一下便知
a* p$ _7 i) t% M 轮子大小 ^, ]! I2 A% K% R5 V3 Y
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)
9 U: B4 F+ ?* {/ I 随动轮的设计' H2 I7 m# l6 a* w- _
随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响
; N7 _/ W( e( \* t7 j& g 重心位置+ r3 B, G4 ?" v& [; n
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小
- P6 K- i- S- n ^2 Z, ? 结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大
3 [+ [: h- L. L" H3 l. Z0 g 轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)2 h" l i% ^# |3 a: Y. U% A
驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动
" W; C, T" |$ E* W; R5 C# T/ e! F8 W0 B- D4 K
控制程序+ T! D; ^ k2 ^# ^5 B
运动过程控制
7 ?! [/ q5 k8 H' g; p1 I: |8 ?0 Q; p 好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)
+ ~ U' l( Q# G4 N7 W9 d0 P
7 y/ {9 Y1 c4 J调试与运行
# q; I' G9 _# l% r 轮子转速调整(车轮大小的设定)
" t$ E, w9 C2 i9 Q$ E% Q 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法' L% o' Z( l3 t7 [' F2 C, L0 ^
起始位置车子正方向和直线间的夹角' X0 N/ A! [3 W t3 C, ~6 G/ w2 K
一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
9 I; t$ t2 h& V! T7 f3 L5 b
+ p6 F/ E; _* S$ q0 K: p; H% f) A7 m
然后上源代码: - 6 k. W6 c5 E+ U* O7 F
- importlejos.robotics.navigation.DifferentialPilot;! \' N0 S$ w9 v, |0 {' U# Q
- import lejos.util.Delay;
; q( q* k+ B& N8 ?- s - 7 T. ^* v0 J3 S3 _# A
- /**
' M5 j; v. A9 b# E1 d' d - *Robot that stops if the buttons are presses before it completes its travel.
0 R \. J" E, X" y, ?5 ?9 ]/ D - */+ w6 N7 b& b$ j5 X* H9 a9 H/ w
8 y$ r: I% d s9 W }
0 c/ q' ^" K2 b- public class Straight {
* c+ v( s; X5 ?* R5 R - DifferentialPilotpilot;
6 ?3 j# x! x/ W - doubleleftD = 4.3, rightD = 4.3;
6 M6 {0 m& n# s" j- R! \
9 W+ E' ^- Y& N7 `' P) M- publicvoid adjust() {
6 w \* [2 e3 A6 G - //wait for release
& h2 g0 M3 R' o9 h3 D; F - while(Button.readButtons() > 0)
3 o0 h& E. q3 e - ;( |* T D A5 A u
- while(Button.readButtons() == 0) {
1 D( d2 W* i) Y. m - //input size by adjusting the wheels) h2 [ M9 m; h& b6 Y7 P2 ^
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;! N- d5 `& ` @
- leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值
4 B$ R4 B! k, n0 N
$ A7 k8 Y! ]9 f8 @- //display the size input+ Z0 g& ^( }0 J8 v
- LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);
) ?, v/ Y" C7 H7 _% ? - LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);% q: u5 ?4 @, ?
- Delay.msDelay(100);
0 x5 c6 L, m- I O c% B - LCD.clear();3 p: m" M; N$ @+ \6 }) E
- }
' }/ c0 F7 W! ] F, y9 B
: N# Q6 r; P& \' i$ @% J! z- }
! H2 `1 T8 x q/ W; q% k4 L1 f% N2 E% p
) @5 T6 z) M3 x. G6 n- publicvoid go() {
4 c% w( ]2 \5 R7 R+ b - //准备起步; y1 K3 n$ g& y3 X
- //wait for release
4 n4 j. {& |4 R( r9 U" C - while(Button.readButtons() > 0)5 O$ K2 ~4 h2 h
- ;
& @! E0 w! e- o" X- m - System.out.println("pressto go");
5 I# o5 [$ y2 S! B
: O" j+ E3 p0 L* F6 u1 A- Button.waitForAnyPress();
. ]7 B: c1 q: O) B - LCD.clear();
) b! L+ w' @- _ - ' o; x( d* c6 S- K* a1 u" f; o
- //初始化2 b% R4 R& u- j& |" W. ^' F
- pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);
! r( f7 d0 N2 y! ?( p& R- _/ n - pilot.reset();# N% |! v4 b) R$ F8 \; [
- F, d) [/ Q1 Y b+ d, Y- Delay.msDelay(3000);
]. O, ~. q; Z# |0 C+ ^! F4 P
_# M+ x9 e4 c3 ]* H- doublespeed = 10;. S1 ?# ?: m E5 V: n9 K
- doubledistance = 0;* o& G$ r$ x* `" _9 W
- ; @5 @( \8 Y l
- //起步加速
5 {% q- b6 F: V- r - pilot.travel(300,true);
7 b! I$ w7 I: h; @" k - ( g8 F8 S8 _' r) C$ W5 v: o% E3 [# k
- while(distance < 20) {
+ D1 |* ^9 g/ y( _ - distance= pilot.getMovement().getDistanceTraveled();
+ W8 `9 Y5 Z4 D: H2 r1 O; m9 J4 q - speed= distance * 5 + 10;/ D$ x9 z& q! F2 T5 u
- pilot.setRotateSpeed(speed);1 d6 ~0 I N: V' x+ L- R6 R
- Delay.msDelay(100);7 V( v4 O S2 P% a* K/ W
- }
5 ]8 k- v0 K! m - for(int i = 0; i < 10; i++) {1 u7 H& c$ ^+ x2 ]) c) v
- speed+= 5;% ^; W) p1 }$ v; r6 U. [' W5 Q
- pilot.setRotateSpeed(speed);, h4 ~* c4 l0 k6 U& U# p& B# r
- Delay.msDelay(200);
! x7 h6 E" f1 l/ d; r+ r - }
1 c1 O2 j0 W2 ]# O4 _/ A
, w. F {" R; w5 t1 U/ M7 f- //TODO 减速
) a( I7 R5 u) Z; Y" G* r1 b) e
$ J3 U/ v) Q( c+ [- @- //停止机能% Z" }9 }8 M W: U, k1 {+ [0 t
- while(pilot.isMoving()) {
* s& |* [: I* Z# P& x& A - if(Button.readButtons() > 0)
! M% A) Q, i& E: x1 E O! D - pilot.stop();1 Z9 v/ t1 l, }. M% E) z+ X: h
- }. `, ?* b4 z! c; a" E$ Y2 c
- , r" c V# M3 u0 x: n1 q
- System.out.println("" + pilot.getMovement().getDistanceTraveled());$ v) l; c) P) |5 m; C
- System.out.println(-Motor.A.getTachoCount());
/ U) V7 B& j, S+ P! {) U' m; C8 ^ - System.out.println(-Motor.B.getTachoCount());
) Y5 A% Z9 @" U( U: H" {7 ? - Button.waitForAnyPress();
! f+ N; X9 J, @; o1 a1 Z% M - }
* O8 c. u# Z3 C# X. R - : h3 Z5 K" `) j8 M% S
- publicstatic void main(String[] args) {. ~" T/ J. \5 R: x$ y4 _
- Straighttraveler = new Straight();
b3 ?* Q' c, F- y - " B, D% A8 R- G8 l! U
- traveler.adjust();5 [4 @) _8 G& F6 h' u6 j
- traveler.go();/ j E7 E) J$ U6 o
- }& i, k5 |- j- x. h z
- }
复制代码 % c! x4 r" K) T! j+ j
|