项目开发中,有个需求是接收udp的组播信息,获取帧并解析其中的内容。之前没有接触过udp的通讯以及数据帧。查阅资料对udp的通讯间的发送与接收实现记录如下。
UDP有三种通讯方式,分别是,单播 、广播、还有组播。
单播: 单机与单机之间的通讯
广播: 当前主机与所在网络中的所有主机通讯
组播: 当前主机与选定的一组主机通讯
一、单播
发送端代码
public class UdpServer {
/**
* 发送端
*/
public static void main(String[] args) throws Exception {
System.out.println("=============发送端启动===========");
// 1.创建发送端对象 参数是绑定本地地址和一个特定的端口号
DatagramSocket socket = new DatagramSocket(6666);
// 2.创建一个数据包对象封装数据
/**
* 参数1:封装要发送的数据
* 参数2:发送数据的大小
* 参数3:服务端的IP地址
* 参数4:服务端的端口
*/
byte[] bytes = "这是一条yyt的测试数据".getBytes();
InetAddress inetAddress = InetAddress.getByName("127.0.0.1");
int port = 8888 ;
DatagramPacket packet = new DatagramPacket(bytes,bytes.length, inetAddress,port);
// 3.发送数据
socket.send(packet);
// 4.关闭管道
socket.close();
}
}
接收端代码
public class UdpClient {
public static void main(String[] args) throws Exception {
System.out.println("=============客户端启动===========");
// 1.创建接受对象 参数是绑定本地地址和一个特定的端口号
DatagramSocket socket = new DatagramSocket(8888);
// 2.创建一个数据包接收数据
byte [] bytes = new byte[1024];
DatagramPacket packet = new DatagramPacket(bytes, bytes.length);
// 3.等待接受数据
socket.receive(packet);
// 4.取出数据
int len = packet.getLength();
String rs = new String(bytes,0,len);
System.out.println("收到的数据:" + rs);
// 获取发送端的ip和端口
String ip = packet.getSocketAddress().toString();
System.out.println("发送端的IP地址: " + ip);
int port = packet.getPort();
System.out.println("发送端端口为: "+port);
// 关闭管道
socket.close();
}
}
二、广播
发送广播消息需要使用广播地址: 255.255.255.255
发送端的数据包的目的地址是广播地址+指定端口号(255.255.255.255,9999)
本机所在网段的其他主机只要匹配到端口即可接受消息(9999)
发送端代码
public class UdpSend {
/**
* 发送端
*/
public static void main(String[] args) throws Exception {
System.out.println("=============发送端启动===========");
// 1.创建发送端对象
DatagramSocket socket = new DatagramSocket(6666);
// 下面通过键盘输入测试,按需读取文件等操作自己修改
Scanner sc = new Scanner(System.in);
while (true) {
System.out.println("请输入您要发送的消息: ");
String msg = sc.nextLine();
if("exit".equals(msg)){
System.out.println("退出成功!");
socket.close();
break;
}
// 2.创建一个数据包对象封装数据
byte[] buffer = msg.getBytes();
DatagramPacket packet = new DatagramPacket(buffer,buffer.length, InetAddress.getByName("255.255.255.255"),9999);
// 3.发送数据
socket.send(packet);
}
}
}
接收端代码
public class UdpReceive {
/**
* 接收端
*/
public static void main(String[] args) throws Exception {
System.out.println("=============客户端启动===========");
// 1.创建接受对象
DatagramSocket socket = new DatagramSocket(9999);
// 2.创建一个数据包接收数据
byte [] buffer = new byte[1024];
DatagramPacket packet = new DatagramPacket(buffer, buffer.length);
while (true) {
// 3.等待接受数据
socket.receive(packet);
// 4.取出数据
int len = packet.getLength();
String rs = new String(buffer,0,len);
System.out.println("收到来自: "+ packet.getAddress()+ ",对方端口号为: "+ packet.getPort()+"的消息: " + rs);
}
}
}
三、组播
使用组播地址: 224.0.0.0~239.255.255.255
发送端的数据包的目的目的地址是组播ip(224.0.1.1,6000)
接收端必须绑定该组播ip(224.0.1.1),端口还要对应发送端的目的端口(6000)
发送端代码
public class UdpGroupServer {
public boolean closed = false;
public String ip = "224.0.0.1";//组播虚拟地址
public int port = 6000;//组播Ip
public int MessageIndex = 0;
// 项目的需求需要我接收数据通过websocket发送给前端 这里我开启一个线程执行其他类定时任务调用该方法
public void start(){
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("UdpTestServer start ");
runServer();
}
}).start();
}
private void runServer(){
// 这里真正是接收组播数据的地方
try {
InetAddress group = InetAddress.getByName(ip);
MulticastSocket s = new MulticastSocket(port);
byte[] arb = new byte[1024];
s.joinGroup(group);//加入该组
while(!closed){
send();
DatagramPacket datagramPacket =new DatagramPacket(arb,arb.length);
s.receive(datagramPacket);
System.out.println("received packet from " + datagramPacket.getAddress().getHostAddress() + " : " + datagramPacket.getPort());
System.out.println(new String(arb));
Thread.sleep(2000);
}
} catch (Exception e) {
e.printStackTrace();
System.out.println("UdpTestServer run Exception: "+e.toString());
}
}
public void send(){
try{
String sendMessage="hello ,message from server,"+MessageIndex++;
byte[] message = sendMessage.getBytes(); //发送信息
InetAddress inetAddress = InetAddress.getByName(ip); //指定组播地址
DatagramPacket datagramPacket = new DatagramPacket(message, message.length, inetAddress, port); //发送数据包囊
MulticastSocket multicastSocket = new MulticastSocket();//创建组播socket
multicastSocket.send(datagramPacket);
}catch (Exception e) {
System.out.println("UdpTestServer send Exception: "+e.toString());
}
if(MessageIndex>=50){
closed = true;
}
}
/**
* @param args
*/
public static void main(String[] args) {
UdpGroupServer server = new UdpGroupServer();
server.start();
}
接收端代码
public class UdpGroupClient {
Logger logger = LoggerFactory.getLogger(UdpGroupClient.class);
private int MessageIndex = 0;
private String ip = "224.0.0.1";//组播地址
private int port = 6000;//指定数据接收端口
private boolean closed = false;
Map<String,Object> map = new HashMap();
// 开启线程实时推送websocket数据 只接收数据的话不用看这部分
public void start() {
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("UpdGroupClient start ");
runClient();
}
}).start();
}
MulticastSocket socket = null;
public void runClient() {
try {
byte[] receiveBuffer = new byte[1024];
DatagramPacket receivePacket = new DatagramPacket(receiveBuffer, receiveBuffer.length);
InetAddress group = InetAddress.getByName(ip);
socket = new MulticastSocket(port);
socket.joinGroup(group);//加入组播地址
while (!closed) {
socket.receive(receivePacket);
System.out.println("received packet from " + receivePacket.getAddress().getHostAddress() + " : " + receivePacket.getPort());
String msg = new String(receivePacket.getData(), receivePacket.getOffset(), receivePacket.getLength());
receiveBuffer = receivePacket.getData();
// 以上是接收的数据,下面是我解析数据帧的内容 就不具体展示了
Map map = new HashMap();
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
// 将解析结果通过websocket发送给前端
WebSocketServer.sendInfo(JSONObject.toJSON(map).toString(),"1");
System.out.println("map长度"+map.size());
System.out.println("*******************"+ JSONObject.toJSON(map).toString());
}else {
logger.info("数据包有错!");
}
Thread.sleep(2000);
}
socket.close();
} catch (Exception e) {
e.printStackTrace();
}
}
四、UDP帧结构
UDP 头定义,共 8 个字节 所以解析数据时一般从第八个字节开始解析自己要取的数据
源端口、目的端口、udp长度、udp校验和占了八个字节。
根据自己需求判断udp是否正确。
以上就是对udp三种通讯方式的简单总结,初步使用,如有错误还望指正。
我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
有时我需要处理键/值数据。我不喜欢使用数组,因为它们在大小上没有限制(很容易不小心添加超过2个项目,而且您最终需要稍后验证大小)。此外,0和1的索引变成了魔数(MagicNumber),并且在传达含义方面做得很差(“当我说0时,我的意思是head...”)。散列也不合适,因为可能会不小心添加额外的条目。我写了下面的类来解决这个问题:classPairattr_accessor:head,:taildefinitialize(h,t)@head,@tail=h,tendend它工作得很好并且解决了问题,但我很想知道:Ruby标准库是否已经带有这样一个类? 最佳
我正在处理旧代码的一部分。beforedoallow_any_instance_of(SportRateManager).toreceive(:create).and_return(true)endRubocop错误如下:Avoidstubbingusing'allow_any_instance_of'我读到了RuboCop::RSpec:AnyInstance我试着像下面那样改变它。由此beforedoallow_any_instance_of(SportRateManager).toreceive(:create).and_return(true)end对此:let(:sport_
rails中是否有任何规定允许站点的所有AJAXPOST请求在没有authenticity_token的情况下通过?我有一个调用Controller方法的JqueryPOSTajax调用,但我没有在其中放置任何真实性代码,但调用成功。我的ApplicationController确实有'request_forgery_protection'并且我已经改变了config.action_controller.consider_all_requests_local在我的environments/development.rb中为false我还搜索了我的代码以确保我没有重载ajaxSend来发送
我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www
给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最
我有一个非常简单的RubyRack服务器,例如:app=Proc.newdo|env|req=Rack::Request.new(env).paramspreq.inspect[200,{'Content-Type'=>'text/plain'},['Somebody']]endRack::Handler::Thin.run(app,:Port=>4001,:threaded=>true)每当我使用JSON对象向服务器发送POSTHTTP请求时:{"session":{"accountId":String,"callId":String,"from":Object,"headers":
我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我