文章目录

  • 前言
  • mysql
  • 表结构设计
  • db_telecom.tb_contacts
  • db_telecom.tb_call
  • db_telecom.tb_dimension_date
  • 建表语句
  • 导入基础数据
  • 姓名手机号映射
  • 时间维度表
  • Meaven依赖
  • HBase
  • Maven依赖
  • mapper
  • reducer
  • 自定义输出
  • OutputFormat
  • MyRecordWriter
  • 私有属性
  • 初始化操作
  • 设置job
  • 运行
  • 博客链接


前言

有了之前存入HBase的数据,现在就可以运用HBase的MR进行数据分析了。

mysql

表结构设计

db_telecom.tb_contacts

用于存放用户手机号码与联系人姓名。


备注

类型

id

自增主键

int(11) NOT NULL

telephone

手机号码

varchar(255) NOT NULL

name

联系人姓名

varchar(255) NOT NULL

db_telecom.tb_call


备注

类型

id

date

contact 复合主键(联系人维度 id,时间维度 id) varchar(255) NOT NULL

id

date

dimension 时间维度 id int(11) NOT NULL

id

contact

查询人的电话号码 int(11) NOT NULL

call

sum

通话次数总和 int(11) NOT NULL DEFAULT 0

call

duration

sum 通话时长总和 int(11)

db_telecom.tb_dimension_date


备注

类型

id

自增主键

int(11)

year

年,当前通话信息所在年

int(11)

month

月,当前通话信息所在月,如果按照年来统计信息,则month 为-1。

int(11) NOT NULL

day

日,当前通话信息所在日,如果是按照月来统计信息,则day为-1。

int(11) NOT

建表语句

我这里简化了约束信息,自己建立的表,需要注意自己的默认字符集,不是utf8请选择下面的完整版,否则可能无法导入中文。

CREATE DATABASE db_telecom;

CREATE TABLE tb_dimension_date(
id int PRIMARY KEY AUTO_INCREMENT,
year int,
month int,
day int);

create table tb_contacts(
id int PRIMARY KEY AUTO_INCREMENT,
telephone char(11),
name varchar(20));

CREATE TABLE tb_call( 
id_date_contact VARCHAR(10) PRIMARY KEY,
id_date_dimension int,
id_contact int, 
call_sum int, 
call_duration_sum int,
FOREIGN KEY(id_date_dimension) REFERENCES tb_dimension_date(id), 
FOREIGN KEY(id_contact) REFERENCES tb_contacts(id));

下面是我老师提供的完整的建表语句

SET FOREIGN_KEY_CHECKS=0;

