网络通信基石Socket (上)

来源:互联网 发布:淘宝网卖家中心入口 编辑:程序博客网 时间:2024/06/09 15:45

socket是什么,socket的原理是什么想必大家都了解个大概了,如果不懂的可以参考上一篇文章 socket原理的简单理解
这里我们也简单回顾一下:
Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口。
在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议。

1.Socket传输模式

1.1 Socket有两种主要操作方式

1.1.1 面向连接(TCP)

面向连接的Socket操作就像一部电话机,必须要等对方接上之后才能通话。所有的数据到达的顺序与它出发时的顺序是一样的。
面向连接的操作使用TCP协议,即此模式下必须先连接上目的地的Socket,连接上后Socket就可以使用一个流接口进行打开、读、写、关闭等操作。所有所发信息都会在另一端以同样的顺序被接收。安全性高,但效率低。

1.1.2 无连接的(UDP)

无连接的就像是一个邮件投递,没有保证,多个邮件到达时的顺序可能与出发时的顺序不一样。
无连接的操作使用数据报协议,一个数据报是一个独立的单元,它包含了这次投递的所有信息。可将其想象成一个信封,这个模式下的Socket不需要连接一个目的Socket,它只是简单的投出数据报。无连接的操作时快速和高效的,但是数据安全性不高。

2.Socket构造

android的编程语言是java,所以java提供的工具包在android都可以使用。
查看官方提供的API文档,在java.net包中提供两个类Socket和ServerSocket,分别用来表示双向连接的客户端和服务器端.
Socket类用来建立client的socket其构造方法如下:

    Socket(InetAddress address,int port);    Socket(InetAddress address,int port,boolean stream);    Socket(String host,int port);    Socket(String host,int port,boolean stream);    Socket(SocketImpl impl);    Socket(String host,int port,InetAddress localAddr,int localPort);    Socket(InetAddress address,int port,InetAddress localAddr,int localPort);

ServerSocket类用来建立server的socket其构造方法如下:

    ServerSocket(int port);    ServerSocket(int port,int backlog);    ServerSocket(int port,int backlog,InetAddress bindAddr);

其中的参数意义:
address :双向连接中另一方的IP地址
host:双向连接中另一方的主机名
port:双向连接中另一方的端口号
stream:指明Socket是流Socket还是数据报Socket
localPort:本地主机的端口号
localAddr和bindAddr是本地机器的地址(ServerSocket的主机地址)
impl 是Socket的父类,即可以用来创建ServerSocket,又可以用来创建Socket

使用

    //client端    Socket client = new Socket("192.168.8.254",9999);    //server端    ServerSocket server = new ServerSocket(9999);

再次强调:0~1023的端口号为系统所保留,选择端口号必须大于1024。

3.异常捕获

在创建Socket时如果发生错误,将产生IOException,所以在创建Socket和ServerSocket时必须捕获或抛出异常。
方法:

