找回密码
 马上注册

QQ登录

只需一步,快速开始

查看: 11422|回复: 7

最直的三百厘米——开环控制的小车

[复制链接]
发表于 2017-7-25 22:09:23 | 显示全部楼层 |阅读模式
本帖最后由 hello怪 于 2017-7-25 22:24 编辑 : c) b0 _( o* F; F; ^% A
% J7 L6 ^2 Z8 h/ J2 ^
(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址:
http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html

* x& z4 Q9 k) V8 @8 O1 \
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:

$ [9 h4 M3 |4 N, Q
场地
* I5 i3 W) ^" I  o  水平度和光滑度
在家里当然瓷砖的效果是最好的0 [' A: K3 ]% I* A( f+ e0 p
  线的直度
虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直
9 L# \1 u7 m  W1 e( ?7 K( i( Y2 J6 M2 ?# O- C4 z
车子结构
: q! p' K( P" |: k& h8 C$ h1 g/ N  轮子间距# A" T* C& U) k9 W
     轮子间距越大越好,稍微想一下便知
8 p* ~% N1 p# U  轮子大小3 O$ _2 t; E+ D7 [2 U
     轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子). D0 c$ T" E2 Y' F  U/ [7 S
  随动轮的设计
4 a+ ^+ U9 w" h: C% G. k, Y     随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响
2 T' g2 \5 z) f7 @3 h+ Z  重心位置
" @$ w8 }& k+ Y- S" c  g     重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小$ d7 ^0 G9 E1 a) q1 `: C2 J
  结构稳定性
不用说,肯定是越结实越好。但是不能造成重量过大
& x7 V, L; p2 \) h  轮子个数
有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)
( _* A5 l, s# G  驱动方式
这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动
+ `9 u' v/ j2 T. F6 B  x2 F' G, y" t/ ^
控制程序0 z+ a: ~7 Z# C* D5 p
  运动过程控制' m0 O) p" K6 B0 a
     好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)9 r  ~6 L, R; e2 P

4 b; K8 r" c! r- f. a2 x' w调试与运行
8 h6 s3 L# a; _  轮子转速调整(车轮大小的设定), x5 v5 a: r# n; |* m  C3 N0 b
      这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法
. _/ U6 z5 i% W1 @4 D5 {  起始位置车子正方向和直线间的夹角
& W, l, v6 C* i6 _7 ?) K      一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点

9 v2 J6 Q0 v; g3 U" Q6 \' h/ r6 v3 W

