我有一个关于在本机c程序中使用以下行的问题:

#include <sys/socket.h>
#include <errno.h>

void testSocket()
{
    int sockfd = socket(AF_INET, SOCK_STREAM, 0);
    int err = errno;
}

无法创建套接字(sockfd = -1)并且errno = 13.我在我的清单中定义了以下内容:

 <uses-permission android:name="android.permission.INTERNET"/>

任何人都可以告诉我他们有任何经验/知道为什么这不起作用? 我现在正在拉空。

#1楼 票数:17 已采纳

从Android应用程序访问Internet需要什么权限?

这个答案的一部分帮助了我:你必须添加这一行:

<uses-permission android:name="android.permission.INTERNET" /> 

在AndroidManifest.xml中的应用程序标记之外

#2楼 票数:12

Application.mk
--------------
APP_PROJECT_PATH := $(call my-dir)
APP_MODULES      := mylib

#--------------------------------------------------------------------------------

【以下是几个c文件】
NativeHello.c ,主要的C文件,调用其它普通linux native c
-----------------------------------------------------
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <jni.h>
#include <assert.h>
#include "HttpGet.h"

JNIEXPORT jstring JNICALL native_hello(JNIEnv *env, jclass clazz)
{
    printf("hello in c native code./n");
    mmain();
    return (*env)->NewStringUTF(env, "-------------hello world mmain().-------------");
}

#define JNIREG_CLASS "org/eshock/jnitest/JNITest"//指定要注册的类
                      //注意:是类名
/** * Table of methods associated with a single class. */

static JNINativeMethod gMethods[] = {
        { "hello", "()Ljava/lang/String;", (void*)native_hello },//绑定
        }; /* * Register several native methods for one class. */

static int registerNativeMethods(JNIEnv* env, const char* className, JNINativeMethod* gMethods, int numMethods)
{
    jclass clazz;
    clazz = (*env)->FindClass(env, className);
    if (clazz == NULL)
    {
        return JNI_FALSE;
    }
    if ((*env)->RegisterNatives(env, clazz, gMethods, numMethods) < 0)
    {
        return JNI_FALSE;
    }

    return JNI_TRUE; }

/* * Register native methods for all classes we know about. */
static int registerNatives(JNIEnv* env)
{
    if (!registerNativeMethods(env, JNIREG_CLASS, gMethods, sizeof(gMethods) / sizeof(gMethods[0])))
    {
        return JNI_FALSE;
    }
    return JNI_TRUE;
}

/* * Set some test stuff up. * * Returns the JNI version on success, -1 on failure.
 *当Android的VM(Virtual Machine)执行到System.loadLibrary()函数时,首先会去执行C组件里的JNI_OnLoad()函数。
 *它的用途有二:
 *(1)【版本】告诉VM此C组件使用那一个JNI版本。如果你的*.so档没有提供JNI_OnLoad()函数,VM会默认该*.so档是使用最老的JNI 1.1版本。
 *   由于新版的JNI做了许多扩充,如果需要使用JNI的新版功能,例如JNI 1.4的java.nio.ByteBuffer,
 *   就必须藉由JNI_OnLoad()函数来告知VM。
 *(2)【初始化】由于VM执行到System.loadLibrary()函数时,就会立即先呼叫JNI_OnLoad(),所以C组件的开发者可以藉由JNI_OnLoad()
 *   来进行C组件内的初期值之设定(Initialization) 。
  * */

JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void* reserved)
{
    JNIEnv* env = NULL;
    jint result = -1;
    if ((*vm)->GetEnv(vm, (void**) &env, JNI_VERSION_1_4) != JNI_OK)
    {
        return -1;
    }
    assert(env != NULL);
    if (!registerNatives(env))//注册
    {
        return -1;
    }
    /* success -- return valid version number */
    result = JNI_VERSION_1_4;
    return result;
}

