本帖最后由 hello怪 于 2017-7-25 22:24 编辑 8 T: F- x/ R" Q' P
0 J3 X, z: [; R1 \6 R( J* Q(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html / D* `( f; ^5 T# Y ?1 R
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
; \9 }1 e; [# E' Y7 t3 | ~" Q场地- O1 X$ E$ w. S
水平度和光滑度
在家里当然瓷砖的效果是最好的0 ~8 k/ r0 m' B2 c, ~; |5 e- A+ [
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直8 B& o5 w4 ?) d7 l! h
( |2 d4 c& p. _$ h1 t+ [. F车子结构3 s0 B- r6 t6 |+ ?1 \$ [6 T$ H
轮子间距& D) Q" |: ]6 W3 C4 t
轮子间距越大越好,稍微想一下便知
- T3 N0 {% h: N0 R, K 轮子大小: k, M1 F# ?, E7 F+ E# W
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子); u1 T1 M' T- [$ O/ p
随动轮的设计* r1 n. C! N# {. l
随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响
! K* ^( S3 j- M; R' |4 z/ T4 \ 重心位置$ Q9 v" \! z: c
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小
% @. y% |0 k! J1 x" ]+ g2 V. Q 结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大
- w, ?2 k" U$ O6 z' j 轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)9 ]* c+ B# U8 n- `0 b, f
驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动
. h# Z+ ]; j$ P8 f9 u7 L( F$ D, t
" s7 P; _: \. c) X: R控制程序
3 k+ {, v# j/ M' N) Y2 m 运动过程控制
. y. F( y' k7 |1 n 好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)9 O" ^, q; I0 d1 [6 c" }4 d8 u, Q# E
4 d8 X; {4 v+ G
调试与运行
8 w; c# m- [! K' A3 E1 G 轮子转速调整(车轮大小的设定), U" N( Z ?# K9 k }/ F
这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法- U4 w0 t3 Q+ }) Y7 L4 A. x
起始位置车子正方向和直线间的夹角
. t2 V. l% Z' d" B: g; S' S 一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点 9 e9 x6 R2 W3 Z; Y# {- i
5 U. @" [, S( @/ ~1 F+ l. x, _
$ W a! Y5 D3 O8 M0 T 然后上源代码:
; X- _# @, l0 ~7 d5 x4 g! P- importlejos.robotics.navigation.DifferentialPilot;( d" w0 o5 ], T
- import lejos.util.Delay;$ b6 k8 U% C9 a3 {" _$ O# ~7 _" n
- & ~4 c# G4 d' u! [, y" m/ A
- /**3 [* b! x1 W& {; o) H# a8 M
- *Robot that stops if the buttons are presses before it completes its travel.
6 V: M8 D7 y$ Q - */! m+ @# _: r) m/ l* z
- # \8 M/ J# ]; L) j
( v6 Z/ h* L& @& ~! K5 |* a2 h- public class Straight {" Y* V. l R6 k# E, L
- DifferentialPilotpilot;2 {/ H* p- _+ ] t" a
- doubleleftD = 4.3, rightD = 4.3;
! I. }0 P, q4 t - / B7 v1 ]! k# U7 L: h. Z( o1 z0 i
- publicvoid adjust() {
! a% M, p+ y# @3 T& d% |6 d - //wait for release
+ H. T x2 }( e" n - while(Button.readButtons() > 0)1 j8 d# f, |& i0 a
- ;8 E. p* [/ L. f* e" m" U# B
- while(Button.readButtons() == 0) {
. O1 r5 R7 r/ m3 q$ D- f - //input size by adjusting the wheels
1 M- B& C: t6 h' ~ - rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
7 r4 D' F2 O; A: K* [7 x - leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值
/ a1 y: s# I( N6 {) s3 h - + A7 c6 h* g' O6 @
- //display the size input
$ L$ s0 G; H l1 i - LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);! q4 H" I' j% O: n+ Q
- LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);
2 S! b* R# [$ t3 y. P1 q% B: P - Delay.msDelay(100);
/ `; h( o2 W- z% W. G - LCD.clear();
! k7 E; I6 x- ^ K - }. c" N8 h* B9 Y J
- - n* m, I7 v/ }$ R
- }
1 l' h5 n0 O, A; E
1 c% I+ r6 ^% T5 S$ O2 J4 I! C- publicvoid go() {$ q8 K: K+ |* a9 V
- //准备起步
6 \' R9 G0 F8 U; R7 C - //wait for release
% @: d) @# A3 e - while(Button.readButtons() > 0)
. E# K5 n! a$ O6 X - ;' N K" V {" w. L
- System.out.println("pressto go");
' Q9 Q/ e' h9 S0 {* _1 z4 ` - 4 v# } W# x3 N: b( ]
- Button.waitForAnyPress();
. Y, s9 X' A; e1 v3 r - LCD.clear();
6 U6 S& ~: A% K: _ - 7 g1 g' W/ g$ l
- //初始化 o! m, u3 s/ M8 O6 K! n0 f1 I8 i4 b
- pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);: i8 t5 Q5 }* F- _, A/ j7 z* f: A1 p5 f
- pilot.reset();
) O7 D) W2 d4 B/ _* P
) G8 j/ G5 |- i3 u4 w- Delay.msDelay(3000);
! w4 p" L; \5 x - : l2 I ^) f5 f4 `- N
- doublespeed = 10;0 y9 A: G) X2 |. \+ l
- doubledistance = 0;& O! n: J0 |2 l! G- N4 @! q' ^
! t* S }' n- l4 c- //起步加速& Z. C( Y4 X/ V( }
- pilot.travel(300,true);
# M: t' U- e \2 e3 I* \* Y/ a
" m6 W0 w4 r% d- _- while(distance < 20) {1 A' u0 m6 z" i! w8 ?/ C* Y) f3 g* d8 n
- distance= pilot.getMovement().getDistanceTraveled();; `5 C/ } v+ G3 p* K6 y* G
- speed= distance * 5 + 10;
+ N# \2 Z! ~1 G1 G - pilot.setRotateSpeed(speed);
. G* \' x, T8 t7 P* V) z! R - Delay.msDelay(100);/ [$ z z( X: M5 p; i
- }
9 t) K- T! D$ z. h - for(int i = 0; i < 10; i++) {
. T% `0 }8 d& F1 [* G - speed+= 5;9 |( a, k! |8 v: J! H# Z5 J
- pilot.setRotateSpeed(speed);
/ Q8 }2 I* `- l# O - Delay.msDelay(200);* p o* ?8 d/ _6 p* ?+ l$ j6 `$ a1 h
- }' _6 w7 O) f, @. Z- c* w
: ?/ |# |6 Q# X( n- //TODO 减速 L) v3 M" ]) @+ ]
- & t2 y. M2 Z3 |; f& e0 M# E
- //停止机能
% ^+ v5 F/ [! |, J - while(pilot.isMoving()) {/ t/ m! S. y0 c, w6 o5 C
- if(Button.readButtons() > 0)
# Z4 r' ?7 f, T( h! \. R, c - pilot.stop();
$ l! P. T( r- i - }
9 l' ]1 c, Y: G9 E
7 Q2 r) b E+ i, ]3 {- System.out.println("" + pilot.getMovement().getDistanceTraveled());
! U/ G! J3 u& g) z: X/ F% ? - System.out.println(-Motor.A.getTachoCount());& j5 x/ _* ]% I. l$ }, y3 H' G) |. p
- System.out.println(-Motor.B.getTachoCount());
3 D9 Q j$ ?/ l* t8 g+ f/ q, e, S - Button.waitForAnyPress();
5 M2 E8 _ B6 h - }
1 i& G5 N$ p8 H' J" y
3 o7 f s, E# U+ e: [5 }, o# [- publicstatic void main(String[] args) {
4 g3 j. z% `- i5 e7 Z& l6 a - Straighttraveler = new Straight();# P( a, j6 Y3 g4 v5 [
- 8 E6 p# [" C4 R- B5 B7 d7 l
- traveler.adjust();
! r$ H2 r# X9 W/ g$ k - traveler.go();
* P& J- _2 [2 V! B/ }8 a - }
% [% \ @- x9 l - }
复制代码 ; M3 b* @, R) a
|