接下来是什么?


15

给定一个用空格分隔的整数列表,您的任务是在序列中查找下一个整数。序列中的每个整数是应用单个数学运算(的结果+-*/)到前一整数,并且每个序列由这样的操作的可变数目(但不超过10个)组成。没有序列会比整数序列的长度长一半,因此您将使每个操作序列至少出现两次以进行确认。
输入将通过stdin(或prompt对于JavaScript解决方案)。

以下是一些说明性示例。

输入:

1 3 5 7 9 11

输出:

13

相当容易,这个。所有值均为先前值+2

输入:

1 3 2 4 3 5 4 6 5 7 6

输出:

8

+2然后按此顺序执行两个步骤-1

输入:

2 6 7 3 9 10 6 18 19 15 45 46

输出:

42

三个步骤- ,*3,。+1-4

测试用例

这里还有一些测试用例:

输入:

1024 512 256 128 64 32 16

输出:

8

输入:

1 3 9 8 24 72 71 213 639

输出:

638

输入:

1 2 3 4 5 2 3 4 5 6 3 4 5 6 7

输出:

4

输入:

1 2 4 1 3 9 5 8 32 27 28 56 53 55 165 161 164 656 651 652 1304

输出:

1301

我有一个待解决的Scala解决方案(42行),将在几天内发布。

这是代码高尔夫球-最短答案胜出。


是否保证划分准确?
彼得·泰勒

@彼得是的。每一步都会得到一个整数。
加雷斯

但是,如果步长为“ / 3”,是否可以保证余数始终为0?
彼得·泰勒

@Peter是的,余数将始终为0。–
Gareth

Answers:


12

Golfscript,203 138个字符

~]0{).2$.);[\(;]zip\/zip{.{~-}%.&({-}+\{;{.0={.~\%{.~%{;0}{~/{/}+}if}{~\/{*}+}if}{~!{;}*}if}%.&(\{;;0}/}{\;}if}%.{!}%1?)!{0}*}do\@)\,@%@=~

if比标准的Golfscript程序使用更多的s,并且它的操作是非常隐秘的,因此这里有一个注释版本(但除了添加空格和注释外,其他注释都没有保留):

~]0
# Loop over guesses L for the length of the sequence
{
    # [x0 ... xN] L-1
    ).
    # [x0 ... xN] L L
    2$.);[\(;]zip
    # [x0 ... xN] L L [[x0 x1][x1 x2]...[x{N-1} xN]]
    \/zip
    # [x0 ... xN] L [[[x0 x1][xL x{L+1}]...][[x1 x2][x{L+1} x{L+2}]...]...]
    {
        # [x0 ... xN] L [[xi x{i+1}][x{i+L} x{i+L+1}]...]
        # Is there an operation which takes the first element of each pair to the second?
        # Copy the pairs, work out each difference, and remove duplicates
        .{~-}%.&
        # Turn the first difference into an operation
        ({-}+\
        # If there was more than one unique difference, look for a multiplication
        {
            ;
            # [x0 ... xN] L [[xi x{i+1}][x{i+L} x{i+L+1}]...]
            # Do something similar, but working out multiplicative factors
            # Note that if we have [0 0] it could be multiplication by anything, so we remove it.
            # However, they can't all be [0 0] or we'd have only one unique difference
            {
                # [0     0   ] =>
                # [0     _   ] => 0     # a "false" value, because it can't possibly be multiplication
                # [a     a.b'] => {b'*}
                # [a'.b  b   ] => {a'/}
                # [_     _   ] => 0     # ditto
                # This is the obvious place to look for further improvements
                .0={.~\%{.~%{;0}{~/{/}+}if}{~\/{*}+}if}{~!{;}*}if
            }%.&
            # If we have one unique multiplication, even if it's false, keep it.
            # Otherwise discard them and replace them with false.
            (\{;;0}/
        }
        # There was only one unique difference. Discard the pairs
        {\;}if
        # operation - one of 0, {d+}, {m*}, {M/}
    }%
    # [x0 ... xN] L [op0 ... op{L-1}]
    # Is any of the operations false, indicating that we couldn't find a suitable operation?
    .{!}%1?
    # [x0 ... xN] L [op0 ... op{L-1}] idxFalse
    # If idxFalse is -1 then all the ops are ok, and we put a false to exit the loop
    # Otherwise one op failed, so we leave the array of ops, which is non-false, to be popped by the do.
    )!{0}*
}do
# [x0 ... xN] L [op0 ... op{L-1}]
\@)\,@%@=~
# op{(len(Xs)-1)%L} (Xs[-1])

