c语言中关于socket函数的例子

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了c语言中关于socket函数的例子相关的知识,希望对你有一定的参考价值。

求一个c语言中关于socket函数的例子 最好有注释 谢谢!

函数说明:socket()用来建立一个新的socket, 也就是向系统注册, 通知系统建立一通信端口. 参数domain 指定使用何种的地址类型, 完整的定义在/usr/include/bits/socket.h 内, 底下是常见的协议:
PF_UNIX/PF_LOCAL/AF_UNIX/AF_LOCAL UNIX 进程通信协议
PF_INET?AF_INET Ipv4 网络协议
PF_INET6/AF_INET6 Ipv6 网络协议
PF_IPX/AF_IPX IPX-Novell 协议
PF_NETLINK/AF_NETLINK 核心用户接口装置
PF_X25/AF_X25 ITU-T X. 25/ISO-8208 协议
PF_AX25/AF_AX25 业余无线AX. 25 协议
PF_ATMPVC/AF_ATMPVC 存取原始 ATM PVCs
PF_APPLETALK/AF_APPLETALK appletalk (DDP)协议
PF_PACKET/AF_PACKET 初级封包接口

参数 type 有下列几种数值:
1、SOCK_STREAM 提供双向连续且可信赖的数据流, 即TCP. 支持 OOB 机制, 在所有数据传送前必须使用connect()来建立连线状态.
2、SOCK_DGRAM 使用不连续不可信赖的数据包连接
3、SOCK_SEQPACKET 提供连续可信赖的数据包连接
4、SOCK_RAW 提供原始网络协议存取
5、SOCK_RDM 提供可信赖的数据包连接
6、SOCK_PACKET 提供和网络驱动程序直接通信. protocol 用来指定socket 所使用的传输协议编号, 通常此参考不用管它, 设为0 即可.

返回值:成功则返回socket 处理代码, 失败返回-1.

错误代码:
1、EPROTONOSUPPORT 参数domain 指定的类型不支持参数type 或protocol 指定的协议
2、ENFILE 核心内存不足, 无法建立新的socket 结构
3、EMFILE 进程文件表溢出, 无法再建立新的socket
4、EACCESS 权限不足, 无法建立type 或protocol 指定的协议
5、ENOBUFS/ENOMEM 内存不足
6、EINVAL 参数domain/type/protocol 不合法
参考技术A 以下是简单socket通信的例子

客户端代码:

// client.c : 定义控制台应用程序的入口点。
//

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <conio.h>
#include <winsock2.h>

#pragma comment(lib,"ws2_32.lib")

int main()

WSADATA wsadata;
WORD socketversion = MAKEWORD(2,2);
if(::WSAStartup(socketversion,&wsadata) != 0)

printf("启动windows socket失败!\n");


SOCKET clientsocket;
if((clientsocket = ::socket(AF_INET,SOCK_STREAM,0)) == INVALID_SOCKET)

printf("创建套接字失败!\n");
::closesocket(clientsocket);
exit(0);


sockaddr_in client;
client.sin_family = AF_INET;
client.sin_port = htons(5051);
client.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
if(::bind(clientsocket,(sockaddr *)&client,sizeof(client)) == SOCKET_ERROR)

printf("绑定地址失败\n");
::closesocket(clientsocket);
exit(0);


sockaddr_in server;
server.sin_family = AF_INET;
server.sin_port = htons(5050);
server.sin_addr.S_un.S_addr = inet_addr("172.19.36.8");
if(::connect(clientsocket,(sockaddr *)&server,sizeof(server)) == SOCKET_ERROR)

printf("服务器连接失败\n");
::closesocket(clientsocket);
exit(0);

printf("服务器连接成功");
while(1)

char buf[1024];
::memset(buf,0,sizeof(buf));
gets(buf);
char *p_c;
int count;
int numchr;
p_c = buf;
//p_c = (LPSTR)(LPCTSTR)m_sendstr;
count = sizeof(buf);
numchr=0;
do

numchr = send(clientsocket,p_c,count,0);
if(numchr == SOCKET_ERROR)

printf("发送错误");

else

p_c+=numchr;
count-=numchr;

while(count>0);

getch();
return 0;

关于C中可变长参数

前言

