本帖最后由 hello怪 于 2017-7-25 22:24 编辑 s5 ~7 V" @0 M( d0 K
: I( j* B9 d! |, A; K
(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html ( ]( C9 V. Y4 b7 r
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
, Y+ n, I0 _: |8 F% n* o场地0 T" u+ p! m8 v; _% q. T
水平度和光滑度
在家里当然瓷砖的效果是最好的
# R0 x; m+ K$ J4 n* a 线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直+ `& L7 q F4 {: Q/ n# f/ V( ~
* I' ]* _/ J; U8 p' ]+ B车子结构1 Q; u0 Y+ k, Q$ G. H1 ~# ] _; @# d
轮子间距4 \* z) X* R3 V' y% h* o
轮子间距越大越好,稍微想一下便知6 e( r" a& h9 W8 h- @
轮子大小! `' D, c& _; R; Z4 @8 J: k$ a- D
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)! c3 M3 y1 J5 \* |1 p
随动轮的设计
# E/ Q# O+ q; R6 h* S' l8 O 随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响; d) u0 ~2 x# u
重心位置& i0 g5 V. E' X+ L- @4 k0 I9 }
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小
- T: N. s. g* R2 P+ L 结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大# H: t# }5 Q- Y
轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)
% ]. x! e2 K( o2 L 驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动4 W5 X3 m) {; w/ O9 L( Y
/ l1 j$ r# b5 b! Q6 g1 h
控制程序
3 ^/ e9 j8 c4 T6 _) f% A: s! S 运动过程控制$ [) F3 f1 v& T7 @$ d# F& a, B7 P
好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)
+ i/ w, d; o( X' L/ _8 H! C; y4 g+ {
7 u& v2 H% c9 q5 X" V调试与运行, E; H# s2 P8 Y6 \5 w f
轮子转速调整(车轮大小的设定)7 V* M$ Z. p J( ]
这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法
8 H2 @1 }* o& {( J. P/ F 起始位置车子正方向和直线间的夹角% N! ^2 `$ |" F0 }4 q
一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
+ G6 j( u- ]( ]+ F8 l$ i9 k8 J: y8 Z+ w8 z
" ]7 M6 {. o& {* X 然后上源代码: - 1 X) I: @$ z/ r$ Z6 W3 Z. X0 j
- importlejos.robotics.navigation.DifferentialPilot;
- S0 B/ c; Y; L - import lejos.util.Delay;
( ~! a1 y/ n- l+ s# Z0 ^; x
# h3 {9 r# y, U% l* _+ V g4 b e- /**% E& {+ V( x, @1 N- ~& x
- *Robot that stops if the buttons are presses before it completes its travel., V) n& E7 n2 Y, n% F6 F# s
- */
6 G( }4 } c9 Y5 I1 h1 ~: z - # K! f2 f* z& d' }6 T" X. _- _" o
- 1 Y" m9 S* b5 `
- public class Straight {; r# o/ Q3 e7 s0 H( N
- DifferentialPilotpilot;8 x& x( o# a: I% t/ r, D" t
- doubleleftD = 4.3, rightD = 4.3;
8 o$ D7 `; i$ {
# i9 ^2 D# f6 o- publicvoid adjust() {
& }" w, j/ K) M$ G' p3 m - //wait for release
) l- c' M) N9 C1 B. \ - while(Button.readButtons() > 0)& ~0 j+ f9 {3 t
- ;
& |( J9 K, L, j' s1 }* _ - while(Button.readButtons() == 0) {* u6 n5 ~* O7 Z3 d' g! @
- //input size by adjusting the wheels5 j1 }4 b# q$ n
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
$ H3 i8 I) O3 ~4 u - leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值) ~+ _8 t7 K% U
- 9 f& H# J7 y8 ^9 d
- //display the size input
. R5 h! E' C% U. H) J - LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);
5 R# ^$ O3 c0 D! f6 T; i* s - LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);2 M5 g4 {5 L0 I9 E1 w
- Delay.msDelay(100);% f4 Z9 [% `# f+ v% U2 L; T
- LCD.clear();3 O. S4 |; H( W
- }
' @( V* N! G' ~* N* F8 h! ^" T - & i) [4 n& e7 M I# I+ f2 c5 ~
- }
$ T! M' ~9 U# T T- M - 8 m3 T9 E( L. t8 Q
- publicvoid go() {4 w# m- q4 y- E* x0 b1 ~
- //准备起步
) ~$ T2 {2 b$ i) M, ?) `4 j - //wait for release1 L! ?7 }* ?$ u4 Q
- while(Button.readButtons() > 0), D$ r: }; y' p
- ;0 R M8 g% @/ k) _
- System.out.println("pressto go");
9 c- g! a* L2 B2 J/ N: A0 M+ n
. R( B* O! n, H! p8 Y$ f- Button.waitForAnyPress();- K! ]( P" x- e% z, P$ b" @7 ^
- LCD.clear();& g# c& }( ]# B( l6 h$ F8 f
- # Q/ b" H" T2 Q0 C$ H6 D
- //初始化
! y, _# Q- Z7 |+ K% ]: \# U - pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);
) w$ [* m2 G# u - pilot.reset();
. y* ^! U9 O; L& J2 |5 Q/ ?% Y
3 o* g! u& o1 N$ Q |; f0 T" e- Delay.msDelay(3000);
. Q7 G z/ j0 J4 ?: O0 P5 z/ h8 E- w: i - 8 F9 R" @, S8 g( t7 P
- doublespeed = 10;) J2 m$ Q0 d2 u5 `& V5 z
- doubledistance = 0;9 E- y. V; Z; H6 X8 K
- + j: d& v7 H+ g3 P3 v' S# _
- //起步加速4 n$ M' L! D7 x
- pilot.travel(300,true);" u( _$ M4 I D) g D: b
) r1 i1 Y. M' I2 C* y4 \/ q- while(distance < 20) {7 |" r- |% m# D9 |+ n. A4 B
- distance= pilot.getMovement().getDistanceTraveled();
) }& P0 @/ \4 }3 D5 W- Q - speed= distance * 5 + 10;
, F- P5 K0 F! V - pilot.setRotateSpeed(speed);; H/ Y0 a `1 {
- Delay.msDelay(100);% s, i; O1 X# m! c# M8 g
- }
' W# z0 A* J5 E. `# ?8 \+ P. T - for(int i = 0; i < 10; i++) {
- K4 y1 p: Q( F& M) [ - speed+= 5;
" Z% N( ~6 P* N9 ?: _ - pilot.setRotateSpeed(speed);
* F- h" u- X; M- I, p: X - Delay.msDelay(200);
! b# v/ g w" H& g/ _% n% _9 \- p - }
# R, A: ]# p2 {+ W; x/ T
: d0 D1 a* `- b6 `- //TODO 减速
# T/ \( }$ m3 V+ }( f; T - 7 C* E) V" Q1 V+ L- X( r% h* N
- //停止机能: m, y$ ~4 f0 z
- while(pilot.isMoving()) {
. C5 }* H9 `& K+ K - if(Button.readButtons() > 0)
- _, p+ l+ ~! i2 O - pilot.stop();7 x4 V: _$ U( q- Y' \0 K
- }" B: T d* X2 C1 `
- F7 y+ |$ Q8 K# ~/ v) w% W# d! D
- System.out.println("" + pilot.getMovement().getDistanceTraveled());
% [: A% ~3 f3 m- J# s - System.out.println(-Motor.A.getTachoCount()); j& P0 R* [% I4 G3 c; ?
- System.out.println(-Motor.B.getTachoCount());
! n0 P5 X' N8 h5 \4 i - Button.waitForAnyPress();
8 F6 f" O; i J: E - }
' t8 i/ s E4 v! E
. x6 f. Y& v: [- l; {- publicstatic void main(String[] args) {4 d* w/ B" P+ [$ A
- Straighttraveler = new Straight();- x6 [! R! r+ Z( U6 k
/ r( G. I$ o# @6 e/ ~- traveler.adjust();
# g2 @2 V( Q/ h/ z - traveler.go();8 L5 P7 Y6 g# v. p
- }
3 z# s1 J: x" A& B! ~5 n+ B - }
复制代码
* ~$ Y# B8 G. q& t a/ V4 Z |