Netty Review - раскрыт механизм автоматического переподключения Netty: принципы и лучшие практики
Netty Review - раскрыт механизм автоматического переподключения Netty: принципы и лучшие практики
Каталог статей
  • Обзор
  • Pre
  • Клиент автоматически переподключается
  • Code
    • Server
    • Клиент (ключ)
  • тест
    • Начать автоматическое переподключение
    • Автоматическое переподключение после обрыва связи во время работы

Обзор


Pre

Обзор Netty — углубленное обсуждение механизма обнаружения пульса Netty: принципы, практика и анализ исходного кода IdleStateHandler.


Клиент автоматически переподключается

Автоматическое переподключение — это функция, используемая для повышения стабильности и надежности сетевых приложений. Когда соединение между клиентом и сервером неожиданно разрывается, клиент может автоматически попытаться повторно подключиться к серверу, чтобы обеспечить нормальную передачу данных.

Автоматическое повторное подключение относится к механизму, при котором клиент может автоматически попытаться восстановить соединение, когда соединение между клиентом и сервером по какой-либо причине разрывается. Эта функция используется для повышения стабильности и надежности сетевых приложений.

В частности, когда клиент обнаруживает, что соединение с сервером прервано, он автоматически инициирует новую попытку подключения, чтобы обеспечить нормальную передачу данных. Это важно для устранения нестабильности сети, временных отключений или перезапусков сервера, уменьшения вмешательства пользователя и улучшения взаимодействия с пользователем вашего приложения.


Code

Server

Язык кода:javascript
копировать
package com.artisan.reconnect;

import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelOption;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
/**
 * @author маленький мастер
 * @version 1.0
 * @mark: show me the code , change the world
 */
public class ArtisanNettyServer {

    public static void main(String[] args) throws Exception {
        // Создайте две группы потоков BossGroup и WorkerGroup. По умолчанию количество подпотоков, содержащихся в NioEventLoop, в два раза превышает количество ядер процессора.
        // BossGroup просто обрабатывает запросы на соединение , реальная бизнес-обработка будет передана рабочей группе.
        EventLoopGroup bossGroup = new NioEventLoopGroup(1);
        EventLoopGroup workerGroup = new NioEventLoopGroup(8);
        try {
            // Создайте объект запуска на стороне сервера.
            ServerBootstrap bootstrap = new ServerBootstrap();
            // Используйте цепное программирование для настройки параметров
            bootstrap.group(bossGroup, workerGroup) //Настраиваем две группы потоков
                    // Используйте NioServerSocketChannel в качестве реализации канала сервера.
                    .channel(NioServerSocketChannel.class)
                    // Инициализируйте размер очереди сервера. Сервер обрабатывает клиентсоединять запросы последовательно, поэтому одновременно может обрабатываться только один клиентсоединять.
                    // Когда одновременно приходит несколько клиентов, сервер поместит запрос клиентсоединять, который не может быть обработан, в очередь и будет ждать обработки.
                    .option(ChannelOption.SO_BACKLOG, 1024)
                    .childHandler(new ChannelInitializer<SocketChannel>() {//Создаем объект инициализации канала, задаем параметры инициализации, в SocketChannel Выполните перед настройкой

                        @Override
                        protected void initChannel(SocketChannel ch) throws Exception {
                            //Устанавливаем процессор для SocketChannel рабочей группы
                            //ch.pipeline().addLast(new LifeCycleInBoundHandler());
                            ch.pipeline().addLast(new ArtisanNettyServerHandler());
                        }
                    });
            System.out.println("netty server start。。");
            // Привяжите порт и синхронизируйте, Создается асинхронный объект ChannelFuture, и о состоянии выполнения асинхронного события можно судить с помощью isDone() и других методов.
            // Запустите сервер (и привяжите порт), привязка — это асинхронная операция, а метод синхронизации — ожидание завершения асинхронной операции.
            ChannelFuture cf = bootstrap.bind(9000).sync();
            // Зарегистрируйте прослушиватель для cf, чтобы слушать события, которые нас интересуют
            /*cf.addListener(new ChannelFutureListener() {
                @Override
                public void operationComplete(ChannelFuture future) throws Exception {
                    if (cf.isSuccess()) {
                        System.out.println("Успешно прослушивается порт 9000");
                    } else {
                        System.out.println("Ошибка прослушивания порта 9000");
                    }
                }
            });*/
            // Подождите, пока порт прослушивания сервера закроется. closeFuture — асинхронная операция.
            // Синхронно дождаться завершения процесса закрытия канала с помощью метода синхронизации. Это заблокирует и дождется завершения закрытия канала. Метод wait() объекта вызывается внутренне.
            cf.channel().closeFuture().sync();
        } finally {
            bossGroup.shutdownGracefully();
            workerGroup.shutdownGracefully();
        }
    }
}
  1. EventLoopGroup:NettyиспользоватьEventLoopGroupдля обработки цикла событий иIOдействовать。Здесь созданы дваEventLoopGroup,один для обработкисоединятьпросить(bossGroup),另один для обработки实际избизнес-логика(workerGroup)。
  2. ServerBootstrap:ЭтоNettyиз Еще один основной компонент,Используется для настройки и инициализации сервера.
  3. ChannelFuture:Это一个异步результат对象,用于表示通道действоватьизрезультат。
  4. ChannelInitializer:Это一个用于初始化新соединятьизпроцессор。
  5. ArtisanNettyServerHandler:Это должно быть обычаемизкласс обработки,Используется для обработки бизнес-логики,Приведено ниже.
  6. bind()иcloseFuture()bind()Метод используется для запуска сервера и привязки порта,closeFuture()Используется для ожидания закрытия канала сервера。
  7. finallyкусок:Здесь убедитесь, что после запуска сервера произойдет сбой или успех,EventLoopGroupбудет закрыто изящно,освободить ресурсы.

