netty基本使用- socket通信

netty基本使用- socket通信,第1张

[netty 基本使用- 作为http服务器][gcssloop]

[gcssloop]: http://www.jianshu.com/p/cd88723c96dc

ServerSocket.java

** ServerInitializer.java **

** ServerHandler.java 处理业务 **

** ClientSocket.java **

** Clientinitializer.java **

**ClientHandler.java 处理业务 **

** 可以多次和服务器端通信的写法 **

netty 常用的处理大数据分包传输问题的解决类。

编码类,自动将

+----------------+

| "HELLO, WORLD" |

+----------------+

格式的数据转换成

+--------+----------------+

+--------+----------------+

格式的数据

[netty 数据分包、组包、粘包处理机制][123]

[123]: http://blog.163.com/linfenliang@126/blog/static/127857195201210821145721/

netty提供了客户端和服务端之间基于socket的服务调用。用于可以定制自己的协议和规范。dubbo框架底层正是使用netty进行相关的数据传输。

下面我们实现客户端像服务器端发送请求,服务器端接受消息后向客户端进行响应。

服务器端自定义handler

客户端自定义handler

分别启动服务器端和客户端,观察控制台输出:

netty v3.9.4

websocket连接建立前,客户端需要与服务器进行握手(http协议) 确认websocket连接,也就是说在处理websocket请求前,必需要处理一些http请求。

websocket到现在为止,已经有多个版本,netty有相应的对应类,这部分处理一般不需要人工干预。

如果运行正常的话,会在页面的文本框中显示1-20记数。

可以通过firefox或chrome的开发人员工具,显看浏览器与服务器的交互。

主要是HttpServerChannelHandler2,加了些注释和少量debug代码。

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

package org.sl.demo.httpserver1

import java.util.List

import java.util.Map

import org.jboss.netty.channel.Channel

import org.jboss.netty.channel.ChannelHandlerContext

import org.jboss.netty.channel.MessageEvent

import org.jboss.netty.channel.SimpleChannelHandler

import org.jboss.netty.handler.codec.http.DefaultHttpResponse

import org.jboss.netty.handler.codec.http.HttpHeaders

import org.jboss.netty.handler.codec.http.HttpMethod

import org.jboss.netty.handler.codec.http.HttpRequest

import org.jboss.netty.handler.codec.http.HttpResponseStatus

import org.jboss.netty.handler.codec.http.HttpVersion

import org.jboss.netty.handler.codec.http.websocketx.CloseWebSocketFrame

import org.jboss.netty.handler.codec.http.websocketx.PingWebSocketFrame

import org.jboss.netty.handler.codec.http.websocketx.PongWebSocketFrame

import org.jboss.netty.handler.codec.http.websocketx.TextWebSocketFrame

import org.jboss.netty.handler.codec.http.websocketx.WebSocketFrame

import org.jboss.netty.handler.codec.http.websocketx.WebSocketServerHandshaker

import org.jboss.netty.handler.codec.http.websocketx.WebSocketServerHandshakerFactory

public class HttpServerChannelHandler2 extends SimpleChannelHandler{

public static boolean debug = true

@Override

public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) {

Channel ch = e.getChannel()

Object msg = e.getMessage()

if(debug){

System.out.println("---------------")

System.out.println("message: "+msg.getClass())

}

//虽然是websocket,但在建立websocket连接前,先进行http握手,所以,这时也要处理http请求

//在http握手完成后,才是websocket下的通信

if(msg instanceof HttpRequest){

processHttpRequest(ch, (HttpRequest)msg)

}else if(msg instanceof WebSocketFrame){

processWebsocketRequest(ch,(WebSocketFrame)msg)

}else{

//未处理的请求类型

}

}

//这个方法:

//1.完成websocket前的http握手

//2.屏蔽掉非websocket握手请求

void processHttpRequest(Channel channel,HttpRequest request){

HttpHeaders headers = request.headers()

if(debug){

List<Map.Entry<String,String>>ls = headers.entries()

for(Map.Entry<String,String>i: ls){

System.out.println("header "+i.getKey()+":"+i.getValue())

}

}

//屏蔽掉非websocket握手请求

//只接受http GET和headers['Upgrade']为'websocket'的http请求

if(!HttpMethod.GET.equals(request.getMethod())

|| !"websocket".equalsIgnoreCase(headers.get("Upgrade"))){

DefaultHttpResponse resp = new DefaultHttpResponse(

HttpVersion.HTTP_1_1,

HttpResponseStatus.BAD_REQUEST)

channel.write(resp)

channel.close()

return

}

WebSocketServerHandshakerFactory wsShakerFactory = new WebSocketServerHandshakerFactory(

"ws://"+request.headers().get(HttpHeaders.Names.HOST),

null,false )

WebSocketServerHandshaker wsShakerHandler = wsShakerFactory.newHandshaker(request)

if(null==wsShakerHandler){

//无法处理的websocket版本

wsShakerFactory.sendUnsupportedWebSocketVersionResponse(channel)

}else{

//向客户端发送websocket握手,完成握手

//客户端收到的状态是101 sitching protocol

wsShakerHandler.handshake(channel, request)

}

}

