图形聊天室

仿照第4篇笔记的形式,笔者决定将GUI和网络编程部分用综合练习的方式来总结。练习项目是有图形界面的聊天室,用到了GUI中的javax.swing包和网络编程中的TCP/socket编程。GUI部分的难点是图形控件的API较为复杂,设置不同属性需要很多的方法和字段,需要参考API手册和网上的一些图形界面作品的代码,不过GUI代码的结构较为固定(相对于聊天室简单的界面而言),容易总结固定格式。网络编程部分的难点是聊天室涉及到多客户端之间通过服务器的通信,不仅服务器要使用多线程,而且每个服务线程都要求能够调取其它服务线程中的socket以便向其它客户端传递消息,这就需要专门的数据容器来储存所有服务线程。网络编程部分的另一个难点是消息的结构设计。由于一个socket只有一对输入输出流,来自客户端和服务器的各种不同类型消息都要通过这对流来传递给对方,所以服务器和客户端都要能根据消息的类型采取不同的动作。这需要仔细考虑消息的形式、结构和解析方法。

编程用了两天时间,下面简要介绍下实现的功能,GUI和网络编程部分的思路,详细说明可以见最后的代码和注释。

1. 实现功能

作为聊天室软件实现了:

(1)可显示并即时更新当前在线列表。当新的客户端连接上服务器,或者在线的客户端退出时,客户端向服务器发送消息,服务器会立刻更新所有客户端的在线列表。在线列表显示当前在线者(不包括自己)的网名、IP和端口;

(2)发送消息时用户从在线列表中选择消息接收者,数量从一个人到所有人皆可;

(3)接收消息时显示发送者IP和端口;

(4)与服务器失去连接时可以在聊天窗口显示异常信息;

(5)服务器用多线程方式工作,有静态容器存储所有服务线程。

2. GUI部分概要

只有客户端需要图形界面。这个界面具有:

(1)聊天窗口:显示自己发送的和收到的信息(包括发信人身份),用设置成不可编辑的JTextArea控件实现,用JScrollPane控件包装来实现滚动条;

(2)打字窗口:输入聊天消息,用设置成可编辑的JTextArea控件实现;

(3)当前在线列表:显示当前在线的所有人的网名,IP和端口,由服务器即时更新。发送聊天消息时需要在表中选择消息接收人,从一个人到所有人皆可。用JTable控件实现,被服务器更新时可动态插入或删除行。用JScrollPane控件包装来实现滚动条;

(4)发送按钮:将打字窗口中的文字按照在线列表中选择的收信人发给服务器,由后者转发给收信人,随即将打字窗口清屏,用JButton控件实现;

(5)清屏按钮:将聊天窗口清屏,用JButton控件实现;

(6)退出按钮:向服务器发出退出消息,关闭此客户端程序。服务器接收到后更新所有客户端的当前在线列表;

(7)收信人标签:与在线列表中选择的收信人一致,起提醒作用。若用户没有选择任何收信人,则不能发送聊天消息。

3. 网络编程部分概要

(1)使用TCP/Socket连接。服务器使用多线程工作,每个客户端都享有一个服务线程;

(2)每个客户端用自己的IP地址和端口号组成一个字符串作为用户标识(uid)

(3)客户端和服务器之间每次通信都是传递一个字符串,这个字符串可能有这几种结构:

     Exit/                          客户端发往服务器,表示该客户端退出

     Chat/收信人地址/聊天内容       客户端发往服务器,表示该客户端要对别的客户端发送聊天消息

     Chat/发信人地址/聊天内容       服务器发往客户端,表示服务器转发给收信客户端的聊天消息

     OnlineListUpdate/在线者地址    服务器发往客户端,表示有客户端加入或退出,要更新所有客户端的当前在线列表

(4)收信人地址,发信人地址,在线者地址字符串都采用以下形式:

     第一个客户端IP地址:第一个客户端端口号,第二个客户端IP地址:第二个客户端端口号,.....

     如果是发信人地址,则只有一个客户端IP地址和端口号

(5)服务器类有两个静态容器:

    一个是String数组,用来储存当前在线的所有人的uid,

    一个是HashTable<String, 服务线程>, 存储所有服务线程,可以根据uid取出对应的服务线程

    当客户端加入或退出时,先更新服务器中的这两个容器,添加或删除相应元素,再向客户端发消息更新其在线列表

(6)服务器用while(true)循环中持续监听客户端消息,根据消息类型作出反应。收到"Exit/"类型消息就向所有客户端发出"OnlineListUpdate/在线者地址"类型消息,

    收到"Chat/收信人地址/聊天内容"类型消息就向收信客户端发出"Chat/发信人地址/聊天内容"类型消息;

