[转]   Visual Studio 2005 项目部署3[http://blog.youkuaiyun.com/yongping8204/archive/2007/06/26/1666893.aspx]

本文介绍如何使用Windows Installer技术发布.NET程序,并自动检测及安装.NET Framework。此外,还提供了使用本机代码实现这一过程的方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

把.NET程序部署到没有安装.NET Framwork的机器上

摘要
本文讨论了如何使用Windows Installer技术发布.NET程序,以及如何使用native代码判断目标机器上是否安装有.NET Framework; 如果没有,将自动安装.NET Framework然后安装作者自己的.NET程序。

目录
  • 本文内容
  • 制作自己的安装程序
  • 发布.NET Framework
  • 制作自己的native安装程序
  • 使用方法
  • 总结
  • 作者

本文内容
1. 使用VS.NET来制作安装程序。
2. 如何把.NET Framework 部署到目标机器。
3. 如何使用Native代码把.NET Framework和自己的安装程序制作为一个统一的安装程序。该程序做到如果目标机器上没有.NET Framework,将自动安装.NET Framework然后再自动安装作者自己的程序。

制作自己的安装程序
在VS.NET中,我们可以通过建立"Setup and Deployment Projects"项目,非常灵活方便的把自己的.NET程序制作为Windows Installer文件。比如,我们可以很方便的定制下面这些选项:
1. 是否在桌面上放置快捷方式。
2. 注册自己的文件类型,可以通过双击该文件来使用自己的程序打开。
3. 注册表的处理
在下面的MSDN站点,我们可以获得在VS.NET中通过Setup and Deployment Projects来制作自己的安装程序的示例:
http://msdn.microsoft.com/library/en-us/vsintro7/html/vbconDeploymentScenarios.asp

发布.NET Framework
.NET Framework 1.0提供一个用来重新部署.NET的exe文件:Dotnetfx.exe. 它包含了Common Language Runtime和其它.NET程序运行时必不可少的内容。
我们可以从下面的站点下载该exe文件:
http://msdn.microsoft.com/downloads/sample.asp?url=/MSDN-FILES/027/001/829/msdncompositedoc.xml
同时,我们也可以在VS.NET安装CD或者DVD中找到该文件。
我们可以通过多种方式来通过运行Dotnetfx.exe把.NET Framework部署到目标机器上:
1. 通过Microsoft Systems Management Server部署。
2. 通过Active Directory部署。
3. 使用第三方工具。
具体的信息,我们可以参阅下面的文章:
http://msdn.microsoft.com/library/en-us/dnnetdep/html/redistdeploy.asp

制作自己的native安装程序
如果我们要把自己的.NET程序发布到目标机器上,同时我们不确定该目标机器是否已经安装了.NET Framework, 那我们就需要自己设计一段unmanaged代码,来判断目标机器是否安装了.NET Framework, 如果没有,则运行Dotnetfx.exe安装.NET Framework, 然后利用Windows Installer安装自己的程序。
在MSDN的下面网页上,我们可以获得一个使用unmanaged C++实现的安装程序和它的源代码:
http://msdn.microsoft.com/downloads/default.asp?URL=/code/sample.asp?url=/msdn-files/027/001/830/msdncompositedoc.xml
1. 在CSettings class中,通过读取"settings.ini",获得您自己的MSI安装文件和dotnetfx.exe的路径,以及其他您自己的设置。(比如.NET Framework的语言版本)
GetCaptionText(void)
GetDialogText(void)
GetErrorCaptionText(void)
GetIniName(void)
GetProductName(void)
Parse()函数用来解析settings.ini文件。
2. 在Main.cpp文件中,全局函数FxInstallRequired()判断是否要在目标机器上安装.NET Framework. FxInstallRequired()会检测下面的注册表键值和dotnetfx.exe的版本和语言设置。
HKLM/SOFTWARE/Microsoft/.NETFramework/policy/v1.0
3.如果需要安装.NET Framework, 在全局函数ExecCmd()中调用下面的命令silent安装dotnetfx.exe:
dotnetfx.exe /q:a /c:"install /l /q"
4。 在ExecCmd()全局函数中调用下面的命令安装您自己的MSI文件:
msiexec /i <your MSI file> REBOOT=ReallySuppress

使用方法
如果我们使用上面的native代码作为自己的安装程序,那么我们可以通过下面的步骤把自己的.NET程序和它结合在一起:
1.把您自己的.NET程序制作成为Windows Installer (.MSI)文件。
2.打开"settings.ini"文件,在"Msi" key中设置您自己的.NET程序的MSI文件路径和文件名;在"FxInstallerPath" Key中,设置dotnetfx.exe的路径。具体的参数信息,您可以从下面文章中得到:
http://msdn.microsoft.com/library/default.asp?url=/library/en-us/dnnetdep/html/redistdeploy.asp
3. 把"setup.exe", "settings.ini", "dotnetfx.exe"和您自己的MSI安装文件,发送到目标机器上,然后运行"setup.exe", 安装程序会自动检测是否有.NET Framwork, 如果没有,将首先运行dotnetfx.exe。
通过上述步骤,您可以将自己的.NET程序成功的部署到没有安装.NET Framwork环境的机器上。

总结
通过上述步骤,您可以将自己的.NET程序成功的部署到没有安装.NET Framwork环境的机器上。 
/** * @file packet_portting.c * @author Lu Yongping (Lucas@hiwonder.com) * @brief 串口协议接口实现 * @version 0.1 * @date 2023-05-23 * * @copyright Copyright (c) 2023 * */ #include "global.h" #include "lwrb.h" #include "usart.h" #include "k210_packet.h" #include "cmsis_os2.h" #include <stdio.h> #include "lwmem_porting.h" #include "k210_packet_handle.h" #define K210_RX_FIFO_BUFFER_SIZE 2048 /* FIFO缓存长度 */ #define K210_RX_DMA_BUFFER_SIZE 256 /* 单个DMA缓存长度 */ /* 对外暴露的变量 */ struct k210_PacketController k210_packet_controller; /* 协议控制器实例 */ /* 内部函数 */ void k210_dma_receive_event_callback(UART_HandleTypeDef *huart, uint16_t length); /* 串口控制依赖的外部变量 */ extern osSemaphoreId_t k210_rx_not_emptyHandle; #if 1 /** * @brief 初始化packet 控制器对象 * @retval void */ void k210_packet_init(void) { memset(&k210_packet_controller, 0, sizeof(k210_packet_controller)); k210_packet_controller.state = K210_CONTROLLER_STATE_STARTBYTE1; k210_packet_controller.data_index = 0; /* DMA 接收缓存初始化 */ static uint8_t rx_dma_buffer1[K210_RX_DMA_BUFFER_SIZE]; static uint8_t rx_dma_buffer2[K210_RX_DMA_BUFFER_SIZE]; k210_packet_controller.rx_dma_buffers[0] = rx_dma_buffer1; k210_packet_controller.rx_dma_buffers[1] = rx_dma_buffer2; k210_packet_controller.rx_dma_buffer_size = K210_RX_DMA_BUFFER_SIZE; k210_packet_controller.rx_dma_buffer_index = 0; /* 接收 FIFO 初始化 */ k210_packet_controller.rx_fifo_buffer = LWMEM_CCM_MALLOC(K210_RX_FIFO_BUFFER_SIZE); k210_packet_controller.rx_fifo = LWMEM_CCM_MALLOC(sizeof(lwrb_t)); lwrb_init(k210_packet_controller.rx_fifo, k210_packet_controller.rx_fifo_buffer, K210_RX_FIFO_BUFFER_SIZE); } /** * @brief 开始串口协议接收 * @retval void */ void k210_start_recv(void) { HAL_UART_RegisterRxEventCallback(&huart2, k210_dma_receive_event_callback); /* 注册接收事件回调 */ /* 使用 ReceiveToIdle_DMA 进行接收, 该函数会在DMA缓存满时中断或在接收空闲时中断并触发接收事件回调 */ HAL_UARTEx_ReceiveToIdle_DMA(&huart2, k210_packet_controller.rx_dma_buffers[k210_packet_controller.rx_dma_buffer_index], K210_RX_DMA_BUFFER_SIZE); /* 开始接收 */ } /** * @brief 串口协议包接收事件回调 * 在 DMA 接收缓存满后或者空闲时触发本函数将接收到的数据压入接收FIFO缓存,再由接收任务完成解析和处理 * @param huart 串口实例 * @param Pos 接收到的字节数 * @retval void */ void k210_dma_receive_event_callback(UART_HandleTypeDef *huart, uint16_t length) { int cur_index = k210_packet_controller.rx_dma_buffer_index; /* 取得当前DMA缓存的索引号 */ k210_packet_controller.rx_dma_buffer_index ^= 1; if(length < K210_RX_DMA_BUFFER_SIZE) { HAL_UART_AbortReceive(&huart2); } HAL_UARTEx_ReceiveToIdle_DMA(&huart2, k210_packet_controller.rx_dma_buffers[k210_packet_controller.rx_dma_buffer_index], K210_RX_DMA_BUFFER_SIZE); lwrb_write(k210_packet_controller.rx_fifo, k210_packet_controller.rx_dma_buffers[cur_index], length); /* 将接收到的数据写入fifo ring */ printf("queue rec\n"); osSemaphoreRelease(k210_rx_not_emptyHandle); /* 置位接收缓存非空信号 */ } /** * @brief 串口协议包的接收任务入口 * 等待串口缓存非空信号量,然后从串口接收FIFO缓存中取出数据进行解析并处理 * 串口缓存非空信号量由串口接收事件置位 * @param argument 保留 * @retval void */ void k210_rx_task_entry(void *argument) { osSemaphoreAcquire(k210_rx_not_emptyHandle, osWaitForever); /* 默认信号不为零,先清除掉 */ __HAL_UNLOCK(&huart2); //注册回调函数 k210_packet_handle_init(); //开始接收 k210_start_recv(); printf("k210_rx_task_entry\n"); for(;;) { osSemaphoreAcquire(k210_rx_not_emptyHandle, osWaitForever); /* 等待接收缓存非空 */ printf("da\n"); k210_packet_recv(&k210_packet_controller); /* 处理接收数据 */ } } #endif
最新发布
03-27
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值