android 之TCP客户端编程
补充,由于这篇文章是自己入门的时候写的,随着Android系统的升级可能有发送需要在任务
中进行,如有问题请百度 thread
或者看下面链接的文章
https://www.cnblogs.com/yangfengwu/category/1187355.html

吸取教训!!!本来花了5个小时写完了,没想到,,,因为没点上面的自动保存查看一下,全没了,重新写呗
关于网络通信:每一台电脑都有自己的ip地址,每台电脑上的网络应用程序都有自己的通信端口,张三的电脑(ip:192.168.1.110)上有一个网络应用程序A(通信端口5000),李四的电脑(ip:192.168.1.220)上有一个网络应用程序B(通信端口8000),张三给李四发消息,首先你要知道李四的ip地址,向指定的ip(李四ip:192.168.1.220)发信息,信息就发到了李四的电脑。再指定一下发送的端口号(通信端口8000),信息就发到了李四电脑的网络应用程序B上。
TCP--一种网络通信方式而已。分为服务器(网络应用程序)和客户端(网络应用程序),TCP通信过程,首先打开服务器,监听自己的网络通信端口(假设为9000),打开客户端,设置好要连接的ip地址和服务器的网络通信端口(9000),这样服务器一旦监听到网络通信端口有连接,二者就建立了连接。
好一步一步写程序(最后有源码!!!!!!!)
怎样建立工程就不说了,本来写好了并贴了图,网络一有问题全没了。抱怨一下,博客传图片真麻烦。竟然不支持复制 粘贴。各位朋友有什么方便的方法请告知。
在布局文件里加入两个按钮(button),一个控制连接,一个控制发送消息;四个输入文本框(edittext),一个填写发送的信息内容,一个显示服务器发来的消息。一个填写要链接的ip地址,一个填写要链接的端口号
布局代码
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="com.wifi123.MainActivity" >以上都不用管的,软件自动生成的,配置界面的
<!--显示的标题:目标IP地址-->
<TextView 
android:textSize="20dp"字体大小
android:id="@+id/IP_tv"    id号
android:text="目标IP地址"  显示的内容
android:layout_width="wrap_content"  宽度随内容而定
android:layout_height="wrap_content"  高度度随内容而定
/>
<!--显示的标题:目标端口号-->
<TextView 
android:textSize="20dp"
android:id="@+id/Port_tv"
android:text="目标端口号"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/IP_tv"  在<!--显示的标题:目标IP地址-->的下面
android:layout_marginTop="30dp"  离它上面那个组件(<!--显示的标题:目标IP地址-->)的距离
/>
<!-- 用于填写ip地址的文本框-->
<EditText 
android:text="192.168.4.1"
android:id="@+id/ip_ET"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_toRightOf="@id/IP_tv"  在<!--显示的标题:目标IP地址-->的右面
/>
<!-- 用于填写端口号的文本框-->
<EditText 
android:text="8080"
android:id="@+id/Port_ET"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_toRightOf="@id/Port_tv"  还是在谁谁谁的右面
android:layout_alignBottom="@id/Port_tv"  本元素的下边缘和某元素的的下边缘对齐 
/>
<!-- 用于发送信息的文本框-->
<EditText 
android:id="@+id/Send_ET"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_below="@id/Port_tv"  在某元素的下方
/>
<!-- 用于连接的按钮-->
<Button 
android:text="连接"
android:id="@+id/Connect_Bt"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="Connect_onClick"设置按钮的动作监听函数,其实有几种写法,就用最简单的一种
android:layout_below="@id/Send_ET"  在某元素的下方
/>
<!-- 用于发送信息的按钮-->
<Button 
android:text="发送"
android:id="@+id/Send_Bt"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="Send_onClick"
android:layout_below="@id/Send_ET"
android:layout_alignParentRight="true"  贴紧父元素的右边缘,指的是整体的界面
/>
<!-- 用于接收信息的文本框-->
<EditText 
android:background="@android:color/darker_gray"
android:id="@+id/Receive_ET"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_below="@id/Connect_Bt"
android:layout_alignParentBottom="true"  贴紧父元素的下边缘 
/>
</RelativeLayout>
看看布局界面

