combineFileInputFormat 中的 isSplitable 不起作用

Posted

技术标签:

【中文标题】combineFileInputFormat 中的 isSplitable 不起作用【英文标题】:isSplitable in combineFileInputFormat does not work 【发布时间】:2014-12-04 15:01:56 【问题描述】:

我有数千个小文件,我想用 combineFileInputFormat 处理它们。

combineFileInputFormat中,一个mapper有多个小文件,每个文件不会被分割。

像这样的小输入文件之一的 sn-p,

vers,3
period,2015-01-26-18-12-00,438469546,449329626,complete
config,libdvm.so,chromeview
pkgproc,com.futuredial.digitchat,10021,,0ns:10860078
pkgpss,com.futuredial.digitchat,10021,,0ns:9:6627:6627:6637:5912:5912:5912
pkgsvc-run,com.futuredial.digitchat,10021,.LiveScreenService,1,0n:10860078
pkgsvc-start,com.futuredial.digitchat,10021,.LiveScreenService,1,0n:10860078
pkgproc,com.google.android.youtube,10103,,0ns:10860078
pkgpss,com.google.android.youtube,10103,,0ns:9:12986:13000:13021:11552:11564:11580
pkgsvc-     run,com.google.android.youtube,10103,com.google.android.apps.youtube.app.offline.transfer.OfflineTransferService,1,0n:10860078
pkgsvc-    start,com.google.android.youtube,10103,com.google.android.apps.youtube.app.offline.transfer.OfflineTransferService,1,0n:10860078

我想将整个文件内容传递给映射器。但是,hadoop 将文件拆分为一半。

比如上面的文件可能会被拆分成

vers,3
period,2015-01-26-18-12-00,438469546,449329626,complete
config,libdvm.so,chromeview
pkgproc,com.futuredial.digitchat,#the line has been cut

但我希望处理整个文件的内容。

这是我的代码,参考Reading file as single record in hadoop

驱动代码

public class CombineSmallfiles 

public static void main(String[] args) throws IOException, ClassNotFoundException, InterruptedException 

    Configuration conf = new Configuration();
    String[] otherArgs = new GenericOptionsParser(conf, args).getRemainingArgs();
    if (otherArgs.length != 2) 
        System.err.println("Usage: conbinesmallfiles <in> <out>");
        System.exit(2);
    

    conf.setInt("mapred.min.split.size", 1);
    conf.setLong("mapred.max.split.size", 26214400); // 25m
    //conf.setLong("mapred.max.split.size", 134217728); // 128m

    //conf.setInt("mapred.reduce.tasks", 5);

    Job job = new Job(conf, "combine smallfiles");
    job.setJarByClass(CombineSmallfiles.class);
    job.setMapperClass(CombineSmallfileMapper.class);
    //job.setReducerClass(IdentityReducer.class);
    job.setNumReduceTasks(0);

    job.setMapOutputKeyClass(Text.class);
    job.setMapOutputValueClass(Text.class);
    job.setOutputKeyClass(Text.class);
    job.setOutputValueClass(Text.class);
    MultipleOutputs.addNamedOutput(job,"pkgproc",TextOutputFormat.class,Text.class,Text.class);
    MultipleOutputs.addNamedOutput(job,"pkgpss",TextOutputFormat.class,Text.class,Text.class);
    MultipleOutputs.addNamedOutput(job,"pkgsvc",TextOutputFormat.class,Text.class,Text.class);

    job.setInputFormatClass(CombineSmallfileInputFormat.class);
    job.setOutputFormatClass(TextOutputFormat.class);

    FileInputFormat.addInputPath(job, new Path(otherArgs[0]));
    FileOutputFormat.setOutputPath(job, new Path(otherArgs[1]));

    int exitFlag = job.waitForCompletion(true) ? 0 : 1;
    System.exit(exitFlag);




我的映射器代码

