1.tcl 1.9 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  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. if {[char_at {*}$forward] == {#}} {
  64. set forward [front_point $x $y [turn_right $direction]]
  65. return [list {*}$forward [turn_right $direction]]
  66. } else {
  67. return [list {*}$forward $direction]
  68. }
  69. }
  70. set state [list {*}$start_point up]
  71. set visited {}
  72. while {![out_of_range {*}[lrange $state 0 1]]} {
  73. dict set visited [lrange $state 0 1] {}
  74. set state [next_state {*}$state]
  75. }
  76. puts [dict size $visited]