我的原始提交内容为以下88个字符:

~]:x;2.{;).(2base(;{[{--}{@*\.!+/}]=}%[x.(;2$]zip\,<{~++}%x,.@[*x\]zip<{~~}%)\x(;=!}do\;

但是,这会尝试从每个操作的第一次出现开始计算操作,因此,如果操作是乘法或除法并且参数的第一轮为0,它将中断。


1
非常感谢您发布解释!我一直在寻找解剖的Golfscript程序,所以我可以尝试使它们更有意义。
migimaru 2011年

6

哈斯克尔,276 261 259 257 243个字符

这是我效率不高的解决方案。它适用于无界(和有界)整数。此解决方案恰好可以与非精确除法一起正常工作(例如:5 / 2 = 2

import Control.Monad
main=interact$show.n.q read.words
f=flip
q=map
_%0=0
x%y=div x y
s b=[1..]>>=q cycle.(f replicateM$[(+),(*),(%)]>>=f q[-b..b].f)
m l x s=[y!!l|y<-[scanl(f($))(x!!0)s],x==take l y]
n x=(s(maximum$q abs x)>>=m(length x)x)!!0

工作原理:我创建(可能)操作的所有可能序列。然后,我针对数字的输入序列进行测试,以查看生成的序列是否会创建输入。如果是这样,则返回序列中的下一个数字。该代码将始终返回从最短的操作序列中得出的答案。发生这种情况是因为操作顺序列表是按该顺序生成的。在决定纽带之间是任意的(但始终如一)。例如代码返回68表示序列2 4

取消高尔夫:

import Control.Monad

main :: IO ()
main = print . next . map read . words =<< getLine

opSequences :: Integral a => a -> [[a -> a]]
opSequences bound = [1 ..] >>= map cycle . (`replicateM` (ops >>= (`map` args) . flip))
  where
    ops = [(+), (*), \x y -> if y == 0 then 0 else div x y]
    args = [-bound .. bound]

match :: (MonadPlus m, Integral a) => [a] -> [a -> a] -> m a
match ns opSeq = guard (ns == take len ms) >> return (ms !! len)
  where
    n0 = ns !! 0
    len = length ns
    ms = scanl (flip ($)) n0 opSeq

next :: Integral a => [a] -> a
next ns = (opSequences bound >>= match ns) !! 0
  where
    bound = maximum $ map abs ns

如何处理非精确除法的好主意。
彼得·泰勒

这是一个偶然的副作用。寻找解决方案只是我最初关于如何解决此问题的想法,对我而言,这比计算答案更容易。
Thomas Eding

Control.Monad -> Monad可能吗 怎么样interact$show.n.q read.words
FUZxxl

6

Python中,333 366 ... 315 303 278 269 261 246个字符

n=map(float,raw_input().split())
y=len(n)-1
l=1
def O(f):
 p,q=n[f:y:l],n[f+1::l]
 for a,b in zip(p,q):
    for x in((b-a).__add__,(b/(a or 1)).__mul__):
     if map(x,p)==q:return x
while 1:
 if all(map(O,range(l))):print int(O(y%l)(n[y]));break
 l+=1

用第一对数字创建运算,然后在其他对上进行检查。存储所有操作,如果所有操作都成功,则将适当的操作应用于列表的最后一个元素。

编辑:通过邪恶测试:-)现在搜索所有位置的操作。


真好 通过了我所有的测试,但没有通过Peter Taylor的特别邪恶的测试:0 0 1 2 3 6 7 14
Gareth

0 0 0 0 1 0 0 0 0 1不输出0
Thomas Eding

@trinithis该输入不符合规范。操作顺序必须至少完全重复一次。
加雷斯

1
哦,这是一个有趣的改进:lambda x:x+b-a-> (b-a).__add__。不幸的是,这只是一个字符,我通过做这些事情对python有了很多了解。
笨拙的

1
制作l:隐含全球节省了大量pastie.org/2416407
懵懵懂懂

4

Python中,309 305 295 279个字符

处理所有原始测试用例,以及彼得·泰勒的陈旧测试用例0 0 1 2 3 6 7 14

l=map(int,raw_input().split())
i=v=0
while v<1:
 v=i=i+1
 for j in range(i):
    p=zip(l[j::i],l[j+1::i])
    d,r=set(q[1]-q[0]for q in p),set(q[1]*1./(q[0]or 1)for q in p if any(q))
    m,n=len(d)<2,len(r)<2
    v*=m+n
    if(len(l)-1)%i==j:s=l[-1]+d.pop()if m else int(l[-1]*r.pop())
print s

放开手脚,提供调试输出(对验证正确性非常有用):

nums = map(int,raw_input().split())
result = None

for i in range(1,len(nums)/2+1):
    print "-- %s --" % i
    valid = 1
    for j in range(i):
        pairs = zip(nums[j::i], nums[j+1::i])
        print pairs

        diffs = [pair[1] - pair[0] for pair in pairs]
        # this is the tough bit: (3, 6) -> 2, (0, 5) -> 5, (5, 0) -> 0, (0, 0) ignored
        ratios = [float(pair[1])/(pair[0] or 1) for pair in pairs if pair[0] != 0 or pair[1] != 0]

        if len(set(diffs))==1:
            print "  can be diff", diffs[0]
            if (len(nums) - 1) % i == j:
                result = nums[-1] + diffs.pop()
        elif len(set(ratios))==1:
            print "  can be ratio", ratios[0]
            if (len(nums) - 1) % i == j:
                result = int(nums[-1]*ratios.pop())
        else:
            print "** invalid period **"
            valid=0
    if valid and result is not None:
        break

print result

用法:

echo 0 0 1 2 3 6 7 14 | python whatcomesnext.py

我已经投票赞成,尽管严格来讲,输入应该通过stdin而不是命令参数。
加雷斯

实际上,这使我可以将程序缩短四个字符:)
笨拙的

几个字符,I = v = 0和当V == 0:
安特

@Ante谢谢,我以为您不能在python中做到这一点,因为赋值不是一种表达方式,但这对打高尔夫球很有帮助。文字标签作为第二级缩进也有帮助。
笨拙的

我不是Pythoner,但您似乎在某些表达式中将布尔值用作整数,但是在while测试中不能将整数v用作布尔值。那正确吗?如果是这样的话,那一定v<1是守卫。
彼得·泰勒

2

红宝石1.9 (437) (521) (447)(477)

适用于所有测试用例,包括“邪恶”的一个。我待会再打高尔夫球。

编辑:我意识到还有另一种情况,我没有正确处理-当继续需要使用“神秘”操作时。该序列2 0 0 -2 -4 -6最初返回0而不是-12。我现在已经解决了。

编辑:修复了几个边缘情况,并将代码减少到447。

编辑:gh。必须添加一些代码来处理其他“邪恶”序列,例如0 0 0 6 18 6 12

def v(c,q);t=*q[0];q.size.times{|i|f=c[z=i%k=c.size]
f=c[z]=(g=q[z+k])==0??_:((h=q[z+k+1])%g==0?"*(#{h/g})":"/(#{g/h})")if f==?_
t<<=f==?_?(a=q[i];b=q[i+1].nil?? 0:q[i+1];(a==0&&b==0)||(a!=0&&(b%a==0||a%b==0))?b:0):eval(t.last.to_s+f)}
*r,s=t
(p s;exit)if q==r
end
s=gets.split.map &:to_i
n=[[]]
((s.size-1)/2).times{|i|a,b=s[i,2]
j=["+(#{b-a})"]
j<<=?_ if a==0&&b==0
j<<="*#{b/a}"if a!=0&&b%a==0
j<<="/#{a/b}"if a*b!=0&&a%b==0
n=n.product(j).map(&:flatten)
n.map{|m|v(m*1,s)}}

1

斯卡拉

这是我想出的解决方案:

object Z extends App{var i=readLine.split(" ").map(_.toInt)
var s=i.size
var(o,v,f)=(new Array[Int](s),new Array[Double](s),1)
def d(p:Int,j:Array[Int]):Unit={if(p<=s/2){def t(){var a=new Array[Int](s+1)
a(0)=i(0)
for(l<-0 to s-1){o(l%(p+1))match{case 0=>a(l+1)=a(l)+v(l%(p+1)).toInt
case _=>a(l+1)=(a(l).toDouble*v(l%(p+1))).toInt}}
if(a.init.deep==i.deep&&f>0){f^=f
println(a.last)}}
o(p)=0 
v(p)=j(1)-j(0)
t
d(p+1,j.tail)
o(p)=1
v(p)=j(1).toDouble/j(0).toDouble
t
d(p+1,j.tail)}}
d(0,i)
i=i.tail
d(0,i)}

取消高尔夫:

object Sequence extends App
{
    var input=readLine.split(" ").map(_.toInt)
    var s=input.size
    var (ops,vals,flag)=(new Array[Int](s),new Array[Double](s),1)
    def doSeq(place:Int,ints:Array[Int]):Unit=
    {
        if(place<=s/2) 
        {
            def trysolution()
            {
                var a=new Array[Int](s+1)
                a(0)=input(0)
                for(loop<-0 to s-1)
                {
                    ops(loop%(place+1))match
                    {
                        case 0=>a(loop+1)=a(loop)+vals(loop%(place+1)).toInt
                        case _=>a(loop+1)=(a(loop).toDouble*vals(loop%(place+1))).toInt
                    }
                }
                if(a.init.deep==input.deep&&flag>0)
                {
                    flag^=flag
                    println(a.last)
                }
            }
            ops(place)=0
            vals(place)=ints(1)-ints(0)
            trysolution
            doSeq(place+1,ints.tail)
            ops(place)=1
            vals(place)=ints(1).toDouble/ints(0).toDouble
            trysolution
            doSeq(place+1,ints.tail)
        }
    }
    doSeq(0,input)
    input=input.tail
    doSeq(0,input)
}

如何调用它?echo "0 0 1 2 3 6 7 14" | scala Sequence使屏幕保持黑屏。
用户未知,

@user unknown scala Sequence,然后输入顺序,然后按Enter。
加雷斯

嗯,您在问题注释中写道,您没有解决这一特定问题-它可以与echo-管道一起解决上述问题。
用户未知

1

斯卡拉936

type O=Option[(Char,Int)]
type Q=(O,O)
type L=List[Q]
val N=None
def t(a:Int,b:Int):Q=if(a>b)(Some('-',a-b),(if(b!=0&&b*(a/b)==a)Some('/',a/b)else N))else
(Some('+',b-a),(if(a!=0&&a*(b/a)==b)Some('*',b/a)else N))
def w(a:Q,b:Q)=if(a._1==b._1&&a._2==b._2)a else
if(a._1==b._1)(a._1,N)else
if(a._2==b._2)(N,a._2)else(N,N)
def n(l:L):Q=l match{case Nil=>(N,N)
case x::Nil=>x
case x::y::Nil=>w(x,y)
case x::y::xs=>n(w(x,y)::xs)} 
def z(l:L,w:Int)=for(d<-1 to w)yield
n(l.drop(d-1).sliding(1,w).flatten.toList)
def h(s:L):Boolean=s.isEmpty||(s(0)!=(N,N))&& h(s.tail)
def j(s:L,i:Int=1):Int=if(h(z(s,i).toList))i else j(s,i+1)
def k(b:Int,o:Char,p:Int)=o match{case'+'=>b+p
case'-'=>b-p
case'*'=>b*p
case _=>b/p}
val e=getLine 
val i=e.split(" ").map(_.toInt).toList
val s=i.sliding(2,1).toList.map(l=>t(l(0),l(1)))
val H=n(s.drop(s.size%j(s)).sliding(1,j(s)).flatten.toList)
val c=H._1.getOrElse(H._2.get)
println (k(i(i.size-1),c._1,c._2))

松开

type O = Option[(Char, Int)]

def stepalize (a: Int, b: Int): (O, O) = (a > b) match {
   case true => (Some('-', a-b), (if (b!=0 && b * (a/b) == a) Some ('/', a/b) else None)) 
   case false=> (Some('+', b-a), (if (a!=0 && a * (b/a) == b) Some ('*', b/a) else None)) }

def same (a: (O, O), b: (O, O)) = {
  if (a._1 == b._1 && a._2 == b._2) a else
  if (a._1 == b._1) (a._1, None) else 
  if (a._2 == b._2) (None, a._2) else 
  (None, None)}

def intersection (lc: List[(O, O)]): (O, O) = lc match {
  case Nil => (None, None)
  case x :: Nil => x
  case x :: y :: Nil => same (x, y)
  case x :: y :: xs  => intersection (same (x, y) :: xs)} 

def seriallen (lc: List[(O, O)], w: Int= 1) =
  for (d <- 1 to w) yield 
    intersection (lc.drop (d-1).sliding (1, w).flatten.toList)

def hit (s: List[(O, O)]) : Boolean = s match {
  case Nil => true 
  case x :: xs => (x != (None, None)) && hit (xs)}

def idxHit (s: List[(O, O)], idx: Int = 1) : Int =
  if (hit (seriallen (s, idx).toList)) idx else 
    idxHit (s, idx+1)

def calc (base: Int, op: Char, param: Int) = op match {
  case '+' => base + param
  case '-' => base - param
  case '*' => base * param
  case _   => base / param}

def getOp (e: String) = {
  val i = e.split (" ").map (_.toInt).toList
  val s = i.sliding (2, 1).toList.map (l => stepalize (l(0), l(1)))
  val w = idxHit (s)
  val hit = intersection (s.drop (s.size % w).sliding (1, w).flatten.toList)
  val ci = hit._1.getOrElse (hit._2.get)
  val base = i(i.size - 1)
  println ("i: " + i + " w: " + w + " ci:" + ci + " " + calc (base, ci._1, ci._2))
}

val a="1 3 5 7 9 11"
val b="1 3 2 4 3 5 4 6 5 7 6"
val c="2 6 7 3 9 10 6 18 19 15 45 46"
val d="1024 512 256 128 64 32 16"
val e="1 3 9 8 24 72 71 213 639"
val f="1 2 3 4 5 2 3 4 5 6 3 4 5 6 7"
val g="1 2 4 1 3 9 5 8 32 27 28 56 53 55 165 161 164 656 651 652 1304"
val h="0 0 1 2 3 6 7 14"
val i="0 0 0 0 1 0 0 0 0 1 0"

List (a, b, c, d, e, f, g, h, i).map (getOp)

彼得·泰勒(Peter Taylor)惨败h,但我认为在合理的时间内无法治愈该程序。


如果您将-视为的特例+/的特例,是否有助于缩小*?我传递彼得·泰勒(和类似的东西)输入的方法是切掉序列中的第一个数字,然后再试一次。我还没来得及研究您的程序如何工作,还不知道对您的程序是否有帮助。
加雷斯

我想这会有所帮助,但仅适用于特殊情况。稍后包含null乘法的序列-1, 0, 0, 1, 2, 3, 6, 7, 14将需要不同的修复。
用户未知,
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.