<>=~/-\d+/;for$y(@a=-$'..$'){print+(map$_|$y?!($t=8*($y>0)+atan2(-$y,$_)/atan2 1,1)&-$&/45==8|$t>=$`/45&$t<=-$&/45?qw(- / | \\)[$t%4]:$":o,@a),$/}
$ echo -n '<>=~/-\d+/;for$y(@a=-$'"'"'..$'"'"'){say map$_|$y?!($t=8*($y>0)+atan2(-$y,$_)/atan2 1,1)&-$&/45==8|$t>=$`/45&$t<=-$&/45?qw(- / | \\)[$t%4]:$":o,@a}' | wc -c
139
$ perl -E '<>=~/-\d+/;for$y(@a=-$'"'"'..$'"'"'){say map$_|$y?!($t=8*($y>0)+atan2(-$y,$_)/atan2 1,1)&-$&/45==8|$t>=$`/45&$t<=-$&/45?qw(- / | \\)[$t%4]:$":o,@a}'
<>=~/-\d+/;for$y(@a=-$'..$'){print$/,map$_|$y?!($t=8*($y>0)+atan2(-$y,$_)/atan2 1,1)&-$&/45==8|$t>=$`/45&$t<=-$&/45?qw(- / | \\)[$t%4]:$":o,@a}
Оригинальная версия прокомментирована:
$_=<>;m/(\d+)-(\d+) (\d+)/;$e=$1/45;$f=$2/45; # parse angles and radius, angles are 0-8
for$y(-$3..$3){ # loop for each row and col
for$x(-$3..$3){
$t=atan2(-$y,$x)/atan2 1,1; # angle of this point
$t+=8if($t<0); # normalize negative angles
@w=split//,"-/|\\"x2; # array of ASCII symbols for enclosing lines
$s.=!$x&&!$y?"o":$t==$e||$t==$f?$w[$t]:$t>$e&&$t<$f?"x":$";
# if it's origin -> "o", if it's enclosing line, get symbol from array
# if it's between enclosing angles "x", otherwise space
}
$s.=$/;
}
print$s;
EDIT 1: Вложенные операторы sub, relational и equality возвращают 0 или 1.
EDIT 2: Добавлена версия с комментариями.
РЕДАКТИРОВАНИЕ 3: Исправлена заключающая линия на 360º. Количество символов значительно увеличилось.
EDIT 4: Добавлена более короткая версия, нарушающая правила.
EDIT 5: Более разумное исправление для ограждающей линии на 360º. Также используйте число в качестве заливки. Обе вещи были очевидны. Meh, я должен больше спать :/
EDIT 6: Удален ненужный m
из оператора match. Удалены некоторые точки с запятой.
EDIT 7: Более умный regexp. Меньше 200 символов!
EDIT 8: Много мелких улучшений:
split
строки -> qw (3 char)
EDIT 9: Небольшая перестановка в условных операторах экономит 2 символа.
EDIT 10: Использование голых слов для символов.
EDIT 11: Переместил print внутрь цикла, вдохновившись ответом Lowjacker'а.
EDIT 12: Добавлена версия с использованием say
.
EDIT 13: Повторное использование угловых символов для символа заполнения, как это сделано в ответе Gwell. Хотя вывод не такой красивый, как у Gwell, это потребует 5 дополнительных символов :) Кроме того, оператору ... не нужны круглые скобки.
EDIT 14: Применять regex непосредственно к <>. Назначьте оператор range переменной, как предложил Адриан в ответе bta. Добавьте версию без последней новой строки. Обновленная say
версия.
EDIT 15: Больше инлайнинга. map{block}@a -> map expr,@a.
Немного злоупотребляет предложением непробельного символа
для создания яркого изображения и, что более важно, для сохранения штрихов.
m=math i=io.read():gmatch("%d+")a=i()/45 b=i()/45 r=i()for y=r,-r,-1 do for x=-r,r do c=m.atan2(y,x)/m.pi*4 c=c<0 and c+8 or c k=1+m.modf(c+.5)io.write(x==0 and y==0 and'o'or c>=a and c<=b and('-/|\\-/|\\-'):sub(k,k)or c==0 and b==8 and'-'or' ')end print()end
Ввод: 45-360 4
\\\|||/// \\\|||// \\\\|// --\\|/ ----o---- --//|\\-- ////|\\\\ ///|||\\\ ///|||\\\
Ввод: 15-75 8
|///// |////// |////// |////// /////// |//////- ////--- //- o
Gnu bc из-за read ()
, else
и логических операторов.
scale=A
a=read()/45
b=read()/45
c=read()
for(y=c;y>=-c;y--){for(x=-c;x<=c;x++){if(x==0)if(y<0)t=-2else t=2else if(x>0)t=a(y/x)/a(1)else if(y<0)t=a(y/x)/a(1)-4else t=a(y/x)/a(1)+4
if(y<0)t+=8
if(x||y)if(t==a||t==b||t==b-8){scale=0;u=(t%4);scale=A;if(u==0)"-";if(u==1)"/";if(u==2)"|";if(u==3)"\"}else if(t>a&&t<b)"x"else" "else"o"};"
"}
quit
Для работы требуется Ruby 1.9
s,e,r=gets.scan(/\d+/).map &:to_i;s/=45;e/=45;G=-r..r;G.map{|y|G.map{|x|a=Math.atan2(-y,x)/Math::PI*4%8;print x|y!=0?a==s||a==e%8?'-/|\\'[a%4]:a<s||a>e ?' ':8:?o};puts}
Версия для чтения:
start, _end, radius = gets.scan(/\d+/).map &:to_i
start /= 45
_end /= 45
(-radius..radius).each {|y|
(-radius..radius).each {|x|
angle = Math.atan2(-y, x)/Math::PI * 4 % 8
print x|y != 0 ? angle==start || angle==_end%8 ? '-/|\\'[angle%4] : angle<start || angle>_end ? ' ' : 8 : ?o
}
puts
}
input ''; [wxr] = strread (ans, '% d-% d% d'); l = '- / | \ - / | \ -'; [ XY] = сетка (-r: r); T = atan2 (-Y, X) / pi * 180; T = T + (T <= 0) * 360; T (T> w & T
%%# Get the string variable (enclose in quotes, e.g. '45-315 4')
input ''
%%# Extract angles and length
[w x r]=strread(ans,'%d-%d%d');
%%# Store characters
l='-/|\-/|\-';
%%# Create the grid
[X Y]=meshgrid(-r:r);
%%# Compute the angles in degrees
T=atan2(-Y,X)/pi*180;
%%# Get all the angles
T=T+(T<=0)*360;
%# Negative numbers indicate valid characters
%%# Add the characters
T(T>w&T<x)=-42;
T(T==w)=-l(1+w/45);
T(T==x)=-l(1+x/45);
%%# Add the origin
T(r+1,r+1)=-'o';
%%# Display
char(-T)
x,y,r=gets.scan(/\d+/).map{|z|z.to_i};s=(-r..r);s.each{|a|s.each{|b|g=Math::atan2(-a,b)/Math::PI*180/1%360;print a|b==0?'o':g==x||g==y%360?'-/|\\'[g/45%4].chr: (x..y)===g ?'*':' '};puts}
Более приятная версия:
x, y, r = gets.scan(/\d+/).map{|z| z.to_i}
s = (-r..r)
s.each {|a|
s.each {|b|
g = (((Math::atan2(-a,b) / Math::PI) * 180) / 1) % 360
print ((a | b) == 0) ? 'o' :
(g == x || g == (y % 360)) ? '-/|\\'[(g / 45) % 4].chr :
((x..y) === g) ? '*' : ' '
}
puts
}
Я уверен, что кто-то, кто спал больше, чем я, может сжать это больше...
Правка 1: Заменил if
операторы во внутреннем цикле на вложенные ? :
оператор
Правка 2: Сохранение диапазона в промежуточную переменную (спасибо Adrian), использование stdin вместо CLI params (спасибо за разъяснение Jon), устранение массива в пользу прямого вывода, исправлена ошибка, когда угол окончания 360 не отображал линию, удалены ненужные круглые скобки, использование деления для округления вместо .round
, использование modulo вместо условного add
Поскольку было бы нечестно ставить задачу, которую я не смог бы решить сам, вот решение, использующее подстановку строк вместо тригонометрических функций, и активно использующее способность вашего дружелюбного соседа Perl рассматривать простые слова как строки. Оно обязательно будет немного длинным, но, возможно, интересным ради уникальности:
($x,$y,$r)=split/\D/,<>;for(0..$r-1){$t=$r-1-$_;
$a.=L x$_.D.K x$t.C.J x$t.B.I x$_."\n";
$b.=M x$t.F.N x$_.G.O x$_.H.P x$t."\n"}
$_=$a.E x$r.o.A x$r."\n".$b;$x/=45;$y/=45;$S=' ';
sub A{$v=$_[0];$x==$v||$y==$v?$_[1]:$x<$v&&$y>$v?x:$S}
sub B{$x<=$_[0]&&$y>$_[0]?x:$S}
@a=!$x||$y==8?'-':$S;
push@a,map{A$_,'\\'.qw(- / | \\)[$_%4]}1..7;
push@a,!$x?x:$S,map{B$_}1..7;
eval"y/A-P/".(join'',@a)."/";print
Все новые строки необязательны. Все довольно просто:
$a
) и нижнюю ($b
) части шаблона. $_
). sub A
, чтобы получить символ заливки для угла. sub B
, чтобы получить символ заливки для области. @a
) символов подстановки, используя A
и B
. Созданный формат выглядит следующим образом, для R
= 4:
DKKKCJJJB LDKKCJJBI LLDKCJBII LLLDCBIII EEEEoAAAA MMMFGHPPP MMFNGOHPP MFNNGOOHP FNNNGOOOH
Где A-H
обозначают углы, а I-P
- области.
(Признаться, это можно было бы сделать и дальше. Операции над @a
дали неверный результат, когда были записаны в виде одного списка, предположительно это связано с тем, как map
играет с $_
)
Начиная с ответа Jacob'а и вдохновленный gwell'ом, использующим любой не пробельный символ для заполнения дуги, я получил следующее решение:
[w x r]=strread(input('','s'),'%d-%d%d');
l='o -/|\-/|\-';
X=meshgrid(-r:r);
T=atan2(-X',X)*180/pi;
T=T+(T<=-~w)*360;
T(T>x|T<w)=-1;
T(r+1,r+1)=-90;
disp(l(fix(3+T/45)))
И немного тестового вывода:
>> arc
0-135 4
\||||////
\|||///-
\||//--
\|/---
o----
Я мог бы сократить его до 156 символов, удалив вызов disp
, но это добавило бы дополнительный ans =
перед выводом (что может нарушить правила форматирования вывода).
Тем не менее, мне кажется, что есть способы уменьшить это еще больше. ;)
Вне конкуренции, потому что графика слишком совершенна :)
f[x_-y_ z_]:=Graphics@Table[
{EdgeForm@Red,Disk[{0,0},r,{x °,y °}],{r,z,1,-1}]
SetAttributes[f,HoldAll]
Вызвать с f[30-70 5]
Result
alt text http://a.imageshack.us/img80/4294/angulosgolf.png
alt text http://a.imageshack.us/img59/7892/angulos2.png
The SetAttributes[f, HoldAll];
необходим, потому что входные данные
f[a-b c]
иначе интерпретируются как
f[(a-b*c)]
using System;class P{static void Main(){var s=Console.ReadLine().Split(' ');
var d=s[0].Split('-');int l=s[1][0]-48,x,y,r,a=int.Parse(d[0]),b=int.Parse(d[1]);
for(y=l;y>=-l;y--)for(x=-l;x<=l;)Console.Write((x==0&&y==0?'o':a<=(r=((int)
(Math.Atan2(y,x)*57.3)+360)%360)&&r<b||r==b%360?
@"-/|\"[r/45%4]:' ')+(x++==l?"\n":""));}}
Новые строки не имеют значения.
Пример ввода / вывода
45-180 8 \||||||||//////// \\|||||||/////// \\\||||||////// \\\\|||||///// \\\\\||||//// \\\\\\|||/// \\\\\\\||// \\\\\\\\|/ --------o
135-360 5 \ \\ \\\ \\\\ \\\\\ -----o----- ----/|\\\\\ ---//||\\\\ --///|||\\\ -////||||\\ /////|||||\
class A{public static void main(String[]a){String[]b=a[0].split("-");int e=new Integer(b[1]),r=new Integer(a[1]),g,x,y=r;for(;y>=-r;y--)for(x=-r;x<=r;)System.out.print((x==0&y==0?'o':new Integer(b[0])<=(g=((int)(Math.atan2(y,x)*57.3)+360)%360)&g<e|g==e%360?"-/|\\".charAt(g/45%4):' ')+(x++<r?"":"\n"));}}
Более читаемая версия:
class A{
public static void main(String[]a){
String[]b=a[0].split("-");
int e=new Integer(b[1]),r=new Integer(a[1]),g,x,y=r;
for(;y>=-r;y--)for(x=-r;x<=r;)System.out.print((
x==0&y==0
?'o'
:new Integer(b[0])<=(g=((int)(Math.atan2(y,x)*57.3)+360)%360)&g<e|g==e%360
?"-/|\\".charAt(g/45%4)
:' '
)+(x++<r?"":"\n"));
}
}
В нем не используются тригонометрические функции (как в исходной версии perl), поэтому он довольно "раздут". Так или иначе, вот мой первый код-гольф:
#define V(r) (4*r*r+6*r+3)
#define F for(i=0;i<r;i++)
#define C ;break;case
#define U p-=2*r+2,
#define D p+=2*r+2,
#define R *++p=
#define L *--p=
#define H *p='|';
#define E else if
#define G(a) for(j=0;j<V(r)-1;j++)if(f[j]==i+'0')f[j]=a;
#define O(i) for(i=0;i<2*r+1;i++){
main(int i,char**v){char*p,f[V(8)];
int j,m,e,s,x,y,r;p=*++v;x=atoi(p);while(*p!=45)p++;
char*h="0123";y=atoi(p+1);r=atoi(*++v);
for(p=f+2*r+1;p<f+V(r);p+=2*r+2)*p=10;
*(p-2*r-2)=0;x=x?x/45:x;y/=45;s=0;e=2*r;m=r;p=f;O(i)O(j)
if(j>e)*p=h[0];E(j>m)*p=h[1];E(j>s)*p=h[2];else*p=h[3];p++;}
if(i+1==r){h="7654";m--;e--;}E(i==r){s--;}E(i>r){s--;e++;}
else{s++;e--;}p++;}for(p=f+V(r)/2-1,i=0;i<r;i++)*++p=48;
for(i=0;i<8;i++)if(i>=x&&i<y){G(64);}else G(32);
y=y==8?0:y;q:p=f+V(r)/2-1;*p='o';switch(x){
C 0:F R 45 C 1:F U R 47 C 2:F U H C 3:F U L 92
C 4:F L 45 C 5:F D L 47 C 6:F D H C 7:F D R 92;}
if(y!=8){x=y;y=8;goto q;}puts(f);}
Кроме того, #define
выглядят довольно уродливо, но они экономят около 200 байт, поэтому я все равно их оставил. Он соответствует ANSI C89/C90 и компилируется с очень небольшим количеством предупреждений (два о atoi
и puts
и два об искаженной форме main
).