当前位置:   article > 正文

用于大规模数据集(大于1TB)的并行运算的MapReduce是怎么实现的?

用于大规模数据集(大于1TB)的并行运算的MapReduce是怎么实现的?

MapReduce 是一种编程模型,用于处理和生成大数据集。MapReduce 分为两个阶段:Map 阶段和 Reduce 阶段。

  1. Map 阶段:在这个阶段,输入数据被拆分成不同的数据块,这些数据块被分发到各个 Map 任务上。每个 Map 任务对输入的数据块进行处理并输出成一组键值对。具体的处理方式取决于编写 Map 任务的方式,这将由程序员决定。

  2. Shuffle 阶段:在 Map 阶段之后,系统会根据 Map 阶段输出的键值对进行排序和分区。这个过程是系统自动完成的,不需要程序员编写代码。

  3. Reduce 阶段:在这个阶段,一组键值对会送到同一个 Reduce 任务上。Reduce 任务会对这些键值对按照键来进行合并,并对每个键的所有值进行处理,输出处理结果。具体的处理方式也是由程序员编写的 Reduce 任务来决定。

通过 Map 和 Reduce 两个步骤,MapReduce 能够将大规模的数据计算工作分解成小规模的计算任务,这些小规模的计算任务可以在不同的计算节点上并行处理。这使得 MapReduce 可以处理非常大规模的数据。

例如,一个简单的MapReduce程序可能会将文本文件分成单词(map),然后计算每个单词的出现次数(reduce)。在这种情况下,Map任务将文本文件切分为单词,并输出每个单词的键值对(键是单词,值是1),然后Reduce任务将这些键值对按照键(单词)合并,并计算每个键(单词)的值(出现次数)的总和。

在真实的环境中,这个模型通常由分布式计算框架(如 Hadoop)来实现,框架会负责任务的调度、数据的分布和容错等工作,使得程序员可以专注于编写处理数据的 Map 和 Reduce 函数。