//websocket通信

void processWebsocketRequest(Channel channel, WebSocketFrame request){

if(request instanceof CloseWebSocketFrame){

channel.close()

}else if(request instanceof PingWebSocketFrame){

channel.write(new PongWebSocketFrame(request.getBinaryData()))

}else if(request instanceof TextWebSocketFrame){

//这个地方 可以根据需求,加上一些业务逻辑

TextWebSocketFrame txtReq = (TextWebSocketFrame) request

if(debug){ System.out.println("txtReq:"+txtReq.getText())}

//向ws客户端发送多个响应

for(int i=1i<=20i++){

channel.write(new TextWebSocketFrame(""+i))

try{Thread.sleep(300)}catch(Exception ex){}

}

}else{

//WebSocketFrame还有一些

}

}

}

其他类跟网上的差不多: http://blog.csdn.net/shagoo/article/details/8028813

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

package org.sl.demo.httpserver1

import java.net.InetSocketAddress

import java.util.concurrent.Executors

import org.jboss.netty.bootstrap.ServerBootstrap

import org.jboss.netty.channel.socket.nio.NioServerSocketChannelFactory

public class HttpServer implements Runnable{

int port = 80

public HttpServer(int port){

this.port = port

}

@Override

public void run() {

ServerBootstrap b = new ServerBootstrap(

new NioServerSocketChannelFactory(

Executors.newCachedThreadPool(),

Executors.newCachedThreadPool()))

b.setPipelineFactory(new HttpServerChannelPipelineFactory())

b.setOption("child.tcpNoDelay", true)

b.setOption("child.keepAlive", true)

b.bind(new InetSocketAddress(port))

}

public static void main(String[] args){

new HttpServer(80).run()

}

}

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

package org.sl.demo.httpserver1

import org.jboss.netty.channel.ChannelPipeline

import org.jboss.netty.channel.ChannelPipelineFactory

import org.jboss.netty.channel.Channels

import org.jboss.netty.handler.codec.http.HttpRequestDecoder

import org.jboss.netty.handler.codec.http.HttpResponseEncoder

import org.jboss.netty.handler.codec.http.websocketx.WebSocket00FrameDecoder

import org.jboss.netty.handler.codec.http.websocketx.WebSocket08FrameDecoder

public class HttpServerChannelPipelineFactory implements ChannelPipelineFactory {

@Override

public ChannelPipeline getPipeline() throws Exception {

ChannelPipeline cp = Channels.pipeline()

cp.addLast("decoder", new HttpRequestDecoder())

// cp.addLast("decoder", new WebSocket00FrameDecoder())

cp.addLast("encoder", new HttpResponseEncoder())

// cp.addLast("downhandler", new HttpServerDownstreamHandler())

// cp.addLast("uphandler", new HttpServerUpstreamHandler())

cp.addLast("handler", new HttpServerChannelHandler2())

return cp

}

}

测试页面:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

<html>

<head>

<script >

function connect1(){

alert('connect1')

var ta = document.getElementById('responseText')

var socket = new WebSocket('ws://127.0.0.1/websocket')

if (window.WebSocket) {

}else{

alert('Your browser does not support Web Socket.')

return

}

socket.onopen = function(event) {

ta.value = "Web Socket opened!"

}

socket.onmessage = function(event) {

ta.value = event.data

}

socket.onclose = function(event) {

ta.value = "Web Socket closed"

}

}

function connect() {

alert('connect')

var socket

if (!window.WebSocket) {

window.WebSocket = window.MozWebSocket

}

if (window.WebSocket) {

socket = new WebSocket("ws://127.0.0.1/websocket")

socket.onmessage = function(event) {

var ta = document.getElementById('responseText')

ta.value = event.data

}

socket.onopen = function(event) {

var ta = document.getElementById('responseText')

ta.value = "Web Socket opened!"

socket.send('hello')

}

socket.onclose = function(event) {

var ta = document.getElementById('responseText')

ta.value = "Web Socket closed"

}

socket.onerror = function(event){

}

} else {

alert("Your browser does not support Web Socket.")

}

}

</script>

</head>

<body>

<input type="button" onclick="connect1()" value="ws connect1"/><br/><br/>

<input type="button" onclick="connect()" value="ws connect"/><br/><br/>

<input id="responseText" type="text" value="123456"/>

</body>

</html>


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/sjk/6664277.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-03-26
下一篇 2023-03-26

发表评论

登录后才能评论

评论列表(0条)

保存