可变长参数指函数的参数个数在调用时才能确定的函数参数。基本上各种语言都支持可变长参数,在特定情形下,可变长参数使用起来非常方便。c语言中函数可变长参数使用“...”来表示,同时可变长参数只能位于固定参数的后面,固定参数的个数至少为1。只要学习过c语言的,应该都知道printf函数,并且见识到了其强大的功能——事实上,迄今为止,我仍认为这是c函数库中最牛逼的函数之一。

一、一个简单的例子 

  1. #include <string> 
  2. #include <stdio> 
  3. ///拼接字符串 
  4. char * JointStr(int Count, ...)
  5. { 
  6.   char * pszBuff = new char[100]; 
  7.   ::memset(pszBuff, 0, 100);   
  8.   va_list vl;  
  9.   va_start(vl, Count); 
  10.   for(int i = 0; i < Count; i++) 
  11.   { 
  12.       strcat(pszBuff, va_arg(vl, char *)); 
  13.   }   
  14.   return pszBuff; 
  15. } 
  16. void main() 
  17. { 
  18.    char * pszStr = JointStr(3, "abc", "123", "[email protected]#"); 
  19.    printf("%s", pszStr); 
  20. }

  执行后,输出:[email protected]#

函数JointStr的功能是指定个数的字符串拼接起来,返回拼接后的字符串的指针。参数Count是字符串的个数,后面跟可变长参数,使用时应该跟Count个char*型参数。使用时,可以随意指定个数。(该例子只是用来说明问题,实际使用时不会用这个函数,可以使用标准库函数中的sprintf函数)。

二、可变长参数的使用方法

首先,必须弄清楚一下三个宏定义:

  • va_arg
  • va_start
  • va_end

以及一个类型:va_list

从c函数库中的头文件中可以看到va_list的定义:

typedef char *  va_list;

也就是说它就是一个指针,那么该指针指向什么地方呢? 这就是va_start的作用了。首先看看这三个宏定义的申明: 

  1. #define va_start(ap,v) 
  2. #define va_arg(ap,t) 
  3. #define va_end(ap)

va_start有两个参数:第一个ap应该填写va_list,第二个v应该填写函数参数列表中(可以认为传给函数的参数是一个列表,一个接一个)的某个参数,例如例子中的Count。其作用是将ap指向函数参数列表中的参数v的位置(msdn是这样说的,我觉得应该指向参数列表中的参数v的下一个参数的开始地址)。

va_end作用是将ap设置为NULL。

va_arg(ap,t)有两个参数:第一个是va_list,第二个是参数类型。其作用是从ap开始取一个t型的值返回,并且自动将ap指向下一个参数。所以如果t即参数类型写错了,例如将char*写成char了,本来要取4个字节,结果只取了一个字节,ap本来要向后面移动4个字节,结果只移动了一个字节,后面的数据就全错了。同时,如果你多取了一次参数,将报内存越界错误,所以使用可变长参数,前面一般都会传一个参数来指定参数的个数。

使用可变长参数的步骤:

  1. 声明va_list变量
  2. 使用va_start指定可变长参数的位置
  3. 使用va_arg来获取参数值
  4. 可选,使用va_end将va_list清零

三、va_list类型作为参数

在c标准库中有一个函数vsprintf,声明如下:

int __cdecl vsprintf(char *_DstBuf, char * _Format, va_list _ArgList);

其第三个参数为va_list类型,我们可以在这样使用: 

char* GetFormatStr(char* format, ...)

{
  char *pszBuff = new char[256];
  va_list vl;
  va_start(vl,format);
  vsprintf(pszBuff, format, vl);
  return pszBuff;
}
void main()
{
  char * str = GetFormatStr("%s is %d year old.", "frank", 25);
  printf("%s", str);
}

 将输出:frank is 25 year old. 

结束语

我们可以认为,我们调用可变长参数的函数时,传递给函数的参数是在堆栈中保存为一个紧挨一个的列表。获取参数的唯一方法就是通过参数列表的指针,取一个参数,移动一个参数长度的指针,实际上如何取参数完全掌握在用户手中,用户应当小心应对。

原文链接:

https://blog.csdn.net/frank_liuxing/article/details/18000825












以上是关于c语言中关于socket函数的例子的主要内容,如果未能解决你的问题,请参考以下文章

C语言socket编程

一起talk C栗子吧(第一百五十五回:C语言实例--获取socket通信地址)

c语言函数的调用和声明

关于C语言结构体构造函数初始化的问题?

一起talk C栗子吧(第一百五十四回:C语言实例--socket通信地址系统调用二)

C语言SOCKET问题