本帖最后由 hello怪 于 2017-7-25 22:24 编辑 ! ?/ |. T% o- K3 w& p% k
9 c$ e" L6 {! k5 q
(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html , u: A% s8 O# F! Y: _6 j
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面: 6 O8 R \- N% K! o3 F
场地6 }( t, O" `- H, v1 |7 |& A/ \
水平度和光滑度 在家里当然瓷砖的效果是最好的$ O7 g) `% ^. a: b' z" ]
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直4 u# i W1 F/ I9 P
/ Z3 U, K4 ^% g- h6 ~* I
车子结构4 s d# {4 D' \/ r
轮子间距 _# R6 H* P) ? _$ R
轮子间距越大越好,稍微想一下便知
7 w0 \; W6 ]% }; m5 q4 z* l% Z7 R 轮子大小) p; f. w1 c0 U" j
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)8 o' ~4 [% P: I
随动轮的设计
4 j; r# m1 ]- @ X8 b 随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响( F6 j9 _* C( k5 U6 d) W' R
重心位置. ^' M0 Q8 V! v( j: {7 R
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小. e( F; w5 A0 _9 ^) M
结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大
! U Q* o& F4 ?/ L ?! @; A* J$ Z 轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)
) h3 e$ f) U9 h' f3 S1 r 驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动) i4 Z9 |7 P) b/ f( @* Z# q
2 I0 J) b1 E! v
控制程序! ^! F; W0 D& d; r5 v; m9 H' L n
运动过程控制
; j+ `# W, n, C* c6 Z* W. ?4 r0 w8 e+ ?1 w 好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)
\4 C7 g9 S# E& Z( G9 I3 C- V2 z- i+ L7 `" Z5 Y
调试与运行
* ]7 k7 j L# V4 T 轮子转速调整(车轮大小的设定)
8 [, y5 \- W& h. Z 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法9 |. n6 V c$ }4 k0 s
起始位置车子正方向和直线间的夹角
* J$ @; W% `) N& C# R3 b 一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
. X' m9 ?% g% ?8 {9 e4 l! y
4 p" J6 f4 _& F0 p& w2 V* e
4 M, n0 \- w. S- X2 n% Z" q 然后上源代码:
" r& C# C1 l# I' l* s- importlejos.robotics.navigation.DifferentialPilot;
) ~& O- O2 b: H+ w3 y/ i - import lejos.util.Delay;
: S" T; f- ^7 Y- u/ }2 @3 L - ' i7 W7 S* n5 R+ X8 s6 o
- /**
2 R4 t& r+ y* G - *Robot that stops if the buttons are presses before it completes its travel.
O! B7 d1 ~0 n8 E- D' E) u1 f) T - */
; ?3 }) z9 Y) ]& A% Y( \0 }
7 C7 ~$ f6 e1 } R8 x7 ~4 L5 Y
^" H/ e: Y/ K; o: ]* f5 w- public class Straight {4 V; c1 b& K/ n
- DifferentialPilotpilot;
2 w( K+ W; @/ y# N5 W2 O# H/ O( c - doubleleftD = 4.3, rightD = 4.3;& s$ M/ T& c# j, n
, P7 Y3 M. n5 |. D4 F' U+ C' h4 i- publicvoid adjust() {
( Q7 O% n. b; _: l - //wait for release
7 d& t3 g3 I( m, _& {& K9 B8 n - while(Button.readButtons() > 0)
6 ]/ y$ l3 Y6 r9 e( C: U - ;
# D C* k5 C/ P2 ~( u6 J: [/ ] - while(Button.readButtons() == 0) {5 q' y8 f; F# k+ B3 P) J
- //input size by adjusting the wheels
2 B/ ^+ h F3 Y, F. ? W - rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
; }/ k% d# J- {& S5 E6 r - leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值9 x( h1 f, @# I' g7 V7 w3 L
- # p' s+ Q7 i: q- K
- //display the size input T' [; a5 F9 R5 X; @- h* W: l: C
- LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);
4 H; j* u3 D$ K. a - LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);: h6 i9 ]1 F4 n3 i0 E! Y
- Delay.msDelay(100);
1 W* z' c- z" L7 k' P' T& ? - LCD.clear();# U5 c9 X+ W& t) Y4 h
- }
6 I2 O& w2 G; {. O9 v; M& x+ Y - 4 S6 Z9 l8 C- f$ W, r" b* x
- }9 {, q9 s- A( N3 a" U
- 1 V' T) o: V# u
- publicvoid go() {
5 o2 b! n& ?+ v* \ - //准备起步
! Q/ [: `* m5 e6 s, E, q - //wait for release6 a" a' A8 k- I5 I, r, Z
- while(Button.readButtons() > 0)
5 Y7 d. w! b+ N2 u! S, C- Q7 O - ;3 b) s% N/ @5 d6 E; @) Y9 B3 z2 M4 B
- System.out.println("pressto go");
- E. M4 V+ Z4 i" ^ I3 q$ G - 2 y* A. G4 R! w9 `; s9 V5 j/ J( @
- Button.waitForAnyPress();
5 {7 E% X/ w* @ - LCD.clear();- ?5 P3 ]1 u! \2 v: o
- ; C+ v+ V0 B t6 P0 q
- //初始化
3 n: Y$ C( u" Y4 B6 I* e8 q - pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);
% V! \3 n* s2 b6 I U6 d - pilot.reset();* v% O. |. f# n. W/ ?9 L9 U( t
- , Q- S8 O: @) j* M0 l
- Delay.msDelay(3000);( s; T. {- p1 j) R5 N3 H. z- o$ o
- ! p3 ?/ ^ ?, o7 e
- doublespeed = 10;
( a8 v6 b t( Y4 u7 U2 a - doubledistance = 0;
9 [( e' z* l* [8 k+ P# ^* ^/ i - & Y* }) L9 f( H( S8 v+ p) J0 }9 U
- //起步加速
' M' x2 u2 t+ g( S2 g7 i4 s) d3 e - pilot.travel(300,true);
! p, c( [9 E& B7 n - - Y/ f2 u, k0 m j
- while(distance < 20) {
# W8 s2 I# `0 m$ d4 x) H7 |1 ? - distance= pilot.getMovement().getDistanceTraveled();* Y0 w* j( {6 G o& t
- speed= distance * 5 + 10;9 @6 N6 E( N+ |: d
- pilot.setRotateSpeed(speed);* f2 t# r* D2 R% y8 u G
- Delay.msDelay(100);4 }- \" j8 j8 A6 t3 G! k
- }$ b8 N1 U' f) d1 E% ^. b' t1 h3 p- N
- for(int i = 0; i < 10; i++) {
9 \7 Q, P& k; h) A9 u9 r9 g) L4 S - speed+= 5;8 x" D8 @5 b- t
- pilot.setRotateSpeed(speed);
0 |; C+ _1 u, ?' k. \% W. L( ^ - Delay.msDelay(200);
5 w5 C9 C; g6 e4 v. A G - }- D! L( E K# b" n8 j
l* ?" S9 T% Q2 {! g& j- //TODO 减速
* @5 z& G7 J3 M* W; F& I: j ]' M - ! y# Z7 ^: r/ s3 O: k
- //停止机能
8 E; Y# o$ V0 z- v+ B) F4 I - while(pilot.isMoving()) {5 J5 s9 a. o) W7 Q( o) y* f7 D
- if(Button.readButtons() > 0)1 O( O& D& f+ @& |) T$ H9 k. S
- pilot.stop();
3 Y) `1 w. {! ^" N& Y - }: D# |) G! M& z4 w6 c g
- 0 o+ ]# Z- R: v
- System.out.println("" + pilot.getMovement().getDistanceTraveled());
# C* b) c/ Z0 f - System.out.println(-Motor.A.getTachoCount());
. w( X. r/ M( r5 }/ u6 D - System.out.println(-Motor.B.getTachoCount());( c, r1 w: C+ l; F6 A& _
- Button.waitForAnyPress();3 w2 y1 R1 X/ y2 b2 r5 D) z
- }
; d9 @. U; b1 B6 u* G( a# Z
: ^0 B q8 w; O, {# N# j; R9 x- publicstatic void main(String[] args) {
. y/ P& d: ?' |$ \ - Straighttraveler = new Straight();
$ @# H' _& j5 v: F& b' q; { - 5 s3 D2 f4 m* l& |% ]
- traveler.adjust();
4 A+ B" M9 j4 V. h# ~; Q - traveler.go();/ q/ a; g' F, Z6 k# e
- }
$ @# q7 G3 z# m* L- A; Q - }
复制代码 6 ?( z2 D3 h( }, C4 C
|