为了账号安全,请及时绑定邮箱和手机立即绑定

MQ消息队列教程:从入门到实践

标签:
运维 中间件
概述

本文档涵盖了MQ消息队列的基础概念、常见系统、生产者与消费者模型以及消息的可靠传输机制。文章详细介绍了如何安装和配置MQ消息队列服务器,并提供了发送和接收消息的步骤,同时还包括了性能优化和常见问题的解决方法。

MQ消息队列教程:从入门到实践
MQ消息队列简介

什么是MQ消息队列

MQ消息队列是一种中间件,用于在不同的应用程序或组件之间传输消息。其主要目的是解耦系统组件,使得不同的应用程序或组件能够通过异步通信方式相互通信。MQ消息队列通过在发送方和接收方之间引入一个中间层(即消息队列),使得发送方不必等待接收方处理完消息,从而提高了系统的可扩展性和灵活性。

MQ消息队列的作用和优点

MQ消息队列的主要作用包括消息的传输、负载均衡、解耦系统组件、异步处理和错误处理等。以下是MQ消息队列的一些优点:

  1. 解耦系统组件:MQ消息队列允许不同应用程序或模块之间进行异步通信。发送方无需等待接收方完成处理,这为系统的可扩展性和灵活性提供了支持。
  2. 负载均衡:MQ消息队列可以实现负载均衡,将消息分配到多个消费者,使得系统能够更好地处理高并发场景。
  3. 异步处理:MQ消息队列支持异步处理,发送方发送消息后可以立即返回,而不需要等待消息处理完成,从而提高了系统的响应速度。
  4. 错误处理:MQ消息队列提供了错误处理机制,如消息重试、延迟处理等,能够更好地处理网络故障或其他异常情况,保证消息的可靠传输。
  5. 可靠性:MQ消息队列可以确保消息传输的可靠性,支持消息持久化和事务功能,确保消息不会丢失。
  6. 灵活性:MQ消息队列可以灵活地配置和扩展,支持多种消息传输协议和编码格式。

常见的MQ消息队列系统

目前市面上有许多MQ消息队列系统,以下是其中一些常用的系统:

  1. RabbitMQ:RabbitMQ是一个开源的消息代理,支持AMQP(高级消息队列协议),具有高可用性、可靠性、灵活性和易用性等特点。
  2. ActiveMQ:ActiveMQ是一个基于JMS(Java消息服务)的消息代理,支持多种消息传输协议,包括AMQP和STOMP,可以集成到多种应用环境中。
  3. Kafka:Kafka是一个分布式流处理平台,主要用于构建实时数据管道和流应用。Kafka具有高吞吐量和持久性的特点,适用于大数据处理场景。
  4. RocketMQ:RocketMQ是一个分布式消息系统,由阿里云开发并维护,支持多种消息传输协议和编码格式,具有高可用性和可靠性。
MQ消息队列的基本概念

生产者与消费者模型

在MQ消息队列中,生产者(Producer)负责生产消息,并将消息发送到消息队列(Queue)。消费者(Consumer)从消息队列中消费消息,并对消息进行处理。生产者和消费者之间通过消息队列进行通信,解耦了发送方和接收方,使得系统能够更好地处理高并发和异步处理等情况。

生产者和消费者之间的交互流程如下:

  1. 生产者创建消息:生产者创建待发送的消息,并将消息发送到消息队列。
  2. 消息队列保存消息:消息队列将接收到的消息进行保存,等待消费者进行消费。
  3. 消费者消费消息:消费者从消息队列中获取消息,并对消息进行处理。
  4. 消息队列删除消息:消息队列在消息被消费者消费后,删除该消息,保证消息不会被重复消费。

消息的可靠传输

消息的可靠传输是MQ消息队列中的一个重要概念,确保消息在传输过程中不会丢失或重复。消息的可靠传输通常通过以下机制实现:

  1. 消息持久化:消息队列支持将消息持久化到磁盘,即使在系统崩溃或重启后,消息也不会丢失。
  2. 事务处理:MQ消息队列支持事务处理机制,确保消息在发送方和接收方之间的同步提交或回滚操作。
  3. 消息确认机制:消费者在消费消息后,需要向消息队列发送确认消息,通知消息队列消息已经被成功处理。如果确认消息失败,则消息队列会重新发送消息给消费者。

