/HelmetClient

UDP发送与接收数据

Primary LanguageJava

你不知道的UDP传输(局域网)

近日闲来无事 特将工作中研究过的小功能提出来,分享给大家。是一篇有关于UDP传输文件、视频的demo。当然 网上肯定已经存在很多了,我也只是将我自己研究的心得来在这里给大家讲一下,说的不好 大家多多担待 !


首先 先为大家简单的讲一下TCP与UDP的区别(真的是简单的讲一下 做个铺垫嘛)

相同:都是传输层

不同:使用TCP协议传输数据,当数据从A端传到B端后,B端会发送一个确认包(ACK包)给A端,告知A端数据我已收到!有重传机制,UDP协议就没有这种确认机制!

UDP 协议是无线连接的数据传输并且无重传机制,很大的可能会造成丢包、收到重复包、乱序的情况,并且无法做这种情况作出处理 只能选择再次发送(如非特殊要求,估计大家不会使用的-但是 我遇到了 哎...)


好了 上面简单的介绍了TCP与UDP的区别,那么下面 咱们该切入正题了

首先 基本配置

    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.RECORD_AUDIO" />
    <uses-permission android:name="android.permission.CHANGE_WIFI_MULTICAST_STATE" />
    <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <uses-permission android:name="android.permission.MOUNT_FORMAT_FILESYSTEMS" />
    <uses-permission android:name="android.permission.CAMERA" />

链接网络(局域网) 、SD卡写入写出、照相机等。

此demo只有一个APP即可 即是客户端也是服务端

首先在MainActivity中启动线程发送自己的信息让另一个手机的demo接收。 整体框架全部基于Thread,Handler,Socket发送与接收消息

	//广播上线(自己)
	private void reBroad() {
		Msg msg = new Msg();
		msg.setSendUserName(user.getName());
		msg.setSendUserIp(user.getIp());
		msg.setReceiveUserIp(Tools.getBroadCastIP());
		msg.setMsgType(Tools.CMD_ONLINE);//通知上线命令
		msg.setDate(Tools.getTimel());
		// 发送广播通知上线
		tools.sendMsg(msg);
	}
  // 心跳响应广播
    class HeartBroadCast extends Thread {
        public void run() {
            while (!mainA.isPaused) {
                try {
                    sleep(10000);
                } catch (InterruptedException e) {
                }
                Msg msgBroad = new Msg();
                msgBroad.setSendUserName(mainA.user.getName());
                msgBroad.setSendUserIp(mainA.user.getIp());
                msgBroad.setMsgType(Tools.CMD_CHECK);
                msgBroad.setReceiveUserIp(Tools.getBroadCastIP());
                msgBroad.setDate(Tools.getTimel());
                // 发送消息
                sendMsg(msgBroad);
            }
        }
    }

    // 检测用户是否在线,如果超过15说明用户已离线,秒则从列表中清除该用户
    class CheckUserOnline extends Thread{
        @Override
        public void run()
        {
            while(!mainA.isPaused)
            {
                for (int i = 0; i < mainA.userList.size(); i++)
                {
                    long cm=System.currentTimeMillis()-mainA.userList.get(i).getOnlineTime();
                    if(cm>15000)
                    {
                        //刷新列表
                        TipsMain(Tools.DESTROYUSER,i);
                    }
                }
                try {
                    sleep(8000);
                    //防掉线,广播
                    //Tips(Tools.CONSTANTBROAD,null);
                } catch (InterruptedException e) {
                }
            }
        }
    }

依照上面大家可以清楚的看到 在起始的MainActivity中首先就去发送自己上线的通知,另一端同样的启动线程随时的接收---保持自己能够随时和B端连接上。大家可以看到几乎所有的发送全部由Thread中进行操作(原因很简单 我就不说明了)。

