搭建第一個(gè)Netty服務(wù)器和客戶端

搭建第一個(gè)Netty應(yīng)用程序

本篇文章主要記錄如何搭建一個(gè)簡單的Netty服務(wù)器和客戶端。

依賴環(huán)境:

  • Java 8
  • Netty4
<dependencies>
    <dependency>
        <groupId>io.netty</groupId>
        <artifactId>netty-all</artifactId>
        <version>4.1.11.Final</version>
    </dependency>
</dependencies>

編寫Echo服務(wù)器

所有的Netty服務(wù)器都需要以下兩個(gè)部分嗽仪。

  • 至少一個(gè)ChannelHandler:用來處理業(yè)務(wù)邏輯
  • 引導(dǎo):配置服務(wù)器啟動(dòng)代碼产舞,將服務(wù)器綁定到監(jiān)聽連接請(qǐng)求的端口上豁辉。

EchoServerHandler類:用來處理核心業(yè)務(wù)邏輯

@ChannelHandler.Sharable
public class EchoServerHandler extends ChannelInboundHandlerAdapter{

    @Override
    public void channelRead(ChannelHandlerContext context, Object msg) {
        ByteBuf in = (ByteBuf) msg;
        System.out.println("server received: " + in.toString(CharsetUtil.UTF_8));
        context.write(in);
    }

    @Override
    public void channelReadComplete(ChannelHandlerContext ctx) {
        ctx.writeAndFlush(Unpooled.EMPTY_BUFFER).addListener(ChannelFutureListener.CLOSE);
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext context, Throwable couse) {
        couse.printStackTrace();
        context.close();
    }
}

EchoServer類:配置服務(wù)器屋确,并將入站消息通知給EchoServerHandler實(shí)例撮奏。

public class EchoServer {

    private final int port;

    public EchoServer(int port) {
        this.port = port;
    }

    public static void main(String[] args) throws Exception {
        System.err.println("Usage: " + EchoServer.class.getSimpleName() + "<port>");
        int port = 8999;
        new EchoServer(port).start();
    }

    public void start() throws Exception{
        final EchoServerHandler serverHandler = new EchoServerHandler();
        EventLoopGroup group = new NioEventLoopGroup();
        try {
            ServerBootstrap b = new ServerBootstrap();
            b.group(group).channel(NioServerSocketChannel.class)
             .localAddress(new InetSocketAddress(port)).childHandler(new ChannelInitializer<SocketChannel>() {
                @Override
                protected void initChannel(SocketChannel socketChannel) throws Exception {
                    socketChannel.pipeline().addLast(serverHandler);
                }
            });
            ChannelFuture future = b.bind().sync();
            future.channel().closeFuture().sync();
        } finally {
            group.shutdownGracefully().sync();
        }
    }
}

服務(wù)器主要的代碼組件

  • EchoServerHandler實(shí)現(xiàn)業(yè)務(wù)邏輯
  • main()引導(dǎo)服務(wù)器

引導(dǎo)過程:

  • 創(chuàng)建一個(gè)ServerBootstap實(shí)例以引導(dǎo)和綁定服務(wù)器隅要;
  • 創(chuàng)建并分配一個(gè)NioEventLoopGroup實(shí)例以進(jìn)行事件的處理报账,如接受新連接以及讀寫數(shù)據(jù)研底;
  • 指定服務(wù)器綁定的本地的InetSocketAddress;
  • 使用一個(gè)Handler實(shí)例初始化新的Channel透罢;
  • 調(diào)用ServerBootstap.bind()方法以綁定服務(wù)器榜晦;

編寫Echo客戶端

Echo客戶端:

  1. 連接服務(wù)器
  2. 發(fā)送一個(gè)或多個(gè)消息
  3. 對(duì)于每個(gè)消息,等待并接受從服務(wù)器發(fā)回的相同的消息
  4. 關(guān)閉連接

ChannelHandler類:實(shí)現(xiàn)客戶端邏輯

  • channelActive()和服務(wù)器建立連接后調(diào)用
  • channelRead()從服務(wù)器接受到一條信息后調(diào)用
  • exceptionCaught()處理過程中出現(xiàn)異常時(shí)被調(diào)用
@ChannelHandler.Sharable
public class EchoClientHandle extends SimpleChannelInboundHandler<ByteBuf>{

