本帖最后由 hello怪 于 2017-7-25 22:24 编辑 3 t1 r7 W! c$ T% ]! p+ g
' s7 Q! ]8 e7 ^8 c+ }
(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html
5 p; I- [8 ?( Y1 h$ x* T k
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
1 n* }/ h# Q7 _8 Y4 x- q; f5 z7 z场地
- j& ~' v$ y8 q$ K8 ^ 水平度和光滑度
在家里当然瓷砖的效果是最好的
# m5 {2 D7 g! C' G& C9 U 线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直% w* i0 o1 U: v" W
) f0 }0 ^' B) D- O车子结构
3 x, A7 ~* A! x' ]! y 轮子间距6 z; k' e! U$ v2 x I3 [4 k1 N4 l* V
轮子间距越大越好,稍微想一下便知
& f' Q( y1 L, X0 c 轮子大小
6 v5 N- r7 d, q) ^- u 轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)$ K6 h+ o& j# V$ Q- I% S3 n
随动轮的设计0 s& h9 O' }. v2 V1 g9 W0 e/ S, i
随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响: ^, B8 ^( s$ f6 ^' {
重心位置
$ ?/ f; ^* x+ [ 重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小% E6 {5 O( b! y! X! X
结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大
5 ?' c+ I9 G4 e8 } 轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)
- B1 R9 L4 O: p5 Y" G 驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动2 M, h' H v4 |( q! [
; N4 B0 E: y# V) D
控制程序
9 I- h* h) x* Z3 o* ] 运动过程控制
# U/ ~$ f6 H2 `" L+ M8 H+ v 好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)
2 Y/ K3 ~ r8 O* Z0 D+ s/ r0 v" ?5 b0 Q$ c: A/ ~8 C- T( F
调试与运行
. I% P* D8 G; m( z 轮子转速调整(车轮大小的设定)
) u( ~0 o& b" }9 m/ y 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法
7 s7 q6 U/ F, K A1 B+ R1 p B 起始位置车子正方向和直线间的夹角
7 j& k& \7 X# k( y" k 一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
3 M3 R3 X) m9 N O+ }
) r; _' Q8 w& H* N0 `. p1 v; E& M* S+ z! ^9 L6 {1 S4 Y
然后上源代码: - " v5 w, a) K2 _# Y+ z. \
- importlejos.robotics.navigation.DifferentialPilot;* Y! t& P0 Y; X3 H% q, z
- import lejos.util.Delay;5 ^% g. e& G0 N& \ \ K4 s
- - r0 Z% z" e3 V, V+ u% u
- /**
& I) c' I7 R" N) L4 l - *Robot that stops if the buttons are presses before it completes its travel.
5 S* v; V& D6 }/ t6 { - */
0 B) |) H9 r/ j: y, m3 R) ?
( J& d: t$ q+ R' z
m3 L! d9 u! V0 f- public class Straight {
. _1 j0 g h# U - DifferentialPilotpilot;
6 U/ d, C+ ~+ Q- I - doubleleftD = 4.3, rightD = 4.3;: R8 i% j8 V( ~! y8 @* e
- ) ]8 ]% P. R* f- |% F1 D; e2 Y
- publicvoid adjust() {! W, y& ~2 f _1 f% k; h# t
- //wait for release2 J4 o- Y+ i4 a- X8 h( ~
- while(Button.readButtons() > 0). `, E, T& K! b- h X
- ;) M$ E( h0 \+ Z1 R4 ^
- while(Button.readButtons() == 0) {; ?4 w( m7 o% `( U3 u* H/ u
- //input size by adjusting the wheels
$ b- m8 b1 X# T' t# S" n - rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3; @( }$ B: d: O8 n7 ^( M
- leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值: Y; V3 C' a& ]. r' W& N# C
8 W9 e) {. u# y3 J' {- //display the size input" i" x8 M9 _1 N
- LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);
4 T# r1 J8 l1 k. D0 r, w - LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);" a/ K* M2 i" r6 z
- Delay.msDelay(100);: A# h% A& h }
- LCD.clear();5 E, p G8 @6 R% w
- }- y% {8 c& y% [$ {" p( c5 j! Q
; V9 @# a* F4 a: C( V* g' q- }! P3 v, ^8 Y2 G9 F; D8 r* \
$ Y" ^% \/ \) w% h0 x% v" J- publicvoid go() {% `; C& f% C: G# [
- //准备起步+ [: U; c- U+ m# Z6 c( u
- //wait for release
5 H' h1 ~# i6 q$ O5 S. f - while(Button.readButtons() > 0)
; [( P8 _' N# N5 D& h( W! y - ;
: E8 {% G0 I- B' H4 w - System.out.println("pressto go");( R0 x" I! Z( V3 X3 N& a" H3 R# ]
: V1 e1 L; `& y! ]# p! u% y- Button.waitForAnyPress();2 X2 ?/ S4 J' w' k
- LCD.clear();: k& ^; o2 ^! ?8 v Z4 l
& r3 r4 M0 k; l1 Y/ M- //初始化
9 A- Z% y; a) T& ] - pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);2 @; H8 d8 r6 y" z
- pilot.reset();: }( _% ]+ B' s6 X
- 1 n4 W8 s8 M' ?7 l% D) {
- Delay.msDelay(3000);
3 H6 M/ L/ l% C - ; z2 M" `& X' E
- doublespeed = 10;( y J6 s) o. i, H: y
- doubledistance = 0;6 \* F6 m. k1 T5 s# J
- 2 F! H/ P: \& Y2 E+ @. b* H f
- //起步加速
4 _% a) q0 k+ Y0 R5 x' O8 Q - pilot.travel(300,true);5 |' W8 l8 a, v) h5 U# f5 C
- 2 _; h/ q6 }$ E2 e
- while(distance < 20) {* l0 m2 b3 G: o, J3 {5 y3 K% c
- distance= pilot.getMovement().getDistanceTraveled();
7 |7 s$ U9 G0 I% E# F# l1 Z - speed= distance * 5 + 10;
P9 l$ ?+ C2 r2 A0 D. N" } - pilot.setRotateSpeed(speed);
! f/ m3 a5 q. f8 H5 }5 ] - Delay.msDelay(100);
- y0 w' x0 w$ o/ x - }
, T* h3 P4 g8 K/ O - for(int i = 0; i < 10; i++) {
1 ~$ u$ @& m6 z: @; W& [, m - speed+= 5;
' A) X* S$ W; B - pilot.setRotateSpeed(speed);8 [8 ]/ e0 x3 r; ]8 N
- Delay.msDelay(200);
2 y, Q/ _6 v' L2 T - }! x9 t" _& |; E5 v0 R
- 0 z8 O; H+ Z$ o
- //TODO 减速
. Z5 ^+ X, t/ a6 }
8 | _# c2 }$ W. j* Z: i0 L- //停止机能
% h' {$ v4 X/ h4 O! h1 a) w+ c - while(pilot.isMoving()) {7 r4 J1 Q6 `& ^# ^. F2 o
- if(Button.readButtons() > 0)
, m, {6 M7 `( h% [7 V, S2 u - pilot.stop();
6 a$ x3 \/ \, F- X- s2 a - }2 n! ]. h4 S, f
1 B: g' M! a7 @3 j9 Q- System.out.println("" + pilot.getMovement().getDistanceTraveled());
' e% X Y& o, H1 C - System.out.println(-Motor.A.getTachoCount());
$ w, P+ N# L+ b# k, @9 A9 ]7 X( S - System.out.println(-Motor.B.getTachoCount());- t+ s- z }* A+ C1 U
- Button.waitForAnyPress();4 O1 Q: e8 v' D) [8 b2 i4 U6 N
- }4 B! W2 j6 D' K6 c) [3 q, a
- % G" ] }( g- @& v/ @! s
- publicstatic void main(String[] args) {& E) u: v5 ^/ ^4 k8 y1 @
- Straighttraveler = new Straight();0 u: ]6 F$ O% |8 H/ u' F9 u0 j9 q
{" S& |3 F% C7 ^2 Z' W- traveler.adjust();% {, C, y/ G5 U
- traveler.go();8 p0 c% ^; m7 P3 a
- }7 ?+ s% y& x e T( R) Z+ G: j' L
- }
复制代码
0 R% U3 x1 D8 E |