使用HDFS完成wordcount词频统计

时间:2021-08-24 01:35:02

任务需求

统计HDFS上文件的wordcount,并将统计结果输出到HDFS

功能拆解

  • 读取HDFS文件
  • 业务处理(词频统计)
  • 缓存处理结果
  • 将结果输出到HDFS

数据准备

  • 事先往HDFS上传需要进行词频统计的文件word.txt、word2.txt(可以是多个)...
  • 假设目录是/user/hadoop/input/...

框架搭建

先把具体的功能框架搭建出来,再进行细节方面的编写。

import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.*;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.URI;

public class HDFSWordCountDemo{
    public static void main(String[] args) throws Exception{
        // 1.读取HDFS文件
        Configuration conf = new Configuration();
        FileSystem fs = FileSystem.get(new URI("hdfs://localhost:9000"), conf, "hadoop");
        // 使用Java API取出HDFS指定目录下所有要进行词频统计的单词文件,false表示不需要递归
        RemoteIterator<LocatedFileStatus> files = fs.listFiles(new Path("input"), false);
        // 用于循环取出多个单词文本
        while (files.hasNext()) {
            LocatedFileStatus file = files.next();
            FSDataInputStream in = fs.open(file.getPath());
            BufferedReader reader = new BufferedReader(new InputStreamReader(in));
            String line = null; // readLine每次读取一行
            // 用于循环取出每个文本的每行内容
            while ((line = reader.readLine()) != null) {
                // 2.业务处理(词频统计)

                /**
                 * 功能:
                 *      此处要进行单词的词频统计功能
                 * 输入:
                 *      每次循环读取的是一个文件,输入的是该文件的一行内容line
                 * 输出:
                 *      把每行内容line按指定分割符分割,成为一个个独立单词,进行累加统计,多个文本累计,返回结果数组
                 */
            }
            reader.close();
            in.close();
        }
        // 3.缓存处理结果:把统计结果写入缓存
        // TODO...
        // 4.将结果输出到HDFS
        // 先在HDFS上创建一个空文本
        FSDataOutputStream out = fs.create(new Path("output/result.txt"));
        // 然后取出缓存中的内容,追加到该HDFS文本即可
        // TODO...
    }
}

词频统计实现

分为两步:1)实现上下文对象,用于保存每次的统计;2)词频统计功能的封装调用

  • 使用Map实现上下文对象
import java.util.HashMap;
import java.util.Map;

/**
 * 自定义上下文对象,其实就是模仿缓存
 */
public class HDFSContext {

    private Map<Object,Object> cacheMap = new HashMap<>();

    // 用于从外部可以直接获取缓存
    public Map<Object,Object> getCacheMap(){
        return cacheMap;
    }

    /**
     * 写数据到缓存
     * @param key
     * @param value
     */
    public void write(Object key,Object value){
        cacheMap.put(key, value);
    }

    /**
     * 从缓存中读取数据
     * @param key
     * @return
     */
    public Object get(Object key){
        return cacheMap.get(key);
    }
}
  • 词频统计逻辑处理
// 自定义一个Mapper接口,封装词频统计功能
public interface HDFSMapper {
    /**
     * @param line 读取到的每一行数据
     * @param context 上下文对象/缓存
     */
    public void map(String line,HDFSContext context);
}

// 接口的功能实现
public class WordCountMapper implements HDFSMapper{
    @Override
    public void map(String line, HDFSContext context) {
        String[] words = line.split(" "); // 按空格切割,words是一行内容的单词数组
        for (String word : words) { // 遍历数组,取出每一个单词
            Object value = context.get(word);   // 取出缓存中的单词,
            if (value == null){             // 如果value为null,则说明缓存中没有该单词
                //不存在这个单词
                context.write(word,1);  // 第一次出现的单词,次数为1,并写入缓存
            }else {
                // 出现次数+1
                int v = Integer.parseInt(value.toString()); // 取出单词的已经出现次数,转成int
                context.write(word,v+1);    // 次数+1,并写入缓存
            }
        }
    }
}
  • 调用
// 先声明类对象
HDFSContext context = new HDFSContext();
HDFSMapper mapper = new WordCountMapper();

// while里调用
while ((line = reader.readLine()) != null) {
    mapper.map(line,context);
}

缓存处理结果

Map<Object,Object> contextMap = context.getCacheMap();

追加结果到HDFS

// 把Map集合转换为Set集合,进行迭代操作
Set<Map.Entry<Object, Object>> entries = contextMap.entrySet();
for (Map.Entry<Object, Object> entry : entries) {
    // 取出key-value,即(word,次数),写入HDFS
    out.write((entry.getKey().toString()+"\t"+entry.getValue()+"\n").getBytes());
}
System.out.println("词频统计运行成功!");
out.close();
fs.close();

完整代码

package com.hadoop.hdfs.wordcount;

import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.*;

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.URI;
import java.util.Map;
import java.util.Set;