public class CombineSmallfileMapper extends Mapper<NullWritable, Text, Text, Text> 

    private Text file = new Text();
    private MultipleOutputs mos;
    private String period;
    private Long elapsed;

    @Override
    public void setup(Context context) throws IOException, InterruptedException 
        mos = new MultipleOutputs(context);
    
    @Override
    protected void map(NullWritable key, Text value, Context context) throws IOException, InterruptedException 
        String file_name = context.getConfiguration().get("map.input.file.name");
        String [] filename_tokens = file_name.split("_");
        String uuid = filename_tokens[0];
        String [] datetime_tokens;
        try
        datetime_tokens = filename_tokens[1].split("-");
        catch(ArrayIndexOutOfBoundsException err)
            throw new ArrayIndexOutOfBoundsException(file_name);
        
        String year,month,day,hour,minute,sec,msec;
        year = datetime_tokens[0];
        month = datetime_tokens[1];
        day = datetime_tokens[2];
        hour = datetime_tokens[3];
        minute = datetime_tokens[4];
        sec = datetime_tokens[5];
        msec = datetime_tokens[6];
        String datetime = year+"-"+month+"-"+"-"+day+" "+hour+":"+minute+":"+sec+"."+msec;
        String content = value.toString();
        String []lines = content.split("\n");
        for(int u = 0;u<lines.length;u++)
            String line = lines[u];
            String []tokens = line.split(",");
            if(tokens[0].equals("period"))
                period = tokens[1];
                try
                long startTime = Long.valueOf(tokens[2]);
                long endTime = Long.valueOf(tokens[3]);
                elapsed = endTime-startTime;
                catch(NumberFormatException err)
                    throw new NumberFormatException(line);
                
            else if(tokens[0].equals("pkgproc"))
                String proc_info = "";
                try
                proc_info += period+","+String.valueOf(elapsed)+","+tokens[2]+","+tokens[3];
                catch(ArrayIndexOutOfBoundsException err)
                    throw new ArrayIndexOutOfBoundsException("pkgproc: "+content+ "line:"+line);
                
                for(int i = 4;i<tokens.length;i++)
                    String []state_info = tokens[i].split(":");
                    String state = "";
                    state += ","+state_info[0].charAt(0)+","+state_info[0].charAt(1)+","+state_info[0].charAt(2)+","+state_info[1];
                    mos.write("pkgproc",new Text(tokens[1]), new Text(proc_info+state+','+uuid+','+datetime));
                
            else if(tokens[0].equals("pkgpss"))
                String proc_info = "";
                proc_info += period+","+String.valueOf(elapsed)+","+tokens[2]+","+tokens[3];
                for(int i = 4;i<tokens.length;i++)
                    String []state_info = tokens[i].split(":");
                    String state = "";
                    state += ","+state_info[0].charAt(0)+","+state_info[0].charAt(1)+","+state_info[0].charAt(2)+","+state_info[1]+","+state_info[2]+","+state_info[3]+","+state_info[4]+","+state_info[5]+","+state_info[6]+","+state_info[7];
                    mos.write("pkgpss",new Text(tokens[1]), new Text(proc_info+state+','+uuid+','+datetime));
                
            else if(tokens[0].startsWith("pkgsvc"))
                String []stateName = tokens[0].split("-");
                String proc_info = "";
                //tokens[2] = uid, tokens[3] =  serviceName
                proc_info += stateName[1]+','+period+","+String.valueOf(elapsed)+","+tokens[2]+","+tokens[3];
                String opcount = tokens[4];
                for(int i = 5;i<tokens.length;i++)
                    String []state_info = tokens[i].split(":");
                    String state = "";
                    state += ","+state_info[0].charAt(0)+","+state_info[0].charAt(1)+","+state_info[1];
                    mos.write("pkgsvc",new Text(tokens[1]), new Text(proc_info+state+','+opcount+','+uuid+','+datetime));
                
            
        
    


我的 CombineFileInputFormat,它覆盖 isSplitable 并返回 false

public class CombineSmallfileInputFormat extends CombineFileInputFormat<NullWritable, Text> 

    @Override
    public RecordReader<NullWritable, Text> createRecordReader(InputSplit split, TaskAttemptContext context) throws IOException 

        return new CombineFileRecordReader<NullWritable,Text>((CombineFileSplit) split,context,WholeFileRecordReader.class);
    
    @Override
    protected boolean isSplitable(JobContext context,Path file )
        return false;
    