    @Override
    public void channelActive(ChannelHandlerContext context) {
        context.writeAndFlush(Unpooled.copiedBuffer("Netty rocks", CharsetUtil.UTF_8));
    }

    @Override
    protected void channelRead0(ChannelHandlerContext channelHandlerContext, ByteBuf byteBuf) throws Exception {
        System.out.println("Client received: " + byteBuf.toString(CharsetUtil.UTF_8));
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext context, Throwable cause) {
        cause.printStackTrace();
        context.close();
    }
}

EchoClient類:引導(dǎo)客戶端

public class EchoClient {

    // 使用的是服務(wù)器的地址
    private final String host;
    private final int port;

    public EchoClient(String host, int port) {
        this.host = host;
        this.port = port;
    }

    public void start() throws Exception{
        EventLoopGroup group = new NioEventLoopGroup();
        try {
            Bootstrap b = new Bootstrap();
            b.group(group)
             .channel(NioSocketChannel.class)
             .remoteAddress(new InetSocketAddress(host, port))
             .handler(new ChannelInitializer<SocketChannel>() {
                 @Override
                 protected void initChannel(SocketChannel socketChannel) throws Exception {
                     socketChannel.pipeline().addLast(new EchoClientHandle());
                 }
             });
            ChannelFuture future = b.connect().sync();
            future.channel().closeFuture().sync();
        } finally {
            group.shutdownGracefully().sync();
        }
    }

    public static void main(String[] args) throws Exception {
        new EchoClient("localhost", 8999).start();
    }
}

GitHub地址:https://github.com/huangbuhuan/netty-demo

參考連接

  • 《Netty In Action》
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末羽圃,一起剝皮案震驚了整個(gè)濱河市乾胶,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌朽寞,老刑警劉巖识窿,帶你破解...
    沈念sama閱讀 218,941評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異脑融,居然都是意外死亡喻频,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,397評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門肘迎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來甥温,“玉大人,你說我怎么就攤上這事妓布∫鲵荆” “怎么了?”我有些...
    開封第一講書人閱讀 165,345評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵匣沼,是天一觀的道長狰挡。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么圆兵? 我笑而不...
    開封第一講書人閱讀 58,851評(píng)論 1 295
  • 正文 為了忘掉前任跺讯,我火速辦了婚禮,結(jié)果婚禮上殉农,老公的妹妹穿的比我還像新娘刀脏。我一直安慰自己,他們只是感情好超凳,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,868評(píng)論 6 392
  • 文/花漫 我一把揭開白布愈污。 她就那樣靜靜地躺著,像睡著了一般轮傍。 火紅的嫁衣襯著肌膚如雪暂雹。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,688評(píng)論 1 305
  • 那天创夜,我揣著相機(jī)與錄音杭跪,去河邊找鬼。 笑死驰吓,一個(gè)胖子當(dāng)著我的面吹牛涧尿,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播檬贰,決...
    沈念sama閱讀 40,414評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼姑廉,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼!你這毒婦竟也來了翁涤?” 一聲冷哼從身側(cè)響起桥言,我...
    開封第一講書人閱讀 39,319評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎葵礼,沒想到半個(gè)月后号阿,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,775評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡鸳粉,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評(píng)論 3 336
  • 正文 我和宋清朗相戀三年倦西,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片赁严。...
    茶點(diǎn)故事閱讀 40,096評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡扰柠,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出疼约,到底是詐尸還是另有隱情卤档,我是刑警寧澤,帶...
    沈念sama閱讀 35,789評(píng)論 5 346
  • 正文 年R本政府宣布程剥,位于F島的核電站劝枣,受9級(jí)特大地震影響汤踏,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜舔腾,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,437評(píng)論 3 331
  • 文/蒙蒙 一溪胶、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧稳诚,春花似錦哗脖、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,993評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至氨距,卻和暖如春桑逝,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背俏让。 一陣腳步聲響...
    開封第一講書人閱讀 33,107評(píng)論 1 271
  • 我被黑心中介騙來泰國打工楞遏, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人首昔。 一個(gè)月前我還...
    沈念sama閱讀 48,308評(píng)論 3 372
  • 正文 我出身青樓寡喝,卻偏偏與公主長得像,于是被迫代替她去往敵國和親沙廉。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,037評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容