--------------------------------------------
HttpGet.c
#include "HttpGet.h"
#include <android/log.h>

int GetHttpResponseHead(int sock,char *buf,int size)
{
    int i;
    char *code,*status;
    memset(buf,0,size);
    for(i=0; i<size-1; i++){
        if(recv(sock,buf+i,1,0)!=1){
            perror("recv error");
            return -1;
        }
        if(strstr(buf,"\r\n\r\n"))
            break;
    }
    if(i >= size-1){
        puts("Http response head too long.");
        return -2;
    }
    code=strstr(buf," 200 ");
    status=strstr(buf,"\r\n");
    if(!code || code>status){
        *status=0;
        printf("Bad http response:\"%s\"\n",buf);
        return -3;
    }
    return i;
}

int HttpGet(const char *server,const char *url)
{
    __android_log_print(ANDROID_LOG_INFO, "-----from--jni-------", "Enter HttpGet function!");
    int sock=socket(PF_INET,SOCK_STREAM,0);
    __android_log_print(ANDROID_LOG_INFO, "-----from--jni-------", "%d",sock);
    struct sockaddr_in peerAddr;
    char buf[2048];
    int ret;
    //const char *filename;
    //FILE *fp=NULL;
    peerAddr.sin_family=AF_INET;
    peerAddr.sin_port=htons(80);
    peerAddr.sin_addr.s_addr=inet_addr(server);
    ret=connect(sock,(struct sockaddr *)&peerAddr,sizeof(peerAddr));
    if(ret != 0){
        perror("connect failed");
        close(sock);
        return -1;
    }
    sprintf(buf,
        "GET %s HTTP/1.1\r\n"
        "Accept: */*\r\n"
        "User-Agent: cnzhuhai@163.com\r\n"
        "Host: %s\r\n"
        "Connection: Close\r\n\r\n",
        url,server);
    send(sock,buf,strlen(buf),0);
    if(GetHttpResponseHead(sock,buf,sizeof(buf))<1){
        close(sock);
        return -1;
    }
    //filename=strrchr(url,'/')+1;
    //if(!filename[0])
        //filename="index.html";
    //fp=fopen(filename,"wb");
    //if(fp)
    //{
        __android_log_print(ANDROID_LOG_INFO, "-----from--jni-------", "Enter HttpGet function's while!");
        while((ret=recv(sock,buf,sizeof(buf),0)) > 0)
        {
            //fwrite(buf,ret,1,fp);
            __android_log_print(ANDROID_LOG_INFO, "-----from--jni-------", "hello, this is my number %s log message!", buf);
        }
    //}
    //fclose(fp);
    shutdown(sock,SHUT_RDWR);
    close(sock);
    return 0;
}

//"http://192.168.7.222/index.html"
int mmain(void)
{
    __android_log_print(ANDROID_LOG_INFO, "-----from--jni-------", "Enter mmain function!");
    char *head,*tail;
    char server[128]={0};
/*  if(argc<2){
        printf("Usage:%s url\n",argv[0]);
        return -1;
    }
    if(strlen(argv[1])>1500){
        puts("Url too long.");
        return -1;
    }*/
    //head=strstr("http://192.168.10.101/gi/1.html","//");
    head=strstr("http://220.181.111.148/index.html","//");
    if(!head){
        puts("Bad url format");
        return -1;
    }
    head+=2;
    tail=strchr(head,'/');
    if(!tail){
        return HttpGet(head,"/");
    }else if(tail-head>sizeof(server)-1){
        puts("Bad url format");
        return -1;
    }else{
        memcpy(server,head,tail-head);
        return HttpGet(server,tail);
    }
}

HttpGet.h
---------------------------
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <fcntl.h>
#include <ctype.h>
#include <unistd.h>
#include <netdb.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

int GetHttpResponseHead(int sock,char *buf,int size);
int HttpGet(const char *server,const char *url);
int mmain(void);

android.mk
--------------------