接着开始编写功能程序


先做点击连接按钮就连接服务器
查看java的API文档,里面封装了专门用于TCP客户端通信的类,和方法

里面有一个类Socket (客服端),有一个它的构造方法
Socket(InetAddress address, int port) 
          创建一个流套接字并将其连接到指定 IP 地址的指定端口号。
意思是Socket socket = new Socket(InetAddress address, int port) ;//创建连接地址和端口,就去连接指定的ip和端口号去了,address填ip地址,port填端口号
只不过InetAddress是一个类,我们打开看一下

那么
InetAddress ipAddress = InetAddress.getByName("192.168.4.1");
socket = new Socket(ipAddress, 8080);//创建连接地址和端口--------------就完了,客户端就去连接了
但是ip地址和端口被我们定死了,,,,可不好玩,我们就设置成获取ip文本框中的ip,端口号文本框中的端口号
InetAddress ipAddress = InetAddress.getByName(IPEditText.getText().toString());
int port =Integer.valueOf(PortText.getText().toString());//获取端口号 
socket = new Socket(ipAddress, port);//创建连接地址和端口-------------------这样就好多了
但是由于在android几开始,不允许在主线程里连接服务器,所以只好让按钮点击后启动一个线程里面写上面的东西
package com.wifi123;
import java.net.InetAddress;
import java.net.Socket;
import java.net.UnknownHostException;
import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;
public class MainActivity extends Activity {
Button ConnectButton;//定义连接按钮
Button SendButton;//定义发送按钮
EditText IPEditText;//定义ip输入框
EditText PortText;//定义端口输入框
EditText MsgText;//定义信息输出框
EditText RrceiveText;//定义信息输入框
Socket socket = null;//定义socket
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ConnectButton = (Button) findViewById(R.id.Connect_Bt);//获得按钮对象
SendButton = (Button) findViewById(R.id.Send_Bt);//获得按钮对象
IPEditText = (EditText) findViewById(R.id.ip_ET);//获得ip文本框对象
PortText = (EditText) findViewById(R.id.Port_ET);//获得端口文本框按钮对象
}
public void Connect_onClick(View v) {
//启动连接线程
Connect_Thread connect_Thread = new Connect_Thread();
connect_Thread.start();
}
class Connect_Thread extends Thread//继承Thread
{
public void run()//重写run方法
{
try 
{
if (socket == null) //如果已经连接上了,就不再执行连接程序
{
//用InetAddress方法获取ip地址
InetAddress ipAddress = InetAddress.getByName(IPEditText.getText().toString());
int port =Integer.valueOf(PortText.getText().toString());//获取端口号 
socket = new Socket(ipAddress, port);//创建连接地址和端口-------------------这样就好多了
}
} 
catch (Exception e) 
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
对了需要添加两个权限,一个是wifi权限,一个是internet




然后下载到手机因为我的电脑的ip为192.168.1.101,所以我把192.168.4.1改了,192.168.4.1是为了做与wifi模块EPS8266通信使得

然后打开网络调试助手,点击连接(可以关闭电脑防火墙),然后点击手机上的连接

好接着,连接按钮按一下连接,再按一下断开连接,并且,连接后按钮上显示断开,断开后按钮上显示连接

按钮事件改为
public void Connect_onClick(View v) {
if (isConnect == true) //标志位 = true表示连接
{
isConnect = false;//置为false
ConnectButton.setText("断开");//按钮上显示--断开
//启动连接线程
Connect_Thread connect_Thread = new Connect_Thread();
connect_Thread.start();
}
else //标志位 = false表示退出连接
{
isConnect = true;//置为true
ConnectButton.setText("连接");//按钮上显示连接
try 
{
socket.close();//关闭连接
socket=null;
} 
catch (IOException e) 
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
你可以试一试了
接着写点击发送按钮把发送信息文本框的内容发送出去

先贴代码
public void Send_onClick(View v) {
try 
{
//获取输出流
outputStream = socket.getOutputStream();
//发送数据
outputStream.write(MsgEditText.getText().toString().getBytes());
//outputStream.write("0".getBytes());
} 
catch (Exception e) 
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
看看javaAPI


所以才有了
//获取输出流
OutputStream outputStream = socket.getOutputStream();
//发送数据
outputStream.write(MsgEditText.getText().toString().getBytes());


接收数据并在信息框显示出来
创建一个接收线程,在连接线程成功建立连接后启动接收线程
//接收线程
class Receive_Thread extends Thread
{
public void run()//重写run方法
{
try
{
while (true)
{
final byte[] buffer = new byte[1024];//创建接收缓冲区
inputStream = socket.getInputStream();
final int len = inputStream.read(buffer);//数据读出来,并且返回数据的长度
runOnUiThread(new Runnable()//不允许其他线程直接操作组件,用提供的此方法可以
{
public void run()
{
// TODO Auto-generated method stub
RrceiveEditText.setText(new String(buffer,0,len));
}
});
}
}
catch (IOException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
//连接线程
class Connect_Thread extends Thread//继承Thread
{
public void run()//重写run方法
{
try
{
if (socket == null)
{
//用InetAddress方法获取ip地址
InetAddress ipAddress = InetAddress.getByName(IPEditText.getText().toString());
int port =Integer.valueOf(PortText.getText().toString());//获取端口号
socket = new Socket(ipAddress, port);//创建连接地址和端口-------------------这样就好多了
//在创建完连接后启动接收线程
Receive_Thread receive_Thread = new Receive_Thread();
receive_Thread.start();
}
}
catch (Exception e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
下面贴全部源码
activity_mian.xml源码
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.wifi123.MainActivity" >
<!--显示的标题:目标IP地址-->
	<TextView 
	    android:textSize="20dp"
	    android:id="@+id/IP_tv"
	    android:text="目标IP地址"
	    android:layout_width="wrap_content"
        android:layout_height="wrap_content"
	    />
	<!--显示的标题:目标端口号-->
	<TextView 
	    android:textSize="20dp"
	    android:id="@+id/Port_tv"
	    android:text="目标端口号"
	    android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_below="@id/IP_tv"
        android:layout_marginTop="30dp"
	    />
<!-- 用于填写ip地址的文本框-->
    <EditText 
        android:text="192.168.1.101"
        android:id="@+id/ip_ET"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_toRightOf="@id/IP_tv"
        />
    <!-- 用于填写端口号的文本框-->
    <EditText 
        android:text="8080"
        android:id="@+id/Port_ET"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_toRightOf="@id/Port_tv"
        android:layout_alignBottom="@id/Port_tv"
        />
     <!-- 用于发送信息的文本框-->
    <EditText 
        android:id="@+id/Send_ET"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_below="@id/Port_tv"
        />
    <!-- 用于连接的按钮-->
    <Button 
        android:text="连接"
        android:id="@+id/Connect_Bt"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:onClick="Connect_onClick"
        android:layout_below="@id/Send_ET"
        />
    <!-- 用于发送信息的按钮-->
    <Button 
        android:text="发送"
        android:id="@+id/Send_Bt"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:onClick="Send_onClick"
        android:layout_below="@id/Send_ET"
        android:layout_alignParentRight="true"
        />
	 <!-- 用于接收信息的文本框-->
    <EditText 
        android:background="@android:color/darker_gray"
        android:id="@+id/Receive_ET"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_below="@id/Connect_Bt"
        android:layout_alignParentBottom="true"
        />
</RelativeLayout>
MainActivity.java源码
package com.wifi123;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.Socket;
import java.net.UnknownHostException;
import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;
public class MainActivity extends Activity {
boolean isConnect=true;//连接还是断开
	Button ConnectButton;//定义连接按钮
	Button SendButton;//定义发送按钮
	EditText IPEditText;//定义ip输入框
	EditText PortText;//定义端口输入框
	EditText MsgEditText;//定义信息输出框
	EditText RrceiveEditText;//定义信息输入框
	Socket socket = null;//定义socket
	private OutputStream outputStream=null;//定义输出流
	private InputStream inputStream=null;//定义输入流
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
ConnectButton = (Button) findViewById(R.id.Connect_Bt);//获得连接按钮对象
        SendButton = (Button) findViewById(R.id.Send_Bt);//获得发送按钮对象
        IPEditText = (EditText) findViewById(R.id.ip_ET);//获得ip文本框对象
        PortText = (EditText) findViewById(R.id.Port_ET);//获得端口文本框按钮对象
        MsgEditText = (EditText) findViewById(R.id.Send_ET);//获得发送消息文本框对象
        RrceiveEditText = (EditText) findViewById(R.id.Receive_ET);//获得接收消息文本框对象
    }
public void Connect_onClick(View v) {
    	if (isConnect == true) //标志位 = true表示连接
		{
			isConnect = false;//置为false
			ConnectButton.setText("断开");//按钮上显示--断开
			//启动连接线程
	    	Connect_Thread connect_Thread = new Connect_Thread();
	    	connect_Thread.start();
		}
		else //标志位 = false表示退出连接
		{
			isConnect = true;//置为true
			ConnectButton.setText("连接");//按钮上显示连接
			try 
			{
socket.close();//关闭连接
				socket=null;
			} 
			catch (IOException e) 
			{
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
		}
	}
public void Send_onClick(View v) {
    	try 
		{
			//获取输出流
			outputStream = socket.getOutputStream();
			//发送数据
			outputStream.write(MsgEditText.getText().toString().getBytes());
			//outputStream.write("0".getBytes());
		} 
		catch (Exception e) 
		{
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
    //连接线程
    class Connect_Thread extends Thread//继承Thread
	{
		public void run()//重写run方法
		{
			try 
			{
				if (socket == null) 
				{
					//用InetAddress方法获取ip地址
					InetAddress ipAddress = InetAddress.getByName(IPEditText.getText().toString());
					int port =Integer.valueOf(PortText.getText().toString());//获取端口号 
					socket = new Socket(ipAddress, port);//创建连接地址和端口-------------------这样就好多了
					//在创建完连接后启动接收线程
					Receive_Thread receive_Thread = new Receive_Thread();
			    	receive_Thread.start();
				}
} 
			catch (Exception e) 
			{
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
		}
	}
	//接收线程
	class Receive_Thread extends Thread
	{
		public void run()//重写run方法
		{
			try 
			{
				while (true) 
				{
					final byte[] buffer = new byte[1024];//创建接收缓冲区
					inputStream = socket.getInputStream();
					final int len = inputStream.read(buffer);//数据读出来,并且返回数据的长度
					runOnUiThread(new Runnable()//不允许其他线程直接操作组件,用提供的此方法可以
						{
							public void run() 
							{	
								// TODO Auto-generated method stub
								RrceiveEditText.setText(new String(buffer,0,len));	
							}						
						});
				}
			} 
			catch (IOException e) 
			{
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
}
	}
}
android 之TCP客户端编程的更多相关文章
- Android之TCP服务器编程
		
推荐一个学java或C++的网站http://www.weixueyuan.net/,本来想自己学了总结出来再写博客,现在没时间,打字太慢!!!!,又想让这好东西让许多人知道. 关于网络通信:每一台电 ...
 - TCP 客户端编程
		
1.Qt中TCP客户端编程 对Qt编程而言,网络只是数据传输的通道: Qt提供了QTcpSocket类(封装了TCP协议细节): 将QTcpSocket的对象当做黑盒使用,进行数据首发. 1.1QTc ...
 - android之TCP客户端框架
		
一.程序框架 1.1 创建方法 onCreate 1.1.1 创建连接按键线程,并使能线程(触发原因:可按键.其他操作,并进行状态判断): Connect_Thread connect_Thread ...
 - python网络编程socketserver模块(实现TCP客户端/服务器)
		
摘录python核心编程 socketserver(python3.x版本重新命名)是标准库中的网络编程的高级模块.通过将创建网络客户端和服务器所必须的代码封装起来,简化了模板,为你提供了各种各样的类 ...
 - 24-ESP8266 SDK开发基础入门篇--Android TCP客户端.控制 Wi-Fi输出PWM的占空比,调节LED亮度
		
https://www.cnblogs.com/yangfengwu/p/11204436.html 刚才有人说需要点鸡汤.... 我想想哈;我还没问关于哪方面的鸡汤呢!!! 我所一直走的路线 第一: ...
 - 23-ESP8266 SDK开发基础入门篇--编写Android TCP客户端 , 加入消息处理
		
https://www.cnblogs.com/yangfengwu/p/11203546.html 先做接收消息 然后接着 public class MainActivity extends App ...
 - 系统编程-网络-tcp客户端服务器编程模型(续)、连接断开、获取连接状态场景
		
相关博文: 系统编程-网络-tcp客户端服务器编程模型.socket.htons.inet_ntop等各API详解.使用telnet测试基本服务器功能 接着该上篇博文,咱们继续,首先,为了内容的完整性 ...
 - --系统编程-网络-tcp客户端服务器编程模型、socket、htons、inet_ntop等各API详解、使用telnet测试基本服务器功能
		
PART1 基础知识 1. 字节序 网络字节序是大端字节序(低地址存放更高位的字节), 所以,对于字节序为小端的机器需要收发网络数据的场景,要对这些数据进行字节序转换. 字节序转换函数,常用的有四个: ...
 - Python--网络编程学习笔记系列02 附:tcp服务端,tcp客户端
		
Python--网络编程学习笔记系列02 TCP和UDP的概述: udp通信模型类似于写信,不需要建立相关链接,只需要发送数据即可(现在几乎不用:不稳定,不安全) tcp通信模型类似于打电话,一定要建 ...
 
随机推荐
- Python sys 模块
			
import sys # 把命令行参数返回一个 List,第一个元素是程序本身的路径 print(sys.argv) # 命令行运行 python3 sys_model.py klvchen hell ...
 - 从零开始学习html(六)开始学习CSS,为网页添加样式
			
一.认识CSS样式 <!DOCTYPE HTML> <html> <head> <meta http-equiv="Content-Type&quo ...
 - 后台返回xml格式转json
			
之前后台做了一个xml格式的数据返回给前端,这个可愁坏了我,不过现在还是解决了,虽然方法有点笨,但没有找到其他的方法,先将就着用吧. 后台返回的是这样的: 那么我们就要这样处理:commonMetho ...
 - Kotlin入门(13)类成员的众生相
			
上一篇文章介绍了类的简单定义及其构造方式,当时为了方便观察演示结果,在示例代码的构造函数中直接调用toast提示方法,但实际开发是不能这么干的.合理的做法是外部访问类的成员属性或者成员方法,从而获得处 ...
 - js replace替换 忽略大小写问题
			
实现就是控制台的内容“abc”,但是后台返回的是“ABC”,这个时候在前台遍历,需要将后台返回的在控制台标红. 当然控制台可以是 abc Abc等大小写混合,以下代码都可替换. var flagnew ...
 - 第三方apk内置因签名导致SystemUI未启动启动问题案例分析
			
这个问题是刷完机正常开机后,发现手机无状态栏,下拉通知栏,按音量键也无法出现VolumeDialog,开始看到这个现象感觉是systemUI未编译到版本中去?或者是在systemserver中syst ...
 - python第五十二天---第九周作业   类 Fabric 主机管理程序
			
类 Fabric 主机管理程序开发:1. 运行程序列出主机组或者主机列表2. 选择指定主机或主机组3. 选择让主机或者主机组执行命令或者向其传输文件(上传/下载)4. 充分使用多线程或多进程5. 不同 ...
 - 描述性统计的matlab实现
			
理论讲的再多不会做也白弄 直接上手 一.针对接近正态分布的(均值,方差,标准差,极差,变异系数,偏度,峰度) 这里我必须提前说明一点就是,你在写好函数后,函数的名是dts,你保存的文件名也必须是dts ...
 - Jmeter参数化方法
			
用Jmeter测试时包含两种情况的参数:一种是在url中,一种是请求中需要发送的参数. 设置参数值的方法有如下几种: 一.函数助手 用Jmeter中的函数获取参数值,__Random,__thread ...
 - MyEclipse10或者eclipse中配置开发Python的Pydev插件安装教程
			
注意使用LiClipse的用户 PyDev已经预装在LiClipse中,所以可以跳过这一步(请注意,如果使用LiClipse,PyDev不能单独安装或更新,因为它必须始终作为一个整体更新). 必需品 ...