C++ 读、写、改、删除本地文件内容

本文详细介绍了C++中对本地文件的读取、写入、修改和删除操作,包括利用ifstream和ofstream进行文件操作,以及处理文件路径、字符串转换、文件定位等技巧。同时,文章还提到了字符串分割、数字转换以及文件内容修改的具体实现方法。
摘要由CSDN通过智能技术生成

在写程序的过程中,有时候需要对程序中的一些参数或者变量在本地读取、存储以及修改等,因此掌握一下C++对于本地文件的一些操作还是挺有必要的。

1、读取本地文件:

C++对于本地文件的读取以及写入都还是挺简单的,主要采用了数据流的形式,读取上来的是一个个行排列的字符串,后面根据自己的需要进行详细的区分,例如字符串转数字、字符串分割等。一个简单的字符串读取函数大概可以写成这个样子:

void local_file_process::ReadLoaclFile()
{
    ifstream fin;  
    string file_path = "xxx";
    fin.open(file_path.c_str());
    string linestr;
    if(fin.is_open())//确认文本存在,is_open()用于确认该文本是否打开
    {
    	while (getline(fin, linestr))//行遍历
    	{
        //在这里进行你需要的数据处理     
    	}
    }
    else{
        ROS_ERROR("can not find locat file,check file path");
    }
    fin.close();
}

这里需要注意的是最好添加一下fin.is_open()判断,因为如果没有这个判断,即使你的路径是不正确的,但是它不会报错,只会读不到东西,这样子可能会导致后面的整个处理都错误。

第二个问题是这里关于打开路径的问题,上面我写成了:

    string file_path = "xxx";
    fin.open(file_path.c_str());

其实这里可以直接写成:

	fin.open("xxx");

但是注意到有时候我们可能处理本地文件的地方不止一个地方,所以如果直接写死的话后面要改起来会比较麻烦还容易漏掉,所以最好的方式是用一个全局变量,这样子我们只要改一个地方就可以了,保证了代码的稳定性以及便捷性。但是这里为什么没有写成:

 	string file_path = "xxx";
    fin.open(file_path);

这样子呢?
这里是由于引用的open函数的问题,open函数传参为open(const char* filepath),如果直接给string类型编译不会报错但是运行时是无法找到地址的。可以写成这样子:

 	const char* s;
    s = file_path.c_str();
    fin.open(s);  
    或者
    fin.open(file_path.c_str());
    c_str()生成一个const char *指针,指向字符串的首地址

但是一定不能直接给字符串。虽然直接给定地址的情况看起来确实很像是传入了一个字符串。

2、写本地文件:

写的方式与读的方式大体类似,具体看参考代码即可:

void local_file_process::AddLoaclFile(vector<string> msg)
{
    //写入到文本最后面
    std::ofstream out(file_path.c_str(),std::ios::app);
    int len = msg.size();
    for(int i=0;i<len;i++)
    {
        out<<msg[i]<<endl;
    }
    out.close();
}

代码非常简单,需要注意一个参数:std::ios::app。这里可以选择std::ios::app以及std::ios::ate。ios::app只在最后写入,但ios::ate默认情况下在末尾读取和写入.另外ios::ate可以在文件中自由搜索,但无论你为写入指针设置什么位置,ios::app都将始终在最后写入。所以在这里执行的时候我将会把新输入的参数放到文件的最后面。

3、修改本地文件:

例如我现在有一个本地文件,我想修改其中的某一行内容,我应该怎么做?在C++中,似乎没有对文本的行替换的功能,但是我们可以通过一个长字符串来解决这个问题:

思路如下:新建一个空字符串,将文本文件一行行读取,不需要修改的就按原来的顺序保存到字符串中,等读到需要修改的行的时候,将新的行保存到字符串中,原来的数据舍弃。然后继续保存后面的内容,直到整个文本读取完成。然后将文本清空,将新字符串保存进去。这样子就完成了某一行的修改。

代码如下:

void local_file_process::ModifyLoaclFile(vector<string> msg)
{
    ifstream fin;  
    fin.open(file_path.c_str());
    string linestr;
    string strFileData = "";//暂存新的数据的地方
    int file_line = 0;
    int line = 1;
    int line_2 = 1;
    //查找需要修改的id是哪一个
    while (getline(fin, linestr))
    {
        file_line ++;
        if(linestr == msg[1])
            break;
        line++;
    }
    fin.close();
    //如果id存在,这里应相等
    if(file_line != line)
    {//如果说目前的库位信息中没有这个id信息,新增一个新的库位信息
        AddLoaclStorage(msg);
        return;
    }
    ifstream in;  
    in.open(file_path.c_str());
    //将需要修改的四行内容保存为新的msg。其他的行不变,暂存到strFileData
    while (getline(in, linestr))
    {
        if(line_2 == line-1)
        {
            strFileData += msg[0];
            strFileData += "\n";
        }
        else if(line_2 == line)
        {
            strFileData += msg[1];
            strFileData += "\n";
        }
        else if(line_2 == line+1)
        {
            strFileData += msg[2];
            strFileData += "\n";
        }
        else if(line_2 == line+2)
        {
            strFileData += msg[3];
            strFileData += "\n";
        }
        else
        {
            strFileData += linestr;
            strFileData += "\n";
        }
        line_2++;
    }
    in.close();
    ofstream out;
    out.open(file_path.c_str());
    out.flush();//清空file内容
    out<<strFileData;//写入修改后的数据
    out.close();
}

