180行ruby代码搞定游戏2048
最今在玩2048这款小游戏,游戏逻辑简单,很适合我这样的对于游戏新入行的人来实现逻辑。于是选择了最拿手的ruby语言来实现这款小游戏的主要逻辑。还是挺简单的,加起来4小时左右搞定。
上代码:
require 'optparse' module Help
HELP_TEXT =<<HELP press buttons for move
l => move to left
r => move to right
t => move to top
b => move to bottom
press e button to exit game you can see this help text if your input ruby ruby_2048.rb --help
HELP
def set_helps
OptionParser.new do |opts|
opts.on_tail("-h", "--help", 'This help text.') do
puts HELP_TEXT
exit!
end
end.parse!
end end class Object
def invoke(need, method)
if need
self.send(method)
else
self
end
end
end class R2048
extend Help attr_reader :chessboard
LEFT = "l"
RIGHT = "r"
TOP = "t"
BOTTOM = "b"
EXIT = "e" def initialize
R2048.set_helps
@chessboard = Array.new(4){|x| Array.new(4){|y| 0}}
@init_moved = false
1.upto(2){|i| generate_init_num}
end def generate_init_num
return false unless @chessboard.flatten.uniq.select{|chess| chess == 0}.count > 0 rand_position = rand(16)
x, y = rand_position/4, rand_position % 4
until @chessboard[x][y] == 0
rand_position = rand(16)
x, y = rand_position/4, rand_position % 4
end
@chessboard[x][y] = [2, 4][rand(2)] end def check_and_merge(transpose, reverse)
moved = false
temp_chessboard = @chessboard.invoke(transpose, :transpose).map do |row|
reversed_row = set_jump_step(row.invoke(reverse, :reverse)).invoke(reverse, :reverse)
moved = true if reversed_row != row.invoke(reverse, :reverse)
reversed_row
end.invoke(transpose, :transpose) if moved
@chessboard = temp_chessboard
true
else
if !@init_moved
@init_moved = true
true
else
false
end
end
end def generate_new_num(transpose, pos)
ungenerated = true right_positions = []
@chessboard.invoke(transpose, :transpose).each_with_index{|row, i| right_positions << i if row[pos] == 0}
right_position = right_positions[rand(right_positions.count)] row_index = 0
@chessboard = @chessboard.invoke(transpose, :transpose).map do |row|
if ungenerated && row_index == right_position
ungenerated = false
row[pos] = [2, 4][rand(2)]
end
row_index += 1
row
end.invoke(transpose, :transpose)
!ungenerated
end def set_jump_step(row)
pured = row.select{|chess| chess != 0 }.inject([]) do |sum, chess|
if sum.last == chess
sum.pop
sum << chess * 2
else
sum << chess
end
end
pured.concat Array.new(4 - pured.count, 0)
end def display
puts "==============================="
@chessboard.each_with_index do |c, row|
puts "#{c[0]} #{c[1]} #{c[2]} #{c[3]}"
puts
end
end def failure_display
puts "you have failed!!!"
end def run
display
key = nil
until key == "e\n"
key = gets
key.gsub!("\n", "")
return if key == EXIT if ![LEFT, RIGHT, TOP, BOTTOM].include? key
puts "input error"
next
end generate = case key
when LEFT
if check_and_merge(false, false)
generate_new_num(false, 3)
else
nil
end
when RIGHT
if check_and_merge(false, true)
generate_new_num(false, 0)
else
nil
end
when TOP
if check_and_merge(true, false)
generate_new_num(true, 3)
else
nil
end
when BOTTOM
if check_and_merge(true, true)
generate_new_num(true, 0)
else
nil
end
end if generate == nil || generate
display
else
failure_display
return
end
end
end
end R2048.new.run
写了一些測试:
require 'ruby_2048' describe R2048 do
before(:each) do
@r2048 = R2048.new
end
it "should jump to [2, 0, 0, 0] when input [0, 0, 0, 2]" do
@r2048.set_jump_step([0, 0, 0, 2]).should == [2, 0, 0, 0]
end it "should jump to [2, 4, 0, 0] when input [2, 0, 4, 0]" do
@r2048.set_jump_step([2, 0, 4, 0]).should == [2, 4, 0, 0]
end it "should jump to [4, 0, 0, 0] when input [2, 0, 2, 0]" do
@r2048.set_jump_step([2, 0, 2, 0]).should == [4, 0, 0, 0]
end it "should jump to [2, 4, 4, 0] when input [2, 4, 2, 2]" do
@r2048.set_jump_step([2, 4, 2, 2]).should == [2, 4, 4, 0]
end it "should jump to [4, 4, 0, 0] when input [2, 2, 2, 2]" do
@r2048.set_jump_step([2, 2, 2, 2]).should == [4, 4, 0, 0]
end it "should + 1 chess if generate_init_num" do
expect { @r2048.generate_init_num }.to change{@r2048.chessboard.flatten.count{|chess| chess!= 0} }.by(1)
end it "should have already have two chess when inited" do
expect{@r2048.count} == 2
end
end
貌似还不错,最新代码请见github:https://github.com/xumc/ruby_2048
后记:
一篇博客介绍c++版命令行2048居然写了500多行,见http://blog.csdn.net/yc7369/article/details/29416819, 恰恰证明了ruby的简洁。
180行ruby代码搞定游戏2048的更多相关文章
- 5行js代码搞定导航吸顶效果
一.HTML布局 首先写HTML布局 <body> <div id="wrap"></div> </body> 二.CSS样式 给点 ...
- 200行Java代码搞定计算器程序
发现了大学时候写的计算器小程序,还有个图形界面,能够图形化展示表达式语法树,哈哈;) 只有200行Java代码,不但能够计算加减乘除,还能够匹配小括号~ 代码点评: 从朴素的界面配色到简单易懂错误提示 ...
- 【备忘】windows环境下20行php代码搞定音频裁剪
先上图,由于最近的需求需要对语音文件进行处理,所以抽空研究了下php处理音/视频文件的处理,简单的demo处理,截取一个音频文件的前20秒,并保存新的媒体文件. 操作步骤: ①在此站点下载所需的辅助程 ...
- 几行JavaScript代码搞定Iframe 自动适应
场景:Iframe嵌入flash,希望flash能随着页面的resize而resize. 主要代码: 代码 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTM ...
- 80行Python代码搞定全国区划代码
微信搜索:码农StayUp 主页地址:https://gozhuyinglong.github.io 源码分享:https://github.com/gozhuyinglong/blog-demos ...
- 3kb jQuery代码搞定各种树形选择。
自制Jquery树形选择插件. 对付各种树形选择(省市,分类..)90行Jquery代码搞定,少说废话直接上插件代码.稍后介绍使用说明.是之前写的一个插件的精简版. 1.Jquery插件代码 /* * ...
- 30行代码搞定WCF并发性能测试
[以下只是个人观点,欢迎交流] 30行代码搞定WCF并发性能 轻量级测试. 1. 调用并发测试接口 static void Main() { List< ...
- 10行代码搞定移动web端自定义tap事件
发发牢骚 移动web端里摸爬滚打这么久踩了不少坑,有一定移动web端经验的同学一定被click困扰过.我也不列外.一路走来被虐的不行,fastclick.touchend.iscroll什么的都用过, ...
- 开源作品ThinkJDBC—一行代码搞定数据库操作
1 简介 ThinkJD,又名ThinkJDBC,一个简洁而强大的开源JDBC操作库.你可以使用Java像ThinkPHP框架的M方法一样,一行代码搞定数据库操作.ThinkJD会自动管理数据库连接, ...
随机推荐
- CSS自学笔记(2):CSS语法
CSS的语法规则主要有两个分构成选择器和声明(声明问一条或者多条). selector {declaration1; declaration2; ... declarationN } 选择器(selc ...
- myeclipse 环境配置优化,不断跟新整理中
myeclipse 环境配置,不断跟新整理中1.General --->Workspace ---> UTF-8 工作环境编码2.General --->Editors --> ...
- ARM Cortex M3(V7-M架构)硬件启动程序 二
解析 STM32 的启动过程 解析STM32的启动过程 当前的嵌入式应用程序开发过程里,并且C语言成为了绝大部分场合的最佳选择.如此一来main函数似乎成为了理所当然的起点——因为C程序往往从main ...
- MyBatis 最常见错误,启动时控制台无限输出日志
你是否遇到过下面的情况,控制台无限的输出下面的日志: Logging initialized using ‘class org.apache.ibatis.logging.log4j.Log4jImp ...
- 解决android开发webservice的发布与数据库连接的问题
由于app后续开发的需要,移植了两次webservice和数据库,遇到了不少问题,也花费了很多时间,实践告诉我要学会寻找问题的根源,这样才能在开发中节省时间,尽快解决问题!好,废话不多说,转入正题…… ...
- Day3_字符串操作与正则表达式
本节课的主要内容有:字符串的格式化.连接与分割.比较.匹配和替换.使用正则表达式 字符串的格式化: 去除空格:trim() 使用html格式化:nl2br() 替换‘\n’为‘<br /> ...
- UberX及以上级别车奖励政策(优步北京第四组)
优步北京第四组: 定义为2015年7月20日至今激活的司机(以优步后台数据显示为准) 滴滴快车单单2.5倍,注册地址:http://www.udache.com/如何注册Uber司机(全国版最新最详细 ...
- Android中的数据存储
Android中的数据存储主要分为三种基本方法: 1.利用shared preferences存储一些轻量级的键值对数据. 2.传统文件系统. 3.利用SQLite的数据库管理系统. 对SharedP ...
- 关于 FPGA 和 外部芯片接口时序设计
在看这篇文章之前, 建议先好好读下这篇文章.http://download.csdn.net/detail/angelbosj/8013827. 因为我不太会用 VISio.要是哪位网友能告诉我.怎么 ...
- 更改DataTable列名方法
1.通过DataAdapter将查询的结果填充到DataSet的表(DataTable)中: 如:dataAdapter.Fill(dataSet),这时dataSet的表名默认为Table 如果使用 ...