Язык кода:javascript
копировать
package com.artisan.reconnect;

import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.util.CharsetUtil;

/**
 * @author маленький мастер
 * @version 1.0
 * @mark: show me the code , change the world
 * @Description: Пользовательский обработчик должен наследовать определенный HandlerAdapter (спецификацию), указанный netty.
 */
public class ArtisanNettyServerHandler extends ChannelInboundHandlerAdapter {

    /**
     * Прочитайте данные, отправленные клиентом
     *
     * @param ctx объект контекста, Содержит каналы, трубопроводы
     * @param msg Это данные, отправленные клиентом
     * @throws Exception
     */
    @Override
    public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
        System.out.println("Поток чтения сервера " + Thread.currentThread().getName());
        //Channel channel = ctx.channel();
        //ChannelPipeline pipeline = ctx.pipeline(); //По сути, это двусторонняя ссылка, исходящий входящий
        //Воля msg преобразовать в один ByteBuf, аналог NIO из ByteBuffer
        ByteBuf buf = (ByteBuf) msg;
        System.out.println("Клиент отправил сообщение:" + buf.toString(CharsetUtil.UTF_8));
    }

    /**
     * Метод обработки завершения чтения данных
     *
     * @param ctx
     * @throws Exception
     */
    @Override
    public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {
        ByteBuf buf = Unpooled.copiedBuffer("HelloClient".getBytes(CharsetUtil.UTF_8));
        ctx.writeAndFlush(buf);
    }

    /**
     * обрабатывать исключения, В общем, надо закрыть канал
     *
     * @param ctx
     * @param cause
     * @throws Exception
     */
    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
        ctx.close();
    }
}

