大数据实验一:大数据系统基本实验(第五部分:MapReduce初级编程)

1)编程实现文件的合并和去重
对于两个输入文件,即文件 A 和文件 B,请编写 MapReduce 程序,对两个文件进行合并, 并剔除其中重复的内容,得到一个新的输出文件 C。

import java.io.IOException;
import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Job;
import org.apache.hadoop.mapreduce.Mapper;
import org.apache.hadoop.mapreduce.Reducer;
import org.apache.hadoop.mapreduce.lib.input.*;
import org.apache.hadoop.mapreduce.lib.output.*;

/*
 * 思路:
 * 数据去重的目标是让原数据中出现次数超过一次的数据在输出文件中只出现一次。
 * 将不同文件中相同内容数据的Key设置成一样的,
 * 在Reduce,不管value-list是什么样,只要在最终结果输出它的key就行了。
 */
public class Merge {
	public static class Map extends Mapper<Object, Text, Text, Text> {// 输入键 输入值 输出键 输出值
		private static Text text = new Text();

		public void map(Object key, Text value, Context context) throws IOException, InterruptedException {
			text = value;
			context.write(text, new Text(""));
		}
	}

	public static class Reduce extends Reducer<Text, Text, Text, Text> {
		public void reduce(Text key, Iterable<Text> values, Context context) throws IOException, InterruptedException {
			context.write(key, new Text(""));
		}
	}

	public static void main(String[] args) throws IOException, ClassNotFoundException, InterruptedException {
		Configuration conf = new Configuration();
		conf.set("fs.defaultFS", "hdfs://localhost:9000");

		String[] otherArgs = new String[] { "hdfs://localhost:9000/MapReduceTest/Merge/input",
				"hdfs://localhost:9000/MapReduceTest/Merge/output" };
		if (otherArgs.length != 2) {
			System.err.println("Usage:Merge and duplicate removal <in> <out>");
			System.exit(2);
		}

		Job job = Job.getInstance(conf, "Merge and duplicate removal");
		job.setJarByClass(Merge.class);
		job.setMapperClass(Map.class);
		job.setReducerClass(Reduce.class);
		job.setOutputKeyClass(Text.class);
		job.setOutputValueClass(Text.class);
		FileInputFormat.addInputPath(job, new Path(otherArgs[0]));
		FileOutputFormat.setOutputPath(job, new Path(otherArgs[1]));
		System.exit(job.waitForCompletion(true) ? 0 : 1);
	}
}

2)编程实现对输入文件的排序
现在有多个输入文件,每个文件中的每行内容均为一个整数。要求读取所有文件中的整数, 进行升序排序后,输出到一个新的文件中,输出的数据格式为每行两个整数,第一个数字为第二个整数的排序位次,第二个整数为原待排列的整数。

import java.io.IOException;
import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.io.IntWritable;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Job;
import org.apache.hadoop.mapreduce.Mapper;
import org.apache.hadoop.mapreduce.Reducer;
import org.apache.hadoop.mapreduce.lib.input.FileInputFormat;
import org.apache.hadoop.mapreduce.lib.output.FileOutputFormat;

/*
 * 思路:
 * MapReduce默认按照key值排序,
 * 如果key为封装int的IntWritable类型,那么MapReduce按照数字大小对key排序。
 * 所以在Map中将读入的数据转化成IntWritable型,然后作为key值输出,value可以随意设。
 * reduce拿到之后,将输入的key作为value输出,并根据value-list中元素的个数决定输出的次数。
 */
public class Sort {
	public static class Map extends Mapper<Object, Text, IntWritable, IntWritable> {
		private static IntWritable data = new IntWritable();

		public void map(Object key, Text value, Context context) throws IOException, InterruptedException {
			String line = value.toString();
			data.set(Integer.parseInt(line));
			context.write(data, new IntWritable(1));
		}
	}

	public static class Reduce extends Reducer<IntWritable, IntWritable, IntWritable, IntWritable> {
		private static IntWritable number = new IntWritable(1);

		public void reduce(IntWritable key, Iterable<IntWritable> values, Context context)
				throws IOException, InterruptedException {
			for (IntWritable num : values) {
				context.write(number, key);
				number = new IntWritable(number.get() + 1);
			}
		}
	}

	public static void main(String[] args) throws IOException, ClassNotFoundException, InterruptedException {
		Configuration conf = new Configuration();
		conf.set("fs.defaultFS", "hdfs://localhost:9000");

		String[] otherArgs = new String[] { "hdfs://localhost:9000/MapReduceTest/Sort/input",
				"hdfs://localhost:9000/MapReduceTest/Sort/output" };
		if (otherArgs.length != 2) {
			System.err.println("Usage:Merge and duplicate removal <in> <out>");
			System.exit(2);
		}

		Job job = Job.getInstance(conf, "Merge and sort");
		job.setJarByClass(Sort.class);
		job.setMapperClass(Map.class);
		job.setReducerClass(Reduce.class);
		job.setOutputKeyClass(IntWritable.class);
		job.setOutputValueClass(IntWritable.class);
		FileInputFormat.addInputPath(job, new Path(otherArgs[0]));
		FileOutputFormat.setOutputPath(job, new Path(otherArgs[1]));
		System.exit(job.waitForCompletion(true) ? 0 : 1);
	}
}


3)对指定的表格进行信息挖掘
给出一个 child-parent 的表格,要求挖掘其中的父子辈关系,给出祖孙辈关系的表格。

import java.io.IOException;
import java.util.*;
import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Job;
import org.apache.hadoop.mapreduce.Mapper;
import org.apache.hadoop.mapreduce.Reducer;
import org.apache.hadoop.mapreduce.lib.input.FileInputFormat;
import org.apache.hadoop.mapreduce.lib.output.FileOutputFormat;

