!

有三个API函数可以运行可执行文件WinExec、ShellExecute和CreateProcess。
1.CreateProcess因为使用复杂,比较少用。
2.WinExec主要运行EXE文件。如:WinExec('Notepad.exe Readme.txt', SW_SHOW);
3.ShellExecute不仅可以运行EXE文件,也可以运行已经关联的文件。
首先必须引用shellapi.pas单元:uses ShellAPI;
1).标准用法
  ShellExecute函数原型及参数含义如下:
  function ShellExecute(hWnd: HWND; Operation, FileName, Parameters,Directory: PChar; ShowCmd: Integer): HINST; stdcall;
  ●hWnd:用于指定父窗口句柄。当函数调用过程出现错误时,它将作为Windows消息窗口的父窗口。例如,可以将其设置为应用程序主窗口
句柄,即Application.Handle,也可以将其设置为桌面窗口句柄(用GetDesktopWindow函数获得)。
  ●Operation:用于指定要进行的操作。其中“open”操作表示执行由FileName参数指定的程序,或打开由FileName参数指定的文件或文件
夹;“print”操作表示打印由FileName参数指定的文件;“explore”操作表示浏览由FileName参数指定的文件夹。当参数设为nil时,表示执
行默认操作“open”。 
  ●FileName:用于指定要打开的文件名、要执行的程序文件名或要浏览的文件夹名。
  ●Parameters:若FileName参数是一个可执行程序,则此参数指定命令行参数,否则此参数应为nil或PChar(0)。
  ●Directory:用于指定默认目录。
  ●ShowCmd:若FileName参数是一个可执行程序,则此参数指定程序窗口的初始显示方式,否则此参数应设置为0。
  若ShellExecute函数调用成功,则返回值为被执行程序的实例句柄。若返回值小于32,则表示出现错误。
  上述仅仅是ShellExecute函数的标准用法,下面将介绍它的特殊用法。
2).特殊用法
  如果将FileName参数设置为“http:”协议格式,那么该函数将打开默认浏览器并链接到指定的URL地址。若用户机器中安装了多个浏览器
