Пиратский клад

18

Найти сокровище, спрятанное пиратами, действительно легко. Все, что вам нужно для этого - карта. Широко известно, что пираты рисуют карты от руки и описывают алгоритм поиска места следующим образом: «Встаньте возле одинокой пальмы, сделайте 30 шагов к лесу, 15 к озеру, ...»

Путешествие по такому маршруту - это, как правило, прекрасная возможность увидеть пейзажи ... Однако сейчас ни у кого нет на это времени. Вот почему искатели сокровищ попросили вас написать программу, которая бы определяла точное местоположение сокровища, используя данную карту.


вход

Ввод состоит из нескольких инструкций <Direction> <Distance>, разделенных запятыми (за которыми следуют по одному пробелу).

Направление одно из следующих:
N- север, S- юг, E- восток, W- запад,
NE- северо-восток, NW- северо-запад, SE- юго-восток, SW- юго-запад.

Расстояние - это целое число (от 1 до 1000).

Выход

Результатом являются координаты, в которых вы заканчиваете после выполнения инструкций, с тремя десятичными знаками, разделенными запятой и пробелом. Стартовая локация имеет нулевые координаты (0, 0).

Первая координата X (Восток означает координаты больше нуля, Запад означает меньше нуля).
Вторая координата Y (север означает больше нуля, юг означает меньше нуля).


Примеры

1. N 3, E 1, N 1, E 3, S 2, W 1

    3.000, 2.000

2. NW 10

    -7.071, 7.071

3. NE 42, NW 42, SE 42, SW 42

    0.000, 0.000


Источник (на украинском). Формат ввода там другой.