(7)客户端用while(true)循环中持续监听服务器消息,根据消息类型作出反应。收到"Chat/发信人地址/聊天内容"类型消息就在聊天窗口中显示发信人地址和聊天内容,

    收到"OnlineListUpdate/在线者地址"类型消息更新在线列表控件显示新的在线列表;

(8)服务器只有在收到客户端消息时才会发送消息;

(9)客户端只有按发送或退出按钮时才会发送消息。

4. 功能示例


5. 服务器代码

import java.io.*;

import java.util.*;

import java.net.*;

import java.text.*;

public class Server

{

    public static void main(String[] args) throws Exception

    {

        //建立服务器ServerSocket

        ServerSocket ss = new ServerSocket(10000);

        //提示Server建立成功

        System.out.println("Server online... " + ss.getInetAddress().getLocalHost().getHostAddress() + ", " + 10000);

        //监听端口,建立连接并开启新的ServerThread线程来服务此连接

        while(true)

        {

            //接收客户端Socket

            Socket s = ss.accept();

            //提取客户端IP和端口

            String ip = s.getInetAddress().getHostAddress();

            int port = s.getPort();

            //建立新的服务器线程, 向该线程提供服务器ServerSocket,客户端Socket,客户端IP和端口

            new Thread(new ServerThread(s, ss, ip, port)).start();

        }

    }

}

class ServerThread implements Runnable

{

    //获取的客户端Socket

    Socket s = null;

    //获取的服务器ServerSocket

    ServerSocket ss = null;

    //获取的客户端IP

    String ip = null;

    //获取的客户端端口

    int port = 0;

    //组合客户端的ip和端口字符串得到uid字符串

    String uid = null;


    //静态ArrayList存储所有uid,uid由ip和端口字符串拼接而成

    static ArrayList<String> uid_arr = new ArrayList<String>();

    //静态HashMap存储所有uid, ServerThread对象组成的对

    static HashMap<String, ServerThread> hm = new HashMap<String, ServerThread>();


    public ServerThread(Socket s, ServerSocket ss, String ip, int port)

    {

        this.s = s;

        this.ss = ss;

        this.ip = ip;

        this.port = port;

        uid = ip + ":" + port;

    }

    @Override

    public void run()

    {

        //将当前客户端uid存入ArrayList

        uid_arr.add(uid);

        //将当前uid和ServerThread对存入HashMap

        hm.put(uid, this);

        //时间显示格式

        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");


        //控制台打印客户端IP和端口

        System.out.println("Client connected: " + uid);

        try

        {

            //获取输入流

            InputStream in = s.getInputStream();

            //获取输出流

            OutputStream out = s.getOutputStream();

            //向当前客户端传输连接成功信息

            String welcome = sdf.format(new Date()) + "\n成功连接服务器...\n服务器IP: " + ss.getInetAddress().getLocalHost().getHostAddress() + ", 端口: 10000\n客户端IP: " + ip + ", 端口: " + port + "\n";

            out.write(welcome.getBytes());

            //广播更新在线名单

            updateOnlineList(out);

            //准备缓冲区

            byte[] buf = new byte[1024];

            int len = 0;


            //持续监听并转发客户端消息

            while(true)

            {

                len = in.read(buf);

                String msg = new String(buf, 0, len);

                System.out.println(msg);

                //消息类型:退出或者聊天

                String type = msg.substring(0, msg.indexOf("/"));

                //消息本体:空或者聊天内容

                String content = msg.substring(msg.indexOf("/") + 1);

                //根据消息类型分别处理

                //客户端要退出

                if(type.equals("Exit"))

                {

                    //更新ArrayList和HashMap, 删除退出的uid和线程

                    uid_arr.remove(uid_arr.indexOf(uid));

                    hm.remove(uid);

                    //广播更新在线名单

                    updateOnlineList(out);

                  //控制台打印客户端IP和端口

                    System.out.println("Client exited: " + uid);

                    //结束循环,结束该服务线程

                    break;

                }

                //客户端要聊天

                else if(type.equals("Chat"))

                {

                    //提取收信者地址

                    String[] receiver_arr = content.substring(0, content.indexOf("/")).split(",");

                    //提取聊天内容

                    String word = content.substring(content.indexOf("/") + 1);

                    //向收信者广播发出聊天信息

                    chatOnlineList(out, uid, receiver_arr, word);

                }

            }

        }

        catch(Exception e){}

    }