在之后的操作 就会全部通过message发送到主线程去进行UI(数据)的更新,给大家看下在发送和接收消息的时候应该如何对的数据进行处理吧。

 //发送消息线程
    class UdpSend extends Thread {
        Msg msg = null;

        UdpSend(Msg msg) {
            this.msg = msg;
        }

        @Override
        public void run() {
            try {
                LogUtil.defLog("Running--"+msg);
                byte[] data = Tools.toByteArray(msg);
                //1、创建DatagramSocket用于UDP数据传送
                DatagramSocket ds = new DatagramSocket(Tools.PORT_SEND);
                //2、创建需要发送的数据包
                DatagramPacket packet = new DatagramPacket(data, data.length, InetAddress.getByName(msg.getReceiveUserIp()), Tools.PORT_RECEIVE);
                //3、发送
                packet.setData(data);
                ds.send(packet);
                //4、关闭连接
                ds.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    //接收消息
    public void receiveMsg() {
        new UdpReceive().start();
    }

    class UdpReceive extends Thread {
        Msg msg = null;

        UdpReceive() {
        }

        public void run() {
            //消息循环
            while (true) {
                try {
                    //1、创建DatagramSocket;
                    DatagramSocket ds = new DatagramSocket(Tools.PORT_RECEIVE);
                    //2、创建数据包,用于接收内容。
                    byte[] data = new byte[1024 * 4];
                    DatagramPacket packet = new DatagramPacket(data, data.length);
                    //3、接收数据
                    packet.setData(data);
                    ds.receive(packet);
                    byte[] data2 = new byte[packet.getLength()];
                    System.arraycopy(data, 0, data2, 0, data2.length);// 得到接收的数据
                    Msg msg = (Msg) Tools.toObject(data2);
                    ds.close();
                    //解析消息
                    parse(msg);
                } catch (Exception e) {
                }
            }
        }
    }

同样需要启动线程,前面介绍了 防止数据在传输过程中出现问题,当然 假如真的出现了问题 我可以告诉你一个更简单的方法 ---重新发送!! 我现在依旧觉得我每一次的成功都是我实验前的祈祷造成的,哈哈 废话不多说了 继续讲解。

Socket是链接中一定要存在的桥梁传送数据全靠他了啊。

然后接到的数据就要开始解析啦

  发送数据将数据压缩成流文件,Socket创建(要接受的id,链接号)
	public void creatClient() throws Exception {
		Socket s = new Socket(msg.getSendUserIp(), 2222);
		// 读文件w
		File file = new File(path);
		BufferedInputStream is = new BufferedInputStream(new FileInputStream(file));
		BufferedOutputStream os =new BufferedOutputStream( s.getOutputStream());
		// 读文件
		double n = 1;
		byte[] data = new byte[Tools.byteSize];// 每次读取的字节数
		int len=-1;
		while ((len=is.read(data))!= -1) {
			os.write(data,0,len); 
			Tools.sendProgress+=len;//进度
		}
		Tools.sendProgress=-1;
		is.close();
		os.flush();
		os.close();
	}
   ServerSocket接收(连接号) 开始对文件进行解压
   
	ServerSocket ss = new ServerSocket(2222);
		Socket s = new Socket();
		s = ss.accept();
		File filesDir = connectB.getExternalFilesDir(null);
		if (!filesDir.exists()) {
			filesDir.mkdirs();
		}
		String name =Tools.newfileName;
		File file = new File(filesDir + name);
		LogUtil.defLog("存数据--" + file.getAbsolutePath());
		if (!file.exists()) {
			file.createNewFile();
		}
		BufferedInputStream is = new BufferedInputStream(s.getInputStream()); // 读进
		BufferedOutputStream os = new BufferedOutputStream(new FileOutputStream(file));// 写出

		Thread.sleep(1000);
		byte[] data = new byte[Tools.byteSize];// 每次读取的字节数
		int len = -1;
		while ((len = is.read(data)) != -1) {
			os.write(data, 0, len);
			Tools.sendProgress += len;// 进度
		}
		Tools.sendProgress = -1;
		is.close();
		os.flush();
		os.close();
		Tools.TipsMain(Tools.INTENT, connectB.getFilesDir().getAbsolutePath());
		s.close();
		Tools.TipsMain(Tools.SHOW, "接收完成" + Tools.newfileName);

大家有木有注意到我将文件存放的位置,没错 我存在了内存中的包里面 简单的原因 存放在这里 我才能将里面的数据显示出来(其他位置可能也行 不过我没有试过)

** 总结:其实UDP传输很简单的 只要将里面的逻辑关系搞清楚 整套的东西很好出来 我会在明年的时候 将点对点的视频通话研究出来 到时这篇文章我在进行更新现在的代码我放进我的guyhub上了 有需要的童鞋可以下载看看 记得拿两个手机哈 一个是没法测试的 .希望你们可以有所学习,另外你会发现更多地惊喜哦 欢迎start**