1、define预处理指令
这种方式很简单,只是简单地将main字符串用宏来代替,或者使用##拼接字符串。示例程序如下:

#include <stdio.h>

#define begin main

int begin(void)

{

printf(
"Hello, World!\n");

return
;

}

#include <stdio.h>

#define begin m##a##i##n

int begin(void)

{

printf(
"Hello, World!\n");

return
;

}

严格来说,这种方式只算是一种技巧......

2、_start函数
_start函数是C程序的入口函数,会调用main函数。在调用main函数之前,会先执行_start函数分配必要的资源,然后再调用main函数。但是在用gcc编译程序时可以使用-nostartfiles选项来重写_start函数。示例程序如下:
#include <stdio.h>


#include <stdlib.h>

_start(void) {

printf(
"Hello, World!\n");


exit(
);

}

编译上面的程序的命令为:
gcc
-nostartfiles _start.c
-o a.out

反汇编生成的可执行程序,如下所示:

a.out
: file format elf64
-x86
-

Disassembly of section .plt
:

<puts@plt
-0x10
>
:

: ff
ea

00 pushq 0x2001ea(
%rip)
# 600510 <_GLOBAL_OFFSET_TABLE_+0x8>

: ff
ec

00 jmpq
*0x2001ec(
%rip)
# 600518 <_GLOBAL_OFFSET_TABLE_+0x10>


40032c
: 0f
1f
00 nopl 0x0(
%rax)

<puts@plt
>
:

: ff
ea

00 jmpq
*0x2001ea(
%rip)
# 600520 <_GLOBAL_OFFSET_TABLE_+0x18>

:
00 00 00 00 pushq $0x0


40033b
: e9 e0 ff ff ff jmpq

<puts@plt
-0x10
>

<
exit@plt
>
:

: ff
e2

00 jmpq
*0x2001e2(
%rip)
# 600528 <_GLOBAL_OFFSET_TABLE_+0x20>

:

00 00 00 pushq $0x1


40034b
: e9 d0 ff ff ff jmpq

<puts@plt
-0x10
>

Disassembly of section .text
:

<_start
>
:

:
push
%rbp

:

e5 mov
%rsp,
%rbp

: bf

00 mov $0x400368,
%edi

: e8 d2 ff ff ff callq

<puts@plt
>


40035e
: bf 00 00 00 00 mov $0x0,
%edi

: e8 d8 ff ff ff callq

exit@plt

上面的结果是完整的反汇编结果,我们可以看到_start函数中只有我们调用printf和exit函数相关的一些指令,并且.txt段中只有_start函数,没有看到main函数。如果将源代码中的_start替换为main,重新编译程序,反汇编的结果中会看到_start函数会调用到main。
另外还有一点需要注意,因为这里重写了_start函数,所以gcc为默认的main函数准备的清理动作就没用上,所以如果退出的时候直接使用return,会导致程序崩溃。所以这里要使用exit()来退出程序。具体的原因可以参见这篇文章
3、gcc的-e选项
示例程序如下:

#include <stdio.h>


#include <stdlib.h>

int nomain(int i, int j, int k) {

printf(
"Hello, World!\n");


exit(
);

}

将上面的程序保存为m.c,编译命令如下所示:
gcc
-nostartfiles
-e nomain m.c
-o a.out

继续使用objdump反汇编生成的可执行程序,结果如下:

a.out
: file format elf64
-x86
-

Disassembly of section .plt
:

<puts@plt
-0x10
>
:

: ff
f2

00 pushq 0x2001f2(
%rip)
# 600518 <_GLOBAL_OFFSET_TABLE_+0x8>

: ff
f4

00 jmpq
*0x2001f4(
%rip)
# 600520 <_GLOBAL_OFFSET_TABLE_+0x10>


40032c
: 0f
1f
00 nopl 0x0(
%rax)

<puts@plt
>
:

: ff
f2

