Swift程序入口深度分析

1.swift为什么不需要main

在c/c++及其它语言中都有一个main函数,程序从main作为起点,开始执行程序,如下:

int main(int argc, const char * argv[]) {
    printf("Hello, World!\n");
    return 0;
}

main函数实际上是一个特殊的函数,为了能找到程序入口,大多楼语言都约定main()函数作为入口。那么为什么在Swift中没有这样的一个函数呢?先看一下官方的解释

Code written at global scope is used as the entry point for the program, so you don’t need a main function. 

大体意思是,在main.swift中的代码是在全局作用域下,这些代码直接作为整个项目的入口,所以这里并不需要main函数。

2.C与Swift的入口对比

在ios/osx系统中,C/Objective-c项目也有main()函数的约定,在项目入口main()代码块打断点

从上图可以看出在执行程序前,先调用了**start()**方法,而后再执行约定的**main()**方法,这种我们能够很好理解,那么在swift下情况如何呢?

从上图可以看出Swift的执行顺序是start()->main()->top_level_code(),相对于C语言项目,多出来top_level_code(),在main.swift中的(非声明)代码会直接作为top_level_code()代码执行。此处要注意在Swift语言本身并不需要入口函数,程序入口是指定为main.swift中的非声明代码。在具体编译环节,ios/osx的入口均采用约定的main()函数,为了兼容以前的入口方法,将Swift语言程序在编译环节处理成隐式入口函数top_level_code(),再由main()调用。

3.代码top_level_code()

在官方解释

> Code written at global scope is used as the entry point for the program, so you don’t need a main function.

提及到全局作用域,main.swift中的声明(如变量、常量、类、结构体、枚举)类代码,会作为全局作用域,在程序的任何都方都可以使用。而其中的非声明(赋值、for循环、if语句等)代码,会作为top_level_code()中代码来执行。这里注意,声明类代码与非声明类代码的作用域并不相同。声明类的作用域是全局作用域,而非声明类代码为top_level_code()作用域。

这里特别注意,只有在main.swift中的代码才可以作为top_level_code来执行。

而在其它文件中,是不能直接在文件中含有非声明类的语句,只能含有声明类的代码。

4.偷天换日,替换隐式入口函数top_level_code()

在编译环节,编译器将main.swift非声明代码作为top_level_code()来执行,为了能够让程序执行我们的入口函数,而不是main.swift代码,需要声明并实现这个特殊的top_level_code()函数

void top_level_code();

下一步如何让编译程序指定我们写的top_level_code(),这里需要注意

* main.swift文件不能删除,如果删除程序直接不能编译通过

* 为了能让编译程序认可我们写的top_level_code()函数,我们需要在main.swift文件中主动使用一次top_level_code()

这里的使用是指两种情况:

1. 在main.swift直调用我们写的top_level_code()函数

2. 在main.swift的声明类、结构体等的方法中调用top_level_code()函数

注意

这两种方法都会导致main.swift中的所有代码不再执行。此处在main.swift加入上述代码的作用就是让编译器改调用我们写的top_level_code函数

具体可以直接下载项目来研究

为了测试里面用到了

1. Swift项目调用C及Objective-c代码的办法

2. Objective-c代码调用Swift代码

3. 类的声明

这部分的知识会在后面一一讲解到,大家只需要了解入口的原理即可。

5.Swift打印入口参数

在C言语中的**main**函数中,有两个参数

* argc:  命令行中字符串数

* argv:  指向字符串的指针数组

这两个参数在Swift中被声明为全局变量,分别为

* C_ARGC

* C_ARGV

我们也可以在Swift中将入口参数打印出来,注意C中类型与Swift的类型转换

//将C语言int型转换为Swift中的Int
let cout = Int(C_ARGC)
println("all->\(cout)")

let end = cout - 1
for index in 0...(end) {
    //获取指定C语言字符串,并将C字符串转换为Swift的String类型
    let str = String.fromCString(C_ARGV[index])
    println("\(str)")
}

点击运行完成后->将项目Products目录入的文件直接拖入终端中->在后面添加空格"aaa"空格"bbb",类似下面的格式

/Users/mac/Library/Developer/Xcode/DerivedData/ILHelloWorld-fvywvzypiomcffbiuxdxwwdaeued/Build/Products/Debug/ILHelloWorld "aaa" "bbb"

打印运行结果如下:

all->3
Optional("/Users/mac/Library/Developer/Xcode/DerivedData/ILHelloWorld-fvywvzypiomcffbiuxdxwwdaeued/Build/Products/Debug/ILHelloWorld")
Optional("aaa")
Optional("bbb")

其中第一个参数为默认的程序路径,第二个及第三个参数为我们在上面输入的**aaa**及**bbb**,加起来共3个参数

时间: 2024-12-11 19:32:31

