多线程的write和read

对于write和read,由于缓冲区不足或者中断等问题,可能导致读不完或写不完,这也是为什么write和read分别返回的是成功的字节数的原因。所以书中给出了另一种操作readn和writen,保证了读写的完整性,但不保证原子性。

  ssize_t /* Read "n" bytes from a descriptor */
    readn(int fd, void *vptr, size_t n)
    {
        size_t nleft;
        ssize_t nread;

        char *ptr;
        ptr = static_cast<char *>(vptr);
        nleft = n;
        while (nleft > 0)
        {
            if ((nread = read(fd, ptr, nleft)) < 0)
            {
                if (errno = EINTR) //the read is interrupted by signal
                    nread = 0;     //call readn() again
                else
                    return -1; /* error, return amount read so far */
            }
            else if (nread == 0)
            {
                break; /* EOF */
            }
            nleft -= nread;
            ptr += nread;
        }
        return (n - nleft); /* return >= 0 */
    }

    ssize_t /* Write "n" bytes to a descriptor */
    writen(int fd, void *vptr, size_t n)
    {
        size_t nleft;
        ssize_t nwritten;
        const char *ptr;

        ptr = static_cast<char *>(vptr);

        nleft = n;
        while (nleft > 0)
        {
            if ((nwritten = write(fd, ptr, nleft)) <= 0)
            {
                if (nwritten < 0 && errno == EINTR)
                    nwritten = 0; /*call write again */
                else
                    return (-1); /* error */
            }
            nleft -= nwritten;
            ptr += nwritten;
        }
        return (n - nleft); /* return >= 0 */
    }

再探Linux内核write系统调用操作的原子性这篇文章中说了,write和read一次,可以保证是原子操作,因为Linux系统内部是对其上锁了的,同时APUE P63介绍了另一种原子操作pread和pwrite。

下面我们看一下write和read的操作的原子性

1、两个线程分别操作同一个文件。

#include <bits/stdc++.h>
#include <initializer_list>
#include <unistd.h>
#include <functional>
#include <stdio.h>
#include "threadpool/threadpool.h"

#include <unistd.h>
#include<stdio.h>
#include<error.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<iostream>
#include<string.h>
using namespace std;

void test3(string input)
{
   int fd=open("test.txt",O_RDWR);
   //int fd=open("test.txt",O_RDWR|O_APPEND);
   char fa[input.size()+1];

    strcpy(fa,input.c_str());


   int num=write(fd,fa,strlen(fa));//这里写的是fa长度
   
   cout<<"write number: "<<num<<endl;
   close(fd);

}


int main()
{
    threadpool::ThreadPool tt;//线程池

    string s1(2000000,'1');
    string s2(2000000,'2');
    
    tt.AddTask(test3,s1);
    tt.AddTask(test3,s2);

    getchar();
    return 0;

}


(1)直接写

int fd=open("test.txt",O_RDWR);

要么全是1,要么全是2

(2)追加写

int fd=open("test.txt",O_RDWR|O_APPEND);

要么是1111…22222,要么是22222…11111,不会出现交错

总结1

对于文件描述符,映射到Linux同一个文件指针,write和read都是原子操作

2、两个线程分别操作同一个文件描述符。

效果和上面情况类似

3、多次写

void test3(int fd,string input,string input2)
{
   
   char fa[input.size()+1];

    strcpy(fa,input.c_str());

       char fa2[input2.size()+1];

    strcpy(fa2,input2.c_str());


   int num=write(fd,fa,strlen(fa));//这里写的是fa长度
   
   cout<<"write number: "<<num<<endl;

    num=write(fd,fa2,strlen(fa2));//这里写的是fa长度
   
   cout<<"write number: "<<num<<endl;

   close(fd);

}

(1)同一个文件,不同文件描述符

不同文件描述符不会相互影响。
如果不添加写,会相互覆盖
在这里插入图片描述

添加写,会在末尾添加
在这里插入图片描述

(2)同一个文件描述符

不管是添加还是不添加,都会在最后的写上出现错误。所以共用一个文件描述符可能会出现冲突。

总结

在这里插入图片描述
这个原因可能是因为同一个文件描述符对应的偏移量是共用的。而对node都会加锁
在这里插入图片描述

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
首先需要使用Qt网络模块中的QNetworkAccessManager类来实现Http请求。下载和上传都可以使用QNetworkAccessManager来实现。 下载: 使用QNetworkAccessManager发送一个GET请求,并控制线程等待下载完成。QNetworkAccessManager下载完成后,会发出finished()信号,可以在此信号槽函数中获取下载的数据并保存到本地文件中。 上传: 使用QNetworkAccessManager发送一个POST请求,并控制线程等待上传完成。QNetworkAccessManager上传完成后,会发出finished()信号,可以在此信号槽函数中处理上传结果。 在多线程环境下,可以使用Qt中的QThread类来实现多线程。我们可以将下载和上传操作分别放在不同的线程中进行,来避免对主线程的阻塞。 示例代码: 下载: ```cpp class DownloadThread : public QThread { Q_OBJECT public: DownloadThread(QObject* parent = nullptr) : QThread(parent) {} signals: void finished(); protected: void run() override { QNetworkAccessManager manager; QNetworkReply* reply = manager.get(QNetworkRequest(QUrl("下载地址"))); QEventLoop loop; connect(reply, &QNetworkReply::finished, &loop, &QEventLoop::quit); loop.exec(); // 等待请求完成 QByteArray data = reply->readAll(); QFile file("本地文件路径"); if (file.open(QIODevice::WriteOnly)) { file.write(data); file.close(); } emit finished(); } }; // 使用方法 DownloadThread thread; connect(&thread, &DownloadThread::finished, [](){ qDebug() << "下载完成"; }); thread.start(); ``` 上传: ```cpp class UploadThread : public QThread { Q_OBJECT public: UploadThread(QObject* parent = nullptr) : QThread(parent) {} signals: void finished(); protected: void run() override { QNetworkAccessManager manager; QNetworkRequest request(QUrl("上传地址")); QHttpMultiPart* multiPart = new QHttpMultiPart(QHttpMultiPart::FormDataType); QHttpPart filePart; filePart.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"file\"; filename=\"文件名\"")); QFile file("本地文件路径"); file.open(QIODevice::ReadOnly); filePart.setBodyDevice(&file); file.setParent(multiPart); // 文件数据在multiPart中管理内存 multiPart->append(filePart); QNetworkReply* reply = manager.post(request, multiPart); QEventLoop loop; connect(reply, &QNetworkReply::finished, &loop, &QEventLoop::quit); loop.exec(); // 等待请求完成 multiPart->setParent(reply); // multiPart的内存释放由reply管理 emit finished(); } }; // 使用方法 UploadThread thread; connect(&thread, &UploadThread::finished, [](){ qDebug() << "上传完成"; }); thread.start(); ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值