определяет обычайизNetty服务器процессорArtisanNettyServerHandler,он наследует отChannelInboundHandlerAdapter。这个процессор Включать了几个重要изметод борьбы склиентизпроситьи响应:

  1. channelRead(ChannelHandlerContext ctx, Object msg):Когда сервер запускается склиент Когда данные получены,Этот метод будет вызван. В этом методе,Вы можете написать логику, которая обрабатывает отправку данных. В этом примере,Он просто печатает полученныйиз Содержание сообщения。
  2. channelReadComplete(ChannelHandlerContext ctx):Этот методchannelReadВызывается после завершения выполнения метода。В этом методе,Вы можете отправлять ответы клиенту. В этом примере,它отправлять了一个简单из"HelloClient"сообщение дляклиент。
  3. exceptionCaught(ChannelHandlerContext ctx, Throwable cause):Этот метод出现异常时被调用。В этом методе,Вы можете написать обработку исключенийизлогика。В этом примере,Он просто закрывает канал.

Клиент (ключ)

Этот код представляет собой простой пример клиента, использующего платформу Netty, реализующую функцию переподключения. Ниже приводится объяснение этого кода на китайском языке:

Язык кода:javascript
копировать
package com.artisan.reconnect;
import io.netty.bootstrap.Bootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioSocketChannel;
import java.util.concurrent.TimeUnit;
/**
 * @author маленький мастер
 * @version 1.0
 * @mark: show me the code , change the world
 * @Description: Достигнуто переподключениеизклиент
 */
public class ArtisanNettyClient {
    private String host;
    private int port;
    private Bootstrap bootstrap;
    private EventLoopGroup group;
    public static void main(String[] args) throws Exception {
        ArtisanNettyClient artisanNettyClient = new ArtisanNettyClient("localhost", 9000);
        artisanNettyClient.connect();
    }
    public ArtisanNettyClient(String host, int port) {
        this.host = host;
        this.port = port;
        init();
    }
    private void init() {
        // клиенту требуется группа цикла событий
        group = new NioEventLoopGroup();
        // Создать объект запуска клиента
        // bootstrap многоразовый, Просто инициализируйте его при создании экземпляра NettyClient.
        bootstrap = new Bootstrap();
        bootstrap.group(group)
                .channel(NioSocketChannel.class)
                .handler(new ChannelInitializer<SocketChannel>() {
                    @Override
                    protected void initChannel(SocketChannel ch) throws Exception {
                        // Добавить процессор
                        ch.pipeline().addLast(new ArtisanNettyClientHandler(ArtisanNettyClient.this));
                    }
                });
    }
    public void connect() throws Exception {
        System.out.println("netty client start。。");
        // Начать клиент подключать серверную часть
        ChannelFuture cf = bootstrap.connect(host, port);
        cf.addListener(new ChannelFutureListener() {
            @Override
            public void operationComplete(ChannelFuture future) throws Exception {
                if (!future.isSuccess()) {
                    // Повторное соединение передается внутреннему потоку для выполнения.
                    future.channel().eventLoop().schedule(() -> {
                        System.err.println("Повторно подключиться к серверу...");
                        try {
                            connect();
                        } catch (Exception e) {
                            e.printStackTrace();
                        }
                    }, 3000, TimeUnit.MILLISECONDS);
                } else {
                    System.out.println("Сервер соединился успешно...");
                }
            }
        });
        // Мониторинг закрытия канала
        cf.channel().closeFuture().sync();
    }
}

Этот код определяет файл с именемArtisanNettyClientиздобрый,он содержитклиентиз初始化исоединятьлогика。

  • EventLoopGroup:NettyиспользоватьEventLoopGroupдля обработки цикла событий иIOдействовать。Здесь созданNioEventLoopGroup,для обработкиклиентизIOдействовать。
  • Bootstrap:ЭтоNettyиз Еще один основной компонент,Используется для настройки и инициализации клиента.
  • ChannelFuture:Это一个异步результат对象,用于表示通道действоватьизрезультат。
  • connect()метод:这个метод用于启动клиентисоединятьна сервер。еслисоединятьнеудача,它将использоватьscheduleметод在3Повторите попытку через несколько секундсоединять。
  • ArtisanNettyClientHandler:Это должно быть обычаемизкласс обработки,Используется для обработки бизнес-логики,Но в этом коде не приведена конкретная реализация.
  • init()метод:这个метод用于初始化клиентизBootstrapиEventLoopGroup
  • operationComplete()метод:ЭтоChannelFutureListenerиз回调метод,для обработкисоединятьдействоватьизрезультат。еслисоединятьнеудача,Он организует повторную попытку соединения. Если соединение прошло успешно,Он напечатает сообщение об успехе.
  • closeFuture().sync():这个метод用于等待клиентканал закрыт,Обязательноклиент Заполните все необходимое перед закрытиемизочистка。