00 jmpq
*0x2001f2(
%rip)
# 600528 <_GLOBAL_OFFSET_TABLE_+0x18>

:
00 00 00 00 pushq $0x0


40033b
: e9 e0 ff ff ff jmpq

<puts@plt
-0x10
>

<
exit@plt
>
:

: ff
ea

00 jmpq
*0x2001ea(
%rip)
# 600530 <_GLOBAL_OFFSET_TABLE_+0x20>

:

00 00 00 pushq $0x1


40034b
: e9 d0 ff ff ff jmpq

<puts@plt
-0x10
>

Disassembly of section .text
:

<nomain
>
:

:
push
%rbp

:

e5 mov
%rsp,
%rbp

:

ec
sub $0x10,
%rsp

:

7d fc mov
%edi,
-0x4(
%rbp)


40035b
:

f8 mov
%esi,
-0x8(
%rbp)


40035e
:

f4 mov
%edx,
-0xc(
%rbp)

: bf

00 mov $0x400375,
%edi

: e8 c5 ff ff ff callq

<puts@plt
>


40036b
: bf 00 00 00 00 mov $0x0,
%edi

: e8 cb ff ff ff callq

<
exit@plt
>

从上面我们可以看到指定的nomain函数位于.text段的开始位置,同样在函数结束的时候没有gcc为main函数准备的清理动作,所以在这里也只能使用exit()来退出程序,而不能使用return。
4、nostartfiles选项
前面已经多次使用了该选项,不过都是配合其他选项使用的,这个选项也可以单独使用,其含义为"Do not use the standard system startup files when linking"。
示例程序如下:

#include <stdio.h>


#include <stdlib.h>

void func() {

printf(
"I am func....\n");

}

int nomain1(int i, int j, int k) {

func();

printf(
"%s: Hello, World!\n", __func__);


exit(
);

}

上面的程序保存为k.c,然后使用下面的命令编译:

[root@CentOS_190
~]
# gcc -nostartfiles p.c


/usr
/bin
/ld
: warning
: cannot find entry symbol _start; defaulting to

在单独使用nostartfiles选项时会报警告,生成的可执行程序可以执行,但是会产生段错误,去掉对func()函数的调用就不会产生段错误了。将生成的可执行程序反汇编,和使用前面的方法生成可执行程序的反汇编结果比较,发现除了函数名不一样外,没有其他区别,不知道为什么会产生段错误。知道的麻烦告知一声,拜谢!

