結果

問題 No.875 Range Mindex Query
ユーザー universatouniversato
提出日時 2020-05-29 18:36:22
言語 Crystal
(1.11.2)
結果
AC  
実行時間 200 ms / 2,000 ms
コード長 2,447 bytes
コンパイル時間 11,384 ms
コンパイル使用メモリ 296,196 KB
実行使用メモリ 14,932 KB
最終ジャッジ日時 2024-06-30 20:11:39
合計ジャッジ時間 13,749 ms
ジャッジサーバーID
(参考情報)
judge5 / judge2
このコードへのチャレンジ
(要ログイン)

テストケース

テストケース表示
入力 結果 実行時間
実行使用メモリ
testcase_00 AC 2 ms
6,816 KB
testcase_01 AC 3 ms
6,816 KB
testcase_02 AC 3 ms
6,944 KB
testcase_03 AC 2 ms
6,940 KB
testcase_04 AC 2 ms
6,944 KB
testcase_05 AC 2 ms
6,944 KB
testcase_06 AC 2 ms
6,940 KB
testcase_07 AC 3 ms
6,944 KB
testcase_08 AC 2 ms
6,944 KB
testcase_09 AC 2 ms
6,944 KB
testcase_10 AC 3 ms
6,940 KB
testcase_11 AC 139 ms
14,612 KB
testcase_12 AC 115 ms
10,368 KB
testcase_13 AC 99 ms
13,716 KB
testcase_14 AC 96 ms
14,932 KB
testcase_15 AC 131 ms
13,712 KB
testcase_16 AC 189 ms
11,840 KB
testcase_17 AC 200 ms
12,336 KB
testcase_18 AC 195 ms
13,896 KB
権限があれば一括ダウンロードができます

ソースコード

diff #

class SegmentTree

  @n : (Int64|Int32)
  @n_1 : (Int64|Int32)
  @nodes : Array(Int64)
  @identity_element : Int64

  def initialize(n, form)
    
    @identity_element =  0_i64
    @func = Proc(Int64, Int64, Int64).new{|x, y| x + y }
    case form
    when "gcd"
      @identity_element = 0_i64
      @func = Proc(Int64, Int64, Int64).new{|x, y| x.gcd y }
    when "min"
      @identity_element =  (1_i64 << 60) - 1
      @func = Proc(Int64, Int64, Int64).new{|x, y| x < y ? x : y }
    when "or"
    #if form == "or"
      @identity_element =  0_i64
      @func = Proc(Int64, Int64, Int64).new{|x, y| x | y }
    end
    
    
    @n = 1
    while @n < n
      @n *= 2
    end
    @n_1 = @n - 1
    @nodes = Array(Int64).new(@n*2-1, @identity_element)
  end

  def all_update
    @n_1.downto(1){ |i| @nodes[i-1] = @func.call(@nodes[2*i-1], @nodes[2*i]) }
  end

  def update(i, val)
    i += @n - 1
    @nodes[i] = val
    while i > 0
      i = ( i - 1 ) >> 1
      @nodes[i] = @func.call(@nodes[2*i+1], @nodes[2*i+2])
    end
  end

  def [](i)
    @nodes[@n_1 + i]
  end

  def []=(i, val)
    @nodes[@n_1 + i] = val
  end
 
  def get(l, r)
    l += @n - 1
    r += @n - 1
    lres, rres = @identity_element, @identity_element
    while l < r
      if 0 == ( l & 1 )
        lres = @func.call(@nodes[l], lres)
        l += 1
      end
      if 0 == ( r & 1 )
        r -= 1
        rres = @func.call(@nodes[r], rres)
      end
      l >>= 1
      r >>= 1
    end
    return @func.call(lres, rres)
  end
end

class Array
  def to_st(form)
    st = SegmentTree.new(size, form)
    each_with_index{ |t, i| st[i] = t }
    st.all_update
    st
  end
end

# yukicoder
n, q = gets.to_s.split.map{|t| t.to_i64 }
a    = gets.to_s.split.map{|t| t.to_i64 }

idx = Array.new(n+1){ 0_i64 }
a.each_with_index{ |t, i| idx[t] = i.to_i64 }

st = a.to_st("min")

q.times do
  x, l, r = gets.to_s.split.map{|t| t.to_i64 }
  l -= 1
  r -= 1

  if x == 1
    nl = st[l]
    nr = st[r]
    idx[nl] = r
    idx[nr] = l
    st.update(l, nr)
    st.update(r, nl)
  else
    puts (idx[st.get(l, r+1)] + 1)
  end
end

#n = gets.to_s.to_i
#s = gets.to_s.chomp
#q = gets.to_s.to_i
# 
## p "a".ord #=> 97
#st = s.chars.map{|c| 1_i64 << (c-'a') }.to_st("or")
#
#q.times do
# 
#  t, x, y = gets.to_s.split
# 
# if t == "1"
#    i, c = x.to_i-1, y[0].ord
#    st.update(i, 1_i64 << (c-97))
#  else
#    l, r = x.to_i-1, y.to_i-1
#    puts st.get(l, r+1).popcount
#  end
#end
0