本帖最后由 hello怪 于 2017-7-25 22:24 编辑 5 A$ _! b9 s& ~3 ]4 @$ q6 p
) Q! A- D+ C9 l2 y(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html ' ~' C! i# H. ]$ b# }4 D% j
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
" b5 C# K. ?3 K; H* o场地% m# }0 I* v7 W7 G
水平度和光滑度
在家里当然瓷砖的效果是最好的
! f7 A4 c0 U$ V' \' J+ b, O8 ? 线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直
$ p' V7 ~# j: e, Y- f( K! F
- U6 p* F, i6 Z) _7 J$ H车子结构* t U9 P% v8 N7 v2 C0 ^6 H
轮子间距
, t: W8 N2 R4 q" c5 r- ] 轮子间距越大越好,稍微想一下便知3 t: a' t0 S4 A' e9 h2 M9 @! L
轮子大小9 E% {/ ^& W$ Y! Y' a1 w
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)
: Y+ H& ~7 T% F* t5 r. T 随动轮的设计3 p6 k, |; m5 l' H6 ]
随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响. k* n9 b4 q3 B( W4 Y6 t
重心位置$ u& z, a2 W! A, L% K
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小
9 ~8 J& f( h* W$ c 结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大7 w' F8 Q$ A+ l- m
轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)) l! m4 x f0 m8 R
驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动
% D7 p; S' _7 I0 z- g; X- G, `( d* M; y$ C( L
控制程序
7 `9 K7 z4 G! \3 r* N 运动过程控制5 v1 D' c" } a# U: g: F
好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)
2 g; h8 d: q5 i( G5 M$ e; `( V, \7 [6 p6 K. k+ {
调试与运行
! [, @3 B3 t* H5 a& c {) _ 轮子转速调整(车轮大小的设定)
) L8 b2 N6 n% Y/ ? 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法- q: P8 k* I% S
起始位置车子正方向和直线间的夹角0 h" ^: p& t J. h: n Z9 i
一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
( ] I2 x2 `3 p; i- n# G* }2 ?* o) k6 c
$ U, @1 } h* \- A: [1 j 然后上源代码: - ' f' s, n" p* N5 T) T$ c9 W
- importlejos.robotics.navigation.DifferentialPilot;
' U& c5 a8 u0 i% m - import lejos.util.Delay;
, V7 F% ]. I! u2 \4 J. f
z" Q: O% U# s- /*** s C6 ?' Z! r9 h4 Q
- *Robot that stops if the buttons are presses before it completes its travel.
, W1 ]- }8 K' C - */) d8 d. g- c" ?+ G# h/ D
8 c, Y( e3 E- Z, j! y& `
, C0 b0 p+ v% k& V1 _- public class Straight {# o; m& s- a @( t( h) N
- DifferentialPilotpilot;
" k9 b# i9 l1 f8 F" {# ] - doubleleftD = 4.3, rightD = 4.3;4 Q) ~/ G- ~: a
/ \$ `& S% a: T/ }) T( G+ D' Y- publicvoid adjust() {
2 S/ C% H; B+ d+ ?1 T - //wait for release' M8 T( K% ]5 u1 N7 a( t/ V9 \
- while(Button.readButtons() > 0)5 ~+ s5 U) H' [# I, M
- ;' V( i6 V/ I7 _( ?: R/ T m( A* Q$ [ e
- while(Button.readButtons() == 0) {
2 r- o) h! D( E/ L! m0 R, |* c - //input size by adjusting the wheels
$ A' z' q& L! G/ P% Q1 c6 b - rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;: b! M: S+ P2 @; p1 h! b
- leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值8 ^4 ?9 N8 H" F4 q0 Q
. e9 S+ y2 }6 F# m1 X- //display the size input' k; U1 \2 s. c
- LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);1 n: k( z( e5 f, {8 M
- LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);/ t5 g; @- b' L
- Delay.msDelay(100);
9 |0 g& Q8 Z( B - LCD.clear();+ P0 P3 s- h! y/ x. c0 s. P
- }) Z) J. M8 b/ a0 J7 @( q% }$ n3 p
- 8 L2 c4 E5 l# S. t
- }0 v) F/ E4 M9 B0 }% o$ ~8 ^
0 h% j3 t) j0 M4 L. a) K, [* p- publicvoid go() {
4 U/ V; G- ?+ f- O - //准备起步- }( z6 Y6 h8 G1 F5 \. C% ~5 N
- //wait for release9 e: z9 e ~3 C& ^' \8 J# S2 P
- while(Button.readButtons() > 0). I$ [9 ^( Q1 ]$ Q* O5 B
- ;
* z4 T' ~8 J- L0 O - System.out.println("pressto go");
& U3 b$ u/ P/ a2 @
) \; r1 c2 K1 ~0 A- Button.waitForAnyPress();
q1 Z5 C; ]/ W3 E) G& c - LCD.clear();
6 b6 o9 A$ \. y9 [) X: e+ g& a8 x! P
( q" H6 E! P2 V- //初始化
5 P0 }( G- x3 g - pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);
$ d0 d' L: z( @7 r+ ? - pilot.reset();
; X0 C/ y0 ~1 _+ `. \; u
7 ?7 C0 S# Y( V& L6 \% r+ }% \8 M- Delay.msDelay(3000);/ ]8 t- C/ M, K9 O8 c) s" X
- 3 S5 R, U" p. B0 T; d1 l
- doublespeed = 10;
/ n- u8 D9 S4 U. { - doubledistance = 0;0 P5 I5 m4 W4 u4 @- _# L5 O6 v7 E5 d* }
& g+ t# h3 o" ~. D- //起步加速
3 N( U5 g, d4 \ d - pilot.travel(300,true);
: S& n' P( N( x/ w* ?
3 F! S9 H0 @" o1 x7 Y- while(distance < 20) {
5 ]2 k# o/ x- j" p7 A: I9 p( p+ c, F2 O0 \ - distance= pilot.getMovement().getDistanceTraveled();
# F& G( G5 Y9 ?: @: B - speed= distance * 5 + 10;! r G) N" p9 m$ o
- pilot.setRotateSpeed(speed);) `. s1 k9 C! W2 N
- Delay.msDelay(100);6 ?# ~) p/ r1 \; b. o) n3 l" M: x
- }' b8 b9 W( z# h
- for(int i = 0; i < 10; i++) {
& t' u/ g R9 K \1 J$ o, y - speed+= 5;
1 @/ g! G8 U4 x7 O - pilot.setRotateSpeed(speed);7 C* C, |, i9 ]) P- D% u, J
- Delay.msDelay(200);& m7 ^( K$ {1 w* \, J$ i9 Q& ~# k% v
- }+ |; V q6 l: N* j- I$ d; o
- 5 C: b: _0 m/ Z/ h; n
- //TODO 减速
" v3 H! q( ]/ f4 { - 1 Y+ E o: C7 U$ h1 b* w
- //停止机能
1 Z. h8 {! J# @5 \- q; T* \+ Z/ Q - while(pilot.isMoving()) {: s T* V6 ]! s' P
- if(Button.readButtons() > 0): g) R7 ^; t& q- f
- pilot.stop();0 n$ u) ?3 Q5 @" z. n
- }7 ^8 b) u$ B: B" c
- : m5 c; k! ]( s5 L8 ~$ n
- System.out.println("" + pilot.getMovement().getDistanceTraveled());) y4 N4 d; H+ u5 u3 W2 {
- System.out.println(-Motor.A.getTachoCount());
" C" @9 w4 e- u - System.out.println(-Motor.B.getTachoCount());
( A5 J% P8 s" ^# F( X - Button.waitForAnyPress();
~' ~1 J7 {, S* f4 l, c, j8 | - }! o! E9 ?8 V' E2 S+ V( n
- 4 g$ D0 o$ V2 k$ v) S% F" F( C
- publicstatic void main(String[] args) {
' n& C5 ~" m" y- ?6 N- v - Straighttraveler = new Straight();( p* g1 N6 c7 I
' a+ N0 I/ c! K- s! j7 e- traveler.adjust();3 [ y6 K; r4 b& U0 Y( m {
- traveler.go();9 y# A( _3 R ]
- }
0 [# J6 ~+ e$ ` - }
复制代码
, U1 I9 P1 f1 H/ z+ { |