try{    //这里执行可能出现异常的程序}catch(IOException e){//catch()中的参数视程序出现的异常而定    //这里可以做相应的异常处理}

想了解try{}catch(){}和有关异常请看
android的异常处理——try、catch、finally、throw、throws
4.输入、输出流
java的数据传输是采用流的机制实现
流:实质是数据的有序排列,流可以看作是数据从某个源出来,输送到某个目的地
根据官方的API文档,java.net.Socket类提供了getInputStream()和getOutPutStream()来得到对应的输入(输出)流以进行读(写)操作,这两个方法分别返回InputStream和OutputStream类对象。
为了便于读(写)数据,可以在返回输入、输出流对象上建立过滤流。如:
字符流对象:
DataInputStream、DataOutPutStream、或PrintStream类对象。
文本方式流对象:
InputStreamReader和OutputStreamWriter、PrintWirter
处理代码如下:

//字符流方式PrintStream os = new PrintStream(new BufferedOutputStream(Socket.getOutputStream()));DataInputStream is = new DataInputStream(socket.getInputStream());//文本方式PrintWriter out = new PrintWriter(socket.getOutStream(),true);BufferedReader in = new ButfferedReader(new InputStreamReader(Socket.getInputStream()));

放上一张java的io流图,这张图描述的已经很清楚,后面可能还会对java的io流资料进行整理,图是出网上摘来,如果涉及版权问题,第一时间处理
java基础之IO流

5.关闭Socket和流
在Socket使用完毕后需要将其关闭,以释放资源。
注意:在关闭Socket之前,应将与Socket相关的所有的输入、输出流先关闭,以释放资源。要注意关闭的顺序。

     os.close();//输出流先关闭     is.close();//输入流其次     socket.close();//最后关闭Socket

下例中 实现一个服务器和客户端通信。客户端发送数据并接受服务器发回的数据。效果图就不上了,放心测试可用。

import java.io.BufferedWriter;import java.io.InputStream;import java.io.OutputStreamWriter;import java.io.PrintWriter;import java.net.ServerSocket;import java.net.Socket;public class server implements Runnable{//服务器实现 注意:该程序需要单独编译,并在命令行模式下启动    String str="";     public void run(){    try{           ServerSocket serverSocket = new ServerSocket(8899);//创建ServerSocket 设置端口号为9999        while (true){               Socket client = serverSocket.accept();//通过accept监听接受客户端请求            System.out.println("accept");            try{                    PrintWriter out = new PrintWriter( new BufferedWriter( new OutputStreamWriter(client.getOutputStream())),true);  //通过PrintWriter向服务器发送消息,但需要通过Socket对象来取得其输出流                out.println("server message 让我放假");                //BufferedReader in = new BufferedReader(new InputStreamReader(client.getInputStream()));//通过BufferedReader对象接收客户端消息                 InputStream in = client.getInputStream();//定义输入流,来自于socket的输入流                 byte[] bytes2 = new byte[20];                 in.read(bytes2);//读取输入流数据                String str = new String(bytes2);//转换成字符串                System.out.println(str);                out.flush();                out.close();//关闭流                in.close();                }catch (Exception e){                    System.out.println(e.getMessage());                    e.printStackTrace();                }finally{                         client.close();//关闭                    System.out.println("close");                }            }            }catch (Exception e){                System.out.println(e.getMessage());            }            //System.out.println(str);            }            public static void main(String a[]){//main函数用来开启服务器                Thread desktopServerThread = new Thread(new server());                desktopServerThread.start();//开启线程        }    }

客户端
简单描述一下:
应用布局只有三个:TextView , EditText , Button
在编辑框输入内容然后点击button就能收到服务器发送来的数据并显示在textView上

import java.io.IOException;import java.io.InputStream;import java.io.OutputStream;import java.net.Socket;import android.os.Bundle;import android.os.Handler;import android.os.Message;import android.app.Activity;import android.view.Menu;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;import android.widget.EditText;import android.widget.TextView;import android.widget.Toast;public class MainActivity extends Activity {    private EditText edt;    private Button but;    private TextView tv;    private Handler handler=null;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        edt = (EditText)findViewById(R.id.editText1);        but = (Button)findViewById(R.id.button1);        tv = (TextView)findViewById(R.id.textView1);        handler = new Handler(){            public void handleMessage(Message msg){                switch(msg.what){                case 1 : tv.setText(msg.obj.toString());                    break;                case 2 : Toast.makeText(MainActivity.this,"Connection successful!",                        Toast.LENGTH_LONG).show();                    break;                }            }        };        but.setOnClickListener(new OnClickListener(){            public void onClick(View v){            new Thread(new Runnable(){                public void run(){                    try {                    @SuppressWarnings("resource")                    Socket socket = new Socket("192.168.8.107",8899);                    //要判断当前的Socket对象是否处于连接状态,必须同时使用isClose和isConnected方法,                    //即只有当isClose返回false,isConnected返回true的时候Socket对象才处于连接状态。                     if (socket.isClosed() == false && socket.isConnected() == true){ //还有一个问题就是&&的用法!!                         /*******注意不能在主线程里里更新UI************///                      Toast.makeText(MainActivity.this,"Connection successful!",//                              Toast.LENGTH_LONG).show();//                         Message message = new Message();                         message.what=2;                         handler.sendMessage(message);                    }                    InputStream in = socket.getInputStream();                    OutputStream output = socket.getOutputStream();                    String str = edt.getText().toString();                    output.write(str.getBytes());                    byte[] buffer = new byte[in.available()];                    in.read(buffer);                    String msg = new String(buffer,"GB2312");//乱码需要加GB2312                    Message message = new Message();                    message.obj=msg;                    message.what=1;                    handler.sendMessage(message);                } catch (IOException e) {                    // TODO Auto-generated catch block                    e.printStackTrace();                }                   }            }).start();            }           });    }    @Override    public boolean onCreateOptionsMenu(Menu menu) {        // Inflate the menu; this adds items to the action bar if it is present.        getMenuInflater().inflate(R.menu.main, menu);        return true;    }}

通过上例总结了一下:
使用Socket实现客户端的步骤;
1、通过IP地址和端口实例化Socket,请求连接服务器
2、获取Socket上的流以进行读写
3、把流包装进OutputStream/InputStream的实例
4、对Socket进行读写
5、关闭打开的流
创建服务器的步骤:
1、指定端口实例化一个ServerSocket
2、调用ServerSocket的accept()以在等待连接期间造成阻塞
3、获取位于该层Socket的流以进行读写操作
4、将数据封装成流
5、对Socket进行读写
6、关闭打开的流
上面这个例子只是能实现单个客户点的发送和接收数据,下一章将对实现多个客户端的通信。

0 0
原创粉丝点击