Bootstrap

Flink中常用物理分区(区别和用法)算子以及Sink(JDBC Connector、Kafka Connector以及自定义Sink)


一、物理分区

[图片]

1、自定义分区+重分区(解决数据倾斜)

重分区能够解决数据倾斜

package com.bigdata.transformation;


import org.apache.flink.api.common.RuntimeExecutionMode;
import org.apache.flink.api.common.functions.Partitioner;
import org.apache.flink.api.common.functions.RichMapFunction;
import org.apache.flink.api.java.functions.KeySelector;
import org.apache.flink.api.java.tuple.Tuple2;
import org.apache.flink.streaming.api.datastream.DataStream;
import org.apache.flink.streaming.api.datastream.DataStreamSource;
import org.apache.flink.streaming.api.environment.StreamExecutionEnvironment;


// 自定义分区
class MyPartitioner implements Partitioner<Long>{

    @Override
    public int partition(Long key, int numPartitions) {
        // 小于等于10000的放到1分区,否则放到2分区
        if (key <= 10000){
            return 0;
        }
        return 1;
    }
}


public class CustomPartition {
    public static void main(String[] args) throws Exception {
        StreamExecutionEnvironment env = StreamExecutionEnvironment.getExecutionEnvironment();
        env.setRuntimeMode(RuntimeExecutionMode.AUTOMATIC);
        env.setParallelism(2);

        DataStreamSource<Long> dataStreamSource = env.fromSequence(1, 15000);

        // 自定义分区API
        DataStream<Long> dataStream = dataStreamSource.partitionCustom(new MyPartitioner(), new KeySelector<Long, Long>() {
            @Override
            public Long getKey(Long value) throws Exception {
                return value;
            }
        });

        // 查看每个分区的数据量
        dataStream.map(new RichMapFunction<Long, Tuple2<Integer,Integer>>() {
            @Override
            public Tuple2<Integer, Integer> map(Long aLong) throws Exception {
                int partitions = getRuntimeContext().getIndexOfThisSubtask();
                return Tuple2.of(partitions,1);
            }
        }).keyBy(new KeySelector<Tuple2<Integer, Integer>, Integer>() {
            @Override
            public Integer getKey(Tuple2<Integer, Integer> integerIntegerTuple2) throws Exception {
                return integerIntegerTuple2.f0;
            }
        }).sum(1).print("前:");

        // 打印自定义分区结果
        // dataStream.print();

        // 进行重分区
        DataStream<Long> rebalance = dataStream.rebalance();

        //查看重分区每个分区的数据量
        rebalance.map(new RichMapFunction<Long, Tuple2<Integer,Integer>>() {
            @Override
            public Tuple2<Integer, Integer> map(Long aLong) throws Exception {
                int partitions = getRuntimeContext().getIndexOfThisSubtask();
                return Tuple2.of(partitions,1);
            }
        }).keyBy(new KeySelector<Tuple2<Integer, Integer>, Integer>() {
            @Override
            public Integer getKey(Tuple2<Integer, Integer> integerIntegerTuple2) throws Exception {
                return integerIntegerTuple2.f0;
            }
        }).sum(1).print("后:");

        env.execute("自定义分区+重分区(解决数据倾斜)");
    }
}

二、Sink

有print、writerAsText(以文本格式输出)、Connectors(连接器)

1、JDBC Connector(JDBC连接器)

导包:

<dependency>
    <groupId>org.apache.flink</groupId>
    <artifactId>flink-connector-jdbc_2.11</artifactId>
    <version>${flink.version}</version>
</dependency>

<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    <version>8.0.25</version>
</dependency>

代码演示:

package com.bigdata.sink;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import org.apache.flink.connector.jdbc.JdbcConnectionOptions;
import org.apache.flink.connector.jdbc.JdbcSink;
import org.apache.flink.connector.jdbc.JdbcStatementBuilder;
import org.apache.flink.streaming.api.datastream.DataStreamSource;
import org.apache.flink.streaming.api.environment.StreamExecutionEnvironment;
import org.apache.flink.api.common.RuntimeExecutionMode;

import java.sql.PreparedStatement;
import java.sql.SQLException;

/**
 @基本功能:
 @program:FlinkDemo
 @author: hang
 @create:2024-11-22 15:55:54
 **/

@Data
@NoArgsConstructor
@AllArgsConstructor
class Student {
    private int id;
    private String name;
    private int age;
}

public class JdbcConnector {

    public static void main(String[] args) throws Exception {

        //1. env-准备环境
        StreamExecutionEnvironment env = StreamExecutionEnvironment.getExecutionEnvironment();
        env.setRuntimeMode(RuntimeExecutionMode.AUTOMATIC);

        //2. source-加载数据
        DataStreamSource<Student> studentDataStreamSource = env.fromElements(
                new Student(1, "zhanngsan", 18),
                new Student(2, "lisi", 19),
                new Student(3, "wangwu", 20)
        );

        //3. transformation-数据处理转换
        //4. sink-数据输出
        JdbcConnectionOptions jdbcConnectionOptions = new JdbcConnectionOptions.JdbcConnectionOptionsBuilder()
                .withDriverName("com.mysql.cj.jdbc.Driver")
                .withUrl("jdbc:mysql://localhost:3306/mydb01")
                .withUsername("root")
                .withPassword("123456").build();


        studentDataStreamSource.addSink(JdbcSink.sink(
                "insert into student values (?,?,?)",
                new JdbcStatementBuilder<Student>() {
                    @Override
                    public void accept(PreparedStatement preparedStatement, Student student) throws SQLException {
                        preparedStatement.setInt(1,student.getId());
                        preparedStatement.setString(2,student.getName());
                        preparedStatement.setInt(3,student.getAge());

                    }
                },
                jdbcConnectionOptions
                // 假如是流的方式可以设置两条插入一次
                //JdbcExecutionOptions.builder().withBatchSize(2).build(),jdbcConnectionOptions
        ));

        //5. execute-执行
        env.execute();
    }
}

