本帖最后由 hello怪 于 2017-7-25 22:24 编辑
: | ?# D: S$ S V6 W2 B8 f) z, f- u1 ]' j0 W0 u
(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html
# I% A: j3 | O2 y) U% J2 C" t
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
9 K2 x( @ a8 j+ `. n; V. ]9 z% g场地
, b! W6 E* q5 Z/ W" t2 r8 F1 v 水平度和光滑度
在家里当然瓷砖的效果是最好的( c2 g' A+ H$ P, l
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直& p% c/ w3 n; z5 f
3 v' E5 ?3 a! r W7 ^
车子结构0 V: _# f; ~1 `, |9 o# x
轮子间距" U( h( |7 a$ Y. o! @* T! }0 {
轮子间距越大越好,稍微想一下便知- _# D4 |4 j! j# j' D
轮子大小- Z9 S/ _ \$ t2 k
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)4 V' M& D/ G0 Q4 x
随动轮的设计- M* f* p; P( p& j g e
随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响
4 f: K, P# i( Z 重心位置9 H7 x+ c- t; C+ K% l! m7 g: w
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小
; A) e) I2 l/ l2 V k7 E' A 结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大
& M; K0 R6 N7 n; n 轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。): i) V( B& e2 I9 `# n2 n: P
驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动 V# G6 v3 R# z- B7 Z) [8 l6 F
& f) I+ p* ?3 q/ p% \! _
控制程序
% w, U% @5 P1 A 运动过程控制# S+ L: o8 f# Y) c+ o* z) x2 H* e9 [
好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)
& J4 G8 z7 Z3 H ~4 S
8 M( C/ [4 c+ z; `2 p4 v! k. }调试与运行- y# K5 p9 i1 Z5 [2 J
轮子转速调整(车轮大小的设定)
& O; M# w) t5 s# m9 c/ W 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法1 Z/ g6 d( O( S% F7 d7 T F
起始位置车子正方向和直线间的夹角" F2 d! F' k3 [" K5 T" w0 c
一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
6 n4 b: o# X' n; h& V6 _! f, e6 J* C7 ]' `
6 w) |) s' S: h: d7 C0 R! V 然后上源代码: - : N/ _/ M( \ M8 y, S' m
- importlejos.robotics.navigation.DifferentialPilot;! [9 J9 y0 W5 R% x/ V
- import lejos.util.Delay;5 g( a. O& A- C5 Z8 y
- # q6 A0 v9 x" M) n1 V* g2 s
- /**
$ z, R2 P/ b- u6 f/ y5 \/ z) L+ h - *Robot that stops if the buttons are presses before it completes its travel.' O& h! S$ l- @) W0 n8 g: ~8 h
- */, H" a. @- n' y9 Y
- 5 D, B7 o _$ F0 t) _) o) f" T
- : b1 k; f: a1 v2 c( v
- public class Straight {4 Q5 W0 X- {! B' U
- DifferentialPilotpilot;
' u$ t3 z! u3 ?: E( _, @( V! B - doubleleftD = 4.3, rightD = 4.3;5 e: u" o% @* P8 v. C7 Z
- ) K N, F1 W, U: K; q! w9 w6 F. I
- publicvoid adjust() {
% Q. N8 W/ K0 a - //wait for release. A/ i) ^* w1 }% N5 l* G
- while(Button.readButtons() > 0)& i4 m7 a1 {+ Y
- ;
/ {8 f8 b9 N2 y" Y& t0 N( t - while(Button.readButtons() == 0) {
3 Q7 c6 [* u$ f - //input size by adjusting the wheels- U3 ]) }1 i$ I0 Z, Y
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
4 [& c; W, f3 |3 v9 j - leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值
- }& \0 d | X$ J" G5 j
; T: ?; T# ^3 z5 q# W- A4 H- //display the size input1 y6 p% c- F% E( J
- LCD.drawString("L:"+ String.valueOf(leftD), 0, 0); R2 i6 O4 D, @4 s% h
- LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);
% l" q7 n4 o: f - Delay.msDelay(100);8 ]: C! z, g1 ~6 a0 r5 I& O, s8 z
- LCD.clear();8 c3 G* e* J$ c2 A R
- }
% K( l5 a$ `9 i) [6 |
5 q) F$ ` g0 ]- n- }2 c% H) p3 W- F- _1 m
+ A7 G+ I p5 T2 q. h- publicvoid go() {
, r8 Z( ^$ o4 f$ `$ U - //准备起步
2 ?& ]6 W1 B: } - //wait for release5 g% z1 P! k6 y1 Z2 j4 c
- while(Button.readButtons() > 0)* T& P `$ ?9 M/ K: T5 j
- ;! H. C- ]: T3 h, t' n
- System.out.println("pressto go");
' ?6 j7 m1 z/ Y+ K5 y
! I/ s$ a& V( [: k8 \/ i7 j- Button.waitForAnyPress();$ h# K, A5 P& X5 N9 H
- LCD.clear();
8 L& |0 L5 ?& s2 }4 V/ k - % Q! U) E* J1 G, O# d" S
- //初始化
3 k- k2 c9 M# t4 H - pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);
: f% W: Y3 o) l" K$ q - pilot.reset();& ~- Z2 {/ _0 X+ B8 M
- 3 h! g, K- L& S
- Delay.msDelay(3000);
1 G7 n0 Q3 \2 ` A) D8 B) U8 O
3 N, a% }% P1 t9 @) X- @- doublespeed = 10;% b6 R3 v6 f- _, ~) ~! [
- doubledistance = 0;* ^5 h: `) G+ |- K
) M* m3 g# u; v: U4 w* e' G- //起步加速5 _8 m3 k$ S9 V3 T* M1 z( o
- pilot.travel(300,true);5 b, a, |2 p! I6 H6 n
5 B: n2 O- C3 y0 d$ b- while(distance < 20) {% g! i( Y# s$ g% r1 q; w: S1 g
- distance= pilot.getMovement().getDistanceTraveled();
# ]) J8 Q3 o w: K - speed= distance * 5 + 10;: P5 H V6 R# M* I7 M/ w
- pilot.setRotateSpeed(speed);
: D& ~- I6 J' g4 m: P( a" x - Delay.msDelay(100);/ O/ G4 L. Q* R) q. b1 T
- }
' y# E9 Z- x) G* P - for(int i = 0; i < 10; i++) {0 ~2 j+ s: X0 O3 U' e5 {* M; ]
- speed+= 5;
4 w) U" k. K. u- c2 t9 K - pilot.setRotateSpeed(speed);
% `: S* [2 G) C* D0 r: ^ - Delay.msDelay(200);2 [" ]7 K) Y( T9 a. i B& m* D0 K: O
- }* E1 a: N% y+ p6 e' z
- 0 j, j0 y/ d- W0 T7 ]! I S
- //TODO 减速
; J9 F+ G3 J; j$ @- O - ; b( D) N7 d; x6 N, _
- //停止机能, c! [/ A- U$ `/ F) r
- while(pilot.isMoving()) {' P4 h8 x& Y N5 N
- if(Button.readButtons() > 0)
4 a6 P; v+ C% Y; { - pilot.stop();, A' K" h6 q' a3 c
- }
# D, F6 E- i) ~# s
: L$ V" J$ n( {8 x, p& }- System.out.println("" + pilot.getMovement().getDistanceTraveled());% h0 x* y3 e3 J3 _4 U! o
- System.out.println(-Motor.A.getTachoCount());
+ B: S5 a# Q# F" v2 |% y$ x - System.out.println(-Motor.B.getTachoCount());- v# e7 F: \: }! k: f
- Button.waitForAnyPress();
9 ?, f$ L, u, [* b - }1 U/ K9 [& Z+ e( w( T
- & E, Y! V0 t2 Y
- publicstatic void main(String[] args) {& K; e% T5 s4 ^2 {2 g( J+ r
- Straighttraveler = new Straight();3 s$ x+ W, y* X* _. r6 K
- 5 F+ k$ B0 Q1 K5 m/ |, V% A; m
- traveler.adjust();
l1 u; w y" |- Q - traveler.go();
1 c" m9 J- L" l/ L7 U2 [ - }# l& ~# O p2 }/ m
- }
复制代码 ; [ |- } P% G
|