2.tcl 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. #!/usr/bin/env tclsh
  2. set fp [open input r]
  3. set map ""
  4. while {1} {
  5. set ret [gets $fp line]
  6. if {$ret == -1} {
  7. break
  8. }
  9. if {[string length $line] == 0} {
  10. break
  11. }
  12. lappend map $line
  13. }
  14. close $fp
  15. proc char_at {x y} {
  16. set line [lindex $::map $y]
  17. return [string index $line $x]
  18. }
  19. set height [llength $map]
  20. set width [string length [lindex $map 0]]
  21. proc find_start {} {
  22. for {set i 0} {$i < $::width} {incr i} {
  23. for {set j 0} {$j < $::height} {incr j} {
  24. if {[char_at $i $j] == "^"} {
  25. return "$i $j"
  26. }
  27. }
  28. }
  29. }
  30. set start_point [find_start]
  31. proc front_point {x y direction} {
  32. if {$direction == "up"} {
  33. return [list $x [expr {$y - 1}]]
  34. } elseif {$direction == "down"} {
  35. return [list $x [expr {$y + 1}]]
  36. } elseif {$direction == "left"} {
  37. return [list [expr {$x - 1}] $y]
  38. } elseif {$direction == "right"} {
  39. return [list [expr {$x + 1}] $y]
  40. }
  41. }
  42. proc out_of_range {x y} {
  43. return [expr {
  44. ($x < 0) || ($x >= $::width) || ($y < 0) || ($y >= $::height)
  45. }]
  46. }
  47. proc turn_right {direction} {
  48. if {$direction == {up}} {
  49. return right
  50. } elseif {$direction == {down}} {
  51. return left
  52. } elseif {$direction == {right}} {
  53. return down
  54. } else {
  55. return up
  56. }
  57. }
  58. proc next_state {x y direction} {
  59. set forward [front_point $x $y $direction]
  60. if {[out_of_range {*}$forward]} {
  61. return [list {*}$forward $direction]
  62. }
  63. while {[char_at {*}$forward] == {#}} {
  64. set forward [front_point $x $y [turn_right $direction]]
  65. set direction [turn_right $direction]
  66. }
  67. return [list {*}$forward $direction]
  68. }
  69. proc will_loop {} {
  70. global start_point
  71. set state [list {*}$start_point up]
  72. set past_states {}
  73. while {1} {
  74. if {[out_of_range {*}[lrange $state 0 1]]} {
  75. return 0
  76. }
  77. if {[dict exists $past_states $state]} {
  78. return 1
  79. }
  80. dict set past_states $state { }
  81. set state [next_state {*}$state]
  82. }
  83. }
  84. proc set_obstruction {x y} {
  85. global map
  86. set orig_line [lindex $map $y]
  87. lset map $y [string replace $orig_line $x $x "#"]
  88. }
  89. proc clear_obstruction {x y} {
  90. global map
  91. set orig_line [lindex $map $y]
  92. lset map $y [string replace $orig_line $x $x "."]
  93. }
  94. set count 0
  95. for {set i 0} {$i < $width} {incr i} {
  96. for {set j 0} {$j < $height} {incr j} {
  97. if {[char_at $i $j] == "^"} {
  98. continue
  99. }
  100. if {[char_at $i $j] == "#"} {
  101. continue
  102. }
  103. set_obstruction $i $j
  104. if {[will_loop]} {
  105. set count [expr {$count + 1}]
  106. }
  107. clear_obstruction $i $j
  108. }
  109. }
  110. puts $count