1.tcl 1.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061
  1. set fp [open "input" r]
  2. set line {}
  3. while {[gets $fp line] >= 0} {
  4. lappend lines $line
  5. }
  6. close $fp
  7. set height [llength $lines]
  8. set width [string length [lindex $lines 0]]
  9. proc out_of_range {x y} {
  10. global height width
  11. if {$x < 0 || $x >= $width} {
  12. return 1
  13. }
  14. if {$y < 0 || $y >= $height} {
  15. return 1
  16. }
  17. return 0
  18. }
  19. proc char_at {x y} {
  20. global lines
  21. if {[out_of_range $x $y]} {
  22. return -1
  23. }
  24. return [string index [lindex $lines $y] $x]
  25. }
  26. set path_cache {}
  27. proc find_path {x y expect} {
  28. global path_cache
  29. if {[out_of_range $x $y]} {
  30. return 0
  31. }
  32. if {$expect != [char_at $x $y]} {
  33. return 0
  34. }
  35. if {$expect == 9} {
  36. dict set path_cache "$x,$y" {}
  37. return 1
  38. }
  39. set next [expr {$expect + 1}]
  40. set ret [expr {
  41. [find_path [expr {$x + 1}] $y $next]
  42. + [find_path [expr {$x - 1}] $y $next]
  43. + [find_path $x [expr {$y + 1}] $next]
  44. + [find_path $x [expr {$y - 1}] $next]
  45. }]
  46. }
  47. set sum 0
  48. for {set x 0} {$x < $width} {incr x} {
  49. for {set y 0} {$y < $height} {incr y} {
  50. set path_cache {}
  51. find_path $x $y 0
  52. incr sum [dict size $path_cache]
  53. }
  54. }
  55. puts $sum