关于socket 两台Android设备上的通信

2024-06-05 12:08

本文主要是介绍关于socket 两台Android设备上的通信,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

没想到这么旧的东西还有人支持。我一开心就把源码传github上了。哈哈。然后下面的代码有两处错误。(一个是start()写成了run(),

另外一处的bundle没加到message里)在github中已经纠正。github是可以直接运行的。需要一台手机或者两台手机都可以。一台手机记得链接之前ip输入127.0.0.1。

哈哈哈 这个是链接 https://github.com/roofroot/SocketTest

另外compose富文本编辑持续更新中,其实做这个是想实现

富文本编辑器的传文件功能。本来是想做外网的。但是既然不能外网就打算直接用蓝牙通讯了。后面蓝牙功能写好会单独开一篇。最近还是在做ui.

 

好久没有看过socket的代码,今天想实现一个两台手机之间互传的功能,本来想用蓝牙,但是想着,蓝牙有距离限制,如果直接用网络通讯。

这样就可以远程发送了啊。于是我想起了socket。经过一上午的时间写了一个demo。然而仅仅实现了局域网的链接。从网上查资料,看了很多关于内网穿透的

文章。似乎虽然能够查询到本机的外网地址,但是其他设备并不能直接通过这个外网的ip地址访问到这个设备。其中涉及到了一个名词叫做内网穿透。由于我不是做

服务器的所以也没太看懂。大概的意思就是外网并不能直接访问内网,需要经过一些端口的映射。而即时通讯类的软件,客户端需要链接到服务器由服务器进行网络地址的映射,

交换数据,经过交换数据成功以后。两台设备链接上了,之后才可以进行点对点的通讯。(也不知道理解的对不对) 虽然说通过一些其他人开发的软件,也能在手机上进行内网穿透,

但是对我来说意义不大,毕竟我想实现的是两台设备的数据互传,而不是必须其中一台当做服务器。所以记录一下我做局域网socket通讯的简单代码吧。

没有实现具体的业务逻辑,所以整个代码非常简单。简单的说一下socket是如果进行通讯的

ScoketServer 初始化后,我们可以开启一个线程用来等待Socket客户端的链接。链接成功以后会返回一个Socket对象,我们就使用这个Socket对象收发消息

我要实现的是两台设备之间的消息互传,所以这两台设备需要一个创建SocketServer用来等待链接,另一个创建Socket连接到这个Service

下面这段代码是Socket创建并链接ServerSocket的方法。需要注意的就是所有的链接与开启输入输出流的代码都不要放主线程就可以了

public class ClientSocketUtil implements Runnable {public static String ip;public static int prot=1040;private Socket clientSocket;private  DataInputStream dataInputStream= null;private  DataOutputStream dataOutputStream = null;private Handler handler;public ClientSocketUtil(Handler handler){
//            new Thread(this).run();new Thread(this).start();}public void sendMsg(String  msg){new Thread(new Runnable() {@Overridepublic void run() {try {Log.e("client","客户端发送消息"+msg);dataOutputStream.writeUTF(msg);} catch (IOException e) {e.printStackTrace();Log.e("client","客户端发送消息出错");Log.e("client",e.getMessage());}}}).start();}public void close(){if(clientSocket!=null){try {clientSocket.close();} catch (IOException e) {e.printStackTrace();}}}@Overridepublic void run() {if(clientSocket==null) {try {InetAddress serverAddr = InetAddress.getByName(ip);clientSocket = new Socket(ip, prot);if(clientSocket.isConnected()) {Log.e("client","客户端链接成功");dataInputStream = new DataInputStream(clientSocket.getInputStream());dataOutputStream = new DataOutputStream(clientSocket.getOutputStream());while (true) {try {if(dataInputStream!=null) {String str = dataInputStream.readUTF();Message message = new Message();Bundle bundle = new Bundle();bundle.putString("msg", str);handler.sendMessage(message);}} catch (IOException e) {e.printStackTrace();}}}else{}} catch (IOException e) {e.printStackTrace();Log.e("client","链接失败");}}}
}