Олег Припин
источник
8
Бонусные баллы за это в логотипе? ;)
Питер Тейлор
@Peter Формат вывода строгий ... Но посмотрим, понравится ли людям картинка: D
Олег Припин
1
Первый пример вывода должен быть -3.000, 2.000.
Lowjacker
с логотипом UCB выходной формат может быть получен как (print (word (form xcor 4 3) ",) (form ycor 4 3)). Но я не уверен, насколько простым будет анализ данных.
Питер Тейлор
@ Lowjacker Да, спасибо. На самом деле, ввод был неверным.
Олег Припин

Ответы:

7

Рубин 1.9, 175 171 162 153 130 120 117

l=0
gets.scan(/(\w+) (\d+)/){|d,n|l+=n.to_i*?i.to_c**%w[E NE N NW W SW S SE].index(d).quo(2)}
puts'%.3f, %.3f'%l.rect
Lowjacker
источник
?i.to_cможно сократить до 1.i-4 байтов.
MegaTom
3

Хаскелл (291)

import Text.Printf
d=sqrt(0.5)
f"N"n(x,y)=(x,y+n)
f"S"n(x,y)=(x,y-n)
f"E"n(x,y)=(x+n,y)
f"W"n(x,y)=(x-n,y)
f[a,b]n(x,y)=(s,t)where(s,_)=f[b](d*n)(x,y);(_,t)=f[a](d*n)(x,y)
p[]=(0,0)
p(a:b:c)=f a(read b::Float)$p c
s(a,b)=printf"%.3f, %.3f"a b
main=getLine>>=putStrLn.s.p.words.filter(/=',')
Мэринус
источник
Как насчет изменения определения f для использования шаблонных охранников? У них есть свойство nice, которое не требует разрыва строки и должно быть более кратким в этом случае. Также используйте интерактив.
FUZxxl
3

C99 (319 символов)

#define B ;break;
#include<math.h>
#include<stdio.h>
float x,y,w,z,j;int
main(void){int
k;char
c[3];while(scanf("%s%d,",c,&k)==2){j=k;w=1;switch(c[1]){case'E':w=3;default:w-=2;j=sqrt(k*k/2)B
case
0:w=z=0;}switch(*c){case'N':z=1
B
case'S':z=-1
B
case'E':w=1
B
default:w=-1;}x+=w*j;y+=z*j;}printf("%5.3f, %5.3f\n",x,y);}

вход в stdin , тестовый прогон на ideone :)

PMG
источник
3

Питон, 158 154 150 символов

p=0j
for s in raw_input().split(','):c,d=s.split();v=sum(dict(N=1j,E=1,S=-1j,W=-1)[x]for x in c);p+=v*int(d)/abs(v)
print'%.3f, %.3f'%(p.real,p.imag)
Кит Рэндалл
источник
На самом деле у вас 157 символов, а не 158.
Lowjacker
Я думаю, мне не нужно, но я обычно считаю завершающий перевод строки.
Кит Рэндалл,
[157] Строка 1: D=dict(N=1j,E=1,S=-1j,W=-1)[153] Строка 2: jдействительно нужна? [152] Строки 3-4: Если вы переключаетесь на Python 3, raw_inputinputи даже после того print, как вам нужно использовать круглые скобки , вы сохраняете 2 символа [150]
Олег Припин
1
@BlaXpirit: Спасибо за оптимизацию. J необходимо в строке 2 в случае , если все направления Е и В. в результате чего р должен быть сложными для .realи .imagк работе.
Кит Рэндалл,
1
В классе int есть и то, imagи другое real...
JBernardo
3

JavaScript, 179 164 170 168 158 156 153 символа

prompt(X=Y=0).replace(/(N|S)?(.)? (\d+)/g,
        function(m,y,x,n){
            n/=x&&y?Math.SQRT2:1
            Y+=y?y<'S'?n:-n:0
            X+=x?x<'W'?n:-n:0
        })
alert(X.toFixed(3)+', '+Y.toFixed(3))
  • 170: исправлена ​​ошибка точности
  • 168: заменено (E|W)в регулярном выражении(.)
  • 158: заменена повторяющаяся логика в функции с переменной d
  • 156: повторно используется nвместо новой переменнойd
  • 153: Лично я думаю, что это редактирование делает его в десять раз ужаснее, но на три символа короче. Он основан на нестандартном поведении, которое вы можете вызывать объекты RegExp как функции: /./g('string')это то же самое, что и /./g.exec('string'):

    for(p=prompt(X=Y=0),R=/(N|S)?(.)? (\d+)/g;[,y,x,n]=R(p)||0;X+=x?x<'W'?n:-n:0)n/=x&&y?Math.SQRT2:1,Y+=y?y<'S'?n:-n:0;alert(X.toFixed(3)+', '+Y.toFixed(3))

Кейси Чу
источник
1
К сожалению, q = 0,707 не удается ввести «NW 10» из-за ошибок округления; Я думаю, что вам нужно "q = Math.SQRT1_2", который добавляет 8 символов. С другой стороны, вы можете заменить "(E | W)?" с "(.)?" так как вы уже установили север / юг / ни один, и вход правильно сформирован, сохраняя 2 символа.
DocMax
Спасибо за регулярное выражение. Что касается точности, я использовал вместо этого SQRT2 и переключил умножение на деление.
Кейси Чу
2

Haskell, 199 символов

import Text.Printf
import Complex
i=0:+(1::Float)
e 'S'= -i
e d=i^mod(fromEnum d-1)4
g p(d:s:t)=g(p+(signum.sum.map e)d*(fst(reads s!!0):+0))t
g(x:+y)[]=printf"%.3f, %.3f"x y
main=interact$g 0.words
Хаммар
источник
1

Скала ( 367 , 332)

var (x,y,s)=(.0,.0,.7071);args.mkString(" ").split(",").foreach{m=>val a=m.trim.split(" ");var (n,u,v)=(a(1).toInt,.0,.0);a(0) match{case "N"=>v=1;case "S"=>v= -1;case "E"=>u=1;case "W"=>u= -1;case "NW"=>{u= -s;v=s};case "NE"=>{u=s;v=s};case "SW"=>{u= -s;v= -s};case "SE"=>{u=s;v= -s}};x += n*u;y += n*v};printf("%1.3f %1.3f\n",x,y)
Лалитх
источник
1

Ява (459) (445) (402) (382) (363) (352)

import java.util.*;class
M{public
static void main(String[]a){double
x=0,y=0;Scanner
s=new
Scanner(System.in);s.useDelimiter("\\W+");while(s.hasNext()){String
d=s.next();double
z=Math.sqrt(d.length());int
w=s.nextInt();y+=(d.contains("N")?w:d.contains("S")?-w:0)/z;x+=(d.contains("E")?w:d.contains("W")?-w:0)/z;}System.out.format("%1.3f %1.3f",x,y);}}

ввод ввода

trutheality
источник
1

PowerShell, 178

$input-split','|%{,@{N=0,1
NE=($s=.707106781186548),$s
E=1,0
SE=$s,-$s
S=0,-1
SW=-$s,-$s
W=-1,0
NW=-$s,$s}[($a=-split$_)[0]]*$a[1]}|%{$x+=$_[0]
$y+=$_[1]}
'{0:N3}, {1:N3}'-f$x,$y

Вероятно, это может привести к потере до 10 символов при уменьшении точности √2 / 2.

детеныш
источник
0

Groovy (215)

x=0.0;y=0.0;args.join(' ').split(', ').each{d=it.split(' ');c=d[0]==~/../?Math.sqrt(2):1;s=d[1] as int;a=['N':1,'S':-1,'E':1,'W':-1];m=d[0]=~/N|S/;y+=m?a[m[0]]*(s/c):0;m=d[0]=~/E|W/;x+=m?s/c*a[m[0]]:0};print "$x,$y"

читает ввод как аргументы программы. Пример:

groovy golf.groovy NW 10, SW 10, W 10
TheBigS
источник