6 f- M0 q; v. O; B# K
然后上源代码:

  1. % m& T! K* ?8 X0 `- a( |: V
  2. importlejos.robotics.navigation.DifferentialPilot;$ g% _3 C( d% y; Q
  3. import lejos.util.Delay;' X5 {/ [9 V  p- H& j5 J  \

  4. # H: U. {- W- T. Q' Y
  5. /**( j9 t: g4 v; R" N3 Q- W" r
  6. *Robot that stops if the buttons are presses before it completes its travel.
    . i  Z8 T3 x5 L( @1 L
  7. */* D7 z+ I% I6 |

  8. . T- k' _# p9 `/ z
  9. 2 ^! q9 R1 k( B/ H2 r
  10. public class Straight {, L  V8 Y+ v$ \- s4 w
  11.   DifferentialPilotpilot;, J$ o! W( I# P% S& s0 g6 |  U
  12.   doubleleftD = 4.3, rightD = 4.3;
    + p/ b2 `' A5 b2 v

  13. ( y! E# y/ f3 V& V& \4 e8 `0 j
  14.   publicvoid adjust() {  ^! _: Z3 Y4 s: d0 c0 w  p
  15.            //wait for release' N4 }1 _( I: D& `
  16.            while(Button.readButtons() > 0)6 Z( M; O# N, i7 \
  17.                    ;
    2 _% N) m6 V; a1 {8 K& E
  18.            while(Button.readButtons() == 0) {- \' a0 I/ e0 j2 U# @6 |
  19.                    //input size by adjusting the wheels/ ]) l/ E/ G2 c
  20.                    rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
    * ~+ k4 d$ ?* f# V5 f
  21.                    leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值
    7 n' p) @9 k- G5 a& A5 {
  22. ( r: g' L. ]- N8 k
  23.                    //display the size input, ^) L7 P/ \5 s) a
  24.                    LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);4 ?! m# W, K) u* \$ @9 B" M
  25.                    LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);4 o9 Q6 D0 e. M
  26.                    Delay.msDelay(100);
    ( P! S7 s3 o1 p' ^1 ]
  27.                    LCD.clear();/ s1 u9 i' l# _6 U
  28.            }9 @" K1 M4 N. l2 H4 e$ u
  29. 6 R4 I! E, t7 \' E  z' n
  30.   }
    ! I% c8 R4 A$ `$ H0 [' A8 L
  31. 3 d# \$ E  w( V& [4 \, [$ t
  32.   publicvoid go() {: s4 @( i. D$ s
  33.            //准备起步
    ! H% P. R% w) b$ ]# |' \
  34.            //wait for release
    ! ?5 G5 c' Y! E2 w9 B& n3 K8 O
  35.            while(Button.readButtons() > 0)" [) h7 {* a& L, z. ^' c' A" ?! l
  36.                    ;
    : F+ u8 G8 S+ q, F+ G8 ]8 a" T" \
  37.            System.out.println("pressto go");
    0 `8 n$ Z+ t. M5 |

  38. " `1 Z. a: q0 c$ W* `- e7 x8 z
  39.            Button.waitForAnyPress();2 U+ {7 c' ?6 y
  40.            LCD.clear();" \! n$ g+ N7 K) |! w
  41. % [5 ~1 Z; S: {
  42.            //初始化
    . |  J4 _  H" P2 n" L. S
  43.            pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);7 j5 [; y# L2 p7 l& t
  44.            pilot.reset();
    7 z; V. e7 H7 Z& Q8 T+ k: \3 g

  45. 5 \( P5 d, o# u2 l
  46.            Delay.msDelay(3000);
    & o0 K& _: E6 n, G; s3 ]

  47. # Y* ]5 b3 t, F" }5 A3 G0 X2 Z+ b
  48.            doublespeed = 10;( H- w! @4 v0 f" \* G' d. u; s1 d; d
  49.            doubledistance = 0;
    2 x! P4 S9 M" ]! S3 `  q9 M9 i
  50. / i, \8 x* _  `8 k0 g
  51.            //起步加速
    $ d/ Q- j$ c7 i8 h7 x0 x
  52.            pilot.travel(300,true);
    . {0 m2 @/ T( j8 X& E1 P2 u& y0 E
  53. , j$ @7 t7 t- h8 U' I1 f; O
  54.            while(distance < 20) {
      A9 `2 F  t: o" A  G/ R4 s
  55.                    distance= pilot.getMovement().getDistanceTraveled();3 q/ y5 {+ ?2 h* K" u1 Y9 N
  56.                    speed= distance * 5 + 10;4 `8 V% H: B: i
  57.                    pilot.setRotateSpeed(speed);
    " m+ {) K8 T! f! d1 {0 X! C2 U
  58.                    Delay.msDelay(100);
      l6 z- [! T& K1 s
  59.            }
    0 H; L7 N* T) t3 r& }
  60.            for(int i = 0; i < 10; i++) {8 D& W# R! C/ E2 M1 |! j& o( U; A, V
  61.                    speed+= 5;
    . M7 f3 h5 _. Y- f
  62.                    pilot.setRotateSpeed(speed);: X. H7 ]( v6 |5 j0 w8 X( q
  63.                    Delay.msDelay(200);
    / z/ m! `1 d1 n/ b: e
  64.            }2 x) w' s% A2 a* u* z

  65. 7 l; ^' M2 {6 s/ p" s+ g
  66.            //TODO 减速" u7 C4 X# {* A
  67. ) Q, L8 `8 _) Z9 W. V7 f
  68.            //停止机能( q+ ^+ h( A- J5 M9 n
  69.            while(pilot.isMoving()) {% h+ K" S% O. ?9 Z" K
  70.                    if(Button.readButtons() > 0)
    3 S! \0 H3 Y5 C, ^8 ]* r
  71.                            pilot.stop();2 X7 t0 ^& R  W; P. P
  72.            }
    " D6 h2 [: j, d% n
  73. ( n* r/ ^1 j% Y* F5 e# o- h5 e7 H
  74.            System.out.println("" + pilot.getMovement().getDistanceTraveled());
    $ E; _! p. T' N# }
  75.            System.out.println(-Motor.A.getTachoCount());
    7 t; r1 W: K# O* X7 i
  76.            System.out.println(-Motor.B.getTachoCount());
    ( i4 g. H$ l% @* a8 B, {! `& c
  77.            Button.waitForAnyPress();0 z8 c! ^! m( m" j
  78.   }6 s* y' L5 z# d+ O6 u
  79. 9 F0 T; U& X& W0 ~) Q( p
  80.   publicstatic void main(String[] args) {/ W& z& w/ K5 m; S
  81.            Straighttraveler = new Straight();
    # a& j! L* q: r
  82. : [# c; [% d2 k; B9 S
  83.            traveler.adjust();
    7 v# P+ E5 a6 |* D/ @+ J
  84.            traveler.go();
    5 d; ^( N" C; y# P
  85.   }) V" ?$ ]7 N& Y
  86. }
复制代码

1 r+ c7 _% ~& {- S1 m
IMG_20170720_111026()().jpg
IMG_20170720_114534()().jpg

轮子外缘与标尺间的距离:半个乐高单位

轮子外缘与标尺间的距离:半个乐高单位
如果您觉得我的帖子对您有用,请不吝给我一个“赞”!
 楼主| 发表于 2017-7-25 22:10:06 | 显示全部楼层
二楼自占备用,欢迎发表评论
如果您觉得我的帖子对您有用,请不吝给我一个“赞”!
回复

使用道具 举报

发表于 2017-7-26 01:34:48 | 显示全部楼层
这个源代码是什么语言呀
如果您觉得我的帖子对您有用,请不吝给我一个“赞”!
回复

使用道具 举报

发表于 2017-7-26 07:22:29 | 显示全部楼层
本帖最后由 clx 于 2017-7-26 07:26 编辑 2 E, w5 y$ T" W$ N/ z% y3 E

8 z' ~) _8 F1 `# [' c8 m这一行行代码,C?,暂还不懂,反正看到写代码的都敬仰,这种程序是不是比G(乐高是不是叫G语言)的严谨,用代码程序的机器人都能很好的实现复杂功能
如果您觉得我的帖子对您有用,请不吝给我一个“赞”!
回复

使用道具 举报

 楼主| 发表于 2017-7-26 10:08:44 | 显示全部楼层
clx 发表于 2017-7-26 07:22
' o7 H0 o& e! K3 k- }7 G这一行行代码,C?,暂还不懂,反正看到写代码的都敬仰,这种程序是不是比G(乐高是不是叫G语言)的严谨, ...
( c1 g% ^* W, @  }% n
我用的是lejos,一种基于Java的nxt专用语言。论严谨的话,程序结构的严谨和语言选择是无关的,但是不同的语言确实各有专长。比如这个lejos,他的函数库就比G大若干倍,有很多别人写过的东西可以直接拿来用。虽然G很适合给小朋友和初学者普及(本来就是针对于此),但是图形编程界面本身不适合大项目的编写,如果想要深入玩乐高的话转向代码是必然的
如果您觉得我的帖子对您有用,请不吝给我一个“赞”!
回复

使用道具 举报

发表于 2017-7-26 10:23:44 | 显示全部楼层
学习了,在乐高深入了解方向还不太明确,转代码确实必不可少
如果您觉得我的帖子对您有用,请不吝给我一个“赞”!
回复

使用道具 举报

发表于 2017-7-29 16:34:41 | 显示全部楼层
车子结构很好,赞
如果您觉得我的帖子对您有用,请不吝给我一个“赞”!
回复

使用道具 举报

发表于 2019-8-16 14:40:13 | 显示全部楼层
如果您觉得我的帖子对您有用,请不吝给我一个“赞”!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 马上注册

本版积分规则

手机版|中文乐高 ( 桂ICP备13001575号-7 )

GMT+8, 2026-1-10 03:22 , Processed in 0.577526 second(s), 19 queries .

Powered by Discuz! X3.5

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表