C语言中没有main函数生成可执行程序的几种方法的更多相关文章

  1. C语言中存储多个字符串的两种方式

    C语言中存储多个字符串的两种方式 方式一    二维字符串数组 声明: char name[][] = { "Justinian", "Momo", " ...

  2. 20155212 C语言实现linux下pwd命令的两种方法

    20155212 C语言实现linux下pwd命令的两种方法 学习pwd命令 通过man pwd命令查看 pwd [OPTION],一般不加参数 -P显示当前目录的物理路径 -L显示当前目录的连接路径 ...

  3. Java中取小数点后两位(四种方法)

    摘自http://irobot.iteye.com/blog/285537 Java中取小数点后两位(四种方法)   一 Long是长整型,怎么有小数,是double吧     java.text.D ...

  4. linux中删除文件内空白行的几种方法。

    linux中删除文件内空白行的几种方法 有时你可能需要在 Linux 中删除某个文件中的空行.如果是的,你可以使用下面方法中的其中一个.有很多方法可以做到,但我在这里只是列举一些简单的方法. 你可能已 ...

  5. (转载)Java中如何遍历Map对象的4种方法

    在Java中如何遍历Map对象 How to Iterate Over a Map in Java 在java中遍历Map有不少的方法.我们看一下最常用的方法及其优缺点. 既然java中的所有map都 ...

  6. 服务器文档下载zip格式 SQL Server SQL分页查询 C#过滤html标签 EF 延时加载与死锁 在JS方法中返回多个值的三种方法(转载) IEnumerable,ICollection,IList接口问题 不吹不擂,你想要的Python面试都在这里了【315+道题】 基于mvc三层架构和ajax技术实现最简单的文件上传 事件管理

    服务器文档下载zip格式   刚好这次项目中遇到了这个东西,就来弄一下,挺简单的,但是前台调用的时候弄错了,浪费了大半天的时间,本人也是菜鸟一枚.开始吧.(MVC的) @using Rattan.Co ...

  7. linux中快速清空文件内容的几种方法

    这篇文章主要介绍了linux中快速清空文件内容的几种方法,需要的朋友可以参考下 $ : > filename $ > filename $ echo "" > f ...

  8. (转)在网页中JS函数自动执行常用三种方法

    原文:http://blog.sina.com.cn/s/blog_6f6b4c3c0100nxx8.html 在网页中JS函数自动执行常用三种方法 在网页中JS函数自动执行常用三种方法 在HTML中 ...

  9. Java 获取*.properties配置文件中的内容 ,常见的两种方法

    import java.io.InputStream; import java.util.Enumeration; import java.util.List; import java.util.Pr ...

随机推荐

  1. 10条影响CSS渲染速度的写法与建议

    1.*{} #zishu *{} 尽量避开由于不同浏览器对HTML标签的解释有差异,所以最终的网页效果在不同的浏览器中可能是不一样的,为了消除这方面的风险,设计者通常会在CSS的一个始就把所有标签的默 ...

  2. JavaScript基础(简介、语法)

    一.JavaScript简介 1.JavaScript是个什么东西? 它是个脚本语言,需要有宿主文件,它的宿主文件是HTML文件. 2.它与Java什么关系? 没有什么直接的联系,Java是Sun公司 ...

  3. Microsoft SQL Server 数据库 错误号大全

    panchzh :Microsoft SQL Server 数据库 错误号大全0 操作成功完成. 1 功能错误. 2 系统找不到指定的文件. 3 系统找不到指定的路径. 4 系统无法打开文件. 5 拒 ...

  4. Winform获取当前程序名称或路径

    以下几种方法获取当前程序名称或路径: // 获取程序的基目录. System.AppDomain.CurrentDomain.BaseDirectory // 获取模块的完整路径.      // 获 ...

  5. c语言中的字符数组与字符串

    1.字符数组的定义与初始化 字符数组的初始化,最容易理解的方式就是逐个字符赋给数组中各元素. char str[10]={ 'I',' ','a','m',' ',‘h’,'a','p','p','y ...

  6. poj 1041 John's trip 欧拉回路

    题目链接 求给出的图是否存在欧拉回路并输出路径, 从1这个点开始, 输出时按边的升序输出. 将每个点的边排序一下就可以. #include <iostream> #include < ...

  7. [LeetCode]题解(python):084-Largest Rectangle in Histogram

    题目来源: https://leetcode.com/problems/largest-rectangle-in-histogram/ 题意分析: 给定一个数组,数组的数字代表这个位置上的bar的高度 ...

  8. IOS 表视图(UITableVIew)的使用方法(5)表视图的编辑功能(删除)

    默认的,如果表视图支持编辑,那用户可以通过两种方式来删除某些行,其一为单击左侧的红色按钮后行右侧显示“Delete”按钮,其二为在单元行上的手指向左滑动,“Delete”按钮也会出现供用户单击.无论哪 ...

  9. 关于JVM的GC机制

    GC优点: 1.提高生产率,不用逐行检查内存是否释放. 2.Java安全策略的一部分,不会使用户错误释放内存而导致JVM崩溃. GC算法基本两点: 1.检测出垃圾对象. 2.回收垃圾对象,释放相应堆空 ...

  10. 关于 free() 函数用法的若干疑问

    <C语言参考手册>中关于 free() 函数有如下描述. (1)free() 函数的原型 void free(void *ptr); (2)free 函数对以前由 malloc.callo ...