根据进程名获取进程ID

include Sys
require 'win32ole'
require 'socket'
require 'date'

# The Sys module serves as a namespace only
module Sys
   
   # The ProcTable class encapsulates process table information
   class ProcTable
      
      # There is no constructor
      private_class_method :new
      
      # Error typically raised if one of the Sys::ProcTable methods fails
      class Error < StandardError; end
      
      # The version of the sys-proctable library
      VERSION = '0.9.0'
      
      # The comm field corresponds to the 'name' field.  The 'cmdline' field
      # is the CommandLine attribute on Windows XP or later, or the 
      # 'executable_path' field on Windows 2000 or earlier.
      #
      @fields = %w/
         caption
         cmdline
         comm
         creation_class_name
         creation_date
         cs_creation_class_name
         cs_name
         description
         executable_path
         execution_state
         handle
         handle_count
         install_date
         kernel_mode_time
         maximum_working_set_size
         minimum_working_set_size
         name
         os_creation_class_name
         os_name
         other_operation_count
         other_transfer_count
         page_faults
         page_file_usage
         ppid
         peak_page_file_usage
         peak_virtual_size
         peak_working_set_size
         priority
         private_page_count
         pid
         quota_non_paged_pool_usage
         quota_paged_pool_usage
         quota_peak_non_paged_pool_usage
         quota_peak_paged_pool_usage
         read_operation_count
         read_transfer_count
         session_id
         status
         termination_date
         thread_count
         user_mode_time
         virtual_size
         windows_version
         working_set_size
         write_operation_count
         write_transfer_count
      /
      
      ProcTableStruct = Struct.new("ProcTableStruct", *@fields)
      
      # call-seq:
      #    ProcTable.fields
      # 
      # Returns an array of fields that each ProcTableStruct will contain.  This
      # may be useful if you want to know in advance what fields are available
      # without having to perform at least one read of the /proc table.
      #
      def self.fields
         @fields
      end
      
      # call-seq:
      #    ProcTable.ps(pid=nil)
      #    ProcTable.ps(pid=nil){ |ps| ... }
      # 
      # In block form, yields a ProcTableStruct for each process entry that you
      # have rights to.  This method returns an array of ProcTableStruct's in
      # non-block form.
      # 
      # If a +pid+ is provided, then only a single ProcTableStruct is yielded or
      # returned, or nil if no process information is found for that +pid+.
      #
      def self.ps(pid=nil, host=Socket.gethostname)
         if pid
            raise TypeError unless pid.kind_of?(Fixnum)
         end

         array  = block_given? ? nil : []
         struct = nil
              
         begin
            wmi = WIN32OLE.connect("winmgmts://#{host}/root/cimv2")
         rescue WIN32OLERuntimeError => e
            raise Error, e # Re-raise as ProcTable::Error
         else
            wmi.InstancesOf("Win32_Process").each{ |wproc|        
               if pid                
                  next unless wproc.ProcessId == pid
               end

               # Some fields are added later, and so are nil initially
               struct = ProcTableStruct.new(
                  wproc.Caption,
                  nil, # Added later, based on OS version
                  wproc.Name,
                  wproc.CreationClassName,
                  self.parse_ms_date(wproc.CreationDate),
                  wproc.CSCreationClassName,
                  wproc.CSName,
                  wproc.Description,
                  wproc.ExecutablePath,
                  wproc.ExecutionState,
                  wproc.Handle,
                  wproc.HandleCount,
                  self.parse_ms_date(wproc.InstallDate),
                  self.convert(wproc.KernelModeTime),
                  wproc.MaximumWorkingSetSize,
                  wproc.MinimumWorkingSetSize,
                  wproc.Name,
                  wproc.OSCreationClassName,
                  wproc.OSName,
                  self.convert(wproc.OtherOperationCount),
                  self.convert(wproc.OtherTransferCount),
                  wproc.PageFaults,
                  wproc.PageFileUsage,
                  wproc.ParentProcessId,
                  self.convert(wproc.PeakPageFileUsage),
                  self.convert(wproc.PeakVirtualSize),
                  self.convert(wproc.PeakWorkingSetSize),
                  wproc.Priority,
                  self.convert(wproc.PrivatePageCount),
                  wproc.ProcessId,
                  wproc.QuotaNonPagedPoolUsage,
                  wproc.QuotaPagedPoolUsage,
                  wproc.QuotaPeakNonPagedPoolUsage,
                  wproc.QuotaPeakPagedPoolUsage,
                  self.convert(wproc.ReadOperationCount),
                  self.convert(wproc.ReadTransferCount),
                  wproc.SessionId,
                  wproc.Status,
                  self.parse_ms_date(wproc.TerminationDate),
                  wproc.ThreadCount,
                  self.convert(wproc.UserModeTime),
                  self.convert(wproc.VirtualSize),
                  wproc.WindowsVersion,
                  self.convert(wproc.WorkingSetSize),
                  self.convert(wproc.WriteOperationCount),
                  self.convert(wproc.WriteTransferCount)
               )
            
               ###############################################################
               # On Windows XP or later, set the cmdline to the CommandLine
               # attribute.  Otherwise, set it to the ExecutablePath
               # attribute.
               ###############################################################
               if wproc.WindowsVersion.to_f < 5.1
                  struct.cmdline = wproc.ExecutablePath
               else
                  struct.cmdline = wproc.CommandLine
               end
               
               struct.freeze # This is read-only data
              
               if block_given?
                  yield struct
               else
                  array << struct
               end
            }
         end

         pid ? struct : array
      end
      
      private
      
      #######################################################################
      # Converts a string in the format '20040703074625.015625-360' into a
      # Ruby Time object.
      #######################################################################
      def self.parse_ms_date(str)
         return if str.nil?
         return Date.parse(str.split('.').first)
      end
      
      #####################################################################
      # There is a bug in win32ole where uint64 types are returned as a
      # String instead of a Fixnum.  This method deals with that for now.
      #####################################################################
      def self.convert(str)
         return nil if str.nil? # Return nil, not 0
         return str.to_i
      end
   end
end

ProcTable.ps{ |p|
  puts p.pid.to_s if p.comm=='explorer.exe'
}

编程技巧