2、Kafka Connector(Kafka连接器)

需求:从Kafka的topic1中消费日志数据,并做实时ETL,将状态为success的数据写入到Kafka的topic2中

package com.bigdata.sink;

import org.apache.flink.api.common.functions.FilterFunction;
import org.apache.flink.api.common.serialization.SimpleStringSchema;
import org.apache.flink.streaming.api.datastream.DataStreamSource;
import org.apache.flink.streaming.api.datastream.SingleOutputStreamOperator;
import org.apache.flink.streaming.api.environment.StreamExecutionEnvironment;
import org.apache.flink.api.common.RuntimeExecutionMode;
import org.apache.flink.streaming.connectors.kafka.FlinkKafkaConsumer;
import org.apache.flink.streaming.connectors.kafka.FlinkKafkaProducer;

import java.util.Properties;

/**
 @基本功能:
 @program:FlinkDemo
 @author: hang
 @create:2024-11-22 16:38:58
 **/
public class KafkaConnector {

    public static void main(String[] args) throws Exception {

        //1. env-准备环境
        StreamExecutionEnvironment env = StreamExecutionEnvironment.getExecutionEnvironment();
        env.setRuntimeMode(RuntimeExecutionMode.AUTOMATIC);

        //2. source-加载数据
        Properties properties = new Properties();
        properties.setProperty("bootstrap.servers","node01:9092");
        properties.setProperty("group.id", "g1");
        FlinkKafkaConsumer<String> kafkaConsumer = new FlinkKafkaConsumer<>("topic1", new SimpleStringSchema(), properties);
        DataStreamSource<String> dataStreamSource = env.addSource(kafkaConsumer);

        //3. transformation-数据处理转换
        SingleOutputStreamOperator<String> success = dataStreamSource.filter(new FilterFunction<String>() {
            @Override
            public boolean filter(String s) throws Exception {
                return s.contains("success");
            }
        });

        //4. sink-数据输出
        FlinkKafkaProducer<String> kafkaProducer = new FlinkKafkaProducer<>("topic2", new SimpleStringSchema(), properties);
        success.addSink(kafkaProducer);

        //5. execute-执行
        env.execute();
    }

}

3、自定义Sink

模拟jdbcSink的实现
jdbcSink官方已经提供过了,此处仅仅是模拟它的实现,从而学习如何自定义sink

package com.bigdata.sink;

import org.apache.flink.api.common.RuntimeExecutionMode;
import org.apache.flink.configuration.Configuration;
import org.apache.flink.streaming.api.datastream.DataStreamSink;
import org.apache.flink.streaming.api.datastream.DataStreamSource;
import org.apache.flink.streaming.api.environment.StreamExecutionEnvironment;
import org.apache.flink.streaming.api.functions.sink.RichSinkFunction;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;

class MyJdbcSink extends RichSinkFunction<Student>{
    Connection conn = null;
    PreparedStatement statement = null;
    @Override
    public void open(Configuration parameters) throws Exception {
        // 注册驱动(安转驱动)  此时这句话可以省略  如果书写的话,mysql8.0 带 cj
        Class.forName("com.mysql.cj.jdbc.Driver");
        // 获取数据库连接对象 Connection
        conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/mydb01","root","123456");
        // 执行sql语句
        statement = conn.prepareStatement("insert into student values (?,?,?)");
    }

    @Override
    public void close() throws Exception {
        // 释放资源
        statement.close();
        conn.close();
    }

    @Override
    public void invoke(Student student, Context context) throws Exception {
        statement.setInt(1,student.getId());
        statement.setString(2,student.getName());
        statement.setInt(3,student.getAge());
        statement.execute();
    }
}

public class jdbcCustomSink {
    public static void main(String[] args) throws Exception {
        //1. env-准备环境
        StreamExecutionEnvironment env = StreamExecutionEnvironment.getExecutionEnvironment();
        env.setRuntimeMode(RuntimeExecutionMode.AUTOMATIC);
        //2. source-加载数据
        DataStreamSource<Student> studentDataStreamSource = env.fromElements(
                new Student(4, "zhaoliu", 18),
                new Student(5, "qianqi", 19),
                new Student(6, "wuba", 20)
        );
        //3. transformation-数据处理转换
        //4. sink-数据输出
        DataStreamSink<Student> studentDataStreamSink = studentDataStreamSource.addSink(new MyJdbcSink());

        //5. execute-执行
        env.execute();
    }
}
;