草庐IT

Android 套接字每 20 秒卡住一次

coder 2023-09-19 原文

我在 Action 游戏中使用 TCP 套接字连接,我们使用锁步同步,客户端每秒将接收 20 - 40 个数据包。游戏在PC上运行时可以正常运行,但在Android设备上运行时, socket 每隔20秒就会卡住

Server sends 5 packets per second

我曾尝试使用 Unity3D 的 C# 套接字、Android Java 套接字和 Android native C 套接字以及阻塞/非阻塞模式,每个数据包的小/大(1 字节/100 字节)数据,每个数据包更少/更多(5/50)第二,使用单线程/主线程,在多个Android设备上,它们都有同样的问题。

PS:20 秒的持续时间似乎是基于设备,而不是我的应用程序或连接;这意味着如果上一次卡住发生在 1:00:00,下一次卡住将发生在 1:00:20,即使我们重新连接或重新启动应用程序也是如此。

Android 原生 C 代码:

    extern "C" JNIEXPORT int JNICALL
Java_com_example_ymoon_sockettest_MainActivity_connect(JNIEnv *env, jobject /* this */)
{
    __android_log_print(ANDROID_LOG_INFO, "CSocket", "Connecting...");

    int sock = socket(AF_INET, SOCK_STREAM, 0);
    if (sock < 0) return -1;

    struct sockaddr_in server;
    memset(&server, 0, sizeof(server));

    server.sin_family = AF_INET;
    server.sin_port   = htons(12350);
    server.sin_addr.s_addr = inet_addr("192.168.3.66");

    if (connect(sock, (struct sockaddr*)&server, sizeof(server)) < 0)
    {
        close(sock);
        return -2;
    }

    __android_log_print(ANDROID_LOG_INFO, "CSocket", "Connected");

    char buf[100];

    int t = -1;
    int received = 0;
    while (true)
    {
        int count = recv(sock, &buf, 100, 0);
        if (count < 1) break;

        received += count;

        struct timeval tv;
        gettimeofday(&tv, NULL);

        int m = tv.tv_sec * 1000 + tv.tv_usec / 1000;
        int diff = m - t;
        t = m;
        std::string p = t < 0 ? "" : diff < 50 ? "-" : diff < 100 ? "-|" : diff < 150 ? "--|" :
            diff < 250 ? "---|" :  diff < 500 ? "----|" : diff < 1000 ? "-----|" : "------|";
        __android_log_print(diff > 500 ? ANDROID_LOG_ERROR : ANDROID_LOG_INFO,
                            "CSocket", "%i | %s %i", received, p.c_str(), diff);
    }

    close(sock);

    return 0;
}

我做错了什么吗? 这是我第一次在 stackoverflow 上提问,抱歉我的英语不好,感谢任何帮助或建议,谢谢。

编辑:添加服务器代码,我重写了一个简单的tcp服务器来测试(Window平台)

 int main()
{

    addrinfo conf, *add = nullptr;
    memset(&conf, 0, sizeof(conf));

    conf.ai_flags    = AI_PASSIVE;
    conf.ai_socktype = SOCK_STREAM;
    conf.ai_family   = AF_INET;

    if (getaddrinfo(nullptr, "12350", &conf, &add)) return 0;

    SOCKET serverSock = socket(AF_INET, SOCK_STREAM, 0);

    char opt = 1;
    if (setsockopt(serverSock, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)) == -1 ||
        setsockopt(serverSock, IPPROTO_TCP, TCP_NODELAY, &opt, sizeof(opt)) == -1 ||
        bind(serverSock, add->ai_addr, add->ai_addrlen) == -1 ||
        listen(serverSock, 0) == -1)
    {
        close(serverSock);
        return 0;
    }

    printf("Listening....\n");

    sockaddr_storage incoming_addr;

    int size = sizeof(incoming_addr);
    SOCKET clientSock = accept(serverSock, (sockaddr*)&incoming_addr, &size);

    printf("Client connected\n");

    char buf[1] = { 0 };
    int sendCount = 0;
    while (true)
    {
        time_t t = time(nullptr);
        tm *lt = localtime(&t);
        printf("%02d:%02d:%02d Send to client %i\n", lt->tm_hour, lt->tm_min, lt->tm_sec, ++sendCount);

        if (send(clientSock, buf, 1, 0) < 1) break;
        Sleep(200);
    }

    close(serverSock);

    return 0;
}

编辑:添加 WireShark 捕获图像: Wireshark shot when stuck happen

最佳答案

我已经在家里测试过(代码贴在这里),在另一个WIFI环境下,它工作正常。唯一不同的是WIFI环境,所以我想可能是我们公司的WIFI网络设置导致了这个问题。

关于Android 套接字每 20 秒卡住一次,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51758450/

