Хэл Фултон - Программирование на языке Ruby
puts "nУстанавливается соединение..."
socket = TCPSocket.new(ipname, PeerPort)
socket.puts "ready"
puts "Ваш противник #{opponent}... у вас черные.n"
who = BLACK
move = nil
board = nil # В этом примере не используется.
num = 0
draw_board(board) # Нарисовать начальное положение.
loop do
num += 1
move = other_move(who, move, num, board, socket)
draw_board(board) # Нарисовать доску после хода белых,
case move
when "resign"
puts "n#{opponent} сдался... вы выиграли!"
break
when /Checkmate/
puts "n#{opponent} поставил вам мат."
break
end
move = my_move(who, move, num, board, socket)
draw_board(board)
case move
when "resign"
puts "nВы сдались. #{opponent} выиграл."
break
when /Checkmate/
puts "n#{opponent} поставил вам мат."
break
end
end
socket.close
end
Я определил этот протокол так, что черные посылают белым сообщение «ready», чтобы партнер знал о готовности начать игру. Затем белые делают первый ход. Ход посылается черным, чтобы клиент мог нарисовать такую же позицию на доске, как у другого игрока.
Повторю, приложение ничего не знает о шахматах. Вместо проверки допустимости хода вставлена заглушка; проверка выполняется локально, то есть на той стороне, где делается ход. Никакой реальной проверки нет — заглушка всегда говорит, что ход допустим. Кроме того, мы хотим, чтобы имитация игры завершалась после нескольких ходов, поэтому мы написали программу так, что черные всегда выигрывают на четвертом ходу. Победа обозначается строкой «Checkmate!» в конце хода. Эта строка печатается на экране соперника и служит признаком выхода из цикла.
Помимо «традиционной» шахматной нотации (например, «P-K4») существует еще «алгебраическая», которую многие предпочитают. Но написанный код вообще не имеет представления о том, какой нотацией мы пользуемся.
Поскольку это было несложно сделать, мы позволяем игроку в любой момент сдаться. Рисование доски тоже заглушено. Желающие могут реализовать грубый рисунок, выполненный ASCII-символами.
Метод my_move всегда относится к локальному концу, метод other_move — к удаленному.
В листинге 18.3 приведен протокол сеанса. Действия клиентов нарисованы друг против друга.
Листинг 18.3. Протокол сеанса шахматной игры% ruby chess.rb Hal % ruby chess.rb
Capablanca:deepthought.org Hal:deepdoodoo.org
Устанавливается соединение... Устанавливается соединение...
Ваш противник Capablanca... у вас белые. Ваш противник Hal... у вас черные.
+------------------------------+ +------------------------------+
| Заглушка! Шахматная доска... | | Заглушка! Шахматная доска... |
+------------------------------+ +------------------------------+
Ваш ход: N-QB3 Противник: N-QB3
+------------------------------+ +------------------------------+
| Заглушка! Шахматная доска... | | Заглушка! Шахматная доска... |
+------------------------------+ +------------------------------+
Противник: P-K4 Ваш ход: P-K4
+------------------------------+ +------------------------------+
| Заглушка! Шахматная доска... | | Заглушка! Шахматная доска... |
+------------------------------+ +------------------------------+
Ваш ход: P-K4 Противник: P-K4
+------------------------------+ +------------------------------+
| Заглушка! Шахматная доска... | | Заглушка! Шахматная доска... |
+------------------------------+ +------------------------------+
Противник: B-QB4 Ваш ход: B-QB4
+------------------------------+ +------------------------------+
| Заглушка! Шахматная доска... | | Заглушка! Шахматная доска... |
+------------------------------+ +------------------------------+
Ваш ход: B-QB4 Противник: B-QB4
+------------------------------+ +------------------------------+
| Заглушка! Шахматная доска... | | Заглушка! Шахматная доска... +
+------------------------------+ +------------------------------+
Противник: Q-KR5 Ваш ход: Q-KR5
+------------------------------+ +------------------------------+
| Заглушка! Шахматная доска... | | Заглушка! Шахматная доска... |
+------------------------------+ +------------------------------+
Ваш ход: N-KB3 Противник: N-KB3
+------------------------------+ +------------------------------+
| Заглушка! Шахматная доска... | | Заглушка! Шахматная доска... |
+------------------------------+ +------------------------------+
Противник: QxP Checkmate! Ваш ход: QxP
+------------------------------+ +------------------------------+
| Заглушка! Шахматная доска... | | Заглушка! Шахматная доска... |
+------------------------------+ +------------------------------+
Capablanca поставил вам мат. Вы поставили мат Hal!
18.2. Сетевые клиенты
Иногда сервер пользуется хорошо известным протоколом — тогда нам надо лишь спроектировать клиента, который общается с сервером на понятном тому языке.
В разделе 18.1 мы видели, что это можно сделать с помощью протоколов TCP или UDP. Но чаще применяются протоколы более высокого уровня, например HTTP или SNMP. Рассмотрим несколько примеров.
18.2.1. Получение истинно случайных чисел из Web
Всякий, кто пытается сгенерировать случайное число, пользуясь детерминированными средствами, безусловно, живет во грехе.
Джон фон НейманВ модуле Kernel есть функция rand, которая возвращает случайное число, но вот беда — число-то не является истинно случайным. Если вы математик, криптограф или еще какой-нибудь педант, то назовете эту функцию генератором псевдослучайных чисел, поскольку она пользуется алгебраическими методами для детерминированного порождения последовательности чисел. Стороннему наблюдателю эти числа представляются случайными и даже обладают необходимыми статистическими свойствами, но рано или поздно последовательность начнет повторяться. Мы можем даже намеренно (или случайно) повторить ее, задав ту же самую затравку.
Но природные процессы считаются истинно случайными. Поэтому при розыгрыше призов в лотерее счастливчики определяются лототроном, который хаотично выбрасывает шары. Другие источники случайности — радиоактивный распад или атмосферный шум.
Есть источники случайных чисел и в Web. Один из них — сайт www.random.org, который мы задействуем в следующем примере.
Программа в листинге 18.4 имитирует подбрасывание пяти обычных (шестигранных) костей. Конечно, игровые фанаты могли бы увеличить число граней до 10 или 20, но тогда стало бы сложно рисовать ASCII-картинки.
Листинг 18.4. Случайное бросание костейrequire 'net/http'
HOST = "www.random.org"
RAND_URL = "/cgi-bin/randnum?col=5&"
def get_random_numbers(count=1, min=0, max=99)
path = RAND_URL + "num=#{count}&min=#{min}&max=#{max}"
connection = Net::HTTP.new(HOST)
response, data = connection.get(path)
if response.code == "200"
data.split.collect { |num| num.to_i }
else
[]
end
end
DICE_LINES = [
"+-----+ +-----+ +-----+ +-----+ +-----+ +-----+ ",
"| | | * | | * | | * * | | * * | | * * | ",
"| * | | | | * | | | | * | | * * | ",
"| | | * | | * | | * * | | * * | | * * | ",
"+-----+ +-----+ +-----+ +-----+ +-----+ +-----+ "
DIE_WIDTH = DICE_LINES[0].length/6
def draw_dice(values)
DICE_LINES.each do | line |
for v in values
print line[(v-1)*DIE_WIDTH, DIE_WIDTH]
print " "
end
puts
end
end
draw_dice(get_random_numbers(5, 1, 6))
Здесь мы воспользовались классом Net::НТТР для прямого взаимодействия с Web-сервером. Считайте, что эта программа — узкоспециализированный браузер. Мы формируем URL и пытаемся установить соединение; когда оно будет установлено, мы получаем ответ, возможно, содержащий некие данные. Если код ответа показывает, что ошибок не было, то можно разобрать полученные данные. Предполагается, что исключения будут обработаны вызывающей программой.
Посмотрим на вариацию этой идеи. Что если вы захотели бы применить случайные числа в каком-нибудь приложении? Поскольку обслуживающая программа на стороне сервера позволяет указать количество возвращаемых чисел, то было бы логично сохранить их в буфере. Учитывая, что при обращении к удаленному серверу задержки неизбежны, следует сразу заполнить буфер во избежание лишних запросов по сети.