先看下socket通信模型:
服务器并不是主动地建立连接,相反地,他们是被动地监听一个客户端的连接请示然后给他们服务,由类ServerSocket来建立的。
下面的程序建立了一个服务器端socket并把它绑定到80端口:
ServerSocket serverSocket = new ServerSocket(80, 5);
第一个参数是服务器要监听的端口,第二个参数是可选的。API文档中说明了这是一个监听时间,但是在传统的socket程序中第二个参数是监听深度。一个服务器可以同时接收多个连接请求,但是每次只能处理一个。监听堆是一个无回答的连接请求队列。上面的请求建立一个连接来处理最后五个请求,如果省略了后面的一个参数则默认值是50。
一旦socket建立了并开始监听连接,进来的连接将会建立并放在监听堆,accetp()方法把在堆中的连接取出来。Socket clientSocket = serverSocket.accept();这个方法返回一个用来与来访者对话的客户端连接。服务器本身不可能建立对话,相反地,服务器socket会使用accept()方法来产生一个新的socket。服务器socket依旧打开并排列新的连接请求, 与客户端socket一样,下面的一步建立输入和输出流:
DataInputStream inbound = new DataInputStream( clientSocket.getInputStream() );
DataOutputStream outbound = new DataOutputStream( clientSocket.getOutputStream() );
一般的I/O操作可以在新建的流中运用。在服务器回应前它等待客户端发送一个空白的行.当会话结束时,服务器关闭流和客户端socket。如果在队列中没有请示将会出现什么情况呢?那个方法将会等待一个的到来。这个行为叫阻塞。accept()方法将会阻塞服务器线程直到一个呼叫到来,当5个连接处理完闭之后服务器退出。任何的在队列中的呼叫将会被取消。
所有的服务器都要有以下的基本的步骤:
1.建立一个服务器socket并开始监听;2.使用accept()方法取得新的连接;3.建立输入和输出流;4.在已有的协议上产生会话;5.关闭客户端流和socket;6.回到第二步或者到第七步;7.关闭服务器socket.下面看例子代码:一个SocketActivity,需安装在手机上调试package com.company.android; import java.io.IOException; import java.io.InputStream; import java.net.DatagramPacket; import java.net.DatagramSocket; import java.net.ServerSocket; import java.net.Socket; import java.net.SocketException; import android.app.Activity; import android.os.Bundle; import android.view.View; import android.view.View.OnClickListener; import android.widget.Button; public class SocketActivity extends Activity { private Button startButton = null; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); startButton = (Button) findViewById(R.id.start); startButton.setOnClickListener(new startOnClickListener()); } class startOnClickListener implements OnClickListener { public void onClick(View arg0) { new ServerThread().start(); } } class ServerThread extends Thread { public void run() { ServerSocket serverSocket = null; try{ serverSocket = new ServerSocket(5037,10); Socket socket = serverSocket.accept(); InputStream inputStream = socket.getInputStream(); byte[] buffer = new byte[1024*1024]; int temp = 0; while((temp = inputStream.read(buffer))!=-1){ System.out.println(new String (buffer,0,temp)); } }catch(IOException e){ e.printStackTrace(); try { serverSocket.close(); } catch (IOException e1) { e1.printStackTrace(); } } } } }
一个客户端,TcpSocket,作为发送clientsocket与服务器通信(从本地文件读取数据,服务端的数据源)import java.io.FileInputStream; import java.io.InputStream; import java.io.OutputStream; import java.net.Socket; public class TcpClient { public static void main(String[] args) { try { // 创建一个socket对象,指定服务端的ip和端口 Socket socket = new Socket("127.0.0.1", 5037); // 使用inputstream读取硬盘上的文件 InputStream inputStream = new FileInputStream("c://test.txt"); // 从socket当中得到outputStream OutputStream outputStream = socket.getOutputStream(); byte[] buffer = new byte[1024 * 1024]; int temp = 0; // 将inputStream中的数据取出,并写入到outputstream当中 while ((temp = inputStream.read(buffer)) != -1) { outputStream.write(buffer, 0, temp); } outputStream.flush(); socket.close(); } catch (Exception e) { e.printStackTrace(); } } }
创建socket时,监听端口必须是打开的,可以在cmd下用netstat -an查看端口,然后用telnet ip -port测试(一开始我随便用一个端口8888,程序报错,拒绝连接:java.net.ConnectException: Connection refused: connect
at java.net.PlainSocketImpl.socketConnect(Native Method)
at java.net.PlainSocketImpl.doConnect(Unknown Source)
at java.net.PlainSocketImpl.connectToAddress(Unknown Source)
at java.net.PlainSocketImpl.connect(Unknown Source)
at java.net.SocksSocketImpl.connect(Unknown Source)
at java.net.Socket.connect(Unknown Source)
at java.net.Socket.connect(Unknown Source)
at java.net.Socket.<init>(Unknown Source)
at java.net.Socket.<init>(Unknown Source)
at TcpClient.main(TcpClient.java:35)
,后来改用另一端口能建立连接)另外,别忘了再manifest.xml中设置访问网络的权限:<uses-permission android:name="android.permission.INTERNET" />