有关Android 套接字每 20 秒卡住一次的更多相关文章

  1. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  2. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

    在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

  3. 网络编程套接字 - 2

    网络编程套接字网络编程基础知识理解源`IP`地址和目的`IP`地址理解源MAC地址和目的MAC地址认识端口号理解端口号和进程ID理解源端口号和目的端口号认识`TCP`协议认识`UDP`协议网络字节序socket编程接口`sockaddr``UDP`网络程序服务器端代码逻辑:需要用到的接口服务器端代码`udp`客户端代码逻辑`udp`客户端代码`TCP`网络程序服务器代码逻辑多个版本服务器单进程版本多进程版本多线程版本线程池版本服务器端代码客户端代码逻辑客户端代码TCP协议通讯流程TCP协议的客户端/服务器程序流程三次握手(建立连接)数据传输四次挥手(断开连接)TCP和UDP对比网络编程基础知识

  4. kvm虚拟机安装centos7基于ubuntu20.04系统 - 2

    需求:要创建虚拟机,就需要给他提供一个虚拟的磁盘,我们就在/opt目录下创建一个10G大小的raw格式的虚拟磁盘CentOS-7-x86_64.raw命令格式:qemu-imgcreate-f磁盘格式磁盘名称磁盘大小qemu-imgcreate-f磁盘格式-o?1.创建磁盘qemu-imgcreate-fraw/opt/CentOS-7-x86_64.raw10G执行效果#ls/opt/CentOS-7-x86_64.raw2.安装虚拟机使用virt-install命令,基于我们提供的系统镜像和虚拟磁盘来创建一个虚拟机,另外在创建虚拟机之前,提前打开vnc客户端,在创建虚拟机的时候,通过vnc

  5. 安卓apk修改(Android反编译apk) - 2

    最近因为项目需要,需要将Android手机系统自带的某个系统软件反编译并更改里面某个资源,并重新打包,签名生成新的自定义的apk,下面我来介绍一下我的实现过程。APK修改,分为以下几步:反编译解包,修改,重打包,修改签名等步骤。安卓apk修改准备工作1.系统配置好JavaJDK环境变量2.需要root权限的手机(针对系统自带apk,其他软件免root)3.Auto-Sign签名工具4.apktool工具安卓apk修改开始反编译本文拿Android系统里面的Settings.apk做demo,具体如何将apk获取出来在此就不过多介绍了,直接进入主题:按键win+R输入cmd,打开命令窗口,并将路

  6. ruby-on-rails - Rake 任务仅调用一次时执行两次 - 2

    我写了一个非常简单的rake任务来尝试找到这个问题的根源。namespace:foodotaskbar::environmentdoputs'RUNNING'endend当在控制台中执行rakefoo:bar时,输出为:RUNNINGRUNNING当我执行任何rake任务时会发生这种情况。有没有人遇到过这样的事情?编辑上面的rake任务就是写在那个.rake文件中的所有内容。这是当前正在使用的Rakefile。requireFile.expand_path('../config/application',__FILE__)OurApp::Application.load_tasks这里

  7. ruby - 是否可以在不实际发送或读取数据的情况下查明 ruby​​ 套接字是否处于 ESTABLISHED 或 CLOSE_WAIT 状态? - 2

    s=Socket.new(Socket::AF_INET,Socket::SOCK_STREAM,0)s.connect(Socket.pack_sockaddr_in('port','hostname'))ssl=OpenSSL::SSL::SSLSocket.new(s,sslcert)ssl.connect从这里开始,如果ssl连接和底层套接字仍然是ESTABLISHED,或者它是否在默认值7200之后进入CLOSE_WAIT,我想检查一个线程几秒钟甚至更糟的是在实际上不需要.write()或.read()的情况下关闭。是用select()、IO.select()还是其他方法完成

  8. ruby - 我怎样才能只写一次 "Text"并同时检查 path_info 是否包含 'A' ? - 2

    -if!request.path_info.include?'A'%{:id=>'A'}"Text"-else"Text"“文本”写了两次。我怎样才能只写一次并同时检查path_info是否包含“A”? 最佳答案 有两种方法可以做到这一点。使用部分,或使用content_forblock:如果“文本”较长,或者是一个重要的子树,您可以将其提取到一个部分。这会使您的代码变干一点。在给出的示例中,这似乎有点矫枉过正。在这种情况下更好的方法是使用content_forblock,如下所示:-if!request.path_info.inc

  9. ruby-on-rails - Ruby 的 'open_uri' 是否在读取或失败后可靠地关闭套接字? - 2

    一段时间以来,我一直在使用open_uri下拉ftp路径作为数据源,但突然发现我几乎连续不断地收到“530抱歉,允许的最大客户端数(95)已经连接。”我不确定我的代码是否有问题,或者是否是其他人在访问服务器,不幸的是,我无法真正确定谁有问题。本质上,我正在读取FTPURI:defself.read_uri(uri)beginuri=open(uri).readuri=="Error"?nil:urirescueOpenURI::HTTPErrornilendend我猜我需要在这里添加一些额外的错误处理代码...我想确保我采取一切预防措施来关闭所有连接,这样我的连接就不是问题所在,但是我

  10. ruby - Faye WebSocket,关闭处理程序被触发后重新连接到套接字 - 2

    我有一个super简单的脚本,它几乎包含了FayeWebSocketGitHub页面上用于处理关闭连接的内容:ws=Faye::WebSocket::Client.new(url,nil,:headers=>headers)ws.on:opendo|event|p[:open]#sendpingcommand#sendtestcommand#ws.send({command:'test'}.to_json)endws.on:messagedo|event|#hereistheentrypointfordatacomingfromtheserver.pJSON.parse(event.d

随机推荐