上述代码实现的功能是:传入一个含有四个参数的容器:type、id、使用数量、总数量。首先根据传入的id判断当前需要修改的位置。如果文件中没有对应的数据则新增一条字段。如果有就开始操作文件,找到这四行代码所在的位置,用新的msg数据替换,其他数据保持不变,这样子就完成了一次对本地文件的修改。

4、删除本地文件:

删除的方式跟修改是一样的思路,找到要删除的位置,将其他数据都缓存到字符串中,需要删除的不存进去。最后清空文件,放入新的字符串。简单实现如下:

void local_file_process::DeleteLoaclFile(string msg)
{
    ROS_INFO("DeleteLoaclStorage");
    ifstream fin;  
    fin.open(file_path.c_str());
    string linestr;
    string strFileData = "";
    int file_line = 0;
    int line = 1;
    int line_2 = 1;
    while (getline(fin, linestr))
    {
        file_line++;
        if(linestr == msg)
            break;
        line++;
    }
    fin.close();
    if(file_line != line)
    {
        cout<<"delete "<<msg<<" failed,local storage has no this id"<<endl;
        return;
    }
    ROS_INFO("debug");
    ifstream in;  
    in.open(file_path.c_str());
    while (getline(in, linestr))
    {
        if(line_2 == line-1)
        {
            line_2++;
            continue;
        }
        else if(line_2 == line)
        {
            line_2++;
            continue;
        }
        else if(line_2 == line+1)
        {
            line_2++;
            continue;
        }
        else if(line_2 == line+2)
        {
            line_2++;
            continue;
        }
        else if(line_2 == line+3)
        {
            line_2++;
            continue;
        }
        else
        {
            strFileData += linestr;
            strFileData += "\n";
        }
        line_2++;
    }
    in.close();
    ofstream out;
    out.open(file_path.c_str());
    out.flush();
    out<<strFileData;
    out.close();
}

上述代码实现的功能是:给入一个传参索引,删除对应位置的5行数据。如果文本中本身没有这个索引,则返回报错。

5、涉及到的头文件:

#include "ros/ros.h"
#include <fstream>
#include <iostream>
#include <string>
#include <sstream>
#include <vector>
#include <cstdio>
#include "std_msgs/String.h"
#include <boost/algorithm/string.hpp>
#include <opencv2/opencv.hpp>
#include <opencv2/imgproc/imgproc.hpp>
#include <opencv2/highgui/highgui.hpp>

6、字符串的一些其他操作:

6.1、字符串分割:

boost库中提供了对字符串按照某个字符分割的函数:

	string linestr;
    vector<string> temp_sig;
	boost::split(temp_sig, linestr, boost::is_any_of( ":" ), boost::token_compress_on );

第一个参数是分割后的字符串存储位置,一个vector类型容器;
第二个参数是需要被分割的字符串,string类型;
第三个参数是按照什么字符分割。

6.2、字符串转数字:
string temp = "123"
int numb = atoi(temp.c_str());
//或者
int numb = stoi(temp.c_str());

stoi()函数将字符串作为参数并返回其值。 atoi()函数将字符数组或字符串文字作为参数并返回其值。atoi()是旧的C样式函数。在C ++ 11中添加了stoi()。
stoi()最多可以包含三个参数,第二个参数用于起始索引,第三个参数用于输入数字的基数。

int stoi(const string&str , size_t* index = 0,int base = 10);

类似地,为了将String转换为Double,可以使用atof()。上面的函数返回转换后的整数作为int值。如果无法执行有效的转换,它将返回零。

6.3、数字转字符串:
int numb = 123;
string temp = to_string(numb);

to_string函数可以实现简单的将数字转字符串的操作。另外,itoa()也可以实现这个功能,但是没有to_string好用。

int n = 100;
char str2[10];
//字符串比较麻烦,所以转字符串三个参数,我是这么记得(手动滑稽) 
itoa(n,str2,10); //第一个参数为整数,第二个为字符串(char*),第三个为进制 
cout << str2 << endl;
6.4、字符串拼接:

拼接两个字符串,简单的“+”就可以了,非常方便:

int a = 123;
string b = "id:";
string result = b + to_string(a);

这样你就可以得到:

result = "id:123";
6.5、字符串搜索:

搜索某个字符串中是否含有某个子串或者字符:find函数,我们一般可以跟if之类的判断语句连用,实现某些条件:

string a = "today is a good day";
if(a.find("good")==string::npos)
{
//如果没有该字符
}
else
{
//如果有该字符
}
  • 11
    点赞
  • 102
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
在Windows中,要删除一个文件,你需要具有适当的权限。你可以使用Windows API来删除文件,其中一个重要的函数是DeleteFile()。但是,如果文件有DACL(禁止访问控制列表)设置,则可能会阻止文件删除。在这种情况下,你需要修DACL以允许文件删除。 以下是在C++中修文件DACL以允许删除文件的示例代码: ```cpp #include <Windows.h> #include <AclAPI.h> int main() { LPCWSTR filePath = L"C:\\example\\file.txt"; // 打开文件的句柄 HANDLE fileHandle = CreateFileW(filePath, GENERIC_READ | DELETE, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_DELETE_ON_CLOSE, NULL); if (fileHandle == INVALID_HANDLE_VALUE) { printf("Failed to open file. Error code: %d\n", GetLastError()); return 1; } // 获取文件的DACL PSECURITY_DESCRIPTOR fileSd; if (!GetSecurityInfo(fileHandle, SE_FILE_OBJECT, DACL_SECURITY_INFORMATION, NULL, NULL, NULL, NULL, &fileSd)) { printf("Failed to get file security information. Error code: %d\n", GetLastError()); CloseHandle(fileHandle); return 1; } // 获取文件DACL PACL fileDacl; BOOL daclPresent; BOOL daclDefaulted; if (!GetSecurityDescriptorDacl(fileSd, &daclPresent, &fileDacl, &daclDefaulted)) { printf("Failed to get file DACL. Error code: %d\n", GetLastError()); LocalFree(fileSd); CloseHandle(fileHandle); return 1; } // 将当前进程的SID添加到文件的DACL中 PSID currentUserSid; if (!ConvertStringSidToSidW(L"S-1-5-20", &currentUserSid)) // 这里使用了本地服务的SID,具体可根据实际情况更 { printf("Failed to convert string SID to SID. Error code: %d\n", GetLastError()); LocalFree(fileSd); CloseHandle(fileHandle); return 1; } EXPLICIT_ACCESSW ea; ZeroMemory(&ea, sizeof(EXPLICIT_ACCESSW)); ea.grfAccessPermissions = DELETE; ea.grfAccessMode = SET_ACCESS; ea.grfInheritance = NO_INHERITANCE; ea.Trustee.TrusteeForm = TRUSTEE_IS_SID; ea.Trustee.TrusteeType = TRUSTEE_IS_USER; ea.Trustee.ptstrName = (LPWSTR)currentUserSid; PACL newDacl; if (ERROR_SUCCESS != SetEntriesInAclW(1, &ea, fileDacl, &newDacl)) { printf("Failed to set ACE in file DACL. Error code: %d\n", GetLastError()); LocalFree(currentUserSid); LocalFree(fileSd); CloseHandle(fileHandle); return 1; } // 将新的DACL设置为文件的安全信息 if (ERROR_SUCCESS != SetSecurityInfo(fileHandle, SE_FILE_OBJECT, DACL_SECURITY_INFORMATION, NULL, NULL, newDacl, NULL)) { printf("Failed to set new file security information. Error code: %d\n", GetLastError()); LocalFree(currentUserSid); LocalFree(fileSd); CloseHandle(fileHandle); return 1; } // 关闭文件句柄,删除文件 CloseHandle(fileHandle); DeleteFileW(filePath); LocalFree(currentUserSid); LocalFree(fileDacl); LocalFree(fileSd); printf("File deleted successfully.\n"); return 0; } ``` 在上面的代码中,我们首先打开一个文件句柄,然后获取文件的DACL。接下来,我们使用ConvertStringSidToSidW()函数将当前进程的SID转换为SID结构体。然后,我们创建一个EXPLICIT_ACCESSW结构体,将当前进程的SID添加到其中,并将其设置为允许删除文件。使用SetEntriesInAclW()函数将新的ACE(访问控制项)添加到文件的DACL中。然后,我们将新的DACL设置为文件的安全信息。最后,我们关闭文件句柄并删除文件。 需要注意的是,这个例子只是在演示如何修文件DACL以允许删除文件。在实际应用中,你需要根据实际情况来修代码。比如,你需要使用适当的SID来代表你的进程,而不是本地服务的SID。你也需要检查函数返回值以处理错误情况。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

一叶执念

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值