public class HDFSWordCountDemo{
    public static void main(String[] args) throws Exception{
        // 1.读取HDFS文件
        Configuration conf = new Configuration();
        FileSystem fs = FileSystem.get(new URI("hdfs://localhost:9000"), conf, "hadoop");
        HDFSContext context = new HDFSContext();
        HDFSMapper mapper = new WordCountMapper();
        // 使用Java API取出HDFS指定目录下所有要进行词频统计的单词文件,false表示不需要递归
        RemoteIterator<LocatedFileStatus> files = fs.listFiles(new Path("input"), false);
        // 用于循环取出多个单词文本
        while (files.hasNext()) {
            LocatedFileStatus file = files.next();
            FSDataInputStream in = fs.open(file.getPath());
            BufferedReader reader = new BufferedReader(new InputStreamReader(in));
            String line = null; // readLine每次读取一行
            // 用于循环取出每个文本的每行内容
            while ((line = reader.readLine()) != null) {
                mapper.map(line,context);
            }
            reader.close();
            in.close();
        }
        // 3.缓存处理结果:把统计结果写入缓存
        Map<Object,Object> contextMap = context.getCacheMap();
        // 4.将结果输出到HDFS
        // 先在HDFS上创建一个空文本
        FSDataOutputStream out = fs.create(new Path("output/result.txt"));
        // 然后取出缓存中的内容,追加到该HDFS文本即可
        Set<Map.Entry<Object, Object>> entries = contextMap.entrySet();
        for (Map.Entry<Object, Object> entry : entries) {
            out.write((entry.getKey().toString()+"\t"+entry.getValue()+"\n").getBytes());
        }
        System.out.println("词频统计运行成功!");
        out.close();
        fs.close();

    }
}

查看运行结果

$ hadoop fs -cat output/*

使用HDFS完成wordcount词频统计的更多相关文章

  1. 初学Hadoop之WordCount词频统计

    1.WordCount源码 将源码文件WordCount.java放到Hadoop2.6.0文件夹中. import java.io.IOException; import java.util.Str ...

  2. Hadoop基础学习(一)分析、编写并执行WordCount词频统计程序

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/jiq408694711/article/details/34181439 前面已经在我的Ubuntu ...

  3. 词频统计小程序-WordCount&period;exe

    一. 背景 ​ 最近顶哥为了完成学历提升学业中的小作业,做了一个词频统计的.exe小程序.因为当时做的时候网上的比较少,因此顶哥决定把自己拙略的作品发出来给需要的人提供一种思路,希望各位看官不要dis ...

  4. 使用SparkSQL编写wordCount的词频统计

    # 使用SparkSQL编写wordCount的词频统计 ## word.txt```hello hello scala sparkjava sql html java hellojack jack ...

  5. Hive简单编程实践-词频统计

    一.使用MapReduce的方式进行词频统计 (1)在HDFS用户目录下创建input文件夹 hdfs dfs -mkdir input 注意:林子雨老师的博客(http://dblab.xmu.ed ...

  6. hive进行词频统计

    统计文件信息: $ /opt/cdh-5.3.6/hadoop-2.5.0/bin/hdfs dfs -text /user/hadoop/wordcount/input/wc.input hadoo ...

  7. Hadoop之词频统计小实验

    声明:    1)本文由我原创撰写,转载时请注明出处,侵权必究. 2)本小实验工作环境为Ubuntu操作系统,hadoop1-2-1,jdk1.8.0. 3)统计词频工作在单节点的伪分布上,至于真正实 ...

  8. Hadoop上的中文分词与词频统计实践 (有待学习 http&colon;&sol;&sol;www&period;cnblogs&period;com&sol;jiejue&sol;archive&sol;2012&sol;12&sol;16&sol;2820788&period;html)

    解决问题的方案 Hadoop上的中文分词与词频统计实践 首先来推荐相关材料:http://xiaoxia.org/2011/12/18/map-reduce-program-of-rmm-word-c ...

  9. MapReduce词频统计

    自定义Mapper实现 import org.apache.hadoop.io.IntWritable; import org.apache.hadoop.io.LongWritable; impor ...

随机推荐

  1. Selenium Remote-Control架构

    Selenium Remote-Control(RC)是一个测试工具,它允许你编写基于JavaScript浏览器的Web UI自动化测试,它支持很多编程语言. Selenium RC包括两部分: 一个 ...

  2. 简单正则匹配QQ邮箱

    <!DOCTYPE html><html> <head> <meta charset="UTF-8"> <script src ...

  3. 解决UDT中内存下不去的问题

         使用UDT库,编写简单的网络通信程序,发现了一个问题,关闭一部分连接后,程序占用内存并没有变化.      比如先连接500个,再连接另500个,先关掉后面500个,程序占用内存降一半,再关 ...

  4. 在oneAPM参加第一个项目小结

    从12月15日开始加入进入oneAPM的第二个项目,也是我真正近距离接触项目的一次吧,到今天差不多接近尾声了,很高心能和大家一起共同改造这个项目,虽然说我做的贡献并不大,但是身临项目真的会收获很多体会 ...

  5. 20145236 《Java程序设计》第7周学习总结

    20145236 <Java程序设计>第7周学习总结 教材学习内容总结 第十三章 时间与日期 认识时间与日期 时间的度量 格林威治标准时间GMT 格林威治标准时间的正午是太阳抵达天空最高点 ...

  6. HDU 5597 GTW likes function 打表

    GTW likes function 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5596 Description Now you are give ...

  7. Sass函数--颜色函数--RGB颜色函数

    RGB颜色函数-RGB()颜色函数 主要分为 RGB , HSL 和 Opacity 三大函数,当然其还包括一些其他的颜色函数,比如说 adjust-color 和 change-color 等.1. ...

  8. CentOS&comma;Ubuntu&comma;Gentoo&comma;Freebsd&comma;RedHat&comma;Debian的区别及选择

    Linux最早由Linus Benedict Torvalds在1991年开始编写.在这之前,Richard Stallman创建了Free Software Foundation(FSF)组织以及G ...

  9. Linux学习---位运算符

    <<.>> ① << 左移  乘以2^n m << n m*(2^n) eg:4: 0 0 1 0 0 8: 0 1 0 0 0 [数据.数字]移位 左 ...

  10. &lbrack;POI2010&rsqb;KLO-Blocks——一道值得思考的题

    题目大意: 给出N个正整数a[1..N],再给出一个正整数k,现在可以进行如下操作:每次选择一个大于k的正整数a[i],将a[i]减去1,选择a[i-1]或a[i+1]中的一个加上1.经过一定次数的操 ...