现在的位置: 首页 > 综合 > 正文

如何编写跨平台应用程序

2013年10月10日 ⁄ 综合 ⁄ 共 6680字 ⁄ 字号 评论关闭

我们使用的许多著名的应用程序,比如VC,WINRAR,SVN等等,都既能在WIN下运行,也支持命令行操作。这是很爽的,习惯在可视化下操作的同志,可以使用GUI界面操作,习惯怀旧或者喜欢命令行操作的同志,可以使用命令操作。同时我们也可以根据不同情况灵活运用,如果你要把硬盘上多于一百个的影视文件分别压缩成压缩包,并加入说明文件,使用可视化操作显示是一件费时费力的工作,但使用命令行写一个简单的批处理循环,就可以轻松完成。这时你会十分感激那些提供操作系统的大佬们,他们在操作系统里集成的一个简单的SHELL的英明的决策使你轻松了多少呀。

这里简单介绍一下如何写一个程序,使我们的程序可以支持跨平台操作,既带命令行,又可以可视操作。

要写一个简单的跨WIN和DOS的程序是很简单的,要是还想能在LIUNX上运行,那可能会复杂一些。

由于对LIUNX知之不多,这里只对制作WIN和DOS程序做一些简单的介绍。制作DOS和WIN下运行的程序,有两种方法。

第一种方法:

先用标准C或者C++写一个DOS程序,然后使用其他可视化编程工具如VC,VB或者DELPHI编写一个界面,然后使用BUTTON通过调用函数WinExec或者ShellExecute或者CreateProcess来执行写的DOS程序。

这里使用转载来的别人的例子来说明如何调用:

三个SDK函数: WinExec,ShellExecute ,CreateProcess可以实现调用其他程序的要求,其中以WinExec最为简单,ShellExecute比WinExec灵活一些,CreateProcess最为复杂。

WinExec 两个参数,前一个指定路径,后一个指定显示方式。

ShellExecute 可以指定工作目录,并且还可以寻找文件的关联直接打开不用加载与文件关联的应用程序,ShellExecute还可以打开网页,启动相应的邮件关联发送邮件等等。

CreateProcess  一共有十个参数,不过大部分都可以用NULL代替,它可以指定进程的安全属性,继承信息,类的优先级等等。如果我们要得到足够多的关于新的进程的信息,控制新的进程的细节属性,若要达到这些目的,我们就需要使用CreateProcess函数了。

三个SDK函数( WinExec、ShellExec、CrateProcess )的语法:

(一)WinExec

这个函数最简单,只有两个参数,原型如下:

      UINT WinExec(

      LPCSTR lpCmdLine,   // 命令路径

      UINT uCmdShow      // 显示方式

      );

使用方法如下:

WinExec("Notepad.exe", SW_SHOW);  // 打开记事本

WinExec("D://Program Files//Test//Test.exe",SW_SHOWMAXIMIZED); // 以最大化的方式打开Test.exe

需要注意的是若用 SW_SHOWMAXMIZED 方式去加载一个无最大化按钮的程序,譬如Neterm,Calc 等等,就不会出现正常的 窗体,但是已经被加到任务列表里了。

这个函数只能打开exe文件。

需要的头文件:windows.h,winbase.h(前者是试验出来,后者是msdn上说明),另外,这两者的先后顺序不能变。

msdn上的说明:[url]http://msdn.microsoft.com/en-us/library/ms687393[/url](VS.85).aspx

(二)ShellExecute

原型如下:

      HINSTANCE ShellExecute(

      HWND hwnd,           //父窗口句柄

      LPCTSTR lpOperation,   //操作, 打开方式 "edit","explore","open","find","print","NULL"

      LPCTSTR lpFile,         //文件名,前面可加路径

      LPCTSTR lpParameters,   //参数

      LPCTSTR lpDirectory,    //默认文件夹

      INT nShowCmd          //显示方式

);

使用方法如下:

ShellExecute(NULL,"open","C://Test.txt",NULL,NULL,SW_SHOWNORMAL); // 打开C:/Test.txt 文件

ShellExecute(NULL, "open", "[url]http://www.google.com/[/url]",  NULL, NULL, SW_SHOWNORMAL); // 打开网页[url]www.google.com[/url]