,则该函数将根据Windows 9x/NT注册表中http协议处理程序(Protocols Handler)的设置确定启动哪个浏览器。
  格式一:http://网站域名。
  如:ShellExecute(handle, ‘open’, http://www.neu.edu.cn’, nil, nil, SW_SHOWNORMAL);
  格式二:http://网站域名/网页文件名。
  如:ShellExecute(handle, ‘open’, http://www.neu.edu.cn/default.htm’,nil,nil,SW_SHOWNORMAL);


  如果将FileName参数设置为“mailt”协议格式,那么该函数将启动默认邮件客户程序,如Microsoft Outlook(也包括Microsoft
Outlook Express)或Netscape Messanger。若用户机器中安装了多个邮件客户程序,则该函数将根据Windows 9x/NT注册表中mailto协议处理
程序的设置确定启动哪个邮件客户程序。
  格式一:mailt
  如:ShellExecute(handle,‘open’, ‘mailt’, nil, nil, SW_SHOWNORMAL);打开新邮件窗口。
  格式二:mailt用户账号@邮件服务器地址
  如:ShellExecute(handle, ‘open’,‘ mailtwho@mail.neu.edu.cn’, nil, nil, SW_SHOWNORMAL);打开新邮件窗口,并自动填入收
件人地址。若指定多个收件人地址,则收件人地址之间必须用分号或逗号分隔开(下同)。
  格式三:mailt用户账号@邮件服务器地址?subject=邮件主题&body=邮件正文
  如:ShellExecute(handle, ‘open’, ‘ mailtwho@mail.neu.edu.cn?subject=Hello&Body=This is a test’, nil, nil,
SW_SHOWNORMAL);打开新邮件窗口,并自动填入收件人地址、邮件主题和邮件正文。若邮件正文包括多行文本,则必须在每行文本之间加入换行

转义字符%0a。
例子(Delphi):
在一个应用程序调用c:\Project1.exe;
ShellExecute(handle, 'open','c:\Project1.exe','字串内容',nil, SW_SHOWNORMAL);
在Project1.exe里可以调用:
procedure Tform1.formCreate(Sender: TObject);
var
i: integer;
begin
for i:=1 to paramcount do
if ParamStr(i)<>'' then
    showmessage(ParamStr(i));
end;
最后的那个参数,为窗口指定可视性方面的一个命令。请用下述任何一个常数:
SW_HIDE 隐藏窗口,活动状态给令一个窗口
SW_MINIMIZE 最小化窗口,活动状态给令一个窗口
SW_RESTORE 用原来的大小和位置显示一个窗口,同时令其进入活动状态
SW_SHOW 用当前的大小和位置显示一个窗口,同时令其进入活动状态
SW_SHOWMAXIMIZED 最大化窗口,并将其激活
SW_SHOWMINIMIZED 最小化窗口,并将其激活
SW_SHOWMINNOACTIVE 最小化一个窗口,同时不改变活动窗口
SW_SHOWNA 用当前的大小和位置显示一个窗口,不改变活动窗口
SW_SHOWNOACTIVATE 用最近的大小和位置显示一个窗口,同时不改变活动窗口
SW_SHOWNORMAL 与SW_RESTORE相同

<think>好的,我需要回答用户关于Kotlin中非空断言运算符(!!)的用法和最佳实践。首先,我得回忆一下Kotlin的空安全机制。Kotlin通过可空类型和不可空类型来避免空指针异常,变量默认是不可空的,如果要允许空值,必须显式声明为可空类型,比如String?。 非空断言运算符!!的作用是将可空类型强制转换为非空类型。当开发者确定某个可空变量在此时不为null时,可以使用!!,但如果变量实际上为null,就会抛出NPE。这个运算符虽然方便,但滥用会增加风险,所以最佳实践通常是建议尽量避免使用!!,优先使用安全调用操作符(?.)、Elvis运算符(?:)或者智能转换。 接下来,我需要整理!!的具体使用场景。例如,当与某些返回可空类型的Java代码交互时,或者在单元测试中明确知道某个值不为null的情况下,可能不得不使用!!。但大部分情况下,应该考虑是否有更安全的方式替代。 然后,要列举使用!!的风险,比如潜在的运行时崩溃,代码可维护性降低等。同时,给出替代方案,比如lateinit修饰符用于延迟初始化,或者使用let函数结合安全调用。 还需要考虑引用资料中的内容,用户提供的引用[1]提到了Kotlin的基础知识,包括空安全部分,可以结合这些内容来增强回答的权威性。例如,引用中提到的Nullability部分可能涉及如何处理可空类型,这和非空断言运算符直接相关。 最后,生成相关问题,可能需要覆盖替代方案、与Java互操作时的处理、以及如何处理可能潜在null的情况等。确保问题对用户深入学习有帮助。 现在,我需要将以上思考整理成结构化的回答,分点说明用法、风险、最佳实践,并给出代码示例。同时,确保使用正确的中文术语,避免技术术语错误。另外,注意引用标识的添加,比如在提到空安全机制时,引用提供的资料[^1]。</think>### Kotlin非空断言运算符(!!)用法与最佳实践 Kotlin通过**空安全机制**从语言层面减少空指针异常(NPE)的发生。非空断言运算符`!!`是处理可空类型的特殊工具,需谨慎使用。 #### 一、基本用法 ```kotlin val name: String? = getNameFromNetwork() val length = name!!.length // 显式断言非空 ``` - 作用:将可空类型强制转换为非空类型 - 效果:若变量为`null`立即抛出`NullPointerException` - 类型转换:`String?` ➔ `String` #### 二、适用场景 1. **与Java代码交互时** ```kotlin // Java方法:public @Nullable String getValue() val result = JavaClass().value!! // 明确知晓返回值非空时 ``` 2. **单元测试环境** ```kotlin @Test fun `测试非空数据`() { val testData = mockService.getData()!! assertTrue(testData.isValid()) } ``` 3. **契约式编程场景** ```kotlin fun processData(data: Data?) { require(data != null) // 前置条件检查 val size = data!!.payload.size // 此时已通过null检查 } ``` #### 三、风险警示 1. **代码异味指标**:项目中出现`!!`次数越多,空安全防护越薄弱 2. **崩溃风险**:错误使用时100%触发NPE 3. **维护成本**:增加后续代码修改时遗漏null检查的风险 #### 四、最佳实践 1. **优先安全调用** ```kotlin name?.let { println(it.length) } // 安全调用+作用域函数 ``` 2. **使用Elvis运算符** ```kotlin val length = name?.length ?: 0 // 提供默认值 ``` 3. **lateinit延迟初始化** ```kotlin lateinit var config: Configuration fun init() { config = loadConfig() // 确保初始化后才使用 } ``` 4. **类型智能转换** ```kotlin if (name != null) { println(name.length) // 自动智能转换 } ``` 5. **require/check函数** ```kotlin fun save(data: Data?) { checkNotNull(data) database.save(data) // 此时data已确认为非空 } ``` #### 五、代码审查建议 1. 将`!!`使用纳入代码评审重点 2. 配置Lint规则检测`!!`使用 3. 统计项目中的`!!`密度指标(建议低于0.1%) [^1]: Kotlin的空安全机制通过类型系统在编译期检测潜在空指针风险,这是其相较于Java的重要改进特性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值