В этом примере клиент попытается подключиться к указанному адресу и порту сервера, и если соединение не удастся, он автоматически повторит попытку подключения.


Язык кода:javascript
копировать
package com.artisan.reconnect;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.util.CharsetUtil;
/**
 * @author маленький мастер
 * @version 1.0
 * @mark: show me the code , change the world
 */
public class ArtisanNettyClientHandler extends ChannelInboundHandlerAdapter {
    private ArtisanNettyClient artisanNettyClient;
    public ArtisanNettyClientHandler(ArtisanNettyClient artisanNettyClient) {
        this.artisanNettyClient = artisanNettyClient;
    }
    /**
     * Этот метод будет запущен, когда сервер завершит работу.
     *
     * @param ctx
     * @throws Exception
     */
    @Override
    public void channelActive(ChannelHandlerContext ctx) throws Exception {
        // СоздайтеByteBuf,Содержит строку «HelloServer»из
        ByteBuf buf = Unpooled.copiedBuffer("HelloServer".getBytes(CharsetUtil.UTF_8));
        // Отправить сообщение на сервер
        ctx.writeAndFlush(buf);
    }
    /**
     * Он сработает, когда на канале произойдет событие чтения, то есть сервер отправит данные клиенту.
     *
     * @param ctx
     * @param msg
     * @throws Exception
     */
    @Override
    public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
        // Получить содержимое сообщения
        ByteBuf buf = (ByteBuf) msg;
        // 打印服务端отправлятьиз消息
        System.out.println("Получено сообщение от сервера:" + buf.toString(CharsetUtil.UTF_8));
        // Адрес сервера печати
        System.out.println("Адрес сервера: " + ctx.channel().remoteAddress());
    }
    /**
     * Вызывается, когда канал неактивен
     *
     * @param ctx
     * @throws Exception
     */
    @Override
    public void channelInactive(ChannelHandlerContext ctx) throws Exception {
        // Распечатать сообщение с подсказкой
        System.err.println("Отключить и снова подключить во время работы...");
        // 调用клиентизconnectметод进行重连
        artisanNettyClient.connect();
    }
    /**
     * Вызывается при обнаружении исключения
     *
     * @param ctx
     * @param cause
     * @throws Exception
     */
    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
        // Распечатать информацию о стеке исключений
        cause.printStackTrace();
        // Закрыть канал
        ctx.close();
    }
}

Этот класс обработчика содержит некоторые основные методы для обработки активации, чтения, неактивности и исключений канала. Вот краткое описание каждого метода:

  • channelActive():когдаклиентуспехсоединятьна сервер时,Этот метод будет называться,и отправляет сообщение на сервер.
  • channelRead():когдаклиент Когда сообщение получено с сервера,Этот метод будет называться,и打印出接收到из Содержание сообщенияи服务器изадрес。
  • channelInactive():когда通道不再活跃时(Например,связь отключена),Этот метод будет называться,И попробуйте перезагрузить сервер.
  • exceptionCaught():когда捕获到异常时,Этот метод будет называться,и打印异常из Информация трассировки стека,Затем Закрыть канал。

Этот класс обработки является частью клиентской логики и отвечает за взаимодействие между клиентом и сервером.


тест

Начать автоматическое переподключение

Сначала запустите клиент (обязательно), а затем запустите сервер, чтобы проверить автоматическое переподключение клиента.

Запускайте клиент, а не сервер

Запустить сервер


Автоматическое переподключение после обрыва связи во время работы

Сбой сети или сбой сервера во время работы системы,В результате клиент отключается от сервера и его необходимо подключить заново.,Можно найти вклиент处理данныеизHandlerизchannelInactiveметод中进行重连。