 下面这段是ServerSocket初始化并等待链接的代码,如果成功链接以后,用另一个工具类来收发消息。这个工具类的代码和上面的Socket工具类基本一致。只是Socket的对象是ServerSocket返回的。只需要这三个类的代码就可以实现最基础的局域网通讯了。由于我要实现的是互传。所以并没有考虑到多链接的场景。应该只需要每一台都执行一下

ServerSocket用来接收链接,并把所有的链接对象都保存起来就可以实现 多对多的通讯方式了。只是回复消息需要使用不同的链接对象,未免太消耗内存。所以手机端还是不适合这么搞的。互传的功能应该优化一下代码,仅接收一个链接

工具类代码

import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.util.Log;import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.Socket;public class KeepSocketUtil implements Runnable {private Socket clientSocket;private static DataInputStream dataInputStream = null;private static DataOutputStream dataOutputStream = null;private Handler handler;public KeepSocketUtil(Socket socket, Handler handler) {this.handler = handler;this.clientSocket = socket;new Thread(this).run();}public void sendMsg(String msg) {new Thread(new Runnable() {@Overridepublic void run() {try {dataOutputStream.writeUTF(msg);} catch (IOException e) {e.printStackTrace();}}}).start();}public void close(){if(clientSocket!=null){try {clientSocket.close();} catch (IOException e) {e.printStackTrace();}}}@Overridepublic void run() {if(clientSocket==null){return;}try {dataInputStream = new DataInputStream(clientSocket.getInputStream());} catch (IOException e) {e.printStackTrace();}try {dataOutputStream = new DataOutputStream(clientSocket.getOutputStream());} catch (IOException e) {e.printStackTrace();}while (true) {try {String str = dataInputStream.readUTF();Log.e("msg","服务器接受消息"+str);Message message = new Message();Bundle bundle = new Bundle();bundle.putString("msg", str);message.setData(bundle);handler.sendMessage(message);} catch (IOException e) {e.printStackTrace();}}}
}

ServerSocket代码


import android.os.Handler;
import android.util.Log;import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;public class ServerSocketUtil implements Runnable{public static int prot=1040;private Socket clientSocket;private ServerSocket serverSocket;private KeepSocketUtil keepSocketUtil;public KeepSocketUtil getKeepSocketUtil() {return keepSocketUtil;}Handler handler;public ServerSocketUtil(Handler handler){this.handler=handler;new Thread(this).start();}public void close(){if(serverSocket!=null) {try {serverSocket.close();} catch (IOException e) {e.printStackTrace();}}}@Overridepublic void run() {if(serverSocket==null) {try {serverSocket = new ServerSocket(prot);Log.e("server","服务器启动成功");} catch (IOException e) {e.printStackTrace();Log.e("server","服务器启动失败"+e.getMessage());}}if(serverSocket!=null) {while (true) {try {this.clientSocket = serverSocket.accept();} catch (IOException e) {Log.e("server", "有客户端链接失败"+e.getMessage());e.printStackTrace();}if (clientSocket != null) {Log.e("server", "有客户端请求链接");keepSocketUtil = new KeepSocketUtil(clientSocket, handler);}}}}
}

 

这篇关于关于socket 两台Android设备上的通信的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1033010

相关文章

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

【STM32】SPI通信-软件与硬件读写SPI

SPI通信-软件与硬件读写SPI 软件SPI一、SPI通信协议1、SPI通信2、硬件电路3、移位示意图4、SPI时序基本单元(1)开始通信和结束通信(2)模式0---用的最多(3)模式1(4)模式2(5)模式3 5、SPI时序(1)写使能(2)指定地址写(3)指定地址读 二、W25Q64模块介绍1、W25Q64简介2、硬件电路3、W25Q64框图4、Flash操作注意事项软件SPI读写W2

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动

如何编写Linux PCIe设备驱动器 之二

如何编写Linux PCIe设备驱动器 之二 功能(capability)集功能(capability)APIs通过pci_bus_read_config完成功能存取功能APIs参数pos常量值PCI功能结构 PCI功能IDMSI功能电源功率管理功能 功能(capability)集 功能(capability)APIs int pcie_capability_read_wo

Android 10.0 mtk平板camera2横屏预览旋转90度横屏拍照图片旋转90度功能实现

1.前言 在10.0的系统rom定制化开发中,在进行一些平板等默认横屏的设备开发的过程中,需要在进入camera2的 时候,默认预览图像也是需要横屏显示的,在上一篇已经实现了横屏预览功能,然后发现横屏预览后,拍照保存的图片 依然是竖屏的,所以说同样需要将图片也保存为横屏图标了,所以就需要看下mtk的camera2的相关横屏保存图片功能, 如何实现实现横屏保存图片功能 如图所示: 2.mtk

android应用中res目录说明

Android应用的res目录是一个特殊的项目,该项目里存放了Android应用所用的全部资源,包括图片、字符串、颜色、尺寸、样式等,类似于web开发中的public目录,js、css、image、style。。。。 Android按照约定,将不同的资源放在不同的文件夹中,这样可以方便的让AAPT(即Android Asset Packaging Tool , 在SDK的build-tools目

Android fill_parent、match_parent、wrap_content三者的作用及区别

这三个属性都是用来适应视图的水平或者垂直大小,以视图的内容或尺寸为基础的布局,比精确的指定视图的范围更加方便。 1、fill_parent 设置一个视图的布局为fill_parent将强制性的使视图扩展至它父元素的大小 2、match_parent 和fill_parent一样,从字面上的意思match_parent更贴切一些,于是从2.2开始,两个属性都可以使用,但2.3版本以后的建议使