0

我有一个硬汉给你。

我有一个程序用“w”类型的 Tile 对象填充二维数组。然后程序在数组中选择一个随机点(中心)并尝试转动距离中心为 4 的半径内的所有对象。不知何故,程序没有这样做,我认为问题出在数学公式上,但我找不到错误。

class Map
  def initialize(d1,d2)
    @data = Array.new(d1) { Array.new(d2)}
  end

  def [](x, y)
   @data[x][y]
  end

  def x(x)
    @data[x]
  end

  def y(y)
    @data[y]
  end

  def []=(x, y, value)
    @data[x][y] = value
  end
end

# TILES
class Tile
  # Types = ["l", "w", "r"]
  attr_accessor :type
  def initialize(type)
    @type = type
  end

  def set(string)
    @type = string
  end
  def to_s
     @type
  end
end

# REAL ACTION HERE
def generate 
  h = 20
  w = 20
  i = 0
  worldmap = Map.new(20, 20)
  while i < h do
    n = 0
    while n < w do
      worldmap.[] = (i, n, Tile.new("w"))
      n = n + 1 
    end
    i = i + 1
  end 

  gen(worldmap)
  look(worldmap)
end

def look(map)
  z = 0
  b = 0
  c = 0
  string = " "
  while b < 20 do
    while z < 20 do  
      string = string + map.[](b, z).to_s
      z = z + 1
    end 
    b = b + 1 
    puts string
    c = c + 1
  end
end

def gen(map)
  circle_amount = 1

  i = 0
  x = 0
  y = 0
  while i < circle_amount do 
    #select a center
   cy = (rand(1..20))
   cx = (rand(1..20))
   center = map.[](cx,cy)
   center.set("C")

   radius = 4
   puts cy.to_s + " " +  cx.to_s

   while y < 20
     while x < 20
       offsetY = y - cy 
       offsetX = x - cx 
       distance = offsetY**2 + offsetX**2
       if distance <= radius**2 then 
           tile = map.[](y, x)
           tile.set("l") # I DID IT
         end
       x = x + 1
     end
      y = y + 1
   end
   i = i + 1
  end
end

generate 

以下是输出示例

wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww wwwwwwwwwwwwlllllwww

C:/Users/borya/Documents/NetBeansProjects/LearningPurporse/lib/main.rb:7:in []': >undefined method[]' for nil:NilClass (NoMethodError)

4

1 回答 1

2
 > m = Map.new(5,5)
 => #<Map:0x007fa5d9838fc0 @data=[[nil, nil, nil, nil, nil], [nil, nil, nil, nil, nil], [nil, nil, nil, nil, nil], [nil, nil, nil, nil, nil], [nil, nil, nil, nil, nil]]> 
ruby-1.9.3-p125 :040 > m[1,1]="hi"
(x=1, y=1) = hi
@data[x][y]=hi
 => "hi" 
 > m
 => #<Map:0x007fa5d9838fc0 @data=[[nil, nil, nil, nil, nil], [nil, "hi", nil, nil, nil], [nil, nil, nil, nil, nil], [nil, nil, nil, nil, nil], [nil, nil, nil, nil, nil]]> 

我可能会编写更接近下面的代码,尽管它有点不合时宜,并且我保留了你的一些原始代码,即使它可能不是我会这样做的。

class Map
  attr_reader :data, :x, :y

  def initialize(x, y)
    @x = x
    @y = y
    @data = Array.new(x) { Array.new(y) }
  end

  def set(&block)
    (0..x-1).each do |i|
      (0..y-1).each do |j|
        result = yield i, j, @data[i][j]
        @data[i][j] = result if result
      end
    end
  end

  def [](x, y)
    @data[x][y]
  end

  def []=(x, y, value)
    @data[x][y] = value
  end

  def dump
    (0..x-1).each do |i|
      string = ""
      (0..y-1).each do |j|
        string += "#{self[i, j]} "
      end
      puts string
    end
  end
end

def generate 
  map = Map.new(20, 20)
  map.set { |x, y, at| Tile.new('w') }
  gen(map)
end

def gen(map)
  3.times do |i|
    cy = rand(0..map.x-1)
    cx = rand(0..map.y-1)
    puts "cx=#{cx}, cy=#{cy}"
    map[cx, cy].set('C')
    map.dump

    radius = 3.0

    map.set do |x, y, at|
      offsetX = x - cx
      offsetY = y - cy
      distance = Math.sqrt(offsetY**2 + offsetX**2)
      distance.abs <= radius ? '.' : false
    end

    puts ''
    map.dump
  end
end

一旦你修正了你的数学,你最终会得到这样的结果:

w w w w w w w w w w w w w w w w w . w w 
w w w w w w w w w w w w w w w . . . . . 
w w w w w w w w w w w w w w w . . . . . 
w w w w w w w w w w w w w w . . . . . . 
w w w w w . w w w w w w w w w . . . . . 
w w w . . . . . w w w w w w w . . . . . 
w w w . . . . . w w w w w w w w w . w w 
w w . . . . . . . w w w w w w w w w w w 
w w w . . . . . w w w w w w w w w w w w 
w w w . . . . . w w w w w w w w w w w w 
w w w w w . w w w w w w w w w w w w w w 
w w w w w w w w w w w w w w w w w w w w 
w w w w w w w w w w w w w w w w w w w w 
w w w w w w w w w w w w w w w w w w w w 
w w w w w w w w w w w w . w w w w w w w 
w w w w w w w w w w . . . . . w w w w w 
w w w w w w w w w w . . . . . w w w w w 
w w w w w w w w w . . . . . . . w w w w 
w w w w w w w w w w . . . . . w w w w w 
w w w w w w w w w w . . . . . w w w w w 

警告:此代码超出您的技能水平,您可能希望将其翻译回具有额外冗长的 while 循环,直到您进一步了解为止。

Boris,您的原始代码很难推理,原因有很多,包括随机缩进/格式和简单的语法错误。一致地格式化代码很重要。这样做可以让您和其他人更轻松地阅读它。

熟悉您使用的语言的范例很重要。一般来说,如果 Ruby 代码有 while 循环、计数器增量等,而它们实际上并没有用于索引以外的任何事情,那就是代码异味。Canonical Ruby 倾向于使用.each、 块等更实用的风格。

你是正确的,你的数学是错误的,尽管我怀疑你不知道它有多么错误。像这样的数学容易调试:手工完成。画一个网格。使用程序(cx, cy)并手动进行计算。做一些健全的检查,以确保你认为你在做的是(a)你实际上在做的,和(b)合理的。

例如,您的距离有时是数百。显然,当您的网格是...时,这是没有意义的(20, 20)……一旦您看到了,解决方案就很明显了——您省略了一个步骤。

相同的逻辑可用于其余的数学问题。

玩电脑……做电脑。按照它的步骤,在纸上。没有比内化“屏幕下方”实际发生的事情更快的理解途径了。

于 2012-10-05T21:50:45.387 回答