/*
 * 思路:
 * 将人名作为键,值分为两种,一个记录子辈一个记录父辈
 * 同一个人的子辈和父辈是祖孙
 */

public class Relation {
	public static class Map extends Mapper<Object, Text, Text, Text> {
		public void map(Object key, Text value, Context context) throws IOException, InterruptedException {
			String child_name = new String();
			String parent_name = new String();
			String line = value.toString();
			int i = 0;
			while (line.charAt(i) != ' ') {
				i++;
			}
			String[] values = { line.substring(0, i), line.substring(i + 1) };// 分割两个名字
			if (values[0].compareTo("child") != 0) {
				child_name = values[0];
				parent_name = values[1];
				context.write(new Text(values[0]), new Text("1-" + child_name + "-" + parent_name));
				context.write(new Text(values[1]), new Text("2-" + child_name + "-" + parent_name));
			}
		}
	}

	public static int head = 0;

	public static class Reduce extends Reducer<Text, Text, Text, Text> {
		public void reduce(Text key, Iterable<Text> values, Context context) throws IOException, InterruptedException {
			if (head == 0) {
				context.write(new Text("grandchild"), new Text("grandparent"));
				head = 1;
			}
			int grandchild_num = 0;
			String grandchild[] = new String[32];
			int grandparent_num = 0;
			String grandparent[] = new String[32];
			Iterator<Text> ite = values.iterator();
			while (ite.hasNext()) {
				String record = ite.next().toString();
				int len = record.length();
				char relation_type = record.charAt(0);
				String child_name = new String();
				String parent_name = new String();
				int i = 2;
				while (record.charAt(i) != '-') {
					child_name = child_name + record.charAt(i);
					i++;
				}
				i++;
				while (i < len) {
					parent_name = parent_name + record.charAt(i);
					i++;
				}
				if (relation_type == '1') {
					grandparent[grandparent_num] = parent_name;
					grandparent_num++;
				} else {
					grandchild[grandchild_num] = child_name;
					grandchild_num++;
				}
			}
			for (int m = 0; m < grandchild_num; m++) {
				for (int n = 0; n < grandparent_num; n++) {
					context.write(new Text(grandchild[m]), new Text(grandparent[n]));
				}
			}
		}
	}

	public static void main(String[] args) throws IOException, ClassNotFoundException, InterruptedException {
		Configuration conf = new Configuration();
		conf.set("fs.defaultFS", "hdfs://localhost:9000");

		String[] otherArgs = new String[] { "hdfs://localhost:9000/MapReduceTest/Relation/input",
				"hdfs://localhost:9000/MapReduceTest/Relation/output" };
		if (otherArgs.length != 2) {
			System.err.println("Usage:Relation <in> <out>");
			System.exit(2);
		}
		Job job = Job.getInstance(conf, "Relation");
		job.setJarByClass(Relation.class);
		job.setMapperClass(Map.class);
		job.setReducerClass(Reduce.class);
		job.setOutputKeyClass(Text.class);
		job.setOutputValueClass(Text.class);
		FileInputFormat.addInputPath(job, new Path(otherArgs[0]));
		FileOutputFormat.setOutputPath(job, new Path(otherArgs[1]));
		System.exit(job.waitForCompletion(true) ? 0 : 1);
	}
}

  • 5
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
好的,MapReduce是一种用于大规模数据处理的编程模型和计算框架。下面是一个简单的MapReduce程序的实现。 假设我们有一个大型的文本文件,里面包含了很多单词。我们想要统计每个单词在文件中出现的次数。这可以通过MapReduce来实现。 Map阶段: 1. 将文件分成多个小块,并分配给不同的计算节点。 2. 对于每个小块,计算节点会读取其中的内容,并将每个单词作为key,出现次数作为value,生成一系列键值对。 3. 计算节点将这些键值对按照key进行排序,并将相同key的value合并成一个列表。 Reduce阶段: 1. 将所有的key-value列表按照key进行排序。 2. 对于每个key,将其对应的value列表合并起来,得到该key在文件中出现的总次数。 下面是一个基于Python的MapReduce程序示例: ```python # mapper.py import sys # 从标准输入读取数据 for line in sys.stdin: # 删除行末尾的换行符 line = line.strip() # 将行按照空格分割成单词 words = line.split() # 对于每个单词,输出一个键值对 for word in words: print(word, 1) # reducer.py import sys current_word = None current_count = 0 # 从标准输入读取数据 for line in sys.stdin: # 删除行末尾的换行符 line = line.strip() # 分割键值对 word, count = line.split('\t', 1) # 将字符串类型的计数器转换为整数 count = int(count) # 如果当前单词与上一个单词不同,就输出上一个单词的统计结果 if current_word and current_word != word: print(current_word, current_count) current_count = 0 # 将计数器累加到当前单词的统计结果中 current_word = word current_count += count # 输出最后一个单词的统计结果 if current_word: print(current_word, current_count) ``` 我们可以使用Hadoop框架来运行这个MapReduce程序。具体的步骤如下: 1. 将文件上传到HDFS中。 2. 启动Hadoop集群。 3. 运行Mapper任务和Reducer任务: ``` $ hadoop jar /path/to/hadoop-streaming.jar \ -mapper "python mapper.py" \ -reducer "python reducer.py" \ -input "/path/to/input/file" \ -output "/path/to/output/folder" ``` 4. 在HDFS中查看输出文件。 这个MapReduce程序可以处理很大的数据集,并且可以在分布式计算环境中高效运行。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值