ShellExecute(NULL,"explore", "D://C++",NULL,NULL,SW_SHOWNORMAL); // 打开目录D:/C++

ShellExecute(NULL,"print","C://Test.txt",NULL,NULL, SW_HIDE); // 打印文件C:/Test.txt

ShellExecute不支持定向输出。

这个函数可以打开任意文件,会调用系统注册的程序来打开对应后缀名的文件。

需要的头文件:windows.h,shellapi.h(前者是我试验出来的,后者是msdn说需要的)。另外这两者的先后顺序不能变。

msdn上说明:[url]http://msdn.microsoft.com/en-us/library/bb762153[/url](VS.85).aspx

(三)CreateProcess

原型如下:

      BOOL CreateProcess(

      LPCTSTR lpApplicationName, //执行程序名

      LPTSTR lpCommandLine,  // 参数行

      //下面两个参数描述了所创建的进程和线程的安全属性,如果为NULL则使用默认的安全属性

      LPSECURITY_ATTRIBUTES lpProcessAttributes,  // process security attributes

      LPSECURITY_ATTRIBUTES lpThreadAttributes,   // thread security attributes

      BOOL bInheritHandles,  // 继承标志

      DWORD dwCreationFlags, // 创建标志

      LPVOID lpEnvironment,  // 环境变量

      LPCTSTR lpCurrentDirectory,   // 运行该进程的初始目录

      LPSTARTUPINFO lpStartupInfo,  // 用于在创建子进程时设置各种属性

      LPPROCESS_INFORMATION lpProcessInformation //在进程创建后接受相关信息

      );

使用方法如下:

             PROCESS_INFORMATION pi;

                       STARTUPINFO si;

                       memset(&si,0,sizeof(si));

                       si.cb=sizeof(si);

                      si.wShowWindow=SW_SHOW;

                       si.dwFlags=STARTF_USESHOWWINDOW;

                       bool fRet=CreateProcess("D://putty.exe",NULL,NULL,FALSE,NULL,NULL,NULL,NULL,&si,&pi);

这个函数可以打开任意文件,会调用系统注册的程序来打开对应后缀名的文件。

需要的头文件:windows.h,winbase.h(前者是试验出来,后者是msdn上说明),另外,这两者的先后顺序不能变。

msdn上的说明:[url]http://msdn.microsoft.com/en-us/library/ms682425[/url](VS.85).aspx

可以看出,通过上面的几个不同的方法,都可以实现在应用程序中打开其他应用程序的目的,其中有些方法可能会麻烦一点,所以就需要我们根据不同的目的去选择最适合自己的方法去实现自己的目的!

关于三个SDK函数: WinExec, ShellExecute,CreateProcess 的其他注意事项:

1、定义头文件

这个是引用新函数都必须注意的内容。但是MS的头文件引用顺序有点怪,比如上面的三种清理。另外,如果用了预编译,那么记得在任何源程序中的#include “stdafx.h”之前的引用都会失效,从其后才生效。(不可否认,预编译有他的好处,尤其当程序很大的时候,但是任何好处都是要付出代价的)

2、定义路径

C++中所表示的路径要用 " // "而不是平常所用的" / ",所以以上三个函数表示路径都为:

Disk://Directory//...//File name

WinExec("D://Program Files//Test//Test.exe",SW_SHOWMAXIMIZED);

ShellExecute(NULL,"open","C://Test.txt",NULL,NULL,SW_SHOWNORMAL);

bool fRet=CreateProcess("D://putty.exe",NULL,NULL,FALSE,NULL,NULL,NULL,NULL,&si,&pi)

3、注意文件的路径

在程序a调用程序b的时候,b原来的默认的当前路径都会变成a的当前路径。所以,一定要注意。可以养成使用绝对路径的习惯,另外,记得打开文件之类的操作,一定要验证是否有错。

重点说一下第二种方法:

 

