从Rails里“借”来的Ruby代码统计程序

以前每做完一个项目,或者完成某个里程碑的时候都喜欢用代码统计工具统计下自己的代码情况,看看多少LOC,多少注释。现在开始用Ruby写程序了,忽然发现竟然找不到支持Ruby的代码统计工具,偏偏自己这次写的东西又与Rails无关,不能用里面的stats……脑子一转,反正有空,就研究下Rails的代码统计代码,整理出来,让它能够单独运行。

由于原来的代码不统计注释,所以我加了对注释和注释率的统计,单行注释和写在代码后的注释都算作一行。另外,把原来的各个Rails目录的统计去掉了,只统计代码和测试,那些controller之类的就不考虑了。基本对原来的程序基本没有什么大修改。

通过命令行方式直接调用,即:ruby code_statistics.rb [<源代码目录> <测试代码目录>]
程序里用以下代码做了简单的判断,默认这两个参数分别是src和test。

 程序的运行效果与Rails里的stats差不多,只是多了些注释方面的统计数据。

显示效果


 code_statistics.rb

class CodeStatistics
  def initialize(src, test)
    @pairs = [
      ["Codes",src + "/"],
      ["Tests",test + "/"]
    ].collect { |name, dir| [ name, "#{dir}" ] }.select { |name, dir| File.directory?(dir) }
    @statistics = calculate_statistics
    @total      = calculate_total if @pairs.length > 1
  end

  def to_s
    print_header
    @pairs.each { |pair| print_line(pair.first, @statistics[pair.first]) }
    print_splitter
 
    if @total
      print_line("Total", @total)
      print_splitter
    end

    print_code_test_stats
  end

  private
    def calculate_statistics
      @pairs.inject({}) { |stats, pair| stats[pair.first] = calculate_directory_statistics(pair.last); stats }
    end

    def calculate_directory_statistics(directory, pattern = /.*/.rb$/)
      stats = { "lines" => 0, "codelines" => 0, "classes" => 0, "methods" => 0, "comments" => 0}

      Dir.foreach(directory) do |file_name|
        if File.stat(directory + "/" + file_name).directory? and (/^/./ !~ file_name)
          newstats = calculate_directory_statistics(directory + "/" + file_name, pattern)
          stats.each { |k, v| stats[k] += newstats[k] }
        end

        next unless file_name =~ pattern

        f = File.open(directory + "/" + file_name)

        while line = f.gets
          stats["lines"]     += 1
          stats["classes"]   += 1 if line =~ /class [A-Z]/
          stats["methods"]   += 1 if line =~ /def [a-z]/
          stats["codelines"] += 1 unless line =~ /^/s*$/ || line =~ /^/s*#/
          stats["comments"]  += 1 if line =~ /^/s*#/ || line =~ /^.*#/
        end
      end

      stats
    end

    def calculate_total
      total = { "lines" => 0, "codelines" => 0, "classes" => 0, "methods" => 0, "comments" => 0 }
      @statistics.each_value { |pair| pair.each { |k, v| total[k] += v } }
      total
    end

    def calculate_code
      code_loc = 0
      @statistics.each { |k, v| code_loc += v['codelines'] unless k == "Tests" }
      code_loc
    end

    def calculate_tests
      test_loc = 0
      @statistics.each { |k, v| test_loc += v['codelines'] if k == "Tests" }
      test_loc
    end
   
    def calculate_comments
      comments = 0
      @statistics.each { |k, v| comments += v['comments']}
      comments
    end

    def print_header
      print_splitter
      puts "| Name  | Lines |   LOC | Classes | Methods | Comments | M/C | LOC/M | Cm/LOC |"
      print_splitter
    end

    def print_splitter
      puts "+-------+-------+-------+---------+---------+----------+-----+-------+--------|"
    end

    def print_line(name, statistics)
      m_over_c    = (statistics["methods"] / statistics["classes"])   rescue m_over_c = 0
      loc_over_m  = (statistics["codelines"] / statistics["methods"]) - 2 rescue loc_over_m = 0
      cm_over_loc = (statistics["comments"].to_f / statistics["codelines"]) rescue cm_over_loc = 0.0

      start = "| #{name.ljust(5)} "

      puts start +
           "| #{statistics["lines"].to_s.rjust(5)} " +
           "| #{statistics["codelines"].to_s.rjust(5)} " +
           "| #{statistics["classes"].to_s.rjust(7)} " +
           "| #{statistics["methods"].to_s.rjust(7)} " +
           "| #{statistics["comments"].to_s.rjust(8)} " +
           "| #{m_over_c.to_s.rjust(3)} " +
           "| #{loc_over_m.to_s.rjust(5)} " +
           "|   #{sprintf("%.2f", cm_over_loc.to_s)} |"
    end

    def print_code_test_stats
      code  = calculate_code
      tests = calculate_tests
      commnets = calculate_comments

      puts "  Code LOC: #{code}   Test LOC: #{tests}   Code:Test = 1:#{sprintf("%.1f", tests.to_f/code)}   LOC:Comments = 1:#{sprintf("%.1f", commnets.to_f/(code + tests))}"
      puts ""
    end
end

unless ARGV[0].nil? && ARGV[1].nil?
  CodeStatistics.new(ARGV[0], ARGV[1]).to_s
else
  CodeStatistics.new("src", "test").to_s
end

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值