include $(CLEAR_VARS)
LOCAL_LDLIBS := -lm -llog  
LOCAL_SHARED_LIBRARIES := liblog libcutils

LOCAL_MODULE    := mylib
LOCAL_SRC_FILES := \
    NativeHello.c \
    HttpGet.h \
    HttpGet.c \ 

include $(BUILD_SHARED_LIBRARY)
#--------------------------------------------------------------------------------

#3楼 票数:2

<uses-permission android:name="android.permission.INTERNET" /> 
the application tag in your AndroidManifest.xml
注意放置的位置;
如下:my AndroidManifest.xml
---------------------------------------
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
      package="org.eshock.jnitest"
      android:versionCode="1"
      android:versionName="1.0">
    <uses-permission android:name="android.permission.INTERNET" />
    <application android:icon="@drawable/icon" android:label="@string/app_name">
        <activity android:name=".JNITest"
                  android:label="@string/app_name">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

    </application>
</manifest>

  ask by Graeme translate from so

未解决问题?本站智能推荐:

1回复

在Android上用NDK使用libvlc

我目前正在使用NDK在Android上使用libvlc搞砸了一下。 好吧,我了解如何在Android应用程序中使用本机C代码。 现在我想使用libvlc库来创建一个简单的播放器,它可以从套接字读取数据。 是否有可能在该级别使用libvlc? 我看到的问题是为libvlc设置一个窗口处理
1回复

是否可以在Android中使用NDK实现原始套接字?

我想制作读取帧的程序(其中包含所有标题,如链接层,IP层,TP标题)。 我认为这在JAVA环境中是不可能的。 但是,有可能在Android上的NDK上制作这些东西吗? 如果不可能如何在android中制作类似wireshark(读取任何标题)的程序?
1回复

我可以在AndroidNDK中使用BSD插槽吗?

我以为我会先在这里问这个问题,因为使用NDK进行任何设置都非常麻烦。 我知道Android使用Linux内核,但就C库而言,它几乎不符合POSIX。 Android OS系列是否实现BSD套接字API? 并且,如果它确实实现了BSD套接字,是否允许我仅按AF_INET和SOCK_STRE
1回复

socket(AF_INET,SOCK_DGRAM,IPPROTO_ICMP)失败,errno=13(权限被拒绝)

在adb shell中运行,由NDK编译。 我的代码: 输出: 有人知道为什么吗?
3回复

Android的巨大UDP延迟/滞后

我正在使用一个Android应用程序,该应用程序通过WLAN向Windows端点发送/接收大量UDP流量(不,我不能使用TCP)。 问题是,当我增加流量时,我开始看到从调用sendto(应用程序使用NDK编写)到数据包到达Windows端点之间的巨大延迟。 在10秒左右 ! 同样的事情也
1回复

Android上的select()始终返回false

我目前正在用C ++建立一个基于UDP的网络库。 我打算使用它们的方法是运行同一个应用程序的两个实例。 一个在Windows上,一个在Android上。 Android实例应该发送一个在Windows上收到的广播消息。 然后在两端打开单播套接字并开始通信。 然而,现在,我正在
1回复

在Android中翻译C++函数

我正在尝试开发一个与用C ++编写的功能相同的Android应用程序,以便在手机和远程以太网接口之间进行通信(我不知道它到底是什么。。。)。 我知道我必须将从十六进制值中检索到的15个字节传递给接口,以控制一些连接到接口的负载。 我已经尝试了几件事,最接近正确解决方案的似乎是这个:
2回复

Android上的getaddrinfo返回错误EAI_BADFLAGS

我们尝试在Android API 14及更高版本上使用getaddrinfo(使用NDK r12的C ++代码)从IPV4地址获取合成的IPV6地址。 这适用于仅具有NAT64的IPV6网络。 但是,如果设置了aiflags,则getaddrinfo返回EAI_BADFLAGS(AI_V4M