-- ----------------------------
-- Table structure for tb_call
-- ----------------------------
DROP TABLE IF EXISTS `tb_call`;
CREATE TABLE `tb_call` (
  `id_date_contact` varchar(100) NOT NULL,
  `id_date_dimension` int(11) NOT NULL,
  `id_contact` int(11) NOT NULL,
  `call_sum` int(11) NOT NULL,
  `call_duration_sum` int(11) NOT NULL,
  PRIMARY KEY (`id_date_contact`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

-- ----------------------------
-- Table structure for tb_contacts
-- ----------------------------
DROP TABLE IF EXISTS `tb_contacts`;
CREATE TABLE `tb_contacts` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `telephone` varchar(11) NOT NULL,
  `name` varchar(50) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

-- ----------------------------
-- Table structure for tb_dimension_date
-- ----------------------------
DROP TABLE IF EXISTS `tb_dimension_date`;
CREATE TABLE `tb_dimension_date` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `year` int(11) NOT NULL,
  `month` int(11) NOT NULL,
  `day` int(11) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

导入基础数据

姓名手机号映射

我先修改了一下之前生成的文件,推荐用notepad++、或者sublime都有列编辑的工具,直接在每行开始添加",再进行局部替换。

copilot 数据分析并展示_copilot 数据分析并展示


可以直接在Navicat中导入csv文件

copilot 数据分析并展示_ide_02


一直下一步到完成就行了,其中需要注意字段是否对应上

copilot 数据分析并展示_ide_03


也可以用mysql命令

mysql --local-infile -uroot -proot db_telecom -e "LOAD DATA LOCAL INFILE 'nam_num.csv' INTO TABLE tb_contacts character set utf8 FIELDS TERMINATED BY ',' OPTIONALLY ENCLOSED BY '"' LINES TERMINATED BY '\n'"

时间维度表

和上一步操作一样,这里不用担心字符集还简单一些

copilot 数据分析并展示_mysql_04

Meaven依赖

如果没有添加mysql驱动支持,需要添加上,之后才能用JDBC

<!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java -->
		<dependency>
			<groupId>mysql</groupId>
			<artifactId>mysql-connector-java</artifactId>
			<version>5.1.20</version>
		</dependency>

HBase

Maven依赖

检查是否有依赖,没有的话添加进去。

<dependency>
			<groupId>org.apache.hbase</groupId>
			<artifactId>hbase-server</artifactId>
			<version>1.4.5</version>
		</dependency>
		<dependency>
			<groupId>org.apache.hbase</groupId>
			<artifactId>hbase-client</artifactId>
			<version>1.4.5</version>
		</dependency>

mapper

这时候也体现出了rowKey设计的好处,直接切分rowKey就能获得需要的数据

public static class HBase2MysqlMapper extends TableMapper<Text, IntWritable>{
		Text outKey = new Text();
		IntWritable outValue = new IntWritable();
		@Override
		//来一个,算三个维度
		protected void map(ImmutableBytesWritable key, Result value,Context context)
				throws IOException, InterruptedException {
			String rowKey = Bytes.toString(key.get());
			String[] strs = rowKey.split("_");
			//用于计算通话时长,只算一次即可
			if(strs[5].equals("01"))
				outValue.set(Integer.valueOf(strs[4]));
			else
				outValue.set(0);
			//日维度
			outKey.set(strs[1]+"_"+strs[2]);
			context.write(outKey, outValue);
			//月维度
			outKey.set(strs[1]+"_"+strs[2].substring(0, 7));
			context.write(outKey, outValue);
			//年维度
			outKey.set(strs[1]+"_"+strs[2].substring(0, 4));
			context.write(outKey, outValue);
		}
	}

reducer

和wordcount差不多,都是求和,key设计好就行了

public static class HBase2MysqlReducer extends Reducer<Text, IntWritable, Text, IntWritable>{
		Text outKey = new Text();
		IntWritable outValue = new IntWritable();
		int count,sum;
		@Override
		protected void reduce(Text key, Iterable<IntWritable> values,Context context)
				throws IOException, InterruptedException {
			count = 0;
			sum = 0;
			for(IntWritable value:values){				
				//累计通话次数
				count += 1;
				//累计通话时长
				sum += value.get();
			}
			outKey.set(key.toString()+"_"+String.valueOf(count));
			outValue.set(sum);
			context.write(outKey, outValue);
		}
	}

自定义输出

这个需要自定义才能输出到Mysql比较麻烦

OutputFormat

不是输出到文件,不必继承FileOutputFormat,而是直接继承OutputFormat;核心是实现RecordWriter,另外两个方法可以参照FileOutputFormat,也可以直接return。

public class MySQLOutputFormat extends OutputFormat<Text, IntWritable>{
	@Override
	public RecordWriter<Text, IntWritable> getRecordWriter(TaskAttemptContext job) throws IOException, InterruptedException {
		RecordWriter<Text, IntWritable> writer = new MyRecordWriter<Text, IntWritable>(job);
		return writer;
	}
	@Override
	public void checkOutputSpecs(JobContext context) throws IOException, InterruptedException {
		return;		
	}

	@Override
	public OutputCommitter getOutputCommitter(TaskAttemptContext context) throws IOException, InterruptedException {
		return new FileOutputCommitter(null, context);
	}

}

MyRecordWriter

私有属性
private Connection conn = null;
	private Statement stmt = null;
初始化操作

记得修改自己的url,还有密码

public MyRecordWriter(TaskAttemptContext context) {
		try {
			Class.forName("com.mysql.jdbc.Driver");
			String url = "jdbc:mysql://node103:3306/db_telecom";
			String username = "root";
			String password = "root";
			conn = DriverManager.getConnection(url, username, password);
			stmt = conn.createStatement();
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		} catch (SQLException e) {
			e.printStackTrace();
		}
	}

这是最关键的部分,就是查询出两个外键,然后构造成主键,插入mysql

@Override
	public void write(Text key, IntWritable value) throws IOException, InterruptedException {
		String[] strs = key.toString().split("_");
		Integer count = Integer.valueOf(strs[2]);
		String[] dimensions = strs[1].split("-");
		String year="-1",month="-1",day="-1";
		if(dimensions.length>2)
			day = dimensions[2];
		if(dimensions.length>1)
			month = dimensions[1];
		if(dimensions.length>0)
			year = dimensions[0];
		try {
			//获得手机号码对应的外键
			String querySql = "SELECT id FROM tb_contacts WHERE telephone = "+strs[0]+";";
			ResultSet rs = stmt.executeQuery(querySql);
			rs.next();
			String id_contact = rs.getString("id");
			//获得日期对应的外键
			querySql = "SELECT id FROM tb_dimension_date WHERE "
					+ "year = "+year+" AND month = "+month
					+" AND day = "+day+";";
			rs = stmt.executeQuery(querySql);
			rs.next();
			String id_date_dimension = rs.getString("id");
			//拼凑为主键
			String id_date_contact = id_date_dimension+"_"+id_contact;
			//插入
			String insertSql = "INSERT INTO tb_call "
			+"(id_date_contact,id_date_dimension,id_contact,"
			+ "call_sum,call_duration_sum) "
			+ "VALUES('"+id_date_contact+"',"+id_date_dimension+","+id_contact
			+","+count+","+value.get()+");";
			stmt.execute(insertSql);
		} catch (SQLException e) {
			e.printStackTrace();
		}
		
	}

设置job

我实现了Tool接口,主要在run中设置

public int run(String[] args) throws Exception {
		Job job = Job.getInstance(this.getConf());
		job.setJarByClass(HBase2MysqlDriver.class);
		TableMapReduceUtil.initTableMapperJob(args[0], new Scan(), HBase2MysqlMapper.class, Text.class, IntWritable.class, job);
		job.setReducerClass(HBase2MysqlReducer.class);
		job.setOutputFormatClass(MySQLOutputFormat.class);
		boolean result = job.waitForCompletion(true);
		return result?0:1;
	}

其他的代码就是走个形式了

private Configuration conf;
	@Override
	public void setConf(Configuration conf) {
		this.conf = conf;
	}

	@Override
	public Configuration getConf() {
		return conf;
	}
public static void main(String[] args) {
		try {
			Configuration conf = HBaseConfiguration.create(); 
			ToolRunner.run(conf, new HBase2MysqlDriver(), args);
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

运行

直接在Eclipse中运行即可,数据库中的结果如下

copilot 数据分析并展示_ide_05