    //向所有已连接的客户端更新在线名单

    public void updateOnlineList(OutputStream out) throws Exception

    {

        for(String tmp_uid : uid_arr)

            {

                //获取广播收听者的输出流

                out = hm.get(tmp_uid).s.getOutputStream();

                //将当前在线名单以逗号为分割组合成长字符串一次传送

                StringBuilder sb = new StringBuilder("OnlineListUpdate/");

                for(String member : uid_arr)

                {

                    sb.append(member);

                    //以逗号分隔uid,除了最后一个

                    if(uid_arr.indexOf(member) != uid_arr.size() - 1)

                        sb.append(",");

                }

                out.write(sb.toString().getBytes());

            }

    }

    //向指定的客户端发送聊天消息

    public void chatOnlineList(OutputStream out, String uid, String[] receiver_arr, String word) throws Exception

    {

        for(String tmp_uid : receiver_arr)

            {

                //获取广播收听者的输出流

                out = hm.get(tmp_uid).s.getOutputStream();

                //发送聊天信息

                out.write(("Chat/" + uid + "/" + word).getBytes());

            }

    }

}

6. 客户端代码

import java.io.*;

import java.net.*;

import javax.swing.*;

import javax.swing.table.*;

import java.awt.*;

import java.awt.event.*;

import java.awt.geom.*;

import java.util.*;

import java.nio.charset.*;

import java.text.*;

public class Client1

{

    //建立客户端Socket

    static Socket s = null;

    //消息接收者uid

    static StringBuilder uidReceiver = null;

    public static void main(String[] args)

    {

        //创建客户端窗口对象

        ClientFrame cframe = new ClientFrame();

        //窗口关闭键无效,必须通过退出键退出客户端以便善后

        cframe.setDefaultCloseOperation(JFrame.DO_NOTHING_ON_CLOSE);

        //获取本机屏幕横向分辨率

        int w = Toolkit.getDefaultToolkit().getScreenSize().width;

        //获取本机屏幕纵向分辨率

        int h = Toolkit.getDefaultToolkit().getScreenSize().height;

        //将窗口置中

        cframe.setLocation((w - cframe.WIDTH)/2, (h - cframe.HEIGHT)/2);

        //设置客户端窗口为可见

        cframe.setVisible(true);


        try

        {

            //连接服务器

            s = new Socket(InetAddress.getLocalHost(), 10000);

            //获取输入流

            InputStream in = s.getInputStream();

            //获取输出流

            OutputStream out = s.getOutputStream();

            //获取服务端欢迎信息

            byte[] buf = new byte[1024];

            int len = in.read(buf);

            //将欢迎信息打印在聊天消息框内

            cframe.jtaChat.append(new String(buf, 0, len));

            cframe.jtaChat.append("\n");

            //持续等待接收服务器信息直至退出

            while(true)

            {

                in = s.getInputStream();

                len = in.read(buf);

                System.out.println(len);

                //处理服务器传来的消息

                String msg = new String(buf, 0, len);

                //消息类型:更新在线名单或者聊天

                String type = msg.substring(0, msg.indexOf("/"));

                //消息本体:更新后的名单或者聊天内容

                String content = msg.substring(msg.indexOf("/") + 1);

                //根据消息类型分别处理

                //更新在线名单

                if(type.equals("OnlineListUpdate"))

                {

                    //提取在线列表的数据模型

                    DefaultTableModel tbm = (DefaultTableModel) cframe.jtbOnline.getModel();

                    //清除在线名单列表

                    tbm.setRowCount(0);

                    //更新在线名单

                    String[] onlinelist = content.split(",");

                    //逐一添加当前在线者

                    for(String member : onlinelist)

                    {

                        String[] tmp = new String[3];

                        //如果是自己则不在名单中显示

                        if(member.equals(InetAddress.getLocalHost().getHostAddress() + ":" + s.getLocalPort()))

                            continue;

                        tmp[0] = "";

                        tmp[1] = member.substring(0, member.indexOf(":"));

                        tmp[2] = member.substring(member.indexOf(":") + 1);

                        //添加当前在线者之一

                        tbm.addRow(tmp);

                    }

                    //提取在线列表的渲染模型

                    DefaultTableCellRenderer tbr = new DefaultTableCellRenderer();

                    //表格数据居中显示

                    tbr.setHorizontalAlignment(JLabel.CENTER);

                    cframe.jtbOnline.setDefaultRenderer(Object.class, tbr);

                }

                //聊天

                else if(type.equals("Chat"))

                {

                    String sender = content.substring(0, content.indexOf("/"));

                    String word = content.substring(content.indexOf("/") + 1);

                    //在聊天窗打印聊天信息

                    cframe.jtaChat.append(cframe.sdf.format(new Date()) + "\n来自 " + sender + ":\n" + word + "\n\n");

                    //显示最新消息

                    cframe.jtaChat.setCaretPosition(cframe.jtaChat.getDocument().getLength());

                }

            }

        }

        catch(Exception e)

        {

            cframe.jtaChat.append("服务器挂了.....\n");

            e.printStackTrace();

        }

    }

}

