Java的Socket网络编程以及多线程

1.Socket是代表两台机器之间网络连接的对象(java.net.Socket)。

   Socket的建立如下,参数分别是服务器端的IP地址和端口号: Socket socket  =  new Socket("167.5.75.1",5000);

2.    客户端(Client)Socket的使用 

 2.1 从Socket读出数据步骤

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// 1.创建Socket连接,告知Server的IP地址以及端口号
Socket socket = new Socket( "127.0.0.1" , 4242 ); 
// 2.创建InputStreamReader,用于读取socket输入流
InputStreamReader stream = new InputStreamReader(socket.getInputStream());
// 3.使用BufferedReader链接输入流
BufferedReader br = new BufferedReader(stream);
 
// 4.读出数据
String line = null ;
while ((line = br.readLine()) != null )
{
System.out.println( "Today's advice is: " + line);
}
 
// 5. 关闭输入流BufferedReader
br.close();

  2.2 向Scoket写入数据步骤: 

?
1
2
3
4
5
6
7
8
9
// 1.创建Socket连接,告知Server的IP地址以及端口号
Socket socket = new Socket( "127.0.0.1" , 4242 );
// 2.创建PrintWriter对象,用以接收socket输出流
PrintWriter writer = new PrintWriter(socket.getOutputStream());
// 3.使用PrintWriter对象写出输出数据
String advice = "Today's advice" ;
writer.println(advice);
// 4. 关闭连接
writer.close();

3.  服务器端(Server)Socket的使用 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// 1.创建一个SercerSocket,使用4242端口监听客户端请求
ServerSocket serverSocket = new ServerSocket( 4242 );
System.out.println( "The server is started, listening on port 4242" );
 
while ( true )
{
// 2.ServerSocket的accept()在等待用户连接的时候闲置;在用户连接上来的时候,返回一个Socket来与客户端通信
Socket socket = serverSocket.accept();
// 3.创建PrintWriter对象,用以接收socket输出流
PrintWriter writer = new PrintWriter(socket.getOutputStream());
// 4.使用PrintWriter对象写出输出数据
String advice = "notifier's blog" ;
writer.println(advice);
 
// 5. 关闭连接
writer.close();
}

4.   线程的状态

       线程总共有5种状态:

  1. 新建 (Thread t = new Thread())

  2. 就绪 (t.start())

  3. 运行

  4. 堵塞

    线程被block的原因很多,比如: 等待IO操作, sleep(), 等待被占用对象释放

     5. 死亡

5.    解决线程同步化问题的方法是: 对使用到共享对象的方法使用synchronized

   需要注意的是:

       虽说是方法进行了synchronized,但锁不是加在方法上的而是对象上的,也就是说,是synchronized方法获取对象锁。如果对象(类)有两个或者多个synchronized方法,就表示两个线程不能同时进入同一个方法,也不能同时进入不同的方法。 因为同一时间,只有一个方法在占有对象锁。

6.    synchronized代码块

       有时候在一个方法中做了很多事情,但只有一部分逻辑是需要synchronized的,这时候我们可以使用synchronized代码块。如下,其中this表示当前对象: 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
public void function()
{
     doSomething();
     
     //以下方法需要同步化
     synchronized ( this )
     {
         doCriticalStuff();
         moreCriticalStuff();
     }
     
     doSomeOtherThing();
}

7. 以下是一个Socket简单的例子:

客户端代码及详细注释:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
/**
  * @author notifier
  * @create 2010-9-25 上午10:12:10
  * @version 1.0
  */
public class DailyAdviceClient
{
     public static void main(String[] args)
     {
         DailyAdviceClient client = new DailyAdviceClient();
         client.receiveMsg();
     }
     
     public void receiveMsg()
     {
         try
         {
             // 1.创建Socket连接,告知Server的IP地址以及端口号
             Socket socket = new Socket( "127.0.0.1" , 4242 );
             // 2.创建InputStreamReader,用于读取socket输入流
             InputStreamReader stream = new InputStreamReader(socket
                     .getInputStream());
             // 3.使用BufferedReader链接输入流
             BufferedReader br = new BufferedReader(stream);
 
             // 4.读出数据
             String line = null ;
             while ((line = br.readLine()) != null )
             {
                 System.out.println( "Today's advice is: " + line);
             }
 
             // 5. 关闭输入流BufferedReader
             br.close();
         } catch (UnknownHostException e)
         {
             e.printStackTrace();
         } catch (IOException e)
         {
             e.printStackTrace();
         }
     }
}

服务器端代码及详细注释:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
/**
  * @author notifier
  * @create 2010-9-25 下午07:06:54
  * @version 1.0
  */
public class SimpleChatServer
{
     // 保存客户端列表
     private ArrayList clientList = new ArrayList();;
     
     public static void main(String[] args)
     {
         new SimpleChatServer().startUp();
     }
     
     /**
      * 负责服务器端的启动
      *
      */
     public void startUp()
     {
         try
         {
             // 创建服务器端ServerSocket连接,监听端口号5000
             ServerSocket serverSocket = new ServerSocket( 5000 );
             
             // 轮询等待客户端请求
             while ( true )
             {
                 // 等待客户端请求,无请求则闲置;有请求到来时,返回一个对该请求的socket连接
                 Socket clientSocket = serverSocket.accept();
                 
                 // 将该客户端加入到列表中
                 PrintWriter writer = new PrintWriter(clientSocket.getOutputStream());
                 clientList.add(writer);
                 
                 // 创建ClientHandler对象,通过socket连接通信
                 Thread t = new Thread( new ClientHandler(clientSocket));
                 t.start();
 
                 System.out.println( "有Client连进来" );
             }
         } catch (Exception e)
         {
             e.printStackTrace();
         }
     }
     
     /**
      * 客户端处理类, 主要负责:
      * 1.接收客户端发来的消息
      * 2.将消息转发其他客户端
      * @author sdniu
      * @create 2010-9-26 上午10:00:18
      * @version 1.0
      */
     public class ClientHandler implements Runnable
     {
         private BufferedReader reader;
         private Socket socket;
         
         /**
          * ClientHandler的构造函数
          * @param clientSocket
          */
         public ClientHandler(Socket clientSocket)
         {
             
             try
             {
                 // 得到socket连接
                 socket = clientSocket;
                 
                 // 得到客户端发来的消息
                 InputStreamReader isReader = new InputStreamReader(socket.getInputStream());
                 reader = new BufferedReader(isReader);
             } catch (IOException e)
             {
                 e.printStackTrace();
             }
         }
         
         public void run()
         {
             String message;
             try
             {
                 while ((message = reader.readLine()) != null )
                 {
                     System.out.println( "客户端消息: " + message);
                     // 将客户端发来的消息转发所有客户端
                     notifyAllClients(message);
                 }
             } catch (IOException e)
             {
                 e.printStackTrace();
             }
         }
     }
     
     /**
      *
      * @param message
      */
     public void notifyAllClients(String message)
     {
         // 得到客户端列表的迭代器,语法格式为 Iterator it = clientList.iterator();
         Iterator it = clientList.iterator();
         
         while (it.hasNext())
         {
             try
             {
                 // 得到的Iterator别忘了强制转换回PrintWriter
                 PrintWriter writer = (PrintWriter) it.next();
                 writer.println(message);
                 writer.flush();
             } catch (Exception e)
             {
                 e.printStackTrace();
             }          
         }
     }
}

8. 多线程Socket编程的例子, 代码比较长, 放在下载里了, 链接如下:

     http://files.cnblogs.com/notifier/SimpleChatroom.7z

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值