以下是一个使用RabbitMQ实现消息持久化和事务处理的示例代码:

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

public class MessageProducer {
    private static final String QUEUE_NAME = "my_queue";

    public static void sendMessage() throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 设置消息队列为持久化
        channel.queueDeclare(QUEUE_NAME, true, false, false, null);

        String message = "Hello, World!";
        // 设置消息持久化
        channel.basicPublish("", QUEUE_NAME, null, message.getBytes());
        System.out.println("Sent '" + message + "'");

        channel.close();
        connection.close();
    }

    public static void main(String[] args) throws Exception {
        try {
            sendMessage();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

交换机、队列和绑定的概念

在MQ消息队列中,交换机(Exchange)、队列(Queue)和绑定(Binding)是三个重要的概念:

  1. 交换机(Exchange):交换机是消息路由的核心,负责接收生产者发送的消息,并根据消息的路由键(Routing Key)将消息路由到相应的队列。
  2. 队列(Queue):队列是消息的存储容器,负责存储和转发消息。队列可以被一个或多个消费者消费。
  3. 绑定(Binding):绑定是交换机和队列之间的连接关系,通过绑定可以将特定的消息路由到相应的队列。绑定通常由消息的路由键来确定。

以下是一个使用RabbitMQ实现交换机、队列和绑定的示例代码:

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

public class MessageRoutingExample {
    private static final String EXCHANGE_NAME = "my_exchange";
    private static final String QUEUE_NAME = "my_queue";

    public static void sendMessage(String message) throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 声明交换机
        channel.exchangeDeclare(EXCHANGE_NAME, "direct", true);
        // 声明队列,并绑定交换机与队列
        channel.queueDeclare(QUEUE_NAME, true, false, false, null);
        channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "my_routing_key");

        // 发送消息
        channel.basicPublish(EXCHANGE_NAME, "my_routing_key", null, message.getBytes());
        System.out.println("Sent '" + message + "'");

        channel.close();
        connection.close();
    }

    public static void main(String[] args) throws Exception {
        try {
            sendMessage("Hello, World!");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
使用MQ消息队列的步骤

安装与配置MQ消息队列服务器

安装MQ消息队列服务器的具体步骤因所使用的MQ消息队列系统而异。以下是一个使用RabbitMQ安装和配置的示例:

  1. 下载并安装RabbitMQ:访问RabbitMQ官网下载最新的安装包,并按照说明进行安装。
  2. 启动RabbitMQ服务:启动RabbitMQ服务,确保RabbitMQ可以正常运行。
  3. 配置RabbitMQ管理界面:RabbitMQ提供了管理界面,方便管理和监控消息队列。默认情况下,管理界面的访问地址为http://localhost:15672,需要使用用户名和密码登录。

创建消息队列和交换机

在创建消息队列和交换机时,需要根据具体的应用需求进行配置。以下是一个使用RabbitMQ创建消息队列和交换机的示例代码:

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

public class QueueAndExchangeSetup {
    private static final String QUEUE_NAME = "my_queue";
    private static final String EXCHANGE_NAME = "my_exchange";

    public static void setup() throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 声明队列
        channel.queueDeclare(QUEUE_NAME, true, false, false, null);
        // 声明交换机
        channel.exchangeDeclare(EXCHANGE_NAME, "direct", true);

        channel.close();
        connection.close();
    }

    public static void main(String[] args) throws Exception {
        try {
            setup();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

发送消息到队列

发送消息到队列的步骤如下:

  1. 创建消息生产者:创建一个消息生产者,负责生产消息并发送到队列。
  2. 发送消息:消息生产者通过MQ消息队列的API将消息发送到指定的队列。

以下是一个使用RabbitMQ发送消息的示例代码:

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

public class MessageProducer {
    private static final String QUEUE_NAME = "my_queue";

    public static void sendMessage(String message) throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 发送消息到队列
        channel.queueDeclare(QUEUE_NAME, true, false, false, null);
        channel.basicPublish("", QUEUE_NAME, null, message.getBytes());
        System.out.println("Sent '" + message + "'");

        channel.close();
        connection.close();
    }

    public static void main(String[] args) throws Exception {
        try {
            sendMessage("Hello, World!");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

接收并处理消息

接收并处理消息的步骤如下:

  1. 创建消息消费者:创建一个消息消费者,负责从队列中接收消息,并对消息进行处理。
  2. 接收消息:消息消费者通过MQ消息队列的API从队列中接收消息。
  3. 处理消息:消息消费者对接收到的消息进行处理,并根据需要执行相应的操作。

以下是一个使用RabbitMQ接收并处理消息的示例代码:

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import com.rabbitmq.client.DeliverCallback;

public class MessageConsumer {
    private static final String QUEUE_NAME = "my_queue";

    public static void consume() throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 声明队列
        channel.queueDeclare(QUEUE_NAME, true, false, false, null);

        // 定义消息接收器
        DeliverCallback deliverCallback = (consumerTag, delivery) -> {
            String message = new String(delivery.getBody(), "UTF-8");
            System.out.println("Received '" + message + "'");
        };

        // 开始接收消息
        channel.basicConsume(QUEUE_NAME, true, deliverCallback, (consumerTag) -> {});
    }

    public static void main(String[] args) throws Exception {
        try {
            consume();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
常见问题与解决方案

MQ消息队列的性能优化

对于MQ消息队列的性能优化,可以从以下几个方面进行:

  1. 消息批量处理:将多个消息合并成一个批次进行处理,可以减少网络通信的开销。
  2. 消息分片:将大消息拆分成多个较小的消息进行处理,避免消息过大导致的性能瓶颈。
  3. 消息压缩:使用压缩算法对消息进行压缩,减少消息传输的网络开销。
  4. 消息缓存:在消息队列的客户端或服务器端缓存消息,减少消息传输的次数。
  5. 集群部署:通过集群部署增加消息队列的节点数量,提高消息处理的能力。

以下是一个使用RabbitMQ进行消息批量处理的示例代码:

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

public class BatchMessageProducer {
    private static final String QUEUE_NAME = "my_queue";

    public static void sendMessage(String message1, String message2) throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 发送多个消息到队列
        channel.queueDeclare(QUEUE_NAME, true, false, false, null);
        channel.basicPublish("", QUEUE_NAME, null, message1.getBytes());
        channel.basicPublish("", QUEUE_NAME, null, message2.getBytes());
        System.out.println("Sent '" + message1 + "' and '" + message2 + "'");

        channel.close();
        connection.close();
    }

    public static void main(String[] args) throws Exception {
        try {
            sendMessage("Hello, World!", "Hello, RabbitMQ!");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

网络连接问题及解决方法

网络连接问题是MQ消息队列中常见的问题之一,可以通过以下方法进行解决:

  1. 检查网络连接:确保MQ消息队列服务器和客户端之间的网络连接是稳定的。
  2. 优化网络配置:配置防火墙和路由规则,确保MQ消息队列的通信端口是开放的。
  3. 增加重试机制:在客户端代码中增加重试机制,当连接失败时自动重试连接。
  4. 使用负载均衡:通过负载均衡设备或配置提高MQ消息队列的网络连接性能。

以下是一个使用RabbitMQ增加重试机制的示例代码:

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import com.rabbitmq.client.Consumer;
import com.rabbitmq.client.DefaultConsumer;
import com.rabbitmq.client.GetResponse;

public class RetryableMessageConsumer {
    private static final String QUEUE_NAME = "my_queue";
    private static final int MAX_RETRIES = 3;

    public static void consume() throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 声明队列
        channel.queueDeclare(QUEUE_NAME, true, false, false, null);

        // 定义消息接收器
        Consumer consumer = new DefaultConsumer(channel) {
            @Override
            public void handleDelivery(String consumerTag, com.rabbitmq.client.Envelope envelope, com.rabbitmq.client.AMQP.BasicProperties properties, byte[] body) throws IOException {
                String message = new String(body, "UTF-8");
                System.out.println("Received '" + message + "'");

                // 模拟处理消息
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        };

        int retries = 0;
        while (retries < MAX_RETRIES) {
            try {
                channel.basicConsume(QUEUE_NAME, true, consumer);
                break;
            } catch (Exception e) {
                retries++;
                if (retries >= MAX_RETRIES) {
                    throw e;
                }
                Thread.sleep(1000);
            }
        }
    }

    public static void main(String[] args) throws Exception {
        try {
            consume();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

消息丢失和重复问题的处理

消息丢失和重复问题是MQ消息队列中常见的问题之一,可以通过以下方法进行解决:

  1. 消息持久化:将消息持久化到磁盘,即使在系统崩溃或重启后,消息也不会丢失。
  2. 事务处理:使用事务处理机制,确保消息在发送方和接收方之间的同步提交或回滚。
  3. 消息确认机制:消费者在消费消息后,需要向消息队列发送确认消息,通知消息队列消息已经被成功处理。
  4. 消息去重:在应用程序层面实现消息去重机制,确保重复消息不会被处理两次。

以下是一个使用RabbitMQ实现消息持久化和事务处理的示例代码:

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

public class PersistentMessageProducer {
    private static final String QUEUE_NAME = "my_queue";

    public static void sendMessage(String message) throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 设置消息队列为持久化
        channel.queueDeclare(QUEUE_NAME, true, false, false, null);

        // 设置消息持久化
        channel.basicPublish("", QUEUE_NAME, com.rabbitmq.client.MessageProperties.PERSISTENT_TEXT_PLAIN, message.getBytes());
        System.out.println("Sent '" + message + "'");

        channel.close();
        connection.close();
    }

    public static void main(String[] args) throws Exception {
        try {
            sendMessage("Hello, World!");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
实战演练:一个简单的MQ消息队列应用

示例应用的需求分析

假设你正在开发一个电商平台,用户可以下单购买商品。在下单过程中,需要将订单信息发送到后台处理系统,并在处理完成后返回处理结果给用户。为了提高系统的可扩展性和灵活性,你可以使用MQ消息队列来解耦下单系统和后台处理系统,使得下单系统和后台处理系统之间可以通过异步通信方式进行通信。

具体的需求如下:

  1. 下单系统:负责接收用户的下单请求,并将订单信息发送到MQ消息队列。
  2. 后台处理系统:从MQ消息队列中获取订单信息,并进行处理,处理完成后将结果发送到MQ消息队列。
  3. 结果返回系统:从MQ消息队列中获取处理结果,并返回给用户。

应用开发与部署流程

开发一个MQ消息队列应用的流程包括以下步骤:

  1. 安装MQ消息队列服务器:根据需求选择合适的MQ消息队列系统,并完成安装和配置。
  2. 创建消息队列和交换机:根据应用需求创建消息队列和交换机,并绑定队列和交换机。
  3. 开发下单系统:开发下单系统,实现下单请求的处理,并将订单信息发送到MQ消息队列。
  4. 开发后台处理系统:开发后台处理系统,从MQ消息队列中获取订单信息,并进行处理,处理完成后将结果发送到MQ消息队列。
  5. 开发结果返回系统:开发结果返回系统,从MQ消息队列中获取处理结果,并返回给用户。
  6. 部署和调试应用:部署应用,并进行测试,确保应用能够正常运行。

应用测试与调试

在测试和调试MQ消息队列应用时,需要注意以下几个方面:

  1. 消息发送和接收:确保消息能够正确地发送到MQ消息队列,并被接收方正确地消费。
  2. 消息确认机制:确保消息被正确地确认,并在确认失败时进行重试。
  3. 消息处理逻辑:确保后台处理系统的消息处理逻辑正确无误。
  4. 性能和稳定性:进行性能测试和稳定性测试,确保应用在高并发和长时间运行的情况下能够正常工作。

应用测试与调试

  • 调试步骤
    • 运行下单系统,并发送订单信息。
    • 查看后台处理系统是否接收到订单信息,并正确处理。
    • 检查结果返回系统是否正确地接收到处理结果,并返回给用户。
    • 使用日志记录关键步骤,确保每一步都按预期执行。

具体部署细节

  • 部署流程
    • 安装MQ消息队列服务器:根据文档说明,确保MQ消息队列服务器安装成功。
    • 创建消息队列和交换机:通过代码或MQ管理界面创建所需的消息队列和交换机。
    • 开发下单系统:开发下单系统的前端和后端逻辑,并确保其能够发送订单信息。
    • 部署后台处理系统:确保后台处理系统能够接收订单信息并返回处理结果。
    • 部署结果返回系统:确保结果返回系统能够接收到处理结果并反馈给用户。
    • 测试应用:进行端到端测试,确保整个流程正常工作。

以下是一个使用RabbitMQ实现下单系统、后台处理系统和结果返回系统的示例代码:

下单系统

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

public class OrderProducer {
    private static final String EXCHANGE_NAME = "order_exchange";
    private static final String ROUTING_KEY = "order_routing_key";

    public static void placeOrder(String orderId, String orderInfo) throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 发送订单信息到MQ消息队列
        channel.exchangeDeclare(EXCHANGE_NAME, "direct", true);
        channel.basicPublish(EXCHANGE_NAME, ROUTING_KEY, null, (orderId + ": " + orderInfo).getBytes());
        System.out.println("Placed order " + orderId);

        channel.close();
        connection.close();
    }

    public static void main(String[] args) throws Exception {
        try {
            placeOrder("12345", "User A ordered a book");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

后台处理系统

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import com.rabbitmq.client.DeliverCallback;

public class OrderProcessor {
    private static final String EXCHANGE_NAME = "order_exchange";
    private static final String ROUTING_KEY = "order_routing_key";
    private static final String RESULT_QUEUE_NAME = "result_queue";

    public static void processOrder() throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 声明队列和交换机
        channel.exchangeDeclare(EXCHANGE_NAME, "direct", true);
        channel.queueDeclare(RESULT_QUEUE_NAME, true, false, false, null);
        channel.queueBind(RESULT_QUEUE_NAME, EXCHANGE_NAME, ROUTING_KEY);

        // 定义消息接收器
        DeliverCallback deliverCallback = (consumerTag, delivery) -> {
            String message = new String(delivery.getBody(), "UTF-8");
            System.out.println("Received order: " + message);

            // 模拟处理订单
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

            // 发送处理结果到MQ消息队列
            String result = "Order processed: " + message;
            channel.basicPublish("", RESULT_QUEUE_NAME, null, result.getBytes());
            System.out.println("Sent result: " + result);
        };

        // 开始接收订单
        channel.basicConsume(RESULT_QUEUE_NAME, true, deliverCallback, (consumerTag) -> {});

        channel.close();
        connection.close();
    }

    public static void main(String[] args) throws Exception {
        try {
            processOrder();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

结果返回系统

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import com.rabbitmq.client.DeliverCallback;

public class OrderResultConsumer {
    private static final String RESULT_QUEUE_NAME = "result_queue";

    public static void consumeResult() throws Exception {
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("localhost");
        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();

        // 声明队列
        channel.queueDeclare(RESULT_QUEUE_NAME, true, false, false, null);

        // 定义消息接收器
        DeliverCallback deliverCallback = (consumerTag, delivery) -> {
            String message = new String(delivery.getBody(), "UTF-8");
            System.out.println("Received result: " + message);
        };

        // 开始接收结果
        channel.basicConsume(RESULT_QUEUE_NAME, true, deliverCallback, (consumerTag) -> {});

        channel.close();
        connection.close();
    }

    public static void main(String[] args) throws Exception {
        try {
            consumeResult();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

总结

通过以上步骤和示例代码的介绍,你可以了解到如何使用MQ消息队列来构建一个简单但功能完整的应用。MQ消息队列能够帮助你提高系统的可扩展性、灵活性和响应速度。在实际开发中,你还需要根据具体的需求和场景进行相应的配置和优化,以确保应用能够高效稳定地运行。更多关于MQ消息队列的详细信息和高级功能,可以参考相关的在线文档和教程。

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消