Zaimplementuj narzędzie Dog Bash

10

dog to narzędzie wiersza polecenia, które pobiera dowolną liczbę argumentów, z których pierwszy to tekst do napisania, a pozostałe to dowolnie wiele plików.

dogNarzędzie zostanie podzielony tekst w równych porcjach tych plików. Jeśli pozostała część n, pierwsze npliki otrzymują dodatkowy bajt

dogjest przeciwieństwem catforallu jako takiego x, co należy przyjąć.

$> dog x a.txt b.txt ...
$> cat a.txt b.txt ...
x$>

Gdzie ...wskazuje dowolnie wiele plików.

Przykład (12 bajtów, 3 pliki, można podzielić równomiernie):

$> ./dog.py "Dogs vs Cats" a.txt b.txt c.txt
$> cat a.txt
Dogs$> cat b.txt
 vs $> cat c.txt
Cats$> cat a.txt b.txt c.txt
Dogs vs Cats$> 

Przykład z resztą (13 bajtów, 5 plików, reszta 3):

9$>./dog.py "0123456789abc" a.txt b.txt c.txt d.txt e.txt
$> cat a.txt
012$> cat b.txt
345$> cat c.txt
678$> cat d.txt
9a$> cat e.txt
bc$> cat a.txt b.txt c.txt d.txt e.txt
0123456789abc$>
Caridorc
źródło
Sugeruje się to, ale należy dwukrotnie sprawdzić: 1) Czy argumenty muszą pochodzić z wiersza poleceń? 2) Czy zawsze musimy wyprowadzać dane do plików?
Sp3000,
@ Sp3000 tak, do 1 i 2
Caridorc
1
@DigitalTrauma jest już odpowiedź, szkoda mi było unieważnić ją przez zmianę reguły
Caridorc
2
Ostatnio dowiedziałem się o dziwnie nazwanych narzędziach UNIX z tej strony (tac, dog, ...).
kirbyfan64sos
1
@ kirbyfan64sos i Caridorc: tacjest prawdziwy .
DLosc

Odpowiedzi:

4

Pyth - 12 bajtów

.wMC,cl.zz.z

Używa wbudowanej funkcji podziału, a następnie używa funkcji splat-map w funkcji zapisu. Nie działa online.

Maltysen
źródło
2

Python - 181 bajtów

import sys
a=sys.argv
l=len
d=a[2:]
s=a[1]
n,r=divmod(l(s),l(d))
p=0
for i in range(l(d)):
    with open(d[i],'w') as f:
        o=n+int(i<=n)
        f.write(s[p:p+o])
        p+=o
Zac Crites
źródło
1

PHP, 107 bajtów

Kod do gry w golfa:

for($i=1;++$i<$argc;fputs(fopen($argv[$i],w),substr($s=$argv[1],($i-2)*$l=ceil(strlen($s)/($argc-2)),$l)));

Szczegółowy kod:

$len = ceil(strlen($argv[1])/($argc - 2));
for ($i = 2; $i < $argc; $i ++) {
    $fh = fopen($argv[$i], 'w');
    fputs($fh, substr($argv[1], ($i - 2) * $len, $len));
    fclose($fh);          // omitted in the golfed version
}
aksjomat
źródło
0

Pure Bash: 97

s=$1;shift;for((l=${#s}/$#,m=${#s}-l*$#,i=1;i<=$#;p+=q,i++)){
printf "${s:p:q=i>m?l:l+1}">${!i};}

Jako funkcja: ( p=jest wymagany tylko przy drugim uruchomieniu)

dog() { p=
    s=$1;shift;for((l=${#s}/$#,m=${#s}-l*$#,i=1;i<=$#;p+=q,i++)){
    printf "${s:p:q=i>m?l:l+1}">${!i};}
}

Testy

$> rm *
$> dog "Dogs vs Cats" a.txt b.txt c.txt
$> ls -l
total 12
-rw-r--r-- 1 user user 4 May 13 22:09 a.txt
-rw-r--r-- 1 user user 4 May 13 22:09 b.txt
-rw-r--r-- 1 user user 4 May 13 22:09 c.txt
$> cat {a,b,c}.txt;echo
Dogs vs Cats
$> 

Wszystkie pliki mają długość 4 bajtów i są łączone w odpowiedniej kolejności, zawierają „Dogs vs. Cats” .

$> rm *
$> dog "$(printf "%s" {0..9} {a..c})" {a..e}.txt 
$> ls -l
total 20
-rw-r--r-- 1 user user 3 May 13 22:09 a.txt
-rw-r--r-- 1 user user 3 May 13 22:09 b.txt
-rw-r--r-- 1 user user 3 May 13 22:09 c.txt
-rw-r--r-- 1 user user 2 May 13 22:09 d.txt
-rw-r--r-- 1 user user 2 May 13 22:09 e.txt
$> cat *;echo
0123456789abc
$> 

Pliki Firsts mają długość 3 bajtów, a ostatnie tylko 2, połączone w kolejności alfabetycznej, zawierają „0123456789abc” .

Objaśnienie (niestosowanie golfa):

Jeśli trafisz: declare -f dog, odpowie:

$> declare -f dog
dog () 
{ 
    p=;
    s=$1;
    shift;
    for ((l=${#s}/$#,m=${#s}-l*$#,i=1; i<=$#; p+=q,i++))
    do
        printf "${s:p:q=i>m?l:l+1}" > ${!i};
    done
}

Można to napisać:

dog2 () 
{ 
    position=0;
    string=$1;
    shift;
    partLen=$((${#string}/$#));
    oneMore=$((${#string}-partLen*$#));
    for ((i=1; i<=$#; i++))
    do
        if ((i<=oneMore)); then
            partQuant=$((partLen+1));
        else
            partQuant=$partLen;
        fi;
        printf "${string:position:partQuant}" > ${!i};
        ((position+=partQuant));
    done
}
F. Hauri
źródło
0

Ruby, 93 87 bajtów

Pełny program przy użyciu argumentów wiersza poleceń.

Gdybym mógł użyć s.slice!do mutowania łańcucha, zrobiłbym to zamiast go używać s[c..-1], ale Ruby nie pozwala ci mutować łańcuchów z argv bez uprzedniego skopiowania ich

s,*t=$*
d,r=s.size.divmod t.size
t.map{|e|open(e,?w)<<s[0,c=(0>r-=1)?d:d+1];s=s[c..-1]}
Wartość tuszu
źródło