Swift程序入口深度分析的相关文章

开源GUI-Microwindows之程序入口分析

*************************************************************************************************************************** 作者:EasyWave                                                               时间:2014.10.05 类别:开源GUI系统-Microwindows之程序入口分析      声明

Hadoop源码分析—— Job任务的程序入口

这篇文章大致介绍Hadoop Job的程序是如何启动的. 通常用Java编写的Hadoop MapReduce程序是通过一个main方法作为程序的整个入口,如下: public static void main(String[] args) throws Exception { int res = ToolRunner.run(new Configuration(), new CalculateSumJob(),args); System.exit(res);} 可以看到这个Job任务的MapR

Java程序运行原理分析

class文件内容 class文件包含Java程序执行的字节码 数据严格按照格式紧凑排列在class文件的二进制流,中间无分割符 文件开头有一个0xcafebabe(16进制)特殊的标志 JVM运行时数据区 线程独占: 每个线程都会有它独立的空间,随线程的生命周而创建和销毁 线程共享: 所有线程都能访问这块内存数据,随虚拟机或GC而创建和销毁 方法区 方法区是各个线程共享的内存区域 用于存储已被虚拟机加载的类信息, 常量,静态变量, 即时编译后的代码等数据 虽然Java虚拟机规范把方法区描述为堆

Socket与系统调用深度分析

Socket与系统调用深度分析 可以想象的是,当应用程序调用socket()接口,请求操作系统提供服务时,必然会系统调用,内核根据发起系统调用时传递的系统调用号,判断要执行的程序,若为socket对应的编号,则执行socket对应的中断服务程序.服务程序内部,又根据你要请求的不同服务,来执行不同服务对应的处理程序.当处理结束,执行返回,从中断服务程序到发起中断的int 0x80,再到用户态我们执行的用户程序,层层返回,socket()也就执行完毕了. 本次,我们关心三个问题: 1.应用程序如何如

深度分析LINUX环境下如何配置multi-path

首先介绍一下什么是多路径(multi-path)?先说说多路径功能产生的背景,在多路径功能出现之前,主机上的硬盘是直接挂接到一个总线(PCI)上,路径是一对一的关系,也就是一条路径指向一个硬盘或是存储设备,这样的一对一关系对于操作系统而言,处理相对简单,但是缺少了可靠性.当出现了光纤通道网络(Fibre Channle)也就是通常所说的SAN网络时,或者由iSCSI组成的IPSAN环境时,由于主机和存储之间通过光纤通道交换机或者多块网卡及IP来连接时,构成了多对多关系的IO通道,也就是说一台主机

【转】iOS平台的应用程序调试与分析

转自:看雪学院的文章 iOS平台的应用程序调试与分析 作者:zhuliang转载请保证文章完整并注明来自看雪或cd-team 本文阐述如何在iOS平台上对应用程序进行调试与分析,旨在指导新手分析iOS程序,高手请无视.内容包括软件硬件的准备.代码的解密.符号信息的获取.用gdb调试等,最后以京东LeBook为例子进行演示.1.为什么要进行调试与分析研究iOS程序有很多用处,比如:找bug或者漏洞,想知道某程序有没有漏洞或者bug.某程序能实现某功能,我想知道如何实现,如ios6发短信功能,还有比

java集合框架(1) hashMap 简单使用以及深度分析(转)

java.util 类 HashMap<K,V>java.lang.Object  java.util.AbstractMap<K,V>      java.util.HashMap<K,V>类型参数:K - 此映射所维护的键的类型V - 所映射值的类型所有已实现的接口: Serializable, Cloneable, Map<K,V> 直接已知子类: LinkedHashMap, PrinterStateReasons -----------------

转:[gevent源码分析] 深度分析gevent运行流程

[gevent源码分析] 深度分析gevent运行流程 http://blog.csdn.net/yueguanghaidao/article/details/24281751 一直对gevent运行流程比较模糊,最近看源码略有所得,不敢独享,故分享之. gevent是一个高性能网络库,底层是libevent,1.0版本之后是libev,核心是greenlet.gevent和eventlet是亲近,唯一不同的是eventlet是自己实现的事件驱动,而gevent是使用libev.两者都有广泛的应

[iOS] 使用xib做为应用程序入口 with Code

[iOS] 使用xib做为应用程序入口 with Code 前言 开发iOS APP的时候,使用storyboard能够快速并且直觉的建立用户界面.但在多人团队开发的情景中,因为storyboard是以单一档案的方式存在,很容易造成签出.签入时,档案被锁定.档案合并冲突等等问题的发生.这时开发人员可以选择使用xib做为用户接口的开发单位,将用户接口拆散为独立存在的xib档案,分散团队成员同时编辑同一文件的风险. 在Xcode中默认是以storyboard做为应用程序入口,变更为使用xib做为应用