Implementare l'utilità dog bash


10

dog è un'utilità della riga di comando che contiene un numero arbitrario di argomenti, il primo dei quali è il testo da scrivere e gli altri sono arbitrariamente molti file.

L' dogutilità suddividerà il testo in parti uguali su questi file. Se è presente un resto n, i primi nfile ottengono un byte aggiuntivo

dogè l'opposto di cat, in quanto tale, xdovrebbe valere quanto segue.

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

Dove ...indica arbitrariamente molti file.

Un esempio (12 byte, 3 file, possono essere divisi equamente):

$> ./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$> 

Un esempio con resto (13 byte, 5 file, resto 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$>

È implicito, ma solo per ricontrollare: 1) Gli argomenti devono entrare dalla riga di comando? 2) Dobbiamo sempre eseguire l'output su file?
Sp3000,

@ Sp3000 sì, a 1 e 2
Caridorc,

1
@DigitalTrauma c'è già una risposta, mi sentirei male per invalidarla con un cambio di regola
Caridorc,

2
Recentemente ho appreso alcune utility UNIX dal nome strano da questo sito (tac, cane, ...).
Kirbyfan64sos,

1
@ kirbyfan64sos e Caridorc: tacè reale .
DLosc,

Risposte:


4

Pyth - 12 byte

.wMC,cl.zz.z

Utilizza la funzione split integrata e quindi utilizza la mappa splat sulla funzione di scrittura. Non funziona online.


2

Python - 181 byte

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

1

PHP, 107 byte

Il codice golf:

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

Il codice dettagliato:

$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
}

0

Bash puro: 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};}

Come funzione: ( p=è richiesto solo per la seconda corsa)

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};}
}

test

$> 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
$> 

Tutti i file sono di 4 byte e concatenati nell'ordine giusto, contengono "Cani contro gatti" .

$> 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
$> 

I primi file sono 3 byte len e durano solo 2, concatenati in ordine alfabetico, contengono "0123456789abc" .

Spiegazione (ungolfing):

Se si preme: declare -f dog, risponderà:

$> 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
}

Questo potrebbe essere scritto:

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
}

0

Rubino, 93 87 byte

Programma completo che utilizza argomenti della riga di comando.

Se potessi usare s.slice!per mutare la stringa, lo farei invece di usarlo s[c..-1], ma Ruby non ti consente di mutare le stringhe da argv senza duplicarle prima

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]}
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.