//客户端窗口

class ClientFrame extends JFrame

{

    //时间显示格式

    SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");

    //窗口宽度

    final int WIDTH = 700;

    //窗口高度

    final int HEIGHT = 700;


    //创建发送按钮

    JButton btnSend = new JButton("发送");

    //创建清除按钮

    JButton btnClear = new JButton("清屏");

    //创建退出按钮

    JButton btnExit = new JButton("退出");

    //创建消息接收者标签

    JLabel lblReceiver = new JLabel("对谁说?");

    //创建文本输入框, 参数分别为行数和列数

    JTextArea jtaSay = new JTextArea();

    //创建聊天消息框

    JTextArea jtaChat = new JTextArea();

    //当前在线列表的列标题

    String[] colTitles = {"网名", "IP", "端口"};

    //当前在线列表的数据

    String[][] rowData = null;

    //创建当前在线列表

    JTable jtbOnline = new JTable

                                (

                                    new DefaultTableModel(rowData, colTitles)

                                    {

                                        //表格不可编辑,只可显示

                                        @Override

                                        public boolean isCellEditable(int row, int column)

                                        {

                                            return false;

                                        }

                                    }

                                );


    //创建聊天消息框的滚动窗

    JScrollPane jspChat = new JScrollPane(jtaChat);

    //创建当前在线列表的滚动窗

    JScrollPane jspOnline = new JScrollPane(jtbOnline);

    //设置默认窗口属性,连接窗口组件

    public ClientFrame()

    {

        //标题

        setTitle("聊天室");

        //大小

        setSize(WIDTH, HEIGHT);

        //不可缩放

        setResizable(false);

        //设置布局:不适用默认布局,完全自定义

        setLayout(null);

        //设置按钮大小和位置

        btnSend.setBounds(20, 600, 100, 60);

        btnClear.setBounds(140, 600, 100, 60);

        btnExit.setBounds(260, 600, 100, 60);

        //设置标签大小和位置

        lblReceiver.setBounds(20, 420, 300, 30);

        //设置按钮文本的字体

        btnSend.setFont(new Font("宋体", Font.BOLD, 18));

        btnClear.setFont(new Font("宋体", Font.BOLD, 18));

        btnExit.setFont(new Font("宋体", Font.BOLD, 18));

        //添加按钮

        this.add(btnSend);

        this.add(btnClear);

        this.add(btnExit);

        //添加标签

        this.add(lblReceiver);

        //设置文本输入框大小和位置

        jtaSay.setBounds(20, 460, 360, 120);

        //设置文本输入框字体

        jtaSay.setFont(new Font("楷体", Font.BOLD, 16));

        //添加文本输入框

        this.add(jtaSay);


        //聊天消息框自动换行

        jtaChat.setLineWrap(true);

        //聊天框不可编辑,只用来显示

        jtaChat.setEditable(false);

        //设置聊天框字体

        jtaChat.setFont(new Font("楷体", Font.BOLD, 16));

        //设置滚动窗的水平滚动条属性:不出现

        jspChat.setHorizontalScrollBarPolicy(JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);

        //设置滚动窗的垂直滚动条属性:需要时自动出现

        jspChat.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED);

        //设置滚动窗大小和位置

        jspChat.setBounds(20, 20, 360, 400);

        //添加聊天窗口的滚动窗

        this.add(jspChat);

        //设置滚动窗的水平滚动条属性:不出现

        jspOnline.setHorizontalScrollBarPolicy(JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);

        //设置滚动窗的垂直滚动条属性:需要时自动出现

