【开源程序(C++)】获取bing图片并自动设置为电…
2018-06-17 23:07:11来源:未知 阅读 ()
众所周知,bing搜索网站首页每日会更新一张图片,张张漂亮(额,也有一些不合我口味的),特别适合用来做电脑壁纸。
我们想要将bing网站背景图片设置为电脑桌面背景的通常做法是:
- 上网,搜索bing
- 找到图片的下载链接,并下载之
- 将图片设置为桌面背景(也有部分浏览器支持直接在网页中右击图片设置为壁纸)
- 可能你还会删除下载下来的图片
作为一枚coder,你不会觉得麻烦嘛?至少需要三个步骤呢!
So...我就用C++语言写了一个小程序,用于获取bing网站的图片,并将其设置为桌面背景。整个过程只需要点击.exe文件,一键完成!
当然,你也可以通过本程序,窥见Windows API的些许用法,活学活用C++知识!
当然网上也存在片片断断的程序,我的部分程序也借鉴了它们的思想,在此感谢互联网和辛勤的贡献者们。
以下是程序设计的技术要点:
--------------------------------------------------------------
技术要点:
1、获取网络地址 直接使用网络地址或下载 注意若下载下来后,要将\转换为/,当然也可以用\\
网络地址可以从这里获取:http://cn.bing.com/HPImageArchive.aspx?format=xml&idx=0&n=1
在返回的xml页面中(images->image->url)找到具体的图片地址(xml解析),拼接到bing域名后面构成完整地址
注:xml解析用了TinyXml2
2、转换图片格式(jpg->bmp),本程序中的SystemParametersInfoA函数只支持bmp
在程序中自动转换(单单改后缀名是没有用的),转换用的程序是从网上下载的,用C语言编写而成
考虑到需要改后缀名,那就直接下载图片好了,顺便存储之
3、图片保存路径为C:/Users/Administrator/bingPicture/,格式为.jpg 方便以后浏览
注意:部分用户电脑可能不存在路径C:/Users/Administrator/,造成程序无法执行,可以直接在C盘根目录下创建路径,如C:/bingPicture/
注意不保存转换后的bmp格式图片(设置背景后即删除),因为体积较大
判断文件夹是否存在,若不存在,则自动创建文件夹
4、注意本程序获取的图片尺寸是1366x768,若你的屏幕分辨率为1920x1080,还需要对Xml解析出来的图片Url地址进行字符串替换(将1366x768换成1920x1080即可)
前提:针对1920x1080尺寸的图片地址存在
5、若此程序若在国际网络下运行,获取的就是国际版bing壁纸;若在国内网络下运行,获取的就是中国版bing壁纸。因此在同一天内,在不同网络环境下获取的图片可能不同
*未实现的功能*:
1、获取每日壁纸的故事(利用bing故事接口) ,更新壁纸后显示在执行框中
2、开机自启动,并隐藏到托盘中(为减少CPU占用并增加趣味性,设置为开机自动启动,提示网络连接,并输入"go"才执行功能)
电脑若未关机,则在24:00自动启动,更换背景
3、软件自动更新版本功能
--------------------------------------------------------------
程序在文章后面提供,源码已注释很详细,不再赘述。
注意,使用程序之前,务必看以下注意事项:
--------------------------------------------------------------
请注意:
1、本软件使用Qt开发,您也可以将文件加入自己的工程,使用其他IDE开发
需要注意的一点是,本软件需要加入URLMON.DLL(源码包中有)
2、本软件开源(源码位于xiaoxi666的博客园以及github,不对其他地址给出的链接负责),仅用于学习交流,请勿用于商业用途
3、为防止软件被加入恶意功能,不提供可执行文件,若需使用请重新编译,编译器需要支持c++11
4、本软件程序中内含删除临时文件功能,请在更改前仔细确认,避免路径错误而删除其他重要文件
--------------------------------------------------------------
然而许多小伙伴要体验效果,我就一并把可执行文件放出来吧(请勿随意传播.exe文件防止有人添加恶意功能。当然源码开放可共享)
下载区:
- 源码
- 可执行文件(可执行文件中图片保存地址为C:/bingPicture/)
但一定要注意核对文件校验码(以保证安全):
- 可执行文件压缩包WallPaper校验码
MD5: 48173BA7DCF2120F2822226A5D4A90CF
SHA1: CEB0ED570AF613EC3829AC8FDE4F8C50DDEF4101
- 可执行文件WallPaper_1366x768_Common校验码(此版本用于1366x768分辨率)
MD5: ACDA8E5E4CF0B2916254B233D1243FD3
SHA1: 6041C813DC8E2AE29EA8675EF279CFC0E7921D53
- 可执行文件WallPaper_1920x1080_Common校验码(此版本用于1920x1080分辨率)
MD5: 474570808A56EFDC7B589F605D08C5B6
SHA1: B78921AD655B35F079076904A22F1A0E5122EB7F
为方便浏览,贴出主要源程序:
main.cpp文件
1 //main.cpp 2 /******************windows桌面背景更换C++程序*********************************************************** 3 功能:获取每日bing搜索主页图片,设置为当日桌面壁纸。并将其下载保存至本地文件夹方便以后浏览 4 作者:xiaoxi666 5 日期:2017/03/12 6 7 技术要点: 8 1、获取网络地址 直接使用网络地址或下载 注意若下载下来后,要将\转换为/,当然也可以用\\ 9 网络地址可以从这里获取:http://cn.bing.com/HPImageArchive.aspx?format=xml&idx=0&n=1 10 在返回 的xml页面中(images->image->url)找到具体的图片地址(xml解析),拼接到bing域名后面,构成完整地址 11 注:xml解析用了TinyXml2 12 13 2、转换图片格式(jpg->bmp),本程序中的SystemParametersInfoA函数只支持bmp 14 在程序中自动转换(单单改后缀名是没有用的),转换用的程序是从网上下载的,用C语言编写而成 15 考虑到需要改后缀名,那就直接下载图片好了,顺便存储之 16 17 3、图片保存路径为C:\bingPicture\,格式为.jpg 方便以后浏览 18 注意不保存转换后的bmp格式图片(设置背景后即删除),因为体积较大 19 判断文件夹是否存在,若不存在,则自动创建文件夹 20 21 *未实现的功能*: 22 ***获取每日壁纸的故事(利用bing故事接口) ,更新壁纸后显示在执行框中 23 24 ***开机自启动,并隐藏到托盘中(为减少CPU占用并增加趣味性,设置为开机自动启动,提示网络连接,并输入"go"才执行功能) 25 电脑若未关机,则在24:00自动启动,更换背景 26 27 ***软件自动更新版本功能 28 29 ******************************************************************************************************/ 30 31 #include <iostream> //输入输出 32 #include <cstring> //文件命名处理需要用字符串 33 #include <windows.h> //调用操作系统各种API 34 #include <ctime> //获取时间,各种文件命名 35 #include <UrlMon.h> //包含提供下载服务的API 36 #include "tinyxml2.h" //解析XML 37 #include <io.h> //判断文件夹是否存在 38 #include <direct.h> //创建文件夹 39 extern "C" 40 { 41 #include "jpeg.h" //转换图片格式jpg->bmp 转换格式的程序使用C语言写的 42 } 43 44 //创建本地bingPicture路径和Tmp路径 45 void createDir() 46 { 47 //本地bingPicture路径 48 std::string LocalFolder="C:/bingPicture/"; 49 50 if(0!=access(LocalFolder.c_str(),0)) //判断文件夹是否存在,若不存在则创建 51 if(0!=mkdir(LocalFolder.c_str())) 52 std::cout<<"创建文件夹bingPicture失败!"<<std::endl; 53 else 54 std::cout<<"创建文件夹bingPicture成功!"<<std::endl; 55 else 56 std::cout<<"文件夹bingPicture已存在!"<<std::endl; 57 58 //本地Tmp路径 59 std::string LocalXmlFolder="C:/bingPicture/Tmp/"; 60 61 if(0!=access(LocalXmlFolder.c_str(),0)) //判断文件夹是否存在,若不存在则创建 62 if(0!=mkdir(LocalXmlFolder.c_str())) 63 std::cout<<"创建临时文件夹Tmp失败!"<<std::endl; 64 else 65 std::cout<<"创建临时文件夹Tmp成功!"<<std::endl; 66 else 67 std::cout<<"临时文件夹Tmp已存在!"<<std::endl; 68 69 } 70 71 /************************************************************************************** 72 首先明白一个概念,即string替换所有字符串,将"12212"这个字符串的所有"12"都替换成"21",结果是什么? 73 可以是22211,也可以是21221,有时候应用的场景不同,就会希望得到不同的结果,所以这两种答案都做了实现。 74 **************************************************************************************/ 75 //替换字符串方法1(完全轮询,替换一次后接着再次扫描,因为替换一次后可能又出现了满足替换条件的字符串) 76 std::string & replace_all(std::string& str,const std::string& old_value,const std::string& new_value) 77 { 78 while(true) { 79 std::string::size_type pos(0); 80 if((pos=str.find(old_value))!=std::string::npos) 81 str.replace(pos,old_value.length(),new_value); 82 else 83 break; 84 } 85 return str; 86 } 87 88 //替换字符串方法2(只替换一次) 本项目中,只替换\为/用方法2即可 89 std::string & replace_all_distinct(std::string& str,const std::string& old_value,const std::string& new_value) 90 { 91 for(std::string::size_type pos(0); pos!=std::string::npos; pos+=new_value.length()) 92 { 93 if((pos=str.find(old_value,pos))!=std::string::npos) 94 str.replace(pos,old_value.length(),new_value); 95 else break; 96 } 97 return str; 98 } 99 100 //获取年月日(命名用) 101 std::string getYearMonthDay() 102 { 103 time_t timer; 104 time(&timer); 105 tm* t_tm = localtime(&timer); 106 107 std::string Year=std::to_string(t_tm->tm_year+1900); 108 std::string Month=std::to_string(t_tm->tm_mon+1); 109 std::string Day=std::to_string(t_tm->tm_mday); 110 std::string PictureName=Year+"_"+Month+"_"+Day; 111 112 return PictureName; 113 } 114 115 //获取图片的Xml并解析图片的url路径 116 std::string getPicTureXmlAndUrl() 117 { 118 //网络上的XML路径 119 std::string WebXmlpath ="http://cn.bing.com/HPImageArchive.aspx?format=xml&idx=0&n=1"; 120 //本地Xml路径 121 std::string LocalXmlFolder="C:/bingPicture/Tmp/"; 122 std::string LocalXmleach=getYearMonthDay(); 123 std::string LocalXmlFullpath=LocalXmlFolder+LocalXmleach+".xml"; 124 125 if(URLDownloadToFileA(NULL, 126 WebXmlpath.c_str(), 127 LocalXmlFullpath.c_str(), 128 0, 129 NULL) 130 ==S_OK) 131 { 132 std::cout<<"Xml下载成功!即将解析今日壁纸Url!"<<std::endl; 133 134 /***************下面开始解析xml中的url路径*******************/ 135 tinyxml2::XMLDocument doc; 136 if(tinyxml2::XML_SUCCESS != doc.LoadFile(LocalXmlFullpath.c_str())) 137 std::cout<<"读取Xml文件异常!"<<std::endl; 138 tinyxml2::XMLElement *images=doc.RootElement(); 139 tinyxml2::XMLElement *image =images->FirstChildElement("image"); 140 141 //图片Url 142 std::string WebPicturedomain="http://cn.bing.com"; 143 std::string WebPictureUrl=""; 144 145 if(image!=NULL) 146 WebPictureUrl=image->FirstChildElement("url")->GetText(); 147 148 std::string WebPictureFullpath1366x768 =WebPicturedomain+WebPictureUrl; 149 std::cout<<"今日壁纸Url解析成功!"<<std::endl; 150 /*********************************************************/ 151 return WebPictureFullpath1366x768; 152 // //将1366x768换成1920x1080 153 // std::string WebPictureFullpath1920x1080 =replace_all_distinct(WebPictureFullpath1366x768,"1366x768","1920x1080"); 154 155 // return WebPictureFullpath1920x1080; 156 } 157 else 158 { 159 std::cout<<"Xml下载失败!无法获取图片Url!请检查网络连接是否正常!"<<std::endl; 160 return "error"; 161 } 162 163 } 164 165 //从网络上下载图片并存储到本地 166 std::string getPicture(std::string WebFullpath) 167 { 168 //本地存储路径 169 std::string LocalFolder="C:/bingPicture/"; 170 std::string Localeach=getYearMonthDay(); 171 std::string LocalFullpath=LocalFolder+Localeach+".jpg"; 172 173 if(URLDownloadToFileA(NULL, 174 WebFullpath.c_str(), 175 LocalFullpath.c_str(), 176 0, 177 NULL) 178 ==S_OK) 179 { 180 std::cout<<"今日壁纸下载成功!"<<std::endl; 181 182 /***************下面转换图片格式jpg->bmp******************/ 183 //临时文件夹Tmp路径 184 std::string TmpFolder="C:/bingPicture/Tmp/"; 185 //.bmp图片路径 186 std::string bmpFolder=TmpFolder+getYearMonthDay()+".bmp"; 187 LoadJpegFile(const_cast<char *>(LocalFullpath.c_str()),const_cast<char *>(bmpFolder.c_str())); 188 /*******************************************************/ 189 return bmpFolder; 190 } 191 else 192 { 193 std::cout<<"壁纸下载失败!请检查网络连接是否正常!"<<std::endl; 194 return "error"; 195 } 196 } 197 198 //改变桌面背景成功后,删除bmp文件和xml文件(只保留jpg文件),此步骤需要小心,避免删除错误路径下的内容 199 void deleteBmpAndXml() 200 { 201 //临时文件夹Tmp路径 202 std::string TmpFolder="C:/bingPicture/Tmp/"; 203 //.bmp图片路径 204 std::string bmpFolder=TmpFolder+getYearMonthDay()+".bmp"; 205 //xml文件路径 206 std::string xmlFolder=TmpFolder+getYearMonthDay()+".xml"; 207 208 if(0==access("C:/bingPicture/Tmp/",0)) //判断文件夹是否存在,若存在则删除 209 { 210 //删除bmp图片 211 if(0==access(bmpFolder.c_str(),0)) 212 { 213 if(0==remove(bmpFolder.c_str())) 214 std::cout<<"删除临时bmp格式图片成功!"<<std::endl; 215 else 216 std::cout<<"删除临时bmp格式图片失败!"<<std::endl; 217 } 218 else 219 std::cout<<"临时bmp格式图片不存在!"<<std::endl; 220 221 //删除xml文件 222 if(0==access(xmlFolder.c_str(),0)) 223 { 224 if(0==remove(xmlFolder.c_str())) 225 std::cout<<"删除xml文件成功!"<<std::endl; 226 else 227 std::cout<<"删除xml文件失败!"<<std::endl; 228 } 229 else 230 std::cout<<"xml文件不存在!"<<std::endl; 231 232 //删除Tmp文件夹(注意此函数只能删除空文件夹,因此要先删除文件夹中的文件) 233 if(0==rmdir(TmpFolder.c_str())) 234 std::cout<<"临时文件夹Tmp已删除!"<<std::endl; 235 else 236 std::cout<<"临时文件夹Tmp删除失败!"<<std::endl; 237 } 238 else 239 std::cout<<"临时文件夹Tmp不存在!"<<std::endl; 240 241 } 242 243 //改变桌面背景(PictureFullpath:图片完整路径) 244 void changePicture(std::string PictureFullpath) 245 { 246 bool result=false; 247 result=SystemParametersInfoA(SPI_SETDESKWALLPAPER, 248 0, 249 (PVOID)PictureFullpath.c_str(), 250 0); 251 if(result==false) 252 { 253 std::cout<<"今日壁纸更新失败!请联系开发人员!"<<std::endl; 254 } 255 256 else 257 { 258 SystemParametersInfoA(SPI_SETDESKWALLPAPER, 259 0, 260 (PVOID)PictureFullpath.c_str(), 261 SPIF_SENDCHANGE); 262 //deleteBmpAndXml(); //windows8及其以上会变成黑色,把这条语句放到main最后面就没问题,具体原因未知 263 system("cls"); 264 std::cout<<"version:1.0.0 (Author:xiaoxi666)"<<std::endl<<std::endl; 265 std::cout<<"今日壁纸更新成功!"<<std::endl<<std::endl; 266 std::cout<<"美好的一天开始啦!用心享受吧!"<<std::endl<<std::endl; 267 } 268 } 269 270 int main() 271 { 272 std::string startOrder=""; 273 std::cout<<"嗨!小伙伴!你的贴心壁纸小助手已启动!将为你设置今日壁纸哦!"<<std::endl<<std::endl; 274 std::cout<<"请确保电脑网络连接状况良好,准备好后输入go"<<std::endl<<std::endl; 275 std::cout<<"请输入指令: "; 276 std::cin>>startOrder; 277 while("go"!=startOrder) 278 { 279 std::cout<<"哎呀输错了呢,重新输入吧: "; 280 std::cin>>startOrder; 281 } 282 if("go"==startOrder) 283 { 284 createDir(); 285 changePicture(getPicture(getPicTureXmlAndUrl())); 286 } 287 288 /*******************************以下为个性化字幕输出,与程序核心功能无关************************/ 289 std::string umua0=" ** ** ******** ******* ******* *** *** "; 290 std::string umua1=" ** ** ******** ******* ******* *** *** "; 291 std::string umua2=" ** ** ** ** ** ** *** ** *** *** "; 292 std::string umua3=" ** ** ** ** ** ** *** ** ** ** "; 293 std::string umua4=" ********* ******** ******* ******* * * "; 294 std::string umua5=" ********* ******** ******* ** ** "; 295 std::string umua6=" ** ** ** ** ** ** ** "; 296 std::string umua7=" ** ** ** ** ** ** ** "; 297 std::string umua8=" ** ** ** ** ** ** ** "; 298 std::string umua9=" ** ** ** ** ** ** ** "; 299 300 #define mua(n) std::cout<<umua##n<<std::endl; 301 std::cout<<std::endl<<std::endl; 302 mua(0);mua(1);mua(2);mua(3);mua(4);mua(5);mua(6);mua(7);mua(8);mua(9); 303 std::cout<<std::endl<<std::endl<<std::endl; 304 system("pause"); 305 /******************************************************************************************/ 306 deleteBmpAndXml(); 307 return 0; 308 }
图片格式转换程序(这个是网上下载的C源码,我改了一下接口,在此感谢)
jpeg.h
//头文件jpeg.h,配合程序jpeg2bmp.c使用 //若用于c++程序中,请用extern "C"包含此头文件 //功能:用于将图片从jpg类型转换为bmp类型,调用函数LoadJpegFile即可,参数1:jpg文件路径;参数2:bmp文件路径 #define M_SOF0 0xc0 #define M_DHT 0xc4 #define M_EOI 0xd9 #define M_SOS 0xda #define M_DQT 0xdb #define M_DRI 0xdd #define M_APP0 0xe0 static int Zig_Zag[8][8]={{0,1,5,6,14,15,27,28}, {2,4,7,13,16,26,29,42}, {3,8,12,17,25,30,41,43}, {9,11,18,24,37,40,44,53}, {10,19,23,32,39,45,52,54}, {20,22,33,38,46,51,55,60}, {21,34,37,47,50,56,59,61}, {35,36,48,49,57,58,62,63} }; #define W1 2841 /* 2048*sqrt(2)*cos(1*pi/16) */ #define W2 2676 /* 2048*sqrt(2)*cos(2*pi/16) */ #define W3 2408 /* 2048*sqrt(2)*cos(3*pi/16) */ #define W5 1609 /* 2048*sqrt(2)*cos(5*pi/16) */ #define W6 1108 /* 2048*sqrt(2)*cos(6*pi/16) */ #define W7 565 /* 2048*sqrt(2)*cos(7*pi/16) */ int LoadJpegFile (char *JpegFileName,char *bmpFileName);
jpeg2bmp.c
//jpeg.c //本程序用C语言编写,若用于c++程序中,请用extern "C"包含头文件jpeg.h //功能:用于将图片从jpg类型转换为bmp类型,调用函数LoadJpegFile即可,参数1:jpg文件路径;参数2:bmp文件路径 #include "jpeg.h" #include "memory.h" #include "math.h" #include "stdio.h" #include "windows.h" //macro definition #define WIDTHBYTES(i) ((i+31)/32*4) #define PI 3.1415926535 //define return value of function #define FUNC_OK 0 #define FUNC_MEMORY_ERROR 1 #define FUNC_FILE_ERROR 2 #define FUNC_FORMAT_ERROR 3 ////////////////////////////////////////////////// //Jpeg functions //int LoadJpegFile (char *JpegFileName,char *bmpFileName); void showerror(int funcret); int InitTag(); void InitTable(); int Decode(); int DecodeMCUBlock(); int HufBlock(unsigned char dchufindex,unsigned char achufindex); int DecodeElement(); void IQtIZzMCUComponent(short flag); void IQtIZzBlock(short *s ,int * d,short flag); void GetYUV(short flag); void StoreBuffer(); BYTE ReadByte(); void Initialize_Fast_IDCT(); void Fast_IDCT(int * block); void idctrow(int * blk); void idctcol(int * blk); ////////////////////////////////////////////////// //global variable declaration BITMAPFILEHEADER bf; BITMAPINFOHEADER bi; HBITMAP hBitmap=NULL; HGLOBAL hImgData=NULL; DWORD NumColors; DWORD LineBytes; DWORD ImgWidth=0 , ImgHeight=0; unsigned int PcxBytesPerLine; LPSTR lpPtr; ////////////////////////////////////////////////// //variables used in jpeg function short SampRate_Y_H,SampRate_Y_V; short SampRate_U_H,SampRate_U_V; short SampRate_V_H,SampRate_V_V; short H_YtoU,V_YtoU,H_YtoV,V_YtoV; short Y_in_MCU,U_in_MCU,V_in_MCU; unsigned char *lpJpegBuf; unsigned char *lp; short qt_table[3][64]; short comp_num; BYTE comp_index[3]; BYTE YDcIndex,YAcIndex,UVDcIndex,UVAcIndex; BYTE HufTabIndex; short *YQtTable,*UQtTable,*VQtTable; BYTE And[9]={0,1,3,7,0xf,0x1f,0x3f,0x7f,0xff}; short code_pos_table[4][16],code_len_table[4][16]; unsigned short code_value_table[4][256]; unsigned short huf_max_value[4][16],huf_min_value[4][16]; short BitPos,CurByte; short rrun,vvalue; short MCUBuffer[10*64]; int QtZzMCUBuffer[10*64]; short BlockBuffer[64]; short ycoef,ucoef,vcoef; BOOL IntervalFlag; short interval=0; int Y[4*64],U[4*64],V[4*64]; DWORD sizei,sizej; short restart; static long iclip[1024]; static long *iclp; //////////////////////////////////////////////////////////////// int LoadJpegFile (char *JpegFileName,char *bmpFileName) { HFILE hfjpg; DWORD ImgSize; DWORD JpegBufSize; HFILE hfbmp; HGLOBAL hJpegBuf; int funcret; LPBITMAPINFOHEADER lpImgData; if((hfjpg=_lopen(JpegFileName,OF_READ))==HFILE_ERROR) { showerror(FUNC_FILE_ERROR); return 0; } //get jpg file length JpegBufSize=_llseek(hfjpg,0L,SEEK_END); //rewind to the beginning of the file _llseek(hfjpg,0L,SEEK_SET); if((hJpegBuf=GlobalAlloc(GHND,JpegBufSize))==NULL) { _lclose(hfjpg); showerror(FUNC_MEMORY_ERROR); return 0; } lpJpegBuf=(unsigned char *)GlobalLock(hJpegBuf); _hread(hfjpg,(unsigned char *)lpJpegBuf,JpegBufSize); _lclose(hfjpg); InitTable(); if((funcret=InitTag())!=FUNC_OK) { GlobalUnlock(hJpegBuf); GlobalFree(hJpegBuf); showerror(funcret); return 0; } //create new bitmapfileheader and bitmapinfoheader memset((char *)&bf,0,sizeof(BITMAPFILEHEADER)); memset((char *)&bi,0,sizeof(BITMAPINFOHEADER)); bi.biSize=(DWORD)sizeof(BITMAPINFOHEADER); bi.biWidth=(LONG)(ImgWidth); bi.biHeight=(LONG)(ImgHeight); bi.biPlanes=1; bi.biBitCount=24; bi.biClrUsed=0; bi.biClrImportant=0; bi.biCompression=BI_RGB; NumColors=0; LineBytes=(DWORD)WIDTHBYTES(bi.biWidth*bi.biBitCount); ImgSize=(DWORD)LineBytes*bi.biHeight; bf.bfType=0x4d42; bf.bfSize=sizeof(BITMAPFILEHEADER)+sizeof(BITMAPINFOHEADER)+NumColors*sizeof(RGBQUAD)+ImgSize; bf.bfOffBits=(DWORD)(NumColors*sizeof(RGBQUAD)+sizeof(BITMAPFILEHEADER)+sizeof(BITMAPINFOHEADER)); if((hImgData=GlobalAlloc(GHND,ImgSize))==NULL) { GlobalUnlock(hJpegBuf); GlobalFree(hJpegBuf); showerror(FUNC_MEMORY_ERROR); return FALSE; } lpImgData=(LPBITMAPINFOHEADER)GlobalLock(hImgData); lpPtr=(char *)lpImgData; if((SampRate_Y_H==0)||(SampRate_Y_V==0)) { GlobalUnlock(hJpegBuf); GlobalFree(hJpegBuf); GlobalUnlock(hImgData); GlobalFree(hImgData); hImgData=NULL; showerror(FUNC_FORMAT_ERROR); return FALSE ; } funcret=Decode(); if(funcret==FUNC_OK) { hfbmp=_lcreat(bmpFileName,0); _lwrite(hfbmp,(LPSTR)&bf,sizeof(BITMAPFILEHEADER)); //写BMP文件头 _lwrite(hfbmp,(LPSTR)&bi,sizeof(BITMAPINFOHEADER)); //写BMP文件信息 _lwrite(hfbmp,(LPSTR)lpImgData,ImgSize); //写BMP位图数据 _lclose(hfbmp); GlobalUnlock(hJpegBuf); GlobalFree(hJpegBuf); GlobalUnlock(hImgData); return TRUE; } else { GlobalUnlock(hJpegBuf); GlobalFree(hJpegBuf); GlobalUnlock(hImgData); GlobalFree(hImgData); hImgData=NULL; showerror(funcret); return FALSE; } } ///////////////////////////////////////////////// void showerror(int funcret) { switch(funcret) { case FUNC_MEMORY_ERROR: printf("Error alloc memory!\n"); exit(1); break; case FUNC_FILE_ERROR: printf("File not found!!\n"); exit(1); break; case FUNC_FORMAT_ERROR: printf("File format error!\n"); exit(1); break; } } //////////////////////////////////////////////////////////////////////////////// int InitTag() { BOOL finish=FALSE; BYTE id; short llength; short i,j,k; short huftab1,huftab2; short huftabindex; BYTE hf_table_index; BYTE qt_table_index; BYTE comnum; unsigned char *lptemp; short ccount; lp=lpJpegBuf+2; while (!finish) { id=*(lp+1); lp+=2; switch (id) { case M_APP0: llength=MAKEWORD(*(lp+1),*lp); lp+=llength; break; case M_DQT: llength=MAKEWORD(*(lp+1),*lp); qt_table_index=(*(lp+2))&0x0f; lptemp=lp+3; if(llength<80) { for(i=0;i<64;i++) qt_table[qt_table_index][i]=(short)*(lptemp++); } else { for(i=0;i<64;i++) qt_table[qt_table_index][i]=(short)*(lptemp++); qt_table_index=(*(lptemp++))&0x0f; for(i=0;i<64;i++) qt_table[qt_table_index][i]=(short)*(lptemp++); } lp+=llength; break; case M_SOF0: llength=MAKEWORD(*(lp+1),*lp); ImgHeight=MAKEWORD(*(lp+4),*(lp+3)); ImgWidth=MAKEWORD(*(lp+6),*(lp+5)); comp_num=*(lp+7); if((comp_num!=1)&&(comp_num!=3)) return FUNC_FORMAT_ERROR; if(comp_num==3) { comp_index[0]=*(lp+8); SampRate_Y_H=(*(lp+9))>>4; SampRate_Y_V=(*(lp+9))&0x0f; YQtTable=(short *)qt_table[*(lp+10)]; comp_index[1]=*(lp+11); SampRate_U_H=(*(lp+12))>>4; SampRate_U_V=(*(lp+12))&0x0f; UQtTable=(short *)qt_table[*(lp+13)]; comp_index[2]=*(lp+14); SampRate_V_H=(*(lp+15))>>4; SampRate_V_V=(*(lp+15))&0x0f; VQtTable=(short *)qt_table[*(lp+16)]; } else { comp_index[0]=*(lp+8); SampRate_Y_H=(*(lp+9))>>4; SampRate_Y_V=(*(lp+9))&0x0f; YQtTable=(short *)qt_table[*(lp+10)]; comp_index[1]=*(lp+8); SampRate_U_H=1; SampRate_U_V=1; UQtTable=(short *)qt_table[*(lp+10)]; comp_index[2]=*(lp+8); SampRate_V_H=1; SampRate_V_V=1; VQtTable=(short *)qt_table[*(lp+10)]; } lp+=llength; break; case M_DHT: llength=MAKEWORD(*(lp+1),*lp); if (llength<0xd0) { huftab1=(short)(*(lp+2))>>4; //huftab1=0,1 huftab2=(short)(*(lp+2))&0x0f; //huftab2=0,1 huftabindex=huftab1*2+huftab2; lptemp=lp+3; for (i=0; i<16; i++) code_len_table[huftabindex][i]=(short)(*(lptemp++)); j=0; for (i=0; i<16; i++) if(code_len_table[huftabindex][i]!=0) { k=0; while(k<code_len_table[huftabindex][i]) { code_value_table[huftabindex][k+j]=(short)(*(lptemp++)); k++; } j+=k; } i=0; while (code_len_table[huftabindex][i]==0) i++; for (j=0;j<i;j++) { huf_min_value[huftabindex][j]=0; huf_max_value[huftabindex][j]=0; } huf_min_value[huftabindex][i]=0; huf_max_value[huftabindex][i]=code_len_table[huftabindex][i]-1; for (j=i+1;j<16;j++) { huf_min_value[huftabindex][j]=(huf_max_value[huftabindex][j-1]+1)<<1; huf_max_value[huftabindex][j]=huf_min_value[huftabindex][j]+code_len_table[huftabindex][j]-1; } code_pos_table[huftabindex][0]=0; for (j=1;j<16;j++) code_pos_table[huftabindex][j]=code_len_table[huftabindex][j-1]+code_pos_table[huftabindex][j-1]; lp+=llength; } //if else { hf_table_index=*(lp+2); lp+=2; while (hf_table_index!=0xff) { huftab1=(short)hf_table_index>>4; //huftab1=0,1 huftab2=(short)hf_table_index&0x0f; //huftab2=0,1 huftabindex=huftab1*2+huftab2; lptemp=lp+1; ccount=0; for (i=0; i<16; i++) { code_len_table[huftabindex][i]=(short)(*(lptemp++)); ccount+=code_len_table[huftabindex][i]; } ccount+=17; j=0; for (i=0; i<16; i++) if(code_len_table[huftabindex][i]!=0) { k=0; while(k<code_len_table[huftabindex][i]) { code_value_table[huftabindex][k+j]=(short)(*(lptemp++)); k++; } j+=k; } i=0; while (code_len_table[huftabindex][i]==0) i++; for (j=0;j<i;j++) { huf_min_value[huftabindex][j]=0; huf_max_value[huftabindex][j]=0; } huf_min_value[huftabindex][i]=0; huf_max_value[huftabindex][i]=code_len_table[huftabindex][i]-1; for (j=i+1;j<16;j++) { huf_min_value[huftabindex][j]=(huf_max_value[huftabindex][j-1]+1)<<1; huf_max_value[huftabindex][j]=huf_min_value[huftabindex][j]+code_len_table[huftabindex][j]-1; } code_pos_table[huftabindex][0]=0; for (j=1;j<16;j++) code_pos_table[huftabindex][j]=code_len_table[huftabindex][j-1]+code_pos_table[huftabindex][j-1]; lp+=ccount; hf_table_index=*lp; } //while } //else break; case M_DRI: llength=MAKEWORD(*(lp+1),*lp); restart=MAKEWORD(*(lp+3),*(lp+2)); lp+=llength; break; case M_SOS: llength=MAKEWORD(*(lp+1),*lp); comnum=*(lp+2); if(comnum!=comp_num) return FUNC_FORMAT_ERROR; lptemp=lp+3; for (i=0;i<comp_num;i++) { if(*lptemp==comp_index[0]) { YDcIndex=(*(lptemp+1))>>4; //Y YAcIndex=((*(lptemp+1))&0x0f)+2; } else{ UVDcIndex=(*(lptemp+1))>>4; //U,V UVAcIndex=((*(lptemp+1))&0x0f)+2; } lptemp+=2; } lp+=llength; finish=TRUE; break; case M_EOI: return FUNC_FORMAT_ERROR; break; default: if ((id&0xf0)!=0xd0) { llength=MAKEWORD(*(lp+1),*lp); lp+=llength; } else lp+=2; break; } //switch } //while return FUNC_OK; } ///////////////////////////////////////////////////////////////// void InitTable() { short i,j; sizei=sizej=0; ImgWidth=ImgHeight=0; rrun=vvalue=0; BitPos=0; CurByte=0; IntervalFlag=FALSE; restart=0; for(i=0;i<3;i++) for(j=0;j<64;j++) qt_table[i][j]=0; comp_num=0; HufTabIndex=0; for(i=0;i<3;i++) comp_index[i]=0; for(i=0;i<4;i++) for(j=0;j<16;j++) { code_len_table[i][j]=0; code_pos_table[i][j]=0; huf_max_value[i][j]=0; huf_min_value[i][j]=0; } for(i=0;i<4;i++) for(j=0;j<256;j++) code_value_table[i][j]=0; for(i=0;i<10*64;i++) { MCUBuffer[i]=0; QtZzMCUBuffer[i]=0; } for(i=0;i<4*64;i++) { Y[i]=0; U[i]=0; V[i]=0; } for(i=0;i<64;i++) BlockBuffer[i]=0; ycoef=ucoef=vcoef=0; } ///////////////////////////////////////////////////////////////////////// int Decode() { int funcret; Y_in_MCU=SampRate_Y_H*SampRate_Y_V; U_in_MCU=SampRate_U_H*SampRate_U_V; V_in_MCU=SampRate_V_H*SampRate_V_V; H_YtoU=SampRate_Y_H/SampRate_U_H; V_YtoU=SampRate_Y_V/SampRate_U_V; H_YtoV=SampRate_Y_H/SampRate_V_H; V_YtoV=SampRate_Y_V/SampRate_V_V; Initialize_Fast_IDCT(); while((funcret=DecodeMCUBlock())==FUNC_OK) { interval++; if((restart)&&(interval % restart==0)) IntervalFlag=TRUE; else IntervalFlag=FALSE; IQtIZzMCUComponent(0); IQtIZzMCUComponent(1); IQtIZzMCUComponent(2); GetYUV(0); GetYUV(1); GetYUV(2); StoreBuffer(); sizej+=SampRate_Y_H*8; if(sizej>=ImgWidth) { sizej=0; sizei+=SampRate_Y_V*8; } if ((sizej==0)&&(sizei>=ImgHeight)) break; } return funcret; } ///////////////////////////////////////////////////////////////////////////////////////// void GetYUV(short flag) { short H,VV; short i,j,k,h; int *buf; int *pQtZzMCU; switch(flag) { case 0: H=SampRate_Y_H; VV=SampRate_Y_V; buf=Y; pQtZzMCU=QtZzMCUBuffer; break; case 1: H=SampRate_U_H; VV=SampRate_U_V; buf=U; pQtZzMCU=QtZzMCUBuffer+Y_in_MCU*64; break; case 2: H=SampRate_V_H; VV=SampRate_V_V; buf=V; pQtZzMCU=QtZzMCUBuffer+(Y_in_MCU+U_in_MCU)*64; break; } for (i=0;i<VV;i++) for(j=0;j<H;j++) for(k=0;k<8;k++) for(h=0;h<8;h++) buf[(i*8+k)*SampRate_Y_H*8+j*8+h]=*pQtZzMCU++; } /////////////////////////////////////////////////////////////////////////////// void StoreBuffer() { short i,j; unsigned char *lpbmp; unsigned char R,G,B; int y,u,v,rr,gg,bb; for(i=0;i<SampRate_Y_V*8;i++) { if((sizei+i)<ImgHeight) { lpbmp=((unsigned char *)lpPtr+(DWORD)(ImgHeight-sizei-i-1)*LineBytes+sizej*3); for(j=0;j<SampRate_Y_H*8;j++) { if((sizej+j)<ImgWidth) { y=Y[i*8*SampRate_Y_H+j]; u=U[(i/V_YtoU)*8*SampRate_Y_H+j/H_YtoU]; v=V[(i/V_YtoV)*8*SampRate_Y_H+j/H_YtoV]; rr=((y<<8)+18*u+367*v)>>8; gg=((y<<8)-159*u-220*v)>>8; bb=((y<<8)+411*u-29*v)>>8; R=(unsigned char)rr; G=(unsigned char)gg; B=(unsigned char)bb; if (rr&0xffffff00) if (rr>255) R=255; else if (rr<0) R=0; if (gg&0xffffff00) if (gg>255) G=255; else if (gg<0) G=0; if (bb&0xffffff00) if (bb>255) B=255; else if (bb<0) B=0; *lpbmp++=B; *lpbmp++=G; *lpbmp++=R; } else break; } } else break; } } /////////////////////////////////////////////////////////////////////////////// int DecodeMCUBlock() { short *lpMCUBuffer; short i,j; int funcret; if (IntervalFlag) { lp+=2; ycoef=ucoef=vcoef=0; BitPos=0; CurByte=0; } switch(comp_num) { case 3: lpMCUBuffer=MCUBuffer; for (i=0;i<SampRate_Y_H*SampRate_Y_V;i++) //Y { funcret=HufBlock(YDcIndex,YAcIndex); if (funcret!=FUNC_OK) return funcret; BlockBuffer[0]=BlockBuffer[0]+ycoef; ycoef=BlockBuffer[0]; for (j=0;j<64;j++) *lpMCUBuffer++=BlockBuffer[j]; } for (i=0;i<SampRate_U_H*SampRate_U_V;i++) //U { funcret=HufBlock(UVDcIndex,UVAcIndex); if (funcret!=FUNC_OK) return funcret; BlockBuffer[0]=BlockBuffer[0]+ucoef; ucoef=BlockBuffer[0]; for (j=0;j<64;j++) *lpMCUBuffer++=BlockBuffer[j]; } for (i=0;i<SampRate_V_H*SampRate_V_V;i++) //V { funcret=HufBlock(UVDcIndex,UVAcIndex); if (funcret!=FUNC_OK) return funcret; BlockBuffer[0]=BlockBuffer[0]+vcoef; vcoef=BlockBuffer[0]; for (j=0;j<64;j++) *lpMCUBuffer++=BlockBuffer[j]; } break; case 1: lpMCUBuffer=MCUBuffer; funcret=HufBlock(YDcIndex,YAcIndex); if (funcret!=FUNC_OK) return funcret; BlockBuffer[0]=BlockBuffer[0]+ycoef; ycoef=BlockBuffer[0]; for (j=0;j<64;j++) *lpMCUBuffer++=BlockBuffer[j]; for (i=0;i<128;i++) *lpMCUBuffer++=0; break; default: return FUNC_FORMAT_ERROR; } return FUNC_OK; } ////////////////////////////////////////////////////////////////// int HufBlock(BYTE dchufindex,BYTE achufindex) { short count=0; short i; int funcret; //dc HufTabIndex=dchufindex; funcret=DecodeElement(); if(funcret!=FUNC_OK) return funcret; BlockBuffer[count++]=vvalue; //ac HufTabIndex=achufindex; while (count<64) { funcret=DecodeElement(); if(funcret!=FUNC_OK) return funcret; if ((rrun==0)&&(vvalue==0)) { for (i=count;i<64;i++) BlockBuffer[i]=0; count=64; } else { for (i=0;i<rrun;i++) BlockBuffer[count++]=0; BlockBuffer[count++]=vvalue; } } return FUNC_OK; } ////////////////////////////////////////////////////////////////////////////// int DecodeElement() { int thiscode,tempcode; unsigned short temp,valueex; short codelen; BYTE hufexbyte,runsize,tempsize,sign; BYTE newbyte,lastbyte; if(BitPos>=1) { BitPos--; thiscode=(BYTE)CurByte>>BitPos; CurByte=CurByte&And[BitPos]; } else { lastbyte=ReadByte(); BitPos--; newbyte=CurByte&And[BitPos]; thiscode=lastbyte>>7; CurByte=newbyte; } codelen=1; while ((thiscode<huf_min_value[HufTabIndex][codelen-1])|| (code_len_table[HufTabIndex][codelen-1]==0)|| (thiscode>huf_max_value[HufTabIndex][codelen-1])) { if(BitPos>=1) { BitPos--; tempcode=(BYTE)CurByte>>BitPos; CurByte=CurByte&And[BitPos]; } else { lastbyte=ReadByte(); BitPos--; newbyte=CurByte&And[BitPos]; tempcode=(BYTE)lastbyte>>7; CurByte=newbyte; } thiscode=(thiscode<<1)+tempcode; codelen++; if(codelen>16) return FUNC_FORMAT_ERROR; } //while temp=thiscode-huf_min_value[HufTabIndex][codelen-1]+code_pos_table[HufTabIndex][codelen-1]; hufexbyte=(BYTE)code_value_table[HufTabIndex][temp]; rrun=(short)(hufexbyte>>4); runsize=hufexbyte&0x0f; if(runsize==0) { vvalue=0; return FUNC_OK; } tempsize=runsize; if(BitPos>=runsize) { BitPos-=runsize; valueex=(BYTE)CurByte>>BitPos; CurByte=CurByte&And[BitPos]; } else { valueex=CurByte; tempsize-=BitPos; while(tempsize>8) { lastbyte=ReadByte(); valueex=(valueex<<8)+(BYTE)lastbyte; tempsize-=8; } //while lastbyte=ReadByte(); BitPos-=tempsize; valueex=(valueex<<tempsize)+(lastbyte>>BitPos); CurByte=lastbyte&And[BitPos]; } //else sign=valueex>>(runsize-1); if(sign) vvalue=valueex; else { valueex=valueex^0xffff; temp=0xffff<<runsize; vvalue=-(short)(valueex^temp); } return FUNC_OK; } ///////////////////////////////////////////////////////////////////////////////////// void IQtIZzMCUComponent(short flag) { short H,VV; short i,j; int *pQtZzMCUBuffer; short *pMCUBuffer; switch(flag) { case 0: H=SampRate_Y_H; VV=SampRate_Y_V; pMCUBuffer=MCUBuffer; pQtZzMCUBuffer=QtZzMCUBuffer; break; case 1: H=SampRate_U_H; VV=SampRate_U_V; pMCUBuffer=MCUBuffer+Y_in_MCU*64; pQtZzMCUBuffer=QtZzMCUBuffer+Y_in_MCU*64; break; case 2: H=SampRate_V_H; VV=SampRate_V_V; pMCUBuffer=MCUBuffer+(Y_in_MCU+U_in_MCU)*64; pQtZzMCUBuffer=QtZzMCUBuffer+(Y_in_MCU+U_in_MCU)*64; break; } for(i=0;i<VV;i++) for (j=0;j<H;j++) IQtIZzBlock(pMCUBuffer+(i*H+j)*64,pQtZzMCUBuffer+(i*H+j)*64,flag); } ////////////////////////////////////////////////////////////////////////////////////////// void IQtIZzBlock(short *s ,int * d,short flag) { short i,j; short tag; short *pQt; int buffer2[8][8]; int *buffer1; short offset; switch(flag) { case 0: pQt=YQtTable; offset=128; break; case 1: pQt=UQtTable; offset=0; break; case 2: pQt=VQtTable; offset=0; break; } for(i=0;i<8;i++) for(j=0;j<8;j++) { tag=Zig_Zag[i][j]; buffer2[i][j]=(int)s[tag]*(int)pQt[tag]; } buffer1=(int *)buffer2; Fast_IDCT(buffer1); for(i=0;i<8;i++) for(j=0;j<8;j++) d[i*8+j]=buffer2[i][j]+offset; } /////////////////////////////////////////////////////////////////////////////// void Fast_IDCT(int * block) { short i; for (i=0; i<8; i++) idctrow(block+8*i); for (i=0; i<8; i++) idctcol(block+i); } /////////////////////////////////////////////////////////////////////////////// BYTE ReadByte() { BYTE i; i=*(lp++); if(i==0xff) lp++; BitPos=8; CurByte=i; return i; } /////////////////////////////////////////////////////////////////////// void Initialize_Fast_IDCT() { short i; iclp = iclip+512; for (i= -512; i<512; i++) iclp[i] = (i<-256) ? -256 : ((i>255) ? 255 : i); } //////////////////////////////////////////////////////////////////////// void idctrow(int * blk) { int x0, x1, x2, x3, x4, x5, x6, x7, x8; //intcut if (!((x1 = blk[4]<<11) | (x2 = blk[6]) | (x3 = blk[2]) | (x4 = blk[1]) | (x5 = blk[7]) | (x6 = blk[5]) | (x7 = blk[3]))) { blk[0]=blk[1]=blk[2]=blk[3]=blk[4]=blk[5]=blk[6]=blk[7]=blk[0]<<3; return; } x0 = (blk[0]<<11) + 128; // for proper rounding in the fourth stage //first stage x8 = W7*(x4+x5); x4 = x8 + (W1-W7)*x4; x5 = x8 - (W1+W7)*x5; x8 = W3*(x6+x7); x6 = x8 - (W3-W5)*x6; x7 = x8 - (W3+W5)*x7; //second stage x8 = x0 + x1; x0 -= x1; x1 = W6*(x3+x2); x2 = x1 - (W2+W6)*x2; x3 = x1 + (W2-W6)*x3; x1 = x4 + x6; x4 -= x6; x6 = x5 + x7; x5 -= x7; //third stage x7 = x8 + x3; x8 -= x3; x3 = x0 + x2; x0 -= x2; x2 = (181*(x4+x5)+128)>>8; x4 = (181*(x4-x5)+128)>>8; //fourth stage blk[0] = (x7+x1)>>8; blk[1] = (x3+x2)>>8; blk[2] = (x0+x4)>>8; blk[3] = (x8+x6)>>8; blk[4] = (x8-x6)>>8; blk[5] = (x0-x4)>>8; blk[6] = (x3-x2)>>8; blk[7] = (x7-x1)>>8; } ////////////////////////////////////////////////////////////////////////////// void idctcol(int * blk) { int x0, x1, x2, x3, x4, x5, x6, x7, x8; //intcut if (!((x1 = (blk[8*4]<<8)) | (x2 = blk[8*6]) | (x3 = blk[8*2]) | (x4 = blk[8*1]) | (x5 = blk[8*7]) | (x6 = blk[8*5]) | (x7 = blk[8*3]))) { blk[8*0]=blk[8*1]=blk[8*2]=blk[8*3]=blk[8*4]=blk[8*5] =blk[8*6]=blk[8*7]=iclp[(blk[8*0]+32)>>6]; return; } x0 = (blk[8*0]<<8) + 8192; //first stage x8 = W7*(x4+x5) + 4; x4 = (x8+(W1-W7)*x4)>>3; x5 = (x8-(W1+W7)*x5)>>3; x8 = W3*(x6+x7) + 4; x6 = (x8-(W3-W5)*x6)>>3; x7 = (x8-(W3+W5)*x7)>>3; //second stage x8 = x0 + x1; x0 -= x1; x1 = W6*(x3+x2) + 4; x2 = (x1-(W2+W6)*x2)>>3; x3 = (x1+(W2-W6)*x3)>>3; x1 = x4 + x6; x4 -= x6; x6 = x5 + x7; x5 -= x7; //third stage x7 = x8 + x3; x8 -= x3; x3 = x0 + x2; x0 -= x2; x2 = (181*(x4+x5)+128)>>8; x4 = (181*(x4-x5)+128)>>8; //fourth stage blk[8*0] = iclp[(x7+x1)>>14]; blk[8*1] = iclp[(x3+x2)>>14]; blk[8*2] = iclp[(x0+x4)>>14]; blk[8*3] = iclp[(x8+x6)>>14]; blk[8*4] = iclp[(x8-x6)>>14]; blk[8*5] = iclp[(x0-x4)>>14]; blk[8*6] = iclp[(x3-x2)>>14]; blk[8*7] = iclp[(x7-x1)>>14]; } //main( ) //{ // LoadJpegFile("test.jpg"); //}
xml解析,我用的是TinyXml2开源库,这个就不贴源码了。
关于Qt项目图标制作
我用的Qt版本是4.8.5。
步骤如下:
- 在Qt工程目录下新建一个文本文件,并将其名称改为 *.rc (名字任取)
- 将你的图标文件.ico添加到项目中
-
打开该 rc 文件,在该rc文件里面加入以下一行文本(将文本中的*换成你的图标的名字):
IDI_ICON ICON DISCARDABLE "*.ico"
- 在项目文件.pro中加入以下文本(将*换成你的rc文件名字):
RC_FILE = \
*.rc
5. 重新编译工程即可
关于Qt项目的发布
Qt项目发布最麻烦的就是动态依赖库,常用的工具是hap-depends,它可以查看软件的依赖库(*.dll),直接用它打开你的.exe文件,查看缺失哪些.dll文件,找到它们后和exe放在相同目录即可。
当然,发布之前需要经过多个平台的测试。
最后祝大家生活愉快!
标签:
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有
- C++ 转换函数搭配友元函数 2020-06-10
- C++ 自动转换和强制类型转换(用户自定义类类型) 2020-06-10
- C++ rand函数 2020-06-10
- C++ 友元函数 2020-06-10
- C++ 运算符重载 2020-06-10
IDC资讯: 主机资讯 注册资讯 托管资讯 vps资讯 网站建设
网站运营: 建站经验 策划盈利 搜索优化 网站推广 免费资源
网络编程: Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术: Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧: 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷
网页制作: FrontPages Dreamweaver Javascript css photoshop fireworks Flash