本帖最后由 hello怪 于 2017-7-25 22:24 编辑 y% o- R9 \( ~
) { ?" [2 {% V. K(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html - i* A, J% c: o, ], u# Y
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
( y9 t2 K: |2 Y% k场地' q. K' U" m9 C. I
水平度和光滑度
在家里当然瓷砖的效果是最好的: D' a" L( z' |& M* q! i) y' }
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直3 Z+ ~: I3 Y Z* n1 W: }
' b; H/ d6 \3 B! f {0 ^% Z _车子结构
) I' t# W- l: t+ C! S3 ]* K 轮子间距
- k- Y p6 {( G8 z 轮子间距越大越好,稍微想一下便知
) G' ?3 T( K6 _8 z 轮子大小
( o% Q, ~2 P" a7 L* } 轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)
, O, v) u% u% b$ @' ^7 Z 随动轮的设计% ^: D0 d: o7 c. D
随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响
1 ^2 K) r' M9 J3 L/ u. t ? 重心位置
9 J7 N; s: s' L4 [ `7 ` K4 J# H5 ~ 重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小
) N! o% t. N7 F/ W3 J 结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大; s6 B/ j. f& _4 E9 W. A
轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)' x8 P9 m( T4 ^' K7 | P
驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动3 Y% p% s8 _% B" Q- A
- u5 q. X$ u' H6 N
控制程序
' |5 D8 v0 [3 ^# I( Y5 L4 y& z 运动过程控制: @) H" a6 a! }& t3 f4 k
好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)
3 [( s9 C. m5 y: l. A: Z: Z9 y. F' z3 Y2 Y
调试与运行3 O( t0 N( H+ w
轮子转速调整(车轮大小的设定)$ `# A: z2 z+ r3 i; i) Q: E: P
这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法
0 r9 h# V( d! O6 v 起始位置车子正方向和直线间的夹角
' P0 z/ B+ |- y2 k1 K/ ~/ ~ 一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
5 b, @1 q0 q0 L2 ^, D+ n& D2 f
/ e* s; _6 N. t
9 s- W. g. {2 |6 k* g( q 然后上源代码: - 6 }! w" _( C+ [% k
- importlejos.robotics.navigation.DifferentialPilot;, ]7 ]: {" ?* _4 ~! [, h. G3 E }% c
- import lejos.util.Delay;: {2 N6 D* \5 N" T; H! [
- 1 \0 E& p/ x% J$ d
- /**
a! q. X+ J9 ]0 N - *Robot that stops if the buttons are presses before it completes its travel.
+ J* i, S1 |5 Y% Q - */' {& v( i" n2 J. \+ _3 s
* c7 c2 n2 P/ D% u! x
3 ?5 b4 U* i' J) k; N; D+ [- public class Straight {: ^1 I5 G7 e5 v, ^
- DifferentialPilotpilot;+ p9 s: q$ z# e+ N
- doubleleftD = 4.3, rightD = 4.3;- w3 Y2 j- {# c, }- R
( G) w, _/ ~% q8 i7 Y2 i, O. y- publicvoid adjust() {
B, `1 [! N5 O8 Z5 p! B Z. @ - //wait for release v% b1 j* x9 Y( y. x9 q+ F) H
- while(Button.readButtons() > 0)% r, `9 w% y' j/ }* y
- ;/ S2 Q+ L& n% T8 n d- ~" N9 n8 E
- while(Button.readButtons() == 0) {
2 L2 i1 e3 Q) m! t% K) ^2 K - //input size by adjusting the wheels# a+ V& o' f6 q# i6 F$ u! j+ a" h
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;* i4 E! m1 f: r4 h A* N
- leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值% @+ n& c9 h, b" c; B6 y
- 8 J( r1 {+ T1 C6 y! o
- //display the size input
, d+ e/ X7 k* Q7 c" C8 N - LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);
* X, j& F. F% F0 L2 q" C3 l - LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);
4 y1 ?# Z0 T) ^4 t! B& e! T, \9 d - Delay.msDelay(100);
/ F: X* `1 f0 t9 G* B - LCD.clear();, o, w; T9 C5 p7 y# L
- }
: }) D. ?- G; M3 l7 B* u; N
4 X1 {* ~! e# L! w) g5 [# J- }+ Y. _+ @# O Y: }0 X# m8 s
+ f5 T x# h" U8 U T( G- publicvoid go() {* z6 R4 a" q7 d8 f/ s. N# G: `
- //准备起步' x2 Z8 R0 \; s
- //wait for release# q9 j$ j8 D. J- E4 G- b8 F! y, [6 F
- while(Button.readButtons() > 0)
. R6 D8 q: _% D, G - ;
5 i4 f, Z3 q* W; d8 l2 ] - System.out.println("pressto go");
9 k7 F* S! A' s4 u9 k) X+ G) x0 q+ |" L
' x3 `5 x' C2 Y" u% F1 m; ]- n- Button.waitForAnyPress();
2 I5 L' k, h# E5 b1 d( q \- o - LCD.clear();
' p0 z, \; E; N8 ^5 h4 Q* b" l - ! ?; P+ c, x+ I, G" V
- //初始化+ x _3 j- Z5 {' M& y
- pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);& |" L% e8 j' F7 C, E7 Q: ]
- pilot.reset();
( [7 g1 q1 h. \' m
( I2 m% s# e" K/ G! l4 Y; e- Delay.msDelay(3000);
+ }3 {: l2 x; g& G h- g* x1 B
8 ^4 p4 P8 N3 m! g2 O- doublespeed = 10;$ F. O* q3 P! ?% u m- a% V
- doubledistance = 0;. ]/ L9 I6 D6 o6 g& C% O
* q* o1 W! L. E7 b ~- //起步加速; r- ~+ S3 N& @0 j5 ~
- pilot.travel(300,true);
1 n/ m4 l, C) _$ x: F5 y% E) j - . Z# p: a+ k) \" u* }4 n/ }
- while(distance < 20) {8 ]% g) c& _+ T2 B
- distance= pilot.getMovement().getDistanceTraveled();
1 s$ S9 q2 W4 ~/ ]; y/ e ^ - speed= distance * 5 + 10;
. N* \+ p% S3 [ |( e - pilot.setRotateSpeed(speed);
9 H8 Q1 [8 {% ~+ C# u - Delay.msDelay(100);
. _ }' Y- d+ z5 o - }
- _! T0 t1 J0 R3 l; U& v o/ n - for(int i = 0; i < 10; i++) {; ?2 c8 d/ ]. q8 P* d1 y8 a/ k
- speed+= 5;
6 V, b1 D9 C. k: I5 e! q% I9 Z' D - pilot.setRotateSpeed(speed);
: m* B) i" \; \. i3 w8 v) X - Delay.msDelay(200);
2 g$ P$ F' c S+ S2 k( y* l - }- P5 c ~8 M) p4 a6 p5 n0 ]3 H0 `
7 h& b# w% M3 P! P, |- //TODO 减速
$ a* g5 y; ]) [# h; l2 g - % j: |# |9 G- X" q! H! [2 |
- //停止机能
. z" {. V, T1 f - while(pilot.isMoving()) {
v* ?: e" g1 r. W* R) ^ - if(Button.readButtons() > 0)
' a9 U: O! G$ j! t3 d - pilot.stop();- I" O) R9 G0 N* B+ c3 d4 U
- }9 g" p! x$ j2 n5 z- g1 [
- $ [. Y+ P# ^! x
- System.out.println("" + pilot.getMovement().getDistanceTraveled());
5 @4 t9 y% k7 v, f - System.out.println(-Motor.A.getTachoCount());( B1 W- l) I1 N: a
- System.out.println(-Motor.B.getTachoCount());
: C' m. T& o9 a( }; H5 w - Button.waitForAnyPress();& Y3 a& t3 G/ d$ d! `9 ~/ d" C# r
- }
! n' ]/ h1 ]8 W: p+ G+ F* ^. z - 4 N9 w \9 r" M. S" q! l% g
- publicstatic void main(String[] args) {- _& p+ ^9 y7 I) e+ m8 p7 H# @
- Straighttraveler = new Straight();
: S% K% U- H( M$ K* x' q. S3 J - : j$ Q/ g6 @; l1 r/ t
- traveler.adjust();+ x% t: o( H w- A0 C7 V
- traveler.go();; x/ v2 u3 w0 ]0 ?3 t
- }
' s* J6 |. `' `6 E6 V9 E - }
复制代码
$ l3 ~. K6 d3 m9 s) Y. D |