        jspOnline.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED);

        //设置当前在线列表滚动窗大小和位置

        jspOnline.setBounds(420, 20, 250, 400);

        //添加当前在线列表

        this.add(jspOnline);

        //添加发送按钮的响应事件

        btnSend.addActionListener

                                (

                                    new ActionListener()

                                    {

                                        @Override

                                        public void actionPerformed(ActionEvent event)

                                        {

                                            //显示最新消息

                                            jtaChat.setCaretPosition(jtaChat.getDocument().getLength());

                                            try

                                            {

                                                //有收信人才发送

                                                if(Client1.uidReceiver.toString().equals("") == false)

                                                {

                                                    //在聊天窗打印发送动作信息

                                                    jtaChat.append(sdf.format(new Date()) + "\n发往 " + Client1.uidReceiver.toString() + ":\n");

                                                    //显示发送消息

                                                    jtaChat.append(jtaSay.getText() + "\n\n");

                                                    //向服务器发送聊天信息

                                                    OutputStream out = Client1.s.getOutputStream();

                                                    out.write(("Chat/" + Client1.uidReceiver.toString() + "/" + jtaSay.getText()).getBytes());

                                                }

                                            }

                                            catch(Exception e){}

                                            finally

                                            {

                                                //文本输入框清除

                                                jtaSay.setText("");

                                            }

                                        }

                                    }

                                );

        //添加清屏按钮的响应事件

        btnClear.addActionListener

                                (

                                    new ActionListener()

                                    {

                                        @Override

                                        public void actionPerformed(ActionEvent event)

                                        {

                                            //聊天框清屏

                                            jtaChat.setText("");

                                        }

                                    }

                                );

        //添加退出按钮的响应事件

        btnExit.addActionListener

                                (

                                    new ActionListener()

                                    {

                                        @Override

                                        public void actionPerformed(ActionEvent event)

                                        {

                                            try

                                            {

                                                //向服务器发送退出信息

                                                OutputStream out = Client1.s.getOutputStream();

                                                out.write("Exit/".getBytes());

                                                //退出

                                                System.exit(0);

                                            }

                                            catch(Exception e){}

                                        }

                                    }

                                );

        //添加在线列表项被鼠标选中的相应事件

        jtbOnline.addMouseListener

                                (

                                    new MouseListener()

                                    {

                                        @Override

                                        public void mouseClicked(MouseEvent event)

                                        {

                                            //取得在线列表的数据模型

                                            DefaultTableModel tbm = (DefaultTableModel) jtbOnline.getModel();

                                            //提取鼠标选中的行作为消息目标,最少一个人,最多全体在线者接收消息

                                            int[] selectedIndex = jtbOnline.getSelectedRows();

                                            //将所有消息目标的uid拼接成一个字符串, 以逗号分隔

                                            Client1.uidReceiver = new StringBuilder("");

                                            for(int i = 0; i < selectedIndex.length; i++)

                                            {

                                                Client1.uidReceiver.append((String) tbm.getValueAt(selectedIndex[i], 1));

                                                Client1.uidReceiver.append(":");

                                                Client1.uidReceiver.append((String) tbm.getValueAt(selectedIndex[i], 2));

                                                if(i != selectedIndex.length - 1)

                                                    Client1.uidReceiver.append(",");

                                            }

                                            lblReceiver.setText("发给:" + Client1.uidReceiver.toString());

                                        }

                                        @Override

                                        public void mousePressed(MouseEvent event){};

                                        @Override

                                        public void mouseReleased(MouseEvent event){};

                                        @Override

                                        public void mouseEntered(MouseEvent event){};

                                        @Override

                                        public void mouseExited(MouseEvent event){};

                                    }

                                );

    }

}<span style="color:#3333ff;">

</span>

7. 总结

聊天室软件综合运用了类集框架,多线程,GUI和网络编程的知识。在写程序时笔者发现两个静态容器非常关键,它们是沟通不同客户端的桥梁。另外应当重视注释,否则像代码稍多的程序维护起来就会很困难。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容

  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,934评论 6 13
  • 计算机网络概述 网络编程的实质就是两个(或多个)设备(例如计算机)之间的数据传输。 按照计算机网络的定义,通过一定...
    蛋炒饭_By阅读 1,215评论 0 10
  • 1.OkHttp源码解析(一):OKHttp初阶2 OkHttp源码解析(二):OkHttp连接的"前戏"——HT...
    隔壁老李头阅读 20,827评论 24 176
  • 远隔千里的朋友今日找我聊天,说近况,说那些她无法告知身边人的情绪。说罢,她说最近很忙,我知道的,她朋友圈步数排行榜...
    小夕有言阅读 347评论 0 1
  • 三十三年前, 我带着希望和梦想, 来到自己执着追求的工作岗位; 三十三年后, 我拿着荣誉和感慨, 回归原本属于自己...
    清风明月X阅读 135评论 2 2