滴水逆向联盟

标题: 一个值得学习的 扫描器c++源码 [打印本页]

作者: dishui001    时间: 2014-5-14 19:50
标题: 一个值得学习的 扫描器c++源码
//首先 引入需要用到的库文件
#include <stdio.h>// 标准输入输出流
#include <winsock2.h>//socket套接字  网络发包性质的 网络通讯
#include <time.h>//时间的头文件
#pragma comment(lib,"ws2_32.lib")//编译链接用的库文件
//只有包含了这些头文件 才能引用里面的 函数方法


char *host;//定义字符型 变量  指针  保存主机IP
int threadnum,maxthread,totalport;//定义线程 端口
long nowport ;//当前端口
TIMEVAL timeout; //定义一个时间结构体的对象
FD_SET    mask;
void display(void)
{
    static int play=0;
    char *plays[12]=
    {
             " | ",
            " / ",
            " - ",
            " \\ ",
            " | ",
            " / ",
            " - ",
            " // ",
            " | ",
            " / ",
            " - ",
            " // ",
    };
printf("=%s=\t%d 线程 %d %s 完成...\r",plays[play],threadnum,nowport*100/(totalport+1),"%");
play=(play==11)?0:play+1;
Sleep(1);

}

void WaitThreadEnd(void)
{
    Sleep(1000);
    printf("\n 线程结束中.....\n");
    while (threadnum>0)
    {
        Sleep(1);
        printf("=|=\t%d 线程数 \r",threadnum);
    }
}
//最终的回调函数 线程
DWORD WINAPI ThreadFunc(LPVOID lp)
{
   int port=*(DWORD*)lp;//得到端口
   SOCKET sockfd;
   struct sockaddr_in addr;// 套接字 地址的一个 结构体的 对象
   u_long value;//定义一个 无符号的long;
   addr.sin_family=AF_INET;
   addr.sin_addr.s_addr=inet_addr(host);
   value=1;//非阻塞模式的参数
   sockfd=socket(AF_INET,SOCK_STREAM,0);//初始化 流模式TCP
   if (sockfd==INVALID_SOCKET)
   {
       printf("套接字错误\n");
       InterlockedExchangeAdd((long*)&threadnum,-1);
       //意思就是 互斥  在线程中  确保数据被一个线程修改。。
       return 0;
   }
   ioctlsocket(sockfd,FIONBIO,&value);
   //套接字的一个 非阻塞模式
   addr.sin_port=htons(port); //转换端口
   connect(sockfd,(struct sockaddr *)&addr,sizeof(addr));
   FD_ZERO(&mask);
   FD_SET(sockfd,&mask);
   value=select(0,NULL,&mask,NULL,&timeout);
   if (value==0||value==-1)
   {
       closesocket(sockfd);//超时了 关闭
       Sleep(50);
       InterlockedExchangeAdd((long *)&threadnum,-1);
       return 0;
   }
   else
   {
       shutdown(sockfd,0);
       printf("\t\t 发现 %d 打开\r\n",port);
       closesocket(sockfd);
       Sleep(50);
       InterlockedExchangeAdd((long *)&threadnum,-1);
   }
   return 0;


}




void usage(char *name)//传递一个参数进来 char类型
{
printf("\t========================猫猫专用扫描器=============================\n");
printf("\t========================QQ 图上有 2012.6.2========================\n\n");
printf("用法: 程序名 ip 开始端口-结束端口 线程\n\n",name);
printf("\t 程序名:  您的程序名%s\n",name);
printf("\t IP:      你想要扫描的IP地址\n");
printf("\t 端口:    (1-65535之间的数)\n");
printf("\t 线程:    最大开到1000\n\n");
printf("使用例子: %s 192.168.1.1 1-65535 500 \n",name);
}

//定义main 主入口函数
void main(int argc,char **argv)
{
//初始化套接字
    WSADATA ws;//初始化WSADATA
    char *p;//一个指向端口的指针 就是我们传递的端口
    int startport,endport;//定义两个整形变量 来保存开始 结束端口
    clock_t start,end; // 检测程序扫描用了多少时间
    float costtime;//花费的时间 float类型 浮点
    if (argc!=4)//如果参数不等于4个
    {
        usage(argv[0]);//调用该函数 。。usage 传递 argv[0]
        //第一个参数 默认指向的 是程序的名字
        return;
    }
    long lresult; //定义长整型的变量
    lresult=WSAStartup(MAKEWORD(2,2),&ws);// 进行开始wsadata
    p=argv[2];//指针指向第三个参数 端口 第一个是程序名 第二个 IP 第三个才是端口
    if (strstr(argv[2],"-"))  //查找这个- 线 查找到之后 进入if体
    {
       startport=atoi(argv[2]);//比如: 222-333
       for (;*p;)if(*(p++)=='-')break;

           //得到的是p的值  其实就是无限循环 *p取出指针对应的值
       endport=atoi(p);
       if (startport<1 || endport>65535)
       {
           printf("端口你写错了吧 1-65535 之间的一个数\n");
           return;
       }
    }
    host=argv[1];//保存主机IP
    maxthread=(atoi(argv[3]-1)>999)?999:atoi(argv[3])-1;
    //三目运算 如果大于1000那么 等于999 否则 等于用户输入的
    usage(argv[0]);//又输出了一次菜单

    timeout.tv_usec=0;//初始化时间0
    if (maxthread>500)timeout.tv_sec=2;
    //线程大 那么时间 给的多点
    else timeout.tv_sec=1;
    start=clock();
    totalport=endport-startport;
    //一共多少个端口
    for (int port=startport;port<endport;port++,nowport++,threadnum++)
    {
        display();
        while (threadnum>maxthread)
        {
            Sleep(10);
        }
        CreateThread(NULL,0,ThreadFunc,&port,0,NULL);//创建线程
    }
    end=clock();
    costtime=(float)(end-start)/CLOCKS_PER_SEC;//换算成时间格式
    printf("消耗时间: %f 秒",costtime);
    printf("\n\n");
    WaitThreadEnd();//等待线程结束
    WSACleanup();//关闭套接字
    return ;//返回







欢迎光临 滴水逆向联盟 (http://dtdebug.com/) Powered by Discuz! X3.2