上demo:

  1. // 导入 IOException 类,用于处理输入/输出错误
  2. import java.io.IOException;
  3. // 导入 StringTokenizer 类,用于解析字符串
  4. import java.util.StringTokenizer;
  5. // 导入 Hadoop 提供的 Configuration 类,用于 Hadoop 配置设置
  6. import org.apache.hadoop.conf.Configuration;
  7. // 导入 Hadoop 提供的 Path 类,处理 Hadoop 文件系统路径
  8. import org.apache.hadoop.fs.Path;
  9. // 导入 Hadoop 提供的 IntWritable 和 Text 类,数据类型转换
  10. import org.apache.hadoop.io.IntWritable;
  11. import org.apache.hadoop.io.Text;
  12. // 导入 Hadoop MapReduce Job 类,用于定义和提交 MapReduce 任务
  13. import org.apache.hadoop.mapreduce.Job;
  14. // 从 Hadoop MapReduce 库导入 Mapper 和 Reducer 类
  15. import org.apache.hadoop.mapreduce.Mapper;
  16. import org.apache.hadoop.mapreduce.Reducer;
  17. // 导入 Hadoop IO 类库,用于 MapReduce 输入输出
  18. import org.apache.hadoop.mapreduce.lib.input.FileInputFormat;
  19. import org.apache.hadoop.mapreduce.lib.output.FileOutputFormat;
  20. // 定义 WordCount 类
  21. public class WordCount {
  22. // 定义公开的静态 TokenizerMapper 类,继承 Hadoop 的 Mapper 类
  23. public static class TokenizerMapper extends Mapper<Object, Text, Text, IntWritable>{
  24. // 定义一个值为 1 的整型变量,作为每个单词计数的值
  25. private final static IntWritable one = new IntWritable(1);
  26. // 定义一个 Text 类型的变量,存储每个即将被计数的单词
  27. private Text word = new Text();
  28. // 重写 map 方法,这个方法会被 MapReduce 框架调用,每读入一行数据调用一次
  29. public void map(Object key, Text value, Context context) throws IOException, InterruptedException {
  30. // 使用 StringTokenizer 将每行的字符串拆解为单个的单词
  31. StringTokenizer itr = new StringTokenizer(value.toString());
  32. // 循环单词字符串
  33. while (itr.hasMoreTokens()) {
  34. // 为 word 对象设值
  35. word.set(itr.nextToken());
  36. // 将每个单词及其数量(默认为 1)输出
  37. context.write(word, one);
  38. }
  39. }
  40. }
  41. // 定义公开的静态 IntSumReducer 类,继承 Hadoop 的 Reducer 类
  42. public static class IntSumReducer extends Reducer<Text,IntWritable,Text,IntWritable> {
  43. // 定义 IntWritable 类型的对象,存储每个单词的计数总和
  44. private IntWritable result = new IntWritable();
  45. // 重写 reduce 方法,这个方法用于汇总每个单词的数量
  46. public void reduce(Text key, Iterable<IntWritable> values, Context context) throws IOException, InterruptedException {
  47. // 定义 sum 变量,用于累加每个单词的数量
  48. int sum = 0;
  49. // 使用 for 循环,遍历当前单词的所有数量值,进行累加
  50. for (IntWritable val : values) {
  51. sum += val.get();
  52. }
  53. // 为结果设值
  54. result.set(sum);
  55. // 将每个单词及其最终计数总和写出
  56. context.write(key, result);
  57. }
  58. }
  59. // 定义主方法,运行 MapReduce 任务的入口
  60. public static void main(String[] args) throws Exception {
  61. // 创建 Hadoop 配置对象
  62. Configuration conf = new Configuration();
  63. // 创建 Job 实例,设置 Job 名称
  64. Job job = Job.getInstance(conf, "word count");
  65. // 设置 Jar 文件 这行代码的作用是设置该作业的 jar 文件。在 Hadoop 中执行一个 MapReduce 作业,需要将其打包为一个 jar 文件并上传到 Hadoop 集群中。这里的 WordCount.class 即是你的作业驱动类, Hadoop 将会通过这个类查找包含该类的 jar 文件,然后使用这个 jar 文件执行整个作业。这样设置可以确保作业在 Hadoop 集群中的每个节点上都可以正确执行。
  66. job.setJarByClass(WordCount.class);
  67. // 设置 Mapper 类 这行代码的作用是设置 MapReduce 作业的 Mapper 类。在 MapReduce 模型中,Mapper 是负责处理输入数据,将输入数据转换为一系列键值对的组件。TokenizerMapper.class 就是你写的处理数据的类,你可以在这个类中定义如何处理和映射输入数据。设置 Mapper 类是进行 MapReduce 作业的关键步骤之一。
  68. job.setMapperClass(TokenizerMapper.class);
  69. // 设置 Combiner 类 当一个特定的 Mapper 结束处理之后,它会输出很多记录,这些记录中可能有很多键是相同的。如果没有 Combiner,这些记录就会直接传递给 Reducer,这样可能会产生大量的网络 I/O,显著地影响性能。
  70. //而有了 Combiner,我们可以在每个 Mapper 所在的节点上先进行一次局部的归约操作,将相同键的值合并后再传输到 Reducer,这样可以大大减少需要传输的数据量,提高任务的运行性能。
  71. job.setCombinerClass(IntSumReducer.class);
  72. // 设置 Reducer 类及其输出键-值对的类型
  73. // 这行代码是设定了该 MapReduce 任务的 Reducer 类是 IntSumReducer。Reducer 类是对所有 Mapper 的输出进行汇总的部分,每一个 MapReduce 任务都需要设定一个 Reducer 类。这里的 IntSumReducer 类定义了如何对数据进行汇总操作。
  74. job.setReducerClass(IntSumReducer.class);
  75. //这行代码是设定了该 MapReduce 任务输出数据的键的类型是 Text。在 MapReduce 中,数据是以键值对的形式存储和传输的。这行代码指明了输出键的数据类型,确保处理结果的格式正确。
  76. job.setOutputKeyClass(Text.class);
  77. //这行代码是设定了该 MapReduce 任务输出数据的值的类型是 IntWritable。这是对job.setOutputKeyClass(Text.class)的补充,指明了输出值的数据类型。
  78. job.setOutputValueClass(IntWritable.class);
  79. // 设置输入数据的路径,取自程序参数
  80. FileInputFormat.addInputPath(job, new Path(args[0]));
  81. // 设置输出数据的路径,取自程序参数
  82. FileOutputFormat.setOutputPath(job, new Path(args[1]));
  83. // 等待任务完成后退出程序 这行代码是 MapReduce 任务的最后一步,作用是让主线程等待这个 MapReduce 任务完成。
  84. //在这段代码中,waitForCompletion方法将主线程挂起,直到 MapReduce 任务完成为止。这里的参数 true 表示会在控制台上显示这个任务的进度,也就是说你运行这个程序的时候,会在控制台上看到任务的完成进度。
  85. //这段代码的含义就是:如果任务成功完成,那么程序正常退出;如果任务执行过程中出现了错误,那么程序异常退出。
  86. System.exit(job.waitForCompletion(true) ? 0 : 1);
  87. }
  88. }
  89. /**
  90. MapReduce 模型的并发性体现在其 Map 和 Reduce 操作的实现中,这两个操作对数据进行分布式处理。在以上代码片段中,没有一行明确地标示了并发操作,因为并发性质是 MapReduce 框架内部管理的,并非是在业务代码中直接表现出来的。
  91. 然而,job.setMapperClass(TokenizerMapper.class); 和 job.setReducerClass(IntSumReducer.class);
  92. 这两行设置了 Mapper 类和 Reducer 类,这两类的实例在运行时会在多个节点上同时处理数据(一个节点处理输入数据的一部分),这便是 MapReduce 的并发性!!!
  93. 具体来说:
  94. 在 Mapper 阶段,输入的数据集会被划分成多个数据块,每一个数据块会被一个 Mapper 实例负责处理,这些 Mapper 实例是在集群的各个计算节点上同时运行的。
  95. Reducer 阶段同样具有并发性,不同的 Reducer 实例会并发地处理 Mapper 输出的数据集中具有相同键的数据子集。
  96. 请注意,这些并发处理是由 Hadoop 框架自动管理的,并不需要在业务代码中明确地进行并发控制,这也是使用这种大数据处理框架的优点之一。
  97. */

这个程序把一个任务分为两个部分:

  1. Mapper (TokenizerMapper class):接受一行文本,拆分成单词,然后返回一系列的<单词,1>这样的键值对。
  2. Reducer (IntSumReducer class):对所有相同的单词计数总和,并返回<单词,总计数>这样的键值对。

执行这个程序,需要有一个 Hadoop 环境并且把待统计名词的文本文件放到应当的路径,并将结果输出的路径设置为期望的路径进行查看结果。

请注意,这些并发处理是由 Hadoop 框架自动管理的,并不需要在业务代码中明确地进行并发控制,这也是使用这种大数据处理框架的优点之一。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小小林熬夜学编程/article/detail/398762
推荐阅读
相关标签
  

闽ICP备14008679号