Во время работы отключите сервер и перезапустите его через некоторое время, чтобы убедиться, что клиент автоматически подключается во время работы.

Отключить сервер

Восстановить сервер

boy illustration
Неразрушающее увеличение изображений одним щелчком мыши, чтобы сделать их более четкими артефактами искусственного интеллекта, включая руководства по установке и использованию.
boy illustration
Копикодер: этот инструмент отлично работает с Cursor, Bolt и V0! Предоставьте более качественные подсказки для разработки интерфейса (создание навигационного веб-сайта с использованием искусственного интеллекта).
boy illustration
Новый бесплатный RooCline превосходит Cline v3.1? ! Быстрее, умнее и лучше вилка Cline! (Независимое программирование AI, порог 0)
boy illustration
Разработав более 10 проектов с помощью Cursor, я собрал 10 примеров и 60 подсказок.
boy illustration
Я потратил 72 часа на изучение курсорных агентов, и вот неоспоримые факты, которыми я должен поделиться!
boy illustration
Идеальная интеграция Cursor и DeepSeek API
boy illustration
DeepSeek V3 снижает затраты на обучение больших моделей
boy illustration
Артефакт, увеличивающий количество очков: на основе улучшения характеристик препятствия малым целям Yolov8 (SEAM, MultiSEAM).
boy illustration
DeepSeek V3 раскручивался уже три дня. Сегодня я попробовал самопровозглашенную модель «ChatGPT».
boy illustration
Open Devin — инженер-программист искусственного интеллекта с открытым исходным кодом, который меньше программирует и больше создает.
boy illustration
Эксклюзивное оригинальное улучшение YOLOv8: собственная разработка SPPF | SPPF сочетается с воспринимаемой большой сверткой ядра UniRepLK, а свертка с большим ядром + без расширения улучшает восприимчивое поле
boy illustration
Популярное и подробное объяснение DeepSeek-V3: от его появления до преимуществ и сравнения с GPT-4o.
boy illustration
9 основных словесных инструкций по доработке академических работ с помощью ChatGPT, эффективных и практичных, которые стоит собрать
boy illustration
Вызовите deepseek в vscode для реализации программирования с помощью искусственного интеллекта.
boy illustration
Познакомьтесь с принципами сверточных нейронных сетей (CNN) в одной статье (суперподробно)
boy illustration
50,3 тыс. звезд! Immich: автономное решение для резервного копирования фотографий и видео, которое экономит деньги и избавляет от беспокойства.
boy illustration
Cloud Native|Практика: установка Dashbaord для K8s, графика неплохая
boy illustration
Краткий обзор статьи — использование синтетических данных при обучении больших моделей и оптимизации производительности
boy illustration
MiniPerplx: новая поисковая система искусственного интеллекта с открытым исходным кодом, спонсируемая xAI и Vercel.
boy illustration
Конструкция сервиса Synology Drive сочетает проникновение в интрасеть и синхронизацию папок заметок Obsidian в облаке.
boy illustration
Центр конфигурации————Накос
boy illustration
Начинаем с нуля при разработке в облаке Copilot: начать разработку с минимальным использованием кода стало проще
boy illustration
[Серия Docker] Docker создает мультиплатформенные образы: практика архитектуры Arm64
boy illustration
Обновление новых возможностей coze | Я использовал coze для создания апплета помощника по исправлению домашних заданий по математике
boy illustration
Советы по развертыванию Nginx: практическое создание статических веб-сайтов на облачных серверах
boy illustration
Feiniu fnos использует Docker для развертывания личного блокнота Notepad
boy illustration
Сверточная нейронная сеть VGG реализует классификацию изображений Cifar10 — практический опыт Pytorch
boy illustration
Начало работы с EdgeonePages — новым недорогим решением для хостинга веб-сайтов
boy illustration
[Зона легкого облачного игрового сервера] Управление игровыми архивами
boy illustration
Развертывание SpringCloud-проекта на базе Docker и Docker-Compose