c/c++开发,无可避免的文件访问开发案例
一、缓存文件系统
ANSI C标准中的C语言库提供了fopen, fclose, fread, fwrite, fgetc, fgets, fputc, fputs, freopen, fseek, ftell, rewind等标准函数,这些函数在不同的操作系统中应该调用不同的内核API,从而支持开发者跨平台实现对文件的访问。
在Linux环境下,fopen是对open的封装,open系列是 POSIX 定义的,是UNIX系统里的system call。Linux环境下,一切设备皆是文件,一切设备皆是以文件的形式进行操作,如网络套接字、硬件设备等,对于Linux系统内核而言,文本文件和二进制代码文件并无区别,而open、write、read这些函数通过操作系统的功能对文件进行读写,是系统级的输入输出,它不设文件结构体指针,只能读写二进制文件。
不同于open、write这些函数,fopen 、fread这些函数是标准的C库函数,更具有可移植性,几乎可以可移植到任何操作系统。在实际项目中,尤其是上位应用中,一般是用fopen 、fread这些函数来操作普通文件,fopen返回一个文件指针,就可以利用文件指针操作文件,其可以读取一个结构数据,并支持缓冲IO。由于支持缓存IO,因此fopen 、fread这些函数在处理稍大的文件时,系统自动分配缓存,则读出此文件等操作要比open、write这些函数效率高。
因此,fopen 、fread是属于缓冲文件系统,系统在内存中开辟一个“缓冲区”,为程序里每一个文件使用,当执行读文件操作时,从磁盘文件将数据先读入内存“缓冲区”,装满后再从内存“缓 冲区”依次读入接收的变量。执行写文件操作时,也是先将数据写入内存“缓冲区”,待内存“缓冲区”装满后再写入文件。
fopen 函数返回的是一个FILE结构指针,借助于文件结构体指针FILE *就可以实现对文件管理,通过文件指针对文件进行访问,即可以读写字符、字符串、格式化数据,也可以读写二进制数据。
fopen 函数原型是:FILE *fopen(const char* file, const char* model),可以指定一个文件路径来打开一个文件,成功时返回非NULL的FILE *。fclose, fread, fwrite, fgetc, fgets, fputc, fputs等函数都是基于该文件指针进行操作。
二、文件访问开发设计
2.1 文件接口服务设计
本文将通过fopen, fclose, fread, fwrite, fgetc, fgets等系列函数,实现跨windows和linux的文件访问,进行数据读取、写入,并在此基础上,提供对文件内容行读取与写入、列分割等快捷功能,同时实现都目录及文件的创建、存在判定、删除、重命名等操作。
例如:
//文件操作相关
bool writeToFile(const char *ptr,const std::string& path, const std::string& mode);//写入一段内容
bool writeListInfo(std::vector<std::string> list,const std::string path);//写入多行内容
bool readFromFile(char *ptr, int &size, int buf_size,const std::string& path, const std::string& mode);//读取一段内容
bool readListInfo(std::vector<std::vector<std::string> > &list,const std::string path);//读取多行内容
bool readListInfo(std::vector<std::vector<std::string> > &list,const std::string path);//读取多行内容,每行按空格分割//目录相关
bool createDir(std::string _dir); //创建目录
bool isExistDir(std::string _dir); //判断指定目录是否存在
bool getCurrentDir(char* dir_, int len_);//获取当前目录路径//文件相关
void renameFile(const std::string& _oldName, const std::string& _newName);//重命名文件
void removeFile(const std::string& _Name); //删除文件
bool isExist(std::string file); //文件(绝对及相对路径名)是否存在
bool isExist(std::string dir,std::string file); //某目录下某文件是否存在
文件的读写伪代码如下:
//打开
FILE *m_fil = ::fopen(path,mode);
//读取
int size = fread(buf, static_cast<int>(sizeof(char)), (size_t)buf_size,m_fil);
//写入
::fwrite(buf, static_cast<int>(sizeof(char)), buf_len,m_fil );
//循环读取每行数据
while(NULL!=::fgets(buf,buf_size,m_fil))
在目录及文件名处理方面伪代码如下:
//获取路径
char* dir = getcwd(dir_, len_);
//访问目录-linux
DIR* dfd = opendir(_dir.c_str());
//访问目录-win
bool ret = _access(_dir.c_str(),0);
//创建目录(win _mkdir)
int ret = mkdir(_dir.c_str(), S_IRWXU | S_IRWXG | S_IRWXO);//文件是否存在-win
_finddata_t fileInfo;
intptr_t hFile = _findfirst(file,&fileInfo);
//文件是否存在-Linux是否能打开文件来判断
::fopen
//重命名文件
int ret = rename(_oldName,_newName);
//删除文件
int ret = remove(_Name);
行内容按空格或“.”划分:
//将采用空格间隔的字符串划分成子字符串集
std::string buffer; // Have a buffer string
std::stringstream _lineStr(_lineInfo); // Insert the string into a stream
std::vector<std::string> _lineList; // Create vector to hold our words
while(_lineStr >> buffer)
{_lineList.push_back(buffer);
}//将采用","间隔的字符串划分成子字符串集
void pyfree::getNumberFromStr(std::vector<std::string> &ids, std::string idsStr)
{if(""==idsStr||"NULL"==idsStr)return;std::string id = "";for(unsigned int i=0,idSize= static_cast<unsigned int>(idsStr.size()); i<idSize; i++){if(','==idsStr[i]){//use str ',' to dirve the idsStr;if(""!=id){ids.push_back(id);id.clear();id = "";}else{ids.push_back(id);}}else if(i==(idSize-1)){if(""!=id){ids.push_back(id);id.clear();id = "";}}else{id.push_back(idsStr[i]);}}
}
在实际项目中,往往会遇到中文情况,尤其是在linux系统下,不少文件是采用ascii或gbk、gb2312等存储编码格式,但在程序内处理数据是大多是采用utf-8编码格式,因此必要时还需要对读取到的内容做编码格式转换。
//增加codef类型参数来确定是否进行编码格式转换
bool readListInfo(std::vector<std::vector<std::string> > &list,const std::string path,int codef=0);//读取多行内容
bool readListInfo(std::vector<std::vector<std::string> > &list,const std::string path,int codef=0);//读取多行内容,每行按空格分割
2.2 工程及代码设计
创建项目目录结构如下:
myfile_testbinbuild_win //win编译过程文件路径build_linux //Linux编译过程文件目录src //文件接口源码test //接口调用示例CMakeLists.txt //cmake工程配置文件编译指令.txt //编辑及运行命令
在src目录下创建File.h/cpp源文件,File.h设计如下,File.cpp见附录:
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000#ifndef _FILE_H_
#define _FILE_H_
/************************************************************************Copyright 2023-02-07, pyfree**File Name : File.h*File Mark : *Summary : *文本读写函数类**Current Version : 1.00*Author : pyfree*FinishDate :**Replace Version :*Author :*FinishDate :************************************************************************/
#include <string>
#include <vector>
#include <stdio.h>namespace pyfree
{/*** 向指定文件写入数据* @param ptr {const char*} 待写入内容* @param path {string} 文件名* @param mode {string} 写入模式,等同于fopen或open函数的写入模式* @return {bool} */bool writeToFile(const char *ptr,const std::string& path, const std::string& mode);/*** 向指定文件追写数据* @param list {vector} 待写入内容容器,每项作为一行写入* @param path {string} 文件名* @return {bool} */bool writeListInfo(std::vector<std::string> list,const std::string path);/*** 从指定文件读取数据* @param ptr {char*} 待读取内容指针* @param size {int&} 实际读取大小* @param buf_size {int} 待读取指针存储空间大小* @param list {vector} 待写入内容容器,每项作为一行写入* @param path {string} 文件名* @param mode {string} 读取模式,等同于fopen或open函数的读取模式* @return {bool} */bool readFromFile(char *ptr, int &size, int buf_size,const std::string& path, const std::string& mode);/*** 从指定文件中读取文件每一行的字符串,如果行中存在空格,划分该行字符串,并装在到容器中* @param list {vector<vector>} 读取内容容器,每一行作为一项载入容器,每行内容分割后容器装载* @param path {string} 文件名* @param codef {int} 读取时编码转换{1:g2u,2:a2u,其他不做处理}* @return {bool} */bool readListInfo(std::vector<std::vector<std::string> > &list,const std::string path,int codef=0);/*** 从指定文件中读取文件每一行的字符串,如果行中存在","间隔,划分该行字符串,并装在到容器中* @param list {vector<vector>} 读取内容容器,每一行作为一项载入容器,每行内容分割后容器装载* @param path {string} 文件名* @param codef {int} 读取时编码转换{1:g2u,2:a2u,其他不做处理}* @return {bool} */bool readListInfo_dot(std::vector<std::vector<std::string> > &list,const std::string path,int codef=0);/*** 从指定文件中读取文件每一行的字符串并装在到容器中* @param list {vector} 读取内容容器,每一行作为一项载入容器* @param path {string} 文件名* @param codef {int} 读取时编码转换{1:g2u,2:a2u,其他不做处理}* @return {bool} */bool readListInfo(std::vector<std::string> &list,const std::string path,int codef=0);/*** 将采用","间隔的字符串划分成子字符串集,被readListInfo_dot调用* @param ids {vector} 读取内容容器* @param idsStr {string} 被检查字符串* @return {void} */void getNumberFromStr(std::vector<std::string> &ids, std::string idsStr);/*** 从指定文件目录下读取指定扩展名的文件名,返回结果包含扩展名* @param directory {string} 文件目录* @param extName {string} 扩展名* @param fileNames {vector} 读取结果* @return {bool} 是否成功*/void getAllFileName_dot(const std::string& directory, const std::string& extName, std::vector<std::string>& fileNames);/*** 从指定文件目录下读取指定扩展名的文件名,返回结果不包含扩展名* @param directory {string} 文件目录* @param extName {string} 扩展名* @param fileNames {vector} 读取结果* @return {bool} 是否成功*/void getAllFileName(const std::string& directory, const std::string& extName, std::vector<std::string>& fileNames);/*** 从指定文件目录下读取指定扩展名的文件名,被getAllFileName_dot和getAllFileName调用* @param directory {string} 文件目录* @param extName {string} 扩展名* @param fileNames {vector} 读取结果* @param f {bool} 是否包含扩展名* @return {bool} 是否成功*/void getAllFileNamePrivate(const std::string& directory, const std::string& extName, std::vector<std::string>& fileNames, bool f=false);/*** 读取文件修改时间* @param file_path {string} 文件名(全路径)* @return {uint} 时间(time_t)*/unsigned int getFileModifyTime(const std::string file_path);/*** 重命名文件* @param _oldName {string} 旧文件名* @param _newName {string} 新文件名* @return {void}*/void renameFile(const std::string& _oldName, const std::string& _newName);/*** 删除文件* @param _Name {string} 文件名(全路径)* @return {void}*/void removeFile(const std::string& _Name);/*** 文件是否存在* @param file {string} 文件名(全路径)* @return {bool}*/bool isExist(std::string file);/*** 文件在指定目录下是否存在* @param dir {string} 指定目录* @param file {string}* @return {bool}*/bool isExist(std::string dir,std::string file);/*** 将多很节点的xml表述文件重新批量重写为单根节点的xml文件集* @param _xmlFile {string} 文件名* @param _xmlDir {string} 目录* @param _xmlDiv {string} 根节点名* @param utfFlag {bool} 是否utf编码* @return {void}*/void divXmlFile(std::string _xmlFile,std::string _xmlDir,std::string _xmlDiv,bool utfFlag);/*** 创建文件目录* @param _dir {string} 目录名* @return {bool}*/bool createDir(std::string _dir);/*** 文件目录是否存在* @param _dir {string} 目录名* @return {bool}*/bool isExistDir(std::string _dir);/*** 获取程序当前所在目录* @param dir_ {char*} 目录名* @return {bool}*/bool getCurrentDir(char* dir_, int len_);
};#endif //_FILE_H_
在src目录下创建strTrim.h,用来去除字符串前后空格。
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000#ifndef _STRTRIM_H_
#define _STRTRIM_H_
/************************************************************************Copyright 2023-02-08, pyfree**File Name : strTrim.h*File Mark : *Summary : *字符串前后空格处置函数**Current Version : 1.00*Author : pyfree*FinishDate :**Replace Version :*Author :*FinishDate :************************************************************************/#include <string>
#include <vector>
#include <algorithm>
#include <functional>namespace pyfree
{
/*** 删除字符串前面的空格* @param ss {string&} 传入字符串* @return {string& } 返回字符串*/
inline std::string& lTrim(std::string &ss)
{ss.erase(0, ss.find_first_not_of(" \t")); //return ss;
}
/*** 删除字符串后面的空格* @param ss {string&} 传入字符串* @return {string& } 返回字符串*/
inline std::string& rTrim(std::string &ss)
{ss.erase(ss.find_last_not_of(" \t") + 1); //return ss;
}
/*** 删除字符串前面和后面的空格* @param ss {string&} 传入字符串* @return {string& } 返回字符串*/
inline std::string& trim(std::string &st)
{lTrim(rTrim(st));return st;
}
/*** 擦除字符串后面的结束字符* @param ss {string&} 传入字符串* @return {string& } 返回字符串*/
inline void trimEnd(std::string &dir)
{if(dir.empty()){return;}//some line is NULLif(dir.size()==1){return;}while(dir[dir.length()-1] == '\n'|| dir[dir.length()-1] == '\r'|| dir[dir.length()-1] == '\t'){dir = dir.substr(0,dir.size()-1);}
};};#endif //
在src目录下,创建strchange.h/cpp,用来处理字符编码格式转换,strchange.h内容如下,strchange.cpp见附录。
#ifdef WIN32
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#endif#ifndef _STR_CHANGE_H_
#define _STR_CHANGE_H_/************************************************************************Copyright 2023-02-07, pyfree**File Name : strchange.h*File Mark : *Summary : *字符编码相关函数集**Current Version : 1.00*Author : pyfree*FinishDate :**Replace Version :*Author :*FinishDate :************************************************************************/#include <string>namespace pyfree
{
#ifdef WIN32
//UTF-8 to Unicode
std::wstring Utf82Unicode(const std::string& utf8string);
//unicode to ascii
std::string WideByte2Acsi(std::wstring& wstrcode);
//ascii to Unicode
std::wstring Acsi2WideByte(std::string& strascii);
//Unicode to Utf8
std::string Unicode2Utf8(const std::wstring& widestring);
//
std::wstring stringToWstring(const std::string& str);
//
std::string wstringToString(const std::wstring& wstr);
#endif#ifdef __linux__
int code_convert(char *from_charset,char *to_charset,char *inbuf,size_t inlen,char *outbuf,size_t outlen);
int u2g(char *inbuf,int inlen,char *outbuf,int outlen);
int g2u(char *inbuf,size_t inlen,char *outbuf,size_t outlen);
int u2a(char *inbuf,int inlen,char *outbuf,int outlen);
int a2u(char *inbuf,int inlen,char *outbuf,int outlen);
int u2k(char *inbuf,int inlen,char *outbuf,int outlen);
int k2u(char *inbuf,int inlen,char *outbuf,int outlen);
#endif//utf-8 to ascii
std::string UTF_82ASCII(std::string& strUtf8Code);
//ascii to Utf8
std::string ASCII2UTF_8(std::string& strAsciiCode);
};#endif //STRCHANGE_HPP
在str目录下,创建pfunc_print.h,通过宏定义用来打印运行日志信息。
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000#ifndef _PFUNC_PRINT_H_
#define _PFUNC_PRINT_H_
/************************************************************************Copyright 2022-11-06, pyfree**File Name : pfunc_print.h*File Mark : *Summary : 打印输出通用宏定义**Current Version : 1.00*Author : pyfree*FinishDate :**Replace Version :*Author :*FinishDate :************************************************************************/
typedef enum PrintLevel
{ LL_NOTICE = 1, //一般输出LL_WARNING = 2, //告警输出LL_TRACE = 3, //追踪调试LL_DEBUG = 4, //软件bugLL_FATAL = 5 //致命错误
}PrintLevel;#define Print_NOTICE(log_fmt,...) \do{ \printf("L(%d)[%s:%d][%s] \n"log_fmt"\n", LL_NOTICE,__FILE__, __LINE__, __FUNCTION__, ##__VA_ARGS__); \}while (0) #define Print_WARN(log_fmt,...) \do{ \printf("L(%d)[%s:%d][%s] \n"log_fmt"\n", LL_WARNING, __FILE__, __LINE__, __FUNCTION__, ##__VA_ARGS__); \}while (0) #define Print_TRACE(log_fmt,...) \do{ \printf("L(%d)[%s:%d][%s] \n"log_fmt"\n", LL_TRACE,__FILE__, __LINE__, __FUNCTION__, ##__VA_ARGS__); \}while (0) #define Print_DEBUG(log_fmt,...) \do{ \printf("L(%d)[%s:%d][%s] \n"log_fmt"\n", LL_DEBUG, __FILE__, __LINE__, __FUNCTION__, ##__VA_ARGS__); \}while (0) #define Print_FATAL(log_fmt,...) \do{ \printf("L(%d)[%s:%d][%s] \n"log_fmt"\n",LL_FATAL, __FILE__, __LINE__, __FUNCTION__, ##__VA_ARGS__); \}while (0) #endif
在test目录下,创建main.cpp,File IO模块调用示例程序
#include "File.h"
#include "pfunc_print.h"int main(int argc, char* argv[])
{std::string dir_ = "test";std::string file_name = "test.txt";std::string file_ = dir_ + "/" + file_name;std::string comment_ = "test\n";//写入信息需要自行加"\n",是因为实际项目使用中,有时写入数据需要连续性std::vector<std::string> comments_ ;comments_.push_back("test1 test10\n");comments_.push_back("test2,test20\n");comments_.push_back("test3,test30\n");//char cur_dir_buf[128]={0};bool ret = pyfree::getCurrentDir(cur_dir_buf,128);//获取当前目录if(!ret){Print_DEBUG("getCurrentDir fail!\n");}else{Print_NOTICE("CurrentDir is %s!\n",cur_dir_buf);}//目录创建测试ret = pyfree::createDir(dir_);//在当前目录创建子目录:dir_if(!ret){Print_DEBUG("createDir %s fail!\n",dir_.c_str());}else{Print_NOTICE("createDir %s success!\n",dir_.c_str());}//目是否存在ret = pyfree::isExistDir(dir_);//判断子目录dir_是否存在(是否创建成功)if(!ret){Print_DEBUG("Dir %s is not exist!\n",dir_.c_str());}else{Print_NOTICE("Dir %s is exist!\n",dir_.c_str());}//单段内容写入测试ret = pyfree::writeToFile(comment_.c_str(),file_,"w");//重新写入,file_不存在将新建if(!ret){Print_DEBUG("writeToFile %s to %s fail!\n",comment_.c_str(),file_.c_str());}else{Print_NOTICE("writeToFile %s to %s success!\n",comment_.c_str(),file_.c_str());}//指定文件路径名判断ret = pyfree::isExist(file_);//判断文件file_是否存在(或是否创建成功)if(!ret){Print_DEBUG("file_path %s is not exist!\n",file_.c_str());}else{Print_NOTICE("file_path %s is exist!\n",file_.c_str());}//指定目录及文件名判断ret = pyfree::isExist(dir_,file_name);//当前目录的子目录dir_下是否存在文件名file_nameif(!ret){Print_DEBUG("Dir %s file %s is not exist!\n",dir_.c_str(),file_name.c_str());}else{Print_NOTICE("Dir %s file %s is exist!\n",dir_.c_str(),file_name.c_str());}//多行写入测试ret = pyfree::writeListInfo(comments_,file_); //追加写入一组内容if(!ret){Print_DEBUG("writeListInfo to %s fail!\n", file_.c_str());}else{Print_NOTICE("writeListInfo to %s success!\n", file_.c_str());}//文件内容读取测试char buf[512]={0};int readLen = 0;ret = pyfree::readFromFile(buf,readLen,512,file_,"r");//从文件读取指定长度数据if(!ret){Print_DEBUG("readFromFile from %s fail!\n", file_.c_str());}else{Print_NOTICE("readFromFile from %s:\n%s\n", file_.c_str(),buf);}//行读取测试std::vector<std::string> list;ret = pyfree::readListInfo(list,file_);//从文件按行读取数据if(!ret){Print_DEBUG("readListInfo from %s fail!\n", file_.c_str());}else{Print_NOTICE("readListInfo from %s success[%d]:", file_.c_str(),(int)list.size());for(unsigned int i=0; i<list.size(); i++){printf("row[%u]:%s\n",i,list.at(i).c_str());}}//空格间隔测试std::vector<std::vector<std::string> > lists;//从文件按行、列读取数据,列按空格分割ret = pyfree::readListInfo(lists,file_);if(!ret){Print_DEBUG("readListInfo from %s fail!\n", file_.c_str());}else{Print_NOTICE("readListInfo from %s success[%d]:", file_.c_str(),(int)lists.size());for(unsigned int i=0; i<lists.size(); i++){for(unsigned int j=0; j<lists.at(i).size(); j++){printf("row[%u] col[%u]:%s",i,j,lists.at(i).at(j).c_str());}printf("\n");}}//,间隔测试lists.clear();ret = pyfree::readListInfo_dot(lists,file_);//从文件按行、列读取数据,列按","分割if(!ret){Print_DEBUG("readListInfo from %s fail!\n", file_.c_str());}else{Print_NOTICE("readListInfo from %s success[%d]:", file_.c_str(),(int)lists.size());for(unsigned int i=0; i<lists.size(); i++){for(unsigned int j=0; j<lists.at(i).size(); j++){printf("row[%u] col[%u]:%s",i,j,lists.at(i).at(j).c_str());}printf("\n");}}std::vector<std::string> fileNames;pyfree::getAllFileName(dir_,"txt",fileNames);//获取子目录dir_下的所有txt格式文件名,文件名不带扩展名if(fileNames.empty()){Print_DEBUG("getAllFileName from %s fail!\n", dir_.c_str());}else{Print_NOTICE("getAllFileName from %s success[%d]:", dir_.c_str(),(int)fileNames.size());for(unsigned int i=0; i<fileNames.size(); i++){printf("fileName[%u]:%s \n",i,fileNames.at(i).c_str());}}fileNames.clear();pyfree::getAllFileName_dot(dir_,"txt",fileNames);//获取子目录dir_下的所有txt格式文件名,文件名带扩展名if(fileNames.empty()){Print_DEBUG("getAllFileName_dot from %s fail!\n", dir_.c_str());}else{Print_NOTICE("getAllFileName_dot2 from %s success[%d]:", dir_.c_str(),(int)fileNames.size());for(unsigned int i=0; i<fileNames.size(); i++){printf("fileName[%u]:%s \n",i,fileNames.at(i).c_str());}}dir_ = "test_s";//行读取测试,ANSI格式list.clear();file_= dir_+"/"+"test1.ini";ret = pyfree::readListInfo(list,file_,2);//从文件按行读取数据if(!ret){Print_DEBUG("readListInfo ANSI fomat from %s fail!\n", file_.c_str());}else{Print_NOTICE("readListInfo ANSI fomat from %s success[%d]:", file_.c_str(),(int)list.size());for(unsigned int i=0; i<list.size(); i++){printf("row[%u]:%s\n",i,list.at(i).c_str());}}//行读取测试,GB格式list.clear();file_= dir_+"/"+"test2.ini";ret = pyfree::readListInfo(list,file_,1);//从文件按行读取数据if(!ret){Print_DEBUG("readListInfo GB fomat from %s fail!\n", file_.c_str());}else{Print_NOTICE("readListInfo GB fomat from %s success[%d]:", file_.c_str(),(int)list.size());for(unsigned int i=0; i<list.size(); i++){printf("row[%u]:%s\n",i,list.at(i).c_str());}}return 0;
}
2.3 编译及测试
完成后,项目文件结构如下:
myfile_testbinbuild_win //win编译过程文件路径build_linux //Linux编译过程文件目录src //文件接口源码File.hFile.cpppfunc_print.hstrTrim.hstrchange.hstrchange.cpptest //接口调用示例main.cppCMakeLists.txt //cmake工程配置文件编译指令.txt //编辑及运行命令
CMakeLists.txt内容如下,基于cmake+vs(win)或+g++(linux)编译工具:
# CMake 最低版本号要求
cmake_minimum_required (VERSION 2.8)
# 项目信息
project (file_test)
#
if(WIN32)message(STATUS "windows compiling...")add_definitions(-D_PLATFORM_IS_WINDOWS_)set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /MT")set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /MTd")set(WIN_OS true)
else(WIN32)message(STATUS "linux compiling...")add_definitions( -D_PLATFORM_IS_LINUX_)add_definitions("-Wno-invalid-source-encoding")set(UNIX_OS true)set(_DEBUG true)endif(WIN32)#
set(EXECUTABLE_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/bin)# 指定源文件的目录,并将名称保存到变量
SET(source_h#${PROJECT_SOURCE_DIR}/src/pfunc_print.h${PROJECT_SOURCE_DIR}/src/strTrim.h${PROJECT_SOURCE_DIR}/src/strchange.h${PROJECT_SOURCE_DIR}/src/File.h)SET(source_cpp#${PROJECT_SOURCE_DIR}/src/strchange.cpp${PROJECT_SOURCE_DIR}/src/File.cpp${PROJECT_SOURCE_DIR}/test/main.cpp)#头文件目录
include_directories(${PROJECT_SOURCE_DIR}/src)if (${UNIX_OS})add_definitions("-W""-fPIC""-Wall"# "-Wall -g""-Werror""-Wshadow""-Wformat""-Wpointer-arith""-D_REENTRANT""-D_USE_FAST_MACRO""-Wno-long-long""-Wuninitialized""-D_POSIX_PTHREAD_SEMANTICS""-DACL_PREPARE_COMPILE""-Wno-unused-parameter""-fexceptions")set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -O0")link_directories()
# 指定生成目标
add_executable(file_test ${source_h} ${source_cpp} )
#link
target_link_libraries(file_test # -lpthread -pthread-lz -lrt -ldl
)endif(${UNIX_OS})if (${WIN_OS})set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /wd4819")add_definitions("-D_CRT_SECURE_NO_WARNINGS""-D_WINSOCK_DEPRECATED_NO_WARNINGS""-DNO_WARN_MBCS_MFC_DEPRECATION""-DWIN32_LEAN_AND_MEAN"
)link_directories()if (CMAKE_BUILD_TYPE STREQUAL "Debug")set(CMAKE_RUNTIME_OUTPUT_DIRECTORY_DEBUG ${PROJECT_SOURCE_DIR}/bin)
# 指定生成目标
add_executable(file_testd ${source_h} ${source_cpp})else(CMAKE_BUILD_TYPE)set(CMAKE_RUNTIME_OUTPUT_DIRECTORY_RELEASE ${PROJECT_SOURCE_DIR}/bin)
# 指定生成目标
add_executable(file_test ${source_h} ${source_cpp})endif (CMAKE_BUILD_TYPE)endif(${WIN_OS})
编译指令如下:
win:
cd myfile_test && mkdir build_win && cd build_win
cmake -G "Visual Studio 14 2015 Win64" -DCMAKE_BUILD_TYPE=Release .. -Wno-dev
#vs 命令窗口
msbuild file_test.sln /p:Configuration="Release" /p:Platform="x64"linux:
cd file_test && mkdir build_linux && cd build_linux
cmake ..
make
按上述指令编译,测试如下:
先在bin目录下创建test_s文件夹,并在该子目录创建test1.ini文件,该文件编码格式为ANSI格式,写上一些内容;在该子目录创建test2.ini文件,该文件编码格式为 GB格式,写上一些中文内容。
启动程序测试如下:
三、源码附录
File.cpp
#include "File.h"#include <stdarg.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sstream>
#ifdef WIN32
#include <io.h>
#include <windows.h>
#include <direct.h>#define getcwd _getcwd
#endif
#ifdef linux
#include <string.h>
#include <errno.h>
#include <dirent.h>
#include <unistd.h>
#endif#include "strchange.h"
#include "strTrim.h"
#include "pfunc_print.h"bool pyfree::writeToFile(const char *ptr,const std::string& path, const std::string& mode)
{FILE *m_fil = NULL;m_fil = ::fopen(path.c_str(),mode.c_str());if(NULL!=m_fil){size_t size = 0;size = ::fwrite(ptr, static_cast<int>(sizeof(char)), strlen(ptr),m_fil);::fclose(m_fil);m_fil = NULL;if(size>0){return true;}else{return false;}}else{m_fil = NULL;return false;}
}bool pyfree::writeListInfo(std::vector<std::string> list,const std::string path)
{size_t newSize = 0;FILE *new_m_fil=NULL;new_m_fil = ::fopen(path.c_str(),"a+");if(NULL!=new_m_fil){for(size_t i=0,lSize=list.size(); i<lSize; i++){newSize += ::fwrite(list.at(i).c_str(), static_cast<int>(sizeof(char)), strlen(list.at(i).c_str()),new_m_fil);}::fclose(new_m_fil);}new_m_fil = NULL;if(newSize>0){return true;}return false;
}bool pyfree::readFromFile(char *ptr, int &size, int buf_size,const std::string& path, const std::string& mode)
{FILE *m_fil=NULL;m_fil = ::fopen(path.c_str(),mode.c_str());if(NULL!=m_fil){size = static_cast<int>( ::fread(ptr, static_cast<int>(sizeof(char)), (size_t)buf_size,m_fil) );::fclose(m_fil);m_fil = NULL;if(size>0){return true;}elsereturn false;}else{m_fil = NULL;return false;}
}//读取文件每一行得字符串,如果行中存在空格,划分该行字符串,并装在到容器中
bool pyfree::readListInfo(std::vector<std::vector<std::string> > &list,const std::string path,int codef)
{FILE *m_fil=NULL;m_fil = ::fopen(path.c_str(),"r");if(NULL!=m_fil){char buf[1024] ={0};while(NULL!=::fgets(buf,1024,m_fil)){std::string _lineInfo = buf;trimEnd(_lineInfo);#ifdef __linux__switch (codef){case 1:{char buff[1024]={0};g2u(buf,strlen(buf),buff,1024);_lineInfo = buff;}break;case 2:{char buff[1024]={0};a2u(buf,strlen(buf),buff,1024);_lineInfo = buff;}break;default:break;}#endifstd::string buffer;// Have a buffer stringstd::stringstream _lineStr(_lineInfo);// Insert the string into a streamstd::vector<std::string> _lineList;// Create vector to hold our wordswhile(_lineStr >> buffer){_lineList.push_back(buffer);}list.push_back(_lineList);}::fclose(m_fil);m_fil = NULL;if(list.empty()){return false;}}else{#ifdef WIN32Print_WARN("con't open %s, %d \n",path.c_str(), ::GetLastError());#endif#ifdef linuxPrint_WARN("con't open %s, %s \n",path.c_str(),strerror(errno));#endifm_fil = NULL;return false;}return true;
}//将采用","间隔的字符串划分成子字符串集
void pyfree::getNumberFromStr(std::vector<std::string> &ids, std::string idsStr)
{if(""==idsStr||"NULL"==idsStr)return;std::string id = "";for(unsigned int i=0,idSize= static_cast<unsigned int>(idsStr.size()); i<idSize; i++){if(','==idsStr[i]){//use str ',' to dirve the idsStr;if(""!=id){ids.push_back(id);id.clear();id = "";}else{ids.push_back(id);}}else if(i==(idSize-1)){if(""!=id){ids.push_back(id);id.clear();id = "";}}else{id.push_back(idsStr[i]);}}
}//读取文件每一行得字符串,如果行中存在",",划分该行字符串,并装在到容器中
bool pyfree::readListInfo_dot(std::vector<std::vector<std::string> > &list,const std::string path,int codef)
{FILE *m_fil=NULL;m_fil = ::fopen(path.c_str(),"r");if(NULL!=m_fil){char buf[1024] ={0};std::string _lineInfo = "";while(NULL!=::fgets(buf,1024,m_fil)){_lineInfo = buf;trim(_lineInfo);#ifdef __linux__switch (codef){case 1:{char buff[1024]={0};g2u(buf,strlen(buf),buff,1024);_lineInfo = buff;}break;case 2:{char buff[1024]={0};a2u(buf,strlen(buf),buff,1024);_lineInfo = buff;}break;default:break;}#endifstd::vector<std::string> _lineList;getNumberFromStr(_lineList,_lineInfo);list.push_back(_lineList);}::fclose(m_fil);m_fil = NULL;if(list.empty()){return false;}}else{#ifdef WIN32Print_WARN("con't open %s, %d \n",path.c_str(), ::GetLastError());#endif#ifdef linuxPrint_WARN("con't open %s, %s \n",path.c_str(),strerror(errno));#endifm_fil = NULL;return false;}return true;
}//读取文件,并将每一行字符串载入容器
bool pyfree::readListInfo(std::vector<std::string> &list,const std::string path,int codef)
{FILE *m_fil=NULL;m_fil = ::fopen(path.c_str(),"r");if(NULL!=m_fil){char buf[512] ={0};std::string _lineInfo="";while(NULL!=::fgets(buf,512,m_fil)){_lineInfo = buf;trimEnd(_lineInfo);#ifdef __linux__switch (codef){case 1:{char buff[1024]={0};g2u(buf,strlen(buf),buff,1024);_lineInfo = buff;}break;case 2:{char buff[1024]={0};a2u(buf,strlen(buf),buff,1024);_lineInfo = buff;}break;default:break;}#endiflist.push_back(_lineInfo);}::fclose(m_fil);m_fil = NULL;if(list.empty()){return false;}}else{#ifdef WIN32Print_WARN("con't open %s, %d \n",path.c_str(), ::GetLastError());#endif#ifdef linuxPrint_WARN("con't open %s, %s \n",path.c_str(),strerror(errno));#endifm_fil = NULL;return false;}return true;
}//获得dire目录下扩展名为extName的所有文件名
void pyfree::getAllFileName_dot(const std::string& directory, const std::string& extName, std::vector<std::string>& fileNames)
{getAllFileNamePrivate(directory,extName,fileNames,true);
}//获得dire目录下扩展名为extName的所有文件名
void pyfree::getAllFileName(const std::string& directory, const std::string& extName, std::vector<std::string>& fileNames)
{getAllFileNamePrivate(directory,extName,fileNames);
}void pyfree::getAllFileNamePrivate(const std::string& directory, const std::string& extName, std::vector<std::string>& fileNames,bool f)
{#ifdef WIN32_finddata_t fileInfo;intptr_t hFile;std::string filter = directory;char bch = static_cast<char>(('//') & 0X00FF);if(filter[filter.size()-1] != bch){filter.push_back(bch);}filter += "*.";filter += extName;hFile = _findfirst(filter.c_str(),&fileInfo);if(hFile == -1){return;}do {if (extName.empty()){if ((fileInfo.attrib & _A_SUBDIR)){if (0==strcmp(fileInfo.name,".") || 0==strcmp(fileInfo.name,"..") ){continue;}fileNames.push_back(fileInfo.name);}}else{std::string name(fileInfo.name);if(f){fileNames.push_back(name);}else{fileNames.push_back(name.substr(0,name.find_last_of('.')));}// fileNames.push_back(name.substr(0,name.find_last_of('.')));}} while (_findnext(hFile,&fileInfo) == 0);_findclose(hFile);#endif#ifdef linuxstd::string curdir = directory;if(curdir[curdir.size()-1] != '/'){curdir.push_back('/');}DIR *dfd;if( (dfd = opendir(curdir.c_str())) == NULL ){Print_WARN("open %s error with msg is: %s\n",curdir.c_str(),strerror(errno));return;}struct dirent *dp;while ((dp = readdir(dfd)) != NULL){if (extName.empty()){if (dp->d_type == DT_DIR){if ( 0==strcmp(dp->d_name, ".") || 0==strcmp(dp->d_name, "..") ){continue;}fileNames.push_back(dp->d_name);}}else{if ( NULL==strstr(dp->d_name, extName.c_str()) ){continue;}std::string name(dp->d_name);if(f){fileNames.push_back(name);}else{fileNames.push_back(name.substr(0,name.find_last_of('.')));}}}if (NULL != dp) {delete dp;dp = NULL;}if (NULL != dfd) {closedir(dfd);dfd = NULL;}#endif
}unsigned int pyfree::getFileModifyTime(const std::string file_path)
{unsigned int ret = 0;
#if defined(WIN32)_finddata_t fileInfo;intptr_t hFile;hFile = _findfirst(file_path.c_str(),&fileInfo);if(hFile != -1){ret = static_cast<unsigned int>(fileInfo.time_write);}else{Print_WARN("get file %s last modify error: %d \n",file_path.c_str(), ::GetLastError());}_findclose(hFile);
#elif defined(linux)struct stat el;if(0==stat(file_path.c_str(), &el)){ret = static_cast<unsigned int>(el.st_mtime);}else{Print_WARN("get file %s last modify time error: %s\n",file_path.c_str(),strerror(errno));}
#elseret = 0;
#endif // WIN32return ret;
}void pyfree::renameFile(const std::string& _oldName, const std::string& _newName)
{#ifdef WIN32_finddata_t fileInfo;intptr_t hFile;hFile = _findfirst(_oldName.c_str(),&fileInfo);if(hFile == -1){Print_WARN("can't find file %s \n",_oldName.c_str());writeToFile("",_newName,"w");return;}// do {// std::string name(fileInfo.name);// fileNames.push_back(name.substr(0,name.find_last_of('.')));// } while (_findnext(hFile,&fileInfo) == 0);_findclose(hFile);#endif#ifdef linuxFILE* m_fil = ::fopen(_oldName.c_str(),"r");if(NULL==m_fil){return;}::fclose(m_fil);m_fil = NULL;#endifPrint_NOTICE("%s, %s\n", _oldName.c_str(),_newName.c_str());int ret = rename(_oldName.c_str(),_newName.c_str());if(0 != ret){Print_WARN("can't rename %s to %s \n",_oldName.c_str(),_newName.c_str());return;}
}void pyfree::removeFile(const std::string& _Name)
{int ret = remove(_Name.c_str());if(0 != ret){Print_WARN("can't remove %s \n",_Name.c_str());return;}
}bool pyfree::isExist(std::string file)
{#ifdef WIN32_finddata_t fileInfo;intptr_t hFile = _findfirst(file.c_str(),&fileInfo);if(hFile == -1){return false;}_findclose(hFile);return true;#elseFILE *m_fil=NULL;m_fil=::fopen(file.c_str(),"rb");if(NULL==m_fil){return false;}else{::fclose(m_fil);m_fil = NULL;}return true;#endif
}bool pyfree::isExist(std::string dir,std::string file)
{std::string path = dir;
#ifdef WIN32path+="\\";
#elsepath+="/";
#endifpath+=file;return isExist(path);
};void pyfree::divXmlFile(std::string _xmlFile,std::string _xmlDir,std::string _xmlDiv,bool utfFlag)
{static int substationIndex = 1;FILE *m_fil=NULL;FILE *new_m_fil=NULL;std::string _xmlNode="<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n";_xmlNode = utfFlag?ASCII2UTF_8(_xmlNode):_xmlNode;std::string _xmlHead = "<"+_xmlDiv;std::string _xmlEnd = "</"+_xmlDiv+">";if(NULL!=(m_fil=fopen(_xmlFile.c_str(),"rb"))){char buf[512] ={0};while(NULL!=::fgets(buf,512,m_fil)){std::string _lineInfo = buf;_lineInfo = utfFlag?ASCII2UTF_8(_lineInfo):_lineInfo;std::string::size_type _posH = _lineInfo.find(_xmlHead);if(std::string::npos != _posH){char newxml[256]={0};sprintf(newxml,"%s\\%s_%d.xml",_xmlDir.c_str(),_xmlDiv.c_str(),substationIndex);if(NULL!=(new_m_fil=::fopen(newxml,"a+"))){int newSize = static_cast<int>(::fwrite(_xmlNode.c_str(), static_cast<int>(sizeof(char)), strlen(_xmlNode.c_str()),new_m_fil));if(newSize<=0){Print_WARN("newxml=%s,newSize=%d write failed\n",newxml,newSize);}}else{Print_WARN("open %s error\n",newxml);}}if(NULL!=new_m_fil){int lineSize = static_cast<int>(::fwrite(_lineInfo.c_str(), static_cast<int>(sizeof(char)), strlen(_lineInfo.c_str()),new_m_fil));if(lineSize<=0){Print_WARN("write _lineInfo=%s, lineSize=%d error\n",_lineInfo.c_str(),lineSize);}}std::string::size_type _posE = _lineInfo.find(_xmlEnd);if(std::string::npos != _posE){if(NULL!=new_m_fil){::fclose(new_m_fil);new_m_fil=NULL;substationIndex+=1;}}}::fclose(m_fil);m_fil=NULL;}else{Print_WARN("con't open %s \n",_xmlFile.c_str());m_fil = NULL;}
};bool pyfree::createDir(std::string _dir)
{if(isExistDir(_dir))return true;
#ifdef WIN32int ret = _mkdir(_dir.c_str());
#elseint ret = mkdir(_dir.c_str(), S_IRWXU | S_IRWXG | S_IRWXO);
#endifif (-1 == ret) {
#ifdef WIN32if (183 != GetLastError()){Print_WARN("create Directory %s is error,%d \r\n", _dir.c_str(), GetLastError());return false;}
#elseif (EEXIST != errno) {Print_WARN("create Directory %s is error, %s ! \r\n", _dir.c_str(), strerror(errno));return false;}else{Print_WARN("dir(%s) has exist,don't create it again!\n",_dir.c_str());}
#endif}return true;
};bool pyfree::isExistDir(std::string _dir)
{
#ifdef WIN32if(!_access(_dir.c_str(),0)){return true;}else{// Print_WARN("%s is not exist!\n",_dir.c_str());return false;}
#elseDIR *dfd;if( (dfd = opendir(_dir.c_str())) == NULL ){return false;}else{return true;}
#endif // WIN32
}bool pyfree::getCurrentDir(char* dir_, int len_)
{char* ret=NULL;ret = getcwd(dir_, len_);return (NULL!=ret);
}
strchange.cpp
#include "strchange.h"#include <stdio.h>
#include <stdlib.h>
#ifdef __linux__
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <iconv.h>
#endif
#ifdef WIN32
#include <iostream>
#include <windows.h>
#endif
#include <vector>#ifdef WIN32
//UTF-8 to Unicode
std::wstring pyfree::Utf82Unicode(const std::string& utf8string)
{int widesize = ::MultiByteToWideChar(CP_UTF8, 0, utf8string.c_str(), -1, NULL, 0);if (widesize == ERROR_NO_UNICODE_TRANSLATION){throw std::exception("Invalid UTF-8 sequence.");}if (widesize == 0){throw std::exception("Error in conversion.");}std::vector<wchar_t> resultstring(widesize);int convresult = ::MultiByteToWideChar(CP_UTF8, 0, utf8string.c_str(), -1, &resultstring[0], widesize);if (convresult != widesize){throw std::exception("La falla!");}return std::wstring(&resultstring[0]);
};//unicode to asciistd::string pyfree::WideByte2Acsi(std::wstring& wstrcode)
{int asciisize = ::WideCharToMultiByte(CP_OEMCP, 0, wstrcode.c_str(), -1, NULL, 0, NULL, NULL);if (asciisize == ERROR_NO_UNICODE_TRANSLATION){throw std::exception("Invalid UTF-8 sequence.");}if (asciisize == 0){throw std::exception("Error in conversion.");}std::vector<char> resultstring(asciisize);int convresult =::WideCharToMultiByte(CP_OEMCP, 0, wstrcode.c_str(), -1, &resultstring[0], asciisize, NULL, NULL);if (convresult != asciisize){throw std::exception("La falla!");}return std::string(&resultstring[0]);
};/////ascii to Unicodestd::wstring pyfree::Acsi2WideByte(std::string& strascii)
{int widesize = MultiByteToWideChar (CP_ACP, 0, (char*)strascii.c_str(), -1, NULL, 0);if (widesize == ERROR_NO_UNICODE_TRANSLATION){throw std::exception("Invalid UTF-8 sequence.");}if (widesize == 0){throw std::exception("Error in conversion.");}std::vector<wchar_t> resultstring(widesize);int convresult = MultiByteToWideChar (CP_ACP, 0, (char*)strascii.c_str(), -1, &resultstring[0], widesize);if (convresult != widesize){throw std::exception("La falla!");}return std::wstring(&resultstring[0]);
};//Unicode to Utf8std::string pyfree::Unicode2Utf8(const std::wstring& widestring)
{int utf8size = ::WideCharToMultiByte(CP_UTF8, 0, widestring.c_str(), -1, NULL, 0, NULL, NULL);if (utf8size == 0){throw std::exception("Error in conversion.");}std::vector<char> resultstring(utf8size);int convresult = ::WideCharToMultiByte(CP_UTF8, 0, widestring.c_str(), -1, &resultstring[0], utf8size, NULL, NULL);if (convresult != utf8size){throw std::exception("La falla!");}return std::string(&resultstring[0]);
};std::wstring pyfree::stringToWstring(const std::string& str)
{LPCSTR pszSrc = str.c_str();int nLen = MultiByteToWideChar(CP_ACP, 0, pszSrc, -1, NULL, 0);if (nLen == 0) return std::wstring(L"");wchar_t* pwszDst = new wchar_t[nLen];if (!pwszDst) return std::wstring(L"");MultiByteToWideChar(CP_ACP, 0, pszSrc, -1, pwszDst, nLen);std::wstring wstr(pwszDst);delete[] pwszDst;pwszDst = NULL;return wstr;
}std::string pyfree::wstringToString(const std::wstring& wstr)
{LPCWSTR pwszSrc = wstr.c_str();int nLen = WideCharToMultiByte(CP_ACP, 0, pwszSrc, -1, NULL, 0, NULL, NULL);if (nLen == 0) return std::string("");char* pszDst = new char[nLen];if (!pszDst) return std::string("");WideCharToMultiByte(CP_ACP, 0, pwszSrc, -1, pszDst, nLen, NULL, NULL);std::string str(pszDst);delete[] pszDst;pszDst = NULL;return str;
}#endif#ifdef __linux__
//
int pyfree::code_convert(char *from_charset,char *to_charset,char *inbuf,size_t inlen,char *outbuf,size_t outlen)
{iconv_t cd;//int rc;char **pin = &inbuf;char **pout = &outbuf;cd = iconv_open(to_charset,from_charset);if (cd==0) return -1;memset(outbuf,0,outlen);if (-1==static_cast<int>(iconv(cd,pin,&inlen,pout,&outlen)))return -1;iconv_close(cd);return 0;
}
//
int pyfree::u2g(char *inbuf,int inlen,char *outbuf,int outlen)
{char _fbuf[32]="utf-8";char _tbuf[32]="gb2312";return code_convert(_fbuf,_tbuf,inbuf,inlen,outbuf,outlen);
}
//
int pyfree::g2u(char *inbuf,size_t inlen,char *outbuf,size_t outlen)
{char _fbuf[32]="gb2312";char _tbuf[32]="utf-8";return code_convert(_fbuf,_tbuf,inbuf,inlen,outbuf,outlen);
}int pyfree::u2a(char *inbuf,int inlen,char *outbuf,int outlen)
{char _fbuf[32]="utf-8";char _tbuf[32]="ascii";return code_convert(_fbuf,_tbuf,inbuf,inlen,outbuf,outlen);
}int pyfree::a2u(char *inbuf,int inlen,char *outbuf,int outlen)
{char _fbuf[32]="ascii";char _tbuf[32]="utf-8";return code_convert(_fbuf,_tbuf,inbuf,inlen,outbuf,outlen);
}int pyfree::u2k(char *inbuf,int inlen,char *outbuf,int outlen)
{char _fbuf[32]="utf-8";char _tbuf[32]="gbk";return code_convert(_fbuf,_tbuf,inbuf,inlen,outbuf,outlen);
}int pyfree::k2u(char *inbuf,int inlen,char *outbuf,int outlen)
{char _fbuf[32]="gbk";char _tbuf[32]="utf-8";return code_convert(_fbuf,_tbuf,inbuf,inlen,outbuf,outlen);
}#endif//utf-8 to asciistd::string pyfree::UTF_82ASCII(std::string& strUtf8Code)
{#ifdef WIN32std::string strRet("");//std::wstring wstr = Utf82Unicode(strUtf8Code);//strRet = WideByte2Acsi(wstr);return strRet;#endif#ifdef __linux__char lpszBuf[1024]={0};u2k(const_cast<char*>(strUtf8Code.c_str()),strUtf8Code.size(),lpszBuf, 1024);return std::string(lpszBuf);#endif
};//ascii to Utf8std::string pyfree::ASCII2UTF_8(std::string& strAsciiCode)
{#ifdef WIN32std::string strRet("");//std::wstring wstr = Acsi2WideByte(strAsciiCode);//strRet = Unicode2Utf8(wstr);return strRet;#endif#ifdef __linux__char lpszBuf[1024]={0};k2u(const_cast<char*>(strAsciiCode.c_str()),strAsciiCode.size(),lpszBuf, 1024);return std::string(lpszBuf);#endif
};
相关文章:
c/c++开发,无可避免的文件访问开发案例
一、缓存文件系统 ANSI C标准中的C语言库提供了fopen, fclose, fread, fwrite, fgetc, fgets, fputc, fputs, freopen, fseek, ftell, rewind等标准函数,这些函数在不同的操作系统中应该调用不同的内核API,从而支持开发者跨平台实现对文件的访问。 在Lin…...
MySQL学习笔记
MySQL学习笔记一、基础配置二、数据库操作三、表的操作1.创建表2.表选项3.查看表4.修改表5.删除表6.复制表7.检查优化修复表四、数据操作基础增删改查五、字符集编码六、数据类型(列类型)1.数值类型2.字符串类型3.日期时间类型4.枚举和集合七、列属性&am…...
ccs导入工程失败的处理方法
文章目录当导入CCS新工程时出现下述错误怎么办?方法一 从TI官网下载安装包进行安装,下载链接:软件下载完成 安装路径为上面的文件夹点击安装完成后,导入安装路径,并点击Refresh按钮,依据路径进行更新&#…...
探针台常见的故障及解决方法
症状、 可能原因、 解决方法 移动样品后画面变模糊 —显微镜不垂直,调垂直显微镜 样品台不水平 —调水平样品台 显微镜视场亮度不足,边缘切割或看不到像—转换器不在定位位置上 把转换器转到定位位置上 管镜转盘不在定位位置上 —把管镜转盘转到定…...
域内资源探测
✅作者简介:CSDN内容合伙人、信息安全专业在校大学生🏆 🔥系列专栏 :内网安全 📃新人博主 :欢迎点赞收藏关注,会回访! 💬舞台再大,你不上台,永远是…...
c# 将数据导出到EXCEL文件
第一步:项目中加入引用。 在鼠标右击项目,点击【添加】弹出菜单列表,选择【项目引用】弹出【引用管理器】对话框,选择【COM】-【Microsoft Excel 16.0 Object Library】,如图所示: 第二步,编辑…...
微服务 分片 运维管理
微服务 分片 运维管理分片分片的概念分片案例环境搭建案例改造成任务分片Dataflow类型调度代码示例运维管理事件追踪运维平台搭建步骤使用步骤分片 分片的概念 当只有一台机器的情况下,给定时任务分片四个,在机器A启动四个线程,分别处理四个…...
批量占满TEMP表空间问题处理与排查
批量占满TEMP表空间问题处理与排查应急处置问题排查查看占用TEMP表空间高的SQL获取目标SQL执行计划方法一:EXPLAIN PLAN FOR方法二:DBMS_XPLAN.DISPLAY_CURSOR方法三:DBMS_XPLAN.DISPLAY_AWR方法四:AUTOTRACE数据库跑批任务占满TE…...
Pytorch中的tensor和variable
Tensor与Variable pytorch两个基本对象:Tensor(张量)和Variable(变量) 其中,tensor不能反向传播,variable可以反向传播(forword)。 反向传播是为了让神经网络更新前面…...
暗月内网渗透实战——项目七
首先环境配置 VMware的网络配置图 环境拓扑图 开始渗透 信息收集 使用kali扫描一下靶机的IP地址 靶机IP:192.168.0.114 攻击机IP:192.168.0.109 获取到了ip地址之后,我们扫描一下靶机开放的端口 靶机开放了21,80,999,3389,5985,6588端口…...
【Java 面试合集】描述下Objec类中常用的方法(未完待续中...)
描述下Objec类中常用的方法 1. 概述 首先我们要知道Object 类是所有的对象的基类,也就是所有的方法都是可以被重写的。 那么到底哪些方法是我们常用的方法呢??? cloneequalsfinalizegetClasshashCodenotifynotifyAlltoStringw…...
SQLSERVER 的 truncate 和 delete 有区别吗?
一:背景 1. 讲故事 在面试中我相信有很多朋友会被问到 truncate 和 delete 有什么区别 ,这是一个很有意思的话题,本篇我就试着来回答一下,如果下次大家遇到这类问题,我的答案应该可以帮你成功度过吧。 二࿱…...
【C++】CC++内存管理
就是你被爱情困住了?Wake up bro! 文章目录一、C/C内存分布二、C语言中动态内存管理方式三、C中内存管理方式1.new和delete操作内置类型2.new和delete操作自定义类型(仅限vs的底层实现机制,new和delete一定要匹配使用,…...
数据预处理之图像去空白
数据预处理之图像去空白图像去空白介绍方法边缘检测阈值处理形态学图像剪切图像去空白 介绍 图像去空白是指在图像处理中去除图像中的空白区域的过程。空白区域通常是指图像中的白色或其他颜色,其不包含有用的信息。去空白的目的是为了节省存储空间、提高图像处理…...
真的麻了,别再为难软件测试员了......
前言 有不少技术友在测试群里讨论,近期的面试越来越难了,要背的八股文越来越多了,考察得越来越细,越来越底层,明摆着就是想让我们徒手造航母嘛!实在是太为难我们这些测试工程师了。 这不,为了帮大家节约时…...
2月9日,30秒知全网,精选7个热点
///货拉拉将推出同城门到门跑腿服务 据介绍,两轮电动车将成为该业务的主要运力,预计将于3月中旬全面开放骑手注册和用户人气征集活动,并根据人气和线上骑手注册情况选择落地城市,于4月正式开放服务和骑手接单 ///三菱、乐天和莱茵…...
球面坐标系下的三重积分
涉及知识点 三重积分球面坐标系点火公式一些常见积分处理手法 球面坐标系定义 球面坐标系由方位角φ\varphiφ、仰角θ\thetaθ和距离rrr构成 直角坐标系(x,y,z)(x,y,z)(x,y,z)到球面坐标系的(r,φ,θ)(r,\varphi,\theta)(r,φ,θ)的转化规则如下: {xrsinφco…...
谷歌 Jason Wei | AI 研究的 4 项基本技能
文章目录 一、前言二、主要内容三、总结CSDN 叶庭云:https://yetingyun.blog.csdn.net/ 一、前言 原文作者为 Jason Wei,2020 年达特茅斯学院本科毕业,之后加入 Google Brain 工作。 Jason Wei 的博客主页:https://www.jasonwei.net/ 其实我不算是一个特别有经验的研究员…...
excel数据整理:合并计算快速查看人员变动
相信大家平时在整理数据时,都会对比数据是否有重复的地方,或者该数据与源数据相比是否有增加或者减少。数据量不大还好,数据量大的话,对比就比较费劲了。接下来我们将进入数据对比系列课程的学习。该系列一共有两篇教程࿰…...
vit-pytorch实现 MobileViT注意力可视化
项目链接 https://github.com/lucidrains/vit-pytorch 注意一下参数设置: Parameters image_size: int. Image size. If you have rectangular images, make sure your image size is the maximum of the width and heightpatch_size: int. Number of patches. im…...
Python将字典转换为csv
大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。喜欢通过博客创作的方式对所学的知识进行总结与归纳,不仅形成深入且独到的理…...
EasyX精准帧率控制打气球小游戏
🎆音乐分享 New Boy —— 房东的猫 之前都用Sleep()来控制画面帧率,忽略了绘制画面的时间 如果绘制画面需要很长的时间,那么就不能忽略了。 并且Sleep()函数也不是特别准确,那么就…...
你知道 GO 中什么情况会变量逃逸吗?
你知道 GO 中什么情况会变量逃逸吗?首先我们先来看看什么是变量逃逸 Go 语言将这个以前我们写 C/C 时候需要做的内存规划和分配,全部整合到了 GO 的编译器中,GO 中将这个称为 变量逃逸 GO 通过编译器分析代码的特征和代码的生命周期&#x…...
一篇文章学懂C++和指针与链表
指针 目录 指针 C的指针学习 指针的基本概念 指针变量的定义和使用 指针的所占的内存空间 空指针和野指针 const修饰指针 指针和数组 指针和函数 指针、数组、函数 接下来让我们开始进入学习吧! C的指针学习 指针的基本概念 指针的作用:可…...
TPGS-cisplatin顺铂修饰维生素E聚乙二醇1000琥珀酸酯
TPGS-cisplatin顺铂修饰维生素E聚乙二醇1000琥珀酸酯(TPGS)溶于大部分有机溶剂,和水有很好的溶解性。 长期保存需要在-20℃,避光,干燥条件下存放,注意取用一定要干燥,避免频繁的溶解和冻干。 维生素E聚乙二醇琥珀酸酯(简称TPGS)是维生素E的水溶性衍生物,由维生素E…...
【20230206-0209】哈希表小结
哈希表一般哈希表都是用来快速判断一个元素是否出现在集合里。哈希函数哈希碰撞--解决方法:拉链法和线性探测法。拉链法:冲突的元素都被存储在链表中线性探测法:一定要保证tableSize大于dataSize,利用哈希表中的空位解决碰撞问题。…...
c++11 标准模板(STL)(std::multimap)(一)
定义于头文件 <map> template< class Key, class T, class Compare std::less<Key>, class Allocator std::allocator<std::pair<const Key, T> > > class multimap;(1)namespace pmr { template <class Key, class T…...
python进阶——自动驾驶寻找车道
大家好,我是csdn的博主:lqj_本人 这是我的个人博客主页: lqj_本人的博客_CSDN博客-微信小程序,前端,python领域博主lqj_本人擅长微信小程序,前端,python,等方面的知识https://blog.csdn.net/lbcyllqj?spm1011.2415.3001.5343哔哩哔哩欢迎关注…...
男,26岁,做了一年多的自动化测试,最近在纠结要不要转行,求指点。?
最近一个粉丝在后台问我,啊大佬我现在26了,做了做了一年多的自动化测试,最近在纠结要不要转行,求指点。首选做IT这条路,就是很普通的技术蓝领。对于大部分来说干一辈子问题不大,但是发不了什么财。如果你在…...
源码级别的讲解JAVA 中的CAS
没有CAS之前实现线程安全 多线程环境不使用原子类保证线程安全(基本数据类型) public class T3 {volatile int number 0;//读取public int getNumber(){return number;}//写入加锁保证原子性public synchronized void setNumber(){number;} }多线程环…...
域名与网站的区别/如何设计与制作网页
以下是设置按钮的右边框和底边框颜色为红色,边框大小为3dp,如下图: 在drawable新建一个 buttonstyle.xml的文件,内容如下: <?xml version"1.0" encoding"UTF-8"?> <layer-list xmlns:…...
做中介平台网站 需要什么/百度上的广告多少钱一个月
使用场景 提针对零售类公司企业、费用报销和资产招采、经营物资招采、合同招标等共用系统的场景。将业务拆解为申请、采购、入库 3大环节,为跨部门合规管理和 IT 运营提供了便利性。 典型用途:IT 人员监控全流程,各个业务部门按需操作自己的…...
太原做网站baidu/免费外链代发
CSS里实现水平居中非常容易,inline元素用text-align:center;,block元素用margin:auto;就行了。但要实现垂直居中确是一大难题。本篇收集了一些已知的方案,整理出来,以备将来取用。代码可以从GitHub上下载。 Flex弹性盒子absolute绝…...
简洁中文网站模板/做网站的软件有哪些
深度神经网络可视化工具1. visdom1.1 通用操作1.1.1 创建/关闭窗口、查询窗口状态1.1.2 更新窗口 update_window_opts1.1.3 不同的update模式1.2 viz.image/images1.2.1 在窗口显示一张图片1.2.2 多张图片在同一窗口(可查看历史图片)1.2.3 多张图片阵列显…...
如何查看网站开发公司/网站推广公司大家好
作为Java中面试常见的锁问题,也是必须了解的。synchronized与ReentrantLock在jdk1.5之前对共享数据访问的保护方式只有两种:synchronized和volatile。在jdk1.5之后增加了新的机制:ReentrantLock。在1.5的时候ReentrantLock的性能要比内置锁sy…...
兰州网站开发哪里可以做/海口seo网络公司
Docker简介Docker 简介 从 2008 年开始进入公众视野,到亚马逊与微软于近期获得 1 亿美元的美国联邦管理局云计算合同,云计算整整走过了 7 年。7 年间,以第一代云计算技术为基础的移动计算、社交网络、大数据等新信息技术在全球商业领域掀起了…...