也许你觉得这样制作的程序交流不便,程序文件太多了,万一传递中丢失了什么文件,就会导致整个程序失效。下面介绍一下如何在VC设计的可视化程序里添加代码实现在DOS下运行。当然不是在真正的DOS下运行,但可以在WIN命令行里运行。在VC生成的默认项目里,一般都会有一个包含项目名字的APP的类,这个类里有一个函数InitInstance,MFC包装了API函数WinMain,在全局对象实例化后(App a; ),就进入WinMain函数,在里面由MFC这个框架已经写好对InitInstance()的调用,我们进入MFC的源码即可看到),因为该函数是一个虚函数,所以我们实例化继承CWinApp类的App类时,会自动调用App::InitInstance();修改该函数,把DOS的处理代码加入进来,然后RETURN,使其不进入创造可视化窗口函数中。

这里简单举例,我要搞一个读取BMP文件内容并写入别的文件的功能,写好了GUI部分后,我修改InitInstance函数。在其内添加如下内容:

CCommandLineInfo   cmdInfo;  
 ParseCommandLine(cmdInfo); 
 if (strlen((char *)m_lpCmdLine) != 0)
 {
  CFile file;
  file.Open(_T("aa.h"), CFile::modeCreate);
  file.Close();
  if (memcmp((unsigned char *)m_lpCmdLine, (unsigned char *)"-all", 4) == 0)
  {
   CmdlineOperateAll();
  }
  else
  {
   CmdlineOperate((LPSTR)(LPCTSTR)(char *)m_lpCmdLine);
  }
  return FALSE;
 }

这段内容一定要加在

#ifdef _AFXDLL
 Enable3dControls();   // Call this when using MFC in a shared DLL
#else
 Enable3dControlsStatic(); // Call this when linking to MFC statically
#endif

之后。我添加的代码是对命令行参数做分析,提供不同的操作。

两个操作的函数内容如下

void CmdlineOperate(CString filename)
{
 // TODO: Add extra validation here
 CString   FileName; 
 CFile   File;
 BITMAPINFO info;
 unsigned char *temp;
 BITMAPFILEHEADER file_head;
 CString m_ret;
 
 FileName = filename;
 if (File.Open(FileName.GetBuffer(_MAX_PATH), CFile::modeRead)) 
 {
  File.Read(&file_head, sizeof(BITMAPFILEHEADER));
  File.Read(&info.bmiHeader, sizeof(BITMAPINFOHEADER));
  
  long width=info.bmiHeader.biWidth;
  long height=info.bmiHeader.biHeight;
  CString com;
  CTime tm;
  CString curtime;
  tm=CTime::GetCurrentTime();
  curtime=tm.Format("日期:%Y年%m月%d日%x  */ /r/n");
  com.Format("/*    the size is :%dX%d, the char is : %s ; ",
   width, height, (File.GetFileName()).Left((File.GetFileName()).GetLength() - 4));
  com += curtime;
  m_ret += com;
  File.Seek(file_head.bfOffBits,CFile::begin);
  temp = new unsigned char[(File.GetLength()-file_head.bfOffBits) +1];
  File.Read(temp, (File.GetLength()-file_head.bfOffBits));
  for (int i = 0, j = 1; i < (int)(File.GetLength()-file_head.bfOffBits); i++, j++)
  {
   CString str;
   str.Format("0X%02X,", temp[i]);
   m_ret += str;
   printf("0X%02X", temp[i]);
   if (j % width == 0)
   {
    m_ret += _T("/r/n");
    printf("/n");
   }
  }
  
  File.Close();
  delete(temp);
 }
 File.Open(_T("aa.h"), CFile::modeNoTruncate|CFile::modeWrite);
 File.SeekToEnd();
 File.Write((LPSTR)(LPCTSTR)m_ret,m_ret.GetLength());
 File.Flush();
 File.Close();
}
void CmdlineOperateAll(void)
{
 // TODO: Add extra validation here
 WIN32_FIND_DATA fd;
 HANDLE hd=::FindFirstFile((LPCTSTR)"*.bmp",&fd);  //开始查找
 
 if(hd==INVALID_HANDLE_VALUE)
 {
  AfxMessageBox("没有找到文件");
  return;
 }
 
 CmdlineOperate(fd.cFileName);
 while(FindNextFile(hd,&fd)) //继续查找
 {
  CmdlineOperate(fd.cFileName);
 }
}

主要是查找文件,内容写入另一个头文件

 

抱歉!评论已关闭.