Socket通信中flush()方法的使用

首先介绍四个方法:readLine(),flush(),close(),println()方法

readLine()方法:实质上readLine()是一个阻塞方法,API上的定义:读取一个文本行。通过下列字符之一即可认为某行已终止:换行 ('\n')、回车 ('\r') 或回车后直接跟着换行。也就是说,如果,readLine()没有读取到'\r','\n'换行符,则会一直阻塞,等待换行符,否则就无法结束读取。

flush()方法:查看源码的解释:flush()这个方法在PrintStream,PrintWriter,OutputStream的子类都有的成员方法,在OutputStream类的构造器中没有指定autoflush,而PrintWriter和PrintStream类中有PrintStream(OutputStream op,boolean autoflush),而没有指定autoflush()默认值是false;也就是无法自动将数据从缓冲区刷出,这也就是为什么在Socket中使用输出流的情况下一般都需要手动刷新。

flush()方法

* Flushes the stream. If the stream has saved any characters from the

* various write() methods in a buffer, write them immediately to their

* intended destination. Then, if that destination is another character or

* byte stream, flush it. Thus one flush() invocation will flush all the

* buffers in a chain of Writers and OutputStreams.

close():将流关闭,关闭前将缓冲区内容刷出到目标文件,或者目标主机地址端口号;

println(String str):API上解释为:打印 String,然后终止该行。(终止该行的意思就是,数据输出完后,自动追加换行符。)

下面上代码:代码实现客户端和服务器循环通讯,当客户端发送exit时,停止通信(注意,必须是交替进行的)

服务器代码:

import java.io.*;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;

/***
 *  通过客户端发送的exit来结束通信
 *  多线程接收和发送
 *
 *  按行输出输出和按行读取readLine()和println()配合使用,
 *  尤其,println后一定要使用flush将缓冲区内容刷新出去
 *
 * */
public class Test12_Server {
    public static void main(String[] args) throws Exception{
        ServerSocket server=new ServerSocket(8888);
        while(true){
            Socket socket = server.accept();
            new Thread(){
                public void run(){
                    Scanner sc = new Scanner(System.in);
                    try {
                        BufferedReader br=new BufferedReader(new InputStreamReader(socket.getInputStream()));
                        PrintStream ps=new PrintStream(socket.getOutputStream());

                        String client_msg=br.readLine();
                        System.out.println(client_msg+"---来自客户端");
                        if("exit".equals(client_msg)){
                            socket.close();
                            System.exit(0);
                        }
                        ps.println(sc.nextLine());
                        
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }.start();
            //socket.close();
        }

    }
}

客户端代码:

import java.io.*;
import java.net.InetAddress;
import java.net.Socket;
import java.util.Scanner;

/**
 * 通过键盘录入发送消息给服务器,直到输入exit后停止发送
 * 多线程发送和接收
 * */
public class Test12_Client {
    public static void main(String[] args) throws Exception{
        while(true){
            Socket socket=new Socket(InetAddress.getLocalHost(),8888);
            new Thread(){
                public void run(){
                    Scanner sc = new Scanner(System.in);
                    try {

                        BufferedReader br=new BufferedReader(new InputStreamReader(socket.getInputStream()));
                        PrintStream ps=new PrintStream(socket.getOutputStream());
                        ps.println(sc.nextLine());
                        String server_msg=br.readLine();
                        ps.flush();
                        System.out.println(server_msg+"---来自服务器");
                        if(null==server_msg){
                            socket.close();
                            System.exit(0);
                        }
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }.start();
            //socket.close();
        }
    }
}

 

  • 2
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
完整代码如下,其释放信号量的部分已经包含在代码: ```cpp #include <QtWidgets> #include <QLocalServer> #include <QSystemSemaphore> int main(int argc, char *argv[]) { QApplication app(argc, argv); // 创建系统信号量 QSystemSemaphore semaphore("MyApplication", 1, QSystemSemaphore::Create); if (!semaphore.acquire()) { QMessageBox::warning(nullptr, "Warning", "Another instance is already running."); return 0; } // 创建本地服务器 QLocalServer localServer; if (!localServer.listen("MyApplication")) { QMessageBox::warning(nullptr, "Warning", "Unable to start the server: " + localServer.errorString()); semaphore.release(); // 释放信号量 return 0; } // 处理连接请求 QObject::connect(&localServer, &QLocalServer::newConnection, [&](){ QLocalSocket *socket = localServer.nextPendingConnection(); socket->write("Hello, world!"); socket->flush(); socket->waitForBytesWritten(); socket->close(); delete socket; }); // 显示主窗口 QMainWindow mainWindow; mainWindow.show(); // 释放信号量 semaphore.release(); return app.exec(); } ``` 在本例,我们在创建系统信号量时指定了初始值为1,表示该信号量最多只能被一个进程同时持有。在程序启动时,我们首先使用`acquire()`函数尝试获取该信号量。如果获取失败,说明已经有一个实例在运行,我们就弹出一个警告窗口并退出程序。如果获取成功,我们就创建一个本地服务器并开始监听连接请求。在本地服务器收到连接请求时,我们会向客户端发送一条消息,然后关闭连接。在程序退出时,我们通过`release()`函数释放信号量。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值