`
flyfoxs
  • 浏览: 298177 次
  • 性别: Icon_minigender_1
  • 来自: 合肥
社区版块
存档分类
最新评论

使用Clojure实现贪吃蛇

 
阅读更多

最近学习Clojure,买了<<Clojure程序设计>>,对里面大部分章节进行了精读. 比如贪吃蛇程序,并且这个程序也比较精简,200行不到.

在读的过程中,对不能一目了然的地方,添加了注释,现发出来,希望能对有的人有用.

 

 

;;
;;Excerpted from "Programming Clojure, Second Edition",
;;published by The Pragmatic Bookshelf.
;;Copyrights apply to this code. It may not be used to create training material, 
;;courses, books, articles, and the like. Contact us if you are in doubt.
;;We make no guarantees that this code is fit for any purpose. 
;;Visit http://www.pragmaticprogrammer.com/titles/shcloj2 for more book information.
;;
; Inspired by the snakes that have gone before:
; Abhishek Reddy's snake: http://www.plt1.com/1070/even-smaller-snake/
; Mark Volkmann's snake: http://www.ociweb.com/mark/programming/ClojureSnake.html 

; The START:/END: pairs are production artifacts for the book and not 
; part of normal Clojure style

(ns examples.snake
  (:import (java.awt Color Dimension) 
   (javax.swing JPanel JFrame Timer JOptionPane)
           (java.awt.event ActionListener KeyListener))
  (:use examples.import-static))
(import-static java.awt.event.KeyEvent VK_LEFT VK_RIGHT VK_UP VK_DOWN)

; ----------------------------------------------------------
; functional model
; ----------------------------------------------------------
(def width 75)
(def height 50)
(def point-size 10)
(def turn-millis 75)
(def win-length 5)
;通过Map定义了4个运动方向
(def dirs { VK_LEFT  [-1  0] 
            VK_RIGHT [ 1  0]
            VK_UP    [ 0 -1] 
    VK_DOWN  [ 0  1]})

;当snake吃到苹果后, snak的长度需要添加1	
(defn add-points [& pts] 
  (vec (apply map + pts)))

  
;pt是 snake的节点坐标(body)信息,或者apple的位置(location)坐标信息, 数据结构是一个2维vector [(rand-int width) (rand-int height)]   ;
;(pt 0):取vector#pt的第一个元素返回,X坐标
;(pt 1):取vector#pt的第二个元素返回,Y坐标
(defn point-to-screen-rect [pt] 
  (map #(* point-size %) 
       [(pt 0) (pt 1) 1 1]))

(defn create-apple [] 
  {:location [(rand-int width) (rand-int height)]
   :color (Color. 210 50 90)
   :type :apple}) 

(defn create-snake []
  {:body (list [1 1]) 
   :dir [1 0]
   :type :snake
   :color (Color. 15 160 70)})

;这种函数定义方式很少见,可参考 http://stuartsierra.com/2010/01/15/keyword-arguments-in-clojure
;参数1是一个Map, 使用了结构的办法直接传递给函数体,
;参数2不是必须的, 是变长参数(在此Demo程序中,传入了一个类型为key的实参,类似于传入了True )  
;业务逻辑就是在snak移动的过程中,判断是否需要增长一个节点
;:keys [body dir]} 从输入Map中,寻找:body所对应的Value并赋值给body
;
(defn move [{:keys [body dir] :as snake} & grow]
  (assoc snake :body (cons (add-points (first body) dir) 
   (if grow body (butlast body)))))

(defn turn [snake newdir] 
  (assoc snake :dir newdir))

(defn win? [{body :body}]
  (>= (count body) win-length))

;检测snake是否出现交叉,当snake长度扩张后容易出现  
(defn head-overlaps-body? [{[head & body] :body}]
  (contains? (set body) head))

(def lose? head-overlaps-body?)

;snake头和苹果重合时, snake吃掉apple
(defn eats? [{[snake-head] :body} {apple :location}]
   (= snake-head apple))

; ----------------------------------------------------------
; mutable model
; ----------------------------------------------------------
;苹果被吃和新苹果显示,需要在一个事务中;
;如果被吃,需要重新生成苹果,然后Move需要考虑蛇的增长
;如果没有被吃,则只需简单Move
(defn update-positions [snake apple]
  (dosync
   (if (eats? @snake @apple)
     (do (ref-set apple (create-apple))
 (alter snake move :grow))
     (alter snake move)))
  nil)

;Snake的:dir因为同时被move函数读取,所以读取:dir与update :dir需要通过事务来隔离.
(defn update-direction [snake newdir]
  (when newdir (dosync (alter snake turn newdir))))

(defn reset-game [snake apple]
  (dosync (ref-set apple (create-apple))
  (ref-set snake (create-snake)))
  nil)

; ----------------------------------------------------------
; gui
; ----------------------------------------------------------
(defn fill-point [g pt color] 
  (let [[x y width height] (point-to-screen-rect pt)]
    (.setColor g color) 
;x,y指定了横纵坐标, 	width,height指定了矩形的宽和高
    (.fillRect g x y width height)))
	
;声明一个多态的方法,具体的实现由Object中的:type决定, 在此Demo中:type有2类, :apple, :snake
(defmulti paint (fn [g object & _] (:type object)))

;多态的实现之一
(defmethod paint :apple [g {:keys [location color]}] ; <label id="code.paint.apple"/>
  (fill-point g location color))

;多态的实现之一
(defmethod paint :snake [g {:keys [body color]}] ; <label id="code.paint.snake"/>
;doseq 使用point 迭代 body里面的每一个元素,然后传递给(fill-point g point color)处理
;确保snake body里面的每一个元素都被打印
  (doseq [point body]
    (fill-point g point color)))

	
;返回的是一个Class,实现了2个接口,覆写父类的部分方法	
(defn game-panel [frame snake apple]
;JPanel是Class, ActionListener KeyListener 是Interface
  (proxy [JPanel ActionListener KeyListener] []
    (paintComponent [g] ; <label id="code.game-panel.paintComponent"/> 重写 JComponent.paintComponent方法
      (proxy-super paintComponent g)
      (paint g @snake)
      (paint g @apple))
   
   (actionPerformed [e] ; <label id="code.game-panel.actionPerformed"/>实现接口ActionListener总的方法
      (update-positions snake apple)
      (when (lose? @snake)
(reset-game snake apple)
(JOptionPane/showMessageDialog frame "You lose!"))
      (when (win? @snake)
(reset-game snake apple)
(JOptionPane/showMessageDialog frame "You win!"))
      (.repaint this))
  
    (keyPressed [e] ; <label id="code.game-panel.keyPressed"/> ;实现接口KeyListener中的方法
      (update-direction snake (dirs (.getKeyCode e)))) ;捕捉键盘方向键,然后转换成之前定义的Map中寻找匹配的Value
   
   (getPreferredSize []  ;重写 JComponent.getPreferredSize 方法
      (Dimension. (* (inc width) point-size) 
  (* (inc height) point-size)))
    
(keyReleased [e]) ;实现接口KeyListener中的方法
    
(keyTyped [e]))) ;实现接口KeyListener中的方法

(defn game [] 
  (let [snake (ref (create-snake)) ; <label id="code.game.let"/>
apple (ref (create-apple))
frame (JFrame. "Snake")
panel (game-panel frame snake apple)
timer (Timer. turn-millis panel)]
;设置Panel对应的监听
    (doto panel ; <label id="code.game.panel"/>
      (.setFocusable true)
      (.addKeyListener panel))
;将游戏Panel关联到frame上面	  
    (doto frame ; <label id="code.game.frame"/>
      (.add panel)
      (.pack)
      (.setVisible true))
    (.start timer) ; <label id="code.game.timer"/>
    [snake, apple, timer])) ; <label id="code.game.return"/>

  

0
0
分享到:
评论

相关推荐

    Lacinia纯Clojure实现的GraphQL

    【Lacinia:纯Clojure实现的GraphQL解析器】 在当今的Web开发中,GraphQL作为一种强大的数据查询语言,已经越来越受到开发者的欢迎。它提供了一种高效、灵活的方式来获取API数据,允许客户端指定他们需要什么数据,...

    [Clojure] 网络应用开发 (Clojure 实现) (英文版)

    [Pragmatic Bookshelf] 网络应用开发 (Clojure 实现) (英文版) [Pragmatic Bookshelf] Web Development with Clojure Build Bulletproof Web Apps with Less Code (E-Book) ☆ 图书概要:☆ If the usual ...

    Python-利用Clojure实现的一个可拖放的看板示例

    标题中的"Python-利用Clojure实现的一个可拖放的看板示例"表明这是一个结合了Python和Clojure技术的项目,旨在创建一个可交互式的看板应用,支持用户通过拖放操作来管理任务或信息。看板在项目管理和敏捷开发中广泛...

    Clojure学习——使用clojure jdbc操作mysql

    标题 "Clojure学习——使用clojure jdbc操作mysql" 指出的是一个关于使用Clojure编程语言通过Java Database Connectivity (JDBC) API来操作MySQL数据库的主题。Clojure是一种基于Lisp的函数式编程语言,它运行在Java...

    Clojure Data Analysis Cookbook

    - **机器学习基础**:介绍如何使用 Clojure 实现简单的机器学习算法,如线性回归、决策树等。 3. **实践篇**:通过具体的项目案例来巩固前面所学的知识。 - **文本分析**:使用自然语言处理技术进行文本挖掘,如...

    蚂蚁群优化算法演示:使用 Clojure 创建_Clojure _代码_下载

    2. **测试文件**: 使用Clojure的`clojure.test`库进行单元测试,确保算法的正确性。 3. **配置文件**: 可能包括算法参数(如蚂蚁数量、信息素蒸发率、启发式信息权重等)的设置。 4. **示例问题**: 提供一些实际的...

    Clojure编程乐趣]+clojure_programming.pdf

    在“Clojure编程乐趣”这本书中,你将学习到如何使用Clojure进行函数式编程,如何利用Clojure的强大工具处理数据,以及如何构建并发和分布式系统。此外,书中可能还会介绍Clojure社区的一些最佳实践和常用库,例如...

    Python-cljcbloom一个用Clojure脚本实现的跨平台布隆过滤器

    Python-cljcbloom使用Clojure的Java绑定特性,使得这个库能够在任何支持JVM的平台上运行,包括Windows、Linux、macOS等,这样Python开发者无需关心底层实现的平台依赖,可以方便地在不同操作系统上使用。...

    clojurec, 在C 之上,一个Clojure实现.zip

    clojurec, 在C 之上,一个Clojure实现 ClojureC这是面向的面向对象编程语言的编译器。 它基于 ClojureScript,并开始于ClojureScript提交 0e0aa7fdd379649bf87f8fff5c6a64e37fe616a4 社区和组织我们使用

    在 Clojure中实现遗传算法的框架_Clojure_代码_下载

    Clojure中的函数式特性使得交叉操作非常直观,可以使用`assoc`、`update`等函数来实现。 6. **变异(Mutation)**:为保持种群多样性,随机改变部分个体的基因。这通常涉及对编码的随机修改,例如在二进制编码中...

    Practical Clojure.pdf

    这是因为Clojure内部实现了高级别的并发抽象,例如软件事务内存(STM)和其他并发原语。 不可变性是Clojure中的另一个核心概念。在Clojure中,数据结构默认是不可变的,这意味着一旦创建了数据结构,就无法更改。这...

    Programming Clojure 英文电子版

    例如,Clojure使用括号来表示函数调用和数据结构,同时保持了良好的可读性。 - **Functional Programming**:在Clojure中,函数式编程是其核心。不可变的数据结构和无副作用的函数使得程序状态的变化更加容易追踪,...

    clojure1.3.0及资料

    这个文件可能是Storm的Clojure实现或相关工具,对于进行大规模实时数据处理的开发者来说很有价值。 综上所述,这个压缩包包含了一系列资源,可以帮助开发者深入理解Clojure语言,尤其是1.3.0版本及其后续演进。书籍...

    clojure电子书

    这本书不仅讲解了Clojure的基本语法和API,而且通过实例展示了如何使用Clojure解决实际问题。Halloway详细阐述了Clojure的REPL(Read-Eval-Print Loop)工作原理,以及如何利用它进行开发和测试。他还讨论了Clojure...

    Clojure入门教程- Clojure – Functional Programming for the JVM中文版

    - **软件事务内存(STM)**: Clojure通过STM机制实现了高度并发的编程模型,它允许开发者以事务的方式更新共享状态,从而简化了并发编程的复杂度。 #### 六、学习资源与社区 - **官方文档**: Clojure官方网站提供了...

    Lightmod一个全功能的Clojure全栈开发工具

    Lightmod是一款专为Clojure全栈开发设计的强大工具,它为开发者提供了全面的功能,使得在...对于想要涉足或已经使用Clojure进行全栈开发的工程师来说,掌握Lightmod的使用将极大地提升他们的工作效率和开发体验。

    clojure eclipse

    在Eclipse中使用Clojure开发,你需要安装Leiningen,这是一个Clojure项目管理工具,它可以帮助你创建、构建和管理Clojure项目。Leiningen能够生成项目的骨架结构,包含必需的依赖和配置文件,使得开发者可以快速开始...

Global site tag (gtag.js) - Google Analytics