WholeFileRecordReader

   public class WholeFileRecordReader extends RecordReader<NullWritable, Text> 
    //private static final Logger LOG = Logger.getLogger(WholeFileRecordReader.class);

      /** The path to the file to read. */
      private final Path mFileToRead;
      /** The length of this file. */
      private final long mFileLength;

      /** The Configuration. */
      private final Configuration mConf;

      /** Whether this FileSplit has been processed. */
      private boolean mProcessed;
      /** Single Text to store the file name of the current file. */
    //  private final Text mFileName;
      /** Single Text to store the value of this file (the value) when it is read. */
      private final Text mFileText;

      /**
       * Implementation detail: This constructor is built to be called via
       * reflection from within CombineFileRecordReader.
       *
       * @param fileSplit The CombineFileSplit that this will read from.
       * @param context The context for this task.
       * @param pathToProcess The path index from the CombineFileSplit to process in this record.
       */
      public WholeFileRecordReader(CombineFileSplit fileSplit, TaskAttemptContext context,
          Integer pathToProcess) 
        mProcessed = false;
        mFileToRead = fileSplit.getPath(pathToProcess);
        mFileLength = fileSplit.getLength(pathToProcess);
        mConf = context.getConfiguration();
        context.getConfiguration().set("map.input.file.name", mFileToRead.getName());

        assert 0 == fileSplit.getOffset(pathToProcess);
        //if (LOG.isDebugEnabled()) 
          //LOG.debug("FileToRead is: " + mFileToRead.toString());
          //LOG.debug("Processing path " + pathToProcess + " out of " + fileSplit.getNumPaths());

          //try 
            //FileSystem fs = FileSystem.get(mConf);
            //assert fs.getFileStatus(mFileToRead).getLen() == mFileLength;
          // catch (IOException ioe) 
            //// oh well, I was just testing.
          //
        //

        //mFileName = new Text();
        mFileText = new Text();
      

      /** @inheritDoc */
      @Override
      public void close() throws IOException 
        mFileText.clear();
      

      /**
       * Returns the absolute path to the current file.
       *
       * @return The absolute path to the current file.
       * @throws IOException never.
       * @throws InterruptedException never.
       */
      @Override
      public NullWritable getCurrentKey() throws IOException, InterruptedException 
        return NullWritable.get();
      

      /**
       * <p>Returns the current value.  If the file has been read with a call to NextKeyValue(),
       * this returns the contents of the file as a BytesWritable.  Otherwise, it returns an
       * empty BytesWritable.</p>
       *
       * <p>Throws an IllegalStateException if initialize() is not called first.</p>
       *
       * @return A BytesWritable containing the contents of the file to read.
       * @throws IOException never.
       * @throws InterruptedException never.
       */
      @Override
      public Text getCurrentValue() throws IOException, InterruptedException 
        return mFileText;
      

      /**
       * Returns whether the file has been processed or not.  Since only one record
       * will be generated for a file, progress will be 0.0 if it has not been processed,
       * and 1.0 if it has.
       *
       * @return 0.0 if the file has not been processed.  1.0 if it has.
       * @throws IOException never.
       * @throws InterruptedException never.
       */
      @Override
      public float getProgress() throws IOException, InterruptedException 
        return (mProcessed) ? (float) 1.0 : (float) 0.0;
      

      /**
       * All of the internal state is already set on instantiation.  This is a no-op.
       *
       * @param split The InputSplit to read.  Unused.
       * @param context The context for this task.  Unused.
       * @throws IOException never.
       * @throws InterruptedException never.
       */
      @Override
      public void initialize(InputSplit split, TaskAttemptContext context)
          throws IOException, InterruptedException 
        // no-op.
      

      /**
       * <p>If the file has not already been read, this reads it into memory, so that a call
       * to getCurrentValue() will return the entire contents of this file as Text,
       * and getCurrentKey() will return the qualified path to this file as Text.  Then, returns
       * true.  If it has already been read, then returns false without updating any internal state.</p>
       *
       * @return Whether the file was read or not.
       * @throws IOException if there is an error reading the file.
       * @throws InterruptedException if there is an error.
       */
      @Override
      public boolean nextKeyValue() throws IOException, InterruptedException 
        if (!mProcessed) 
          if (mFileLength > (long) Integer.MAX_VALUE) 
            throw new IOException("File is longer than Integer.MAX_VALUE.");
          
          byte[] contents = new byte[(int) mFileLength];

          FileSystem fs = mFileToRead.getFileSystem(mConf);
          FSDataInputStream in = null;
          try 
            // Set the contents of this file.
            in = fs.open(mFileToRead);
            IOUtils.readFully(in, contents, 0, contents.length);
            mFileText.set(contents, 0, contents.length);

           finally 
            IOUtils.closeQuietly(in);
          
          mProcessed = true;
          return true;
        
        return false;
      


我希望每个mapper都解析多个小文件,每个小文件不能被拆分。

但是,上面的代码将剪切(拆分)我的输入文件并引发解析错误(因为我的解析器会将行拆分为标记)。

在我的概念中,combineFileInputFormat 会将多个文件收集到一个拆分中,每个拆分都将馈送到一个映射器中。因此,一个映射器可以处理多个文件。

在我的代码中,最大输入拆分设置为 25MB,所以我认为问题在于 combineFileInputFormat 将拆分输入拆分的小文件的最后一部分以满足拆分大小限制。

但是,我已经覆盖 isSplitable 并返回 false,但它仍然拆分小文件。

这样做的正确方法是什么?

我不确定是否可以为映射器指定文件数量,而不是指定输入拆分大小?

【问题讨论】:

【参考方案1】:

在构造函数代码中使用 setMaxSplitSize() 方法,它应该可以工作, 理想情况下,它会告诉拆分大小,

public class CFInputFormat extends CombineFileInputFormat<FileLineWritable, Text> 
  public CFInputFormat()
    super();
    setMaxSplitSize(67108864); // 64 MB, default block size on hadoop
  
  public RecordReader<FileLineWritable, Text> createRecordReader(InputSplit split, TaskAttemptContext context) throws IOException
    return new CombineFileRecordReader<FileLineWritable, Text>((CombineFileSplit)split, context, CFRecordReader.class);
  
  @Override
  protected boolean isSplitable(JobContext context, Path file)
    return false;
  

【讨论】:

即使我使用 setMaxSplitSize(67108864); 仍然会拆分我的小文件;

以上是关于combineFileInputFormat 中的 isSplitable 不起作用的主要内容,如果未能解决你的问题,请参考以下文章

学习记录CombineFileInputFormat类

MapReduce小文件处理之CombineFileInputFormat实现

Hadoop调优

大数据面试题:Hadoop优化核心高频面试题

如何优化mapreduce job的运行效率

iOS Xcode 中的 UIViewController 中的 UIView 中的 UITableView