使用C語(yǔ)言實(shí)現(xiàn)本地socke通訊的方法
一、概述
使用本地socket通訊可以實(shí)現(xiàn)進(jìn)程之間的通訊。
相關(guān)函數(shù)描述如下:
int socket(int domain, int type, int protocol);
函數(shù)說(shuō)明: 創(chuàng)建本地域socket
函數(shù)參數(shù):
domain: AF_UNIX or AF_LOCAL
type: SOCK_STREAM或者SOCK_DGRAM
protocol: 0 表示使用默認(rèn)協(xié)議
函數(shù)返回值:
成功: 返回文件描述符.
失敗: 返回-1, 并設(shè)置errno值.
創(chuàng)建socket成功以后, 會(huì)在內(nèi)核創(chuàng)建緩沖區(qū), 下圖是客戶端和服務(wù)端內(nèi)核緩沖區(qū)示意圖.
int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
函數(shù)說(shuō)明: 綁定套接字
函數(shù)參數(shù):
socket: 由socket函數(shù)返回的文件描述符
addr: 本地地址
addlen: 本地地址長(zhǎng)度
函數(shù)返回值:
成功: 返回文件描述符.
失敗: 返回-1, 并設(shè)置errno值.
需要注意的是: bind函數(shù)會(huì)自動(dòng)創(chuàng)建socket文件, 若在調(diào)用bind函數(shù)之前socket文件已經(jīng)存在, 則調(diào)用bind會(huì)報(bào)錯(cuò), 可以使用unlink函數(shù)在bind之前先刪除文件.
struct sockaddr_un {
sa_family_t sun_family; /* AF_UNIX or AF_LOCAL*/
char sun_path[108]; /* pathname */
};
整體使用步驟和網(wǎng)絡(luò)通訊的socket是差不多的,如下所示:
tcp的本地套接字服務(wù)器流程:
創(chuàng)建套接字 socket(AF_UNIX,SOCK_STREAM,0)
綁定 struct sockaddr_un &強(qiáng)轉(zhuǎn)
偵聽 listen
獲得新連接 accept
循環(huán)通信 read-write
關(guān)閉文件描述符 close
tcp本地套接字客戶端流程:
調(diào)用socket創(chuàng)建套接字
調(diào)用bind函數(shù)將socket文件描述和socket文件進(jìn)行綁定.
不是必須的, 若無(wú)顯示綁定會(huì)進(jìn)行隱式綁定,但服務(wù)器不知道誰(shuí)連接了.
調(diào)用connect函數(shù)連接服務(wù)端
循環(huán)通信read-write
關(guān)閉文件描述符 close
二、代碼示例
1.服務(wù)端代碼示例
//本地socket通訊服務(wù)端
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/un.h>
int main(){
//創(chuàng)建socket
int lfd = socket(AF_UNIX,SOCK_STREAM,0);
if(lfd<0){
perror("socket error");
return -1;
}
//刪除socket文件,避免bind失敗
unlink("./server.sock");
//綁定
struct sockaddr_un serv;
bzero(&serv,sizeof(serv));
serv.sun_family = AF_UNIX;
strcpy(serv.sun_path,"./server.sock");
int ret = bind(lfd,(struct sockaddr *)&serv,sizeof(serv));
if(ret<0){
perror("bind error");
return -1;
}
//監(jiān)聽
listen(lfd,10);
//接收新的鏈接-accept
struct sockaddr_un client;
bzero(&client,sizeof(client));
socklen_t len = sizeof(client);
int cfd = accept(lfd,(struct sockaddr*)&client,&len);
if(cfd<0){
perror("accept error");
return -1;
}
printf("cient->[%s]\n",client.sun_path);
int n;
char buf[1024];
while(1){
//讀取數(shù)據(jù)
memset(buf,0x00,sizeof(buf));
n = read(cfd,buf,sizeof(buf));
if(n<=0){
printf("read error or client close ,n=[%d]\n",n);
break;
}
printf("n=[%d],buf=[%s]\n",n,buf);
//發(fā)送數(shù)據(jù)
write(cfd,buf,n);
}
//關(guān)閉套接字
close(lfd);
return 0;
}
2.客戶端代碼示例
//本地socket通信客戶端
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/un.h>
int main(){
//創(chuàng)建socket
int cfd = socket(AF_UNIX,SOCK_STREAM,0);
if(cfd<0){
perror("socket error");
return -1;
}
//刪除socket文件,避免bind失敗
unlink("./client.sock");
//綁定
struct sockaddr_un client;
bzero(&client,sizeof(client));
client.sun_family= AF_UNIX;
strcpy(client.sun_path,"./client.sock");
int ret = bind(cfd,(struct sockaddr*)&client,sizeof(client));
if(ret<0){
perror("bind error");
return -1;
}
struct sockaddr_un serv;
bzero(&serv,sizeof(serv));
serv.sun_family = AF_UNIX;
strcpy(serv.sun_path,"./server.sock");
ret = connect(cfd,(struct sockaddr*)&serv,sizeof(serv));
if(ret<0){
perror("connect error");
return -1;
}
int n;
char buf[1024];
while(1){
memset(buf,0x00,sizeof(buf));
n = read(STDIN_FILENO,buf,sizeof(buf));
//發(fā)送數(shù)據(jù)
write(cfd,buf,n);
//讀取數(shù)據(jù)
memset(buf,0x00,sizeof(buf));
n = read(cfd,buf,sizeof(buf));
if(n<=0){
printf("read error or client close ,n=[%d]",n);
break;
}
printf("n=[%d],buf=[%s]",n,buf);
}
close(cfd);
return 0;
}
到此這篇關(guān)于 使用C語(yǔ)言實(shí)現(xiàn)本地socke通訊的文章就介紹到這了,更多相關(guān)C語(yǔ)言本地socke通訊內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語(yǔ)言實(shí)現(xiàn)的雙鏈表功能完整示例
這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)的雙鏈表功能,結(jié)合完整實(shí)例形式分析了基于C語(yǔ)言實(shí)現(xiàn)的雙鏈表定義、添加、刪除、排序等相關(guān)操作實(shí)現(xiàn)技巧,需要的朋友可以參考下2018-04-04
C語(yǔ)言循環(huán)隊(duì)列的表示與實(shí)現(xiàn)實(shí)例詳解
這篇文章主要介紹了C語(yǔ)言循環(huán)隊(duì)列的表示與實(shí)現(xiàn),對(duì)于數(shù)據(jù)結(jié)構(gòu)與算法的研究很有幫助,需要的朋友可以參考下2014-07-07
C語(yǔ)言實(shí)現(xiàn)學(xué)生成績(jī)管理系統(tǒng)課程設(shè)計(jì)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)學(xué)生成績(jī)管理系統(tǒng)課程設(shè)計(jì),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07
C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之棧與隊(duì)列的相互實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了如何利用C語(yǔ)言相互實(shí)現(xiàn)數(shù)據(jù)結(jié)構(gòu)中的棧與隊(duì)列,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2022-07-07
JetBrains?CLion永久激活超詳細(xì)教程(最新激活方法)
JetBrains?Clion?是一款專為?C/C++?開發(fā)所設(shè)計(jì)的跨平臺(tái)?IDE,本文適用?JetBrains?CLion?v2019.3/3.1/3.2/3.3?永久激活,附破解補(bǔ)丁和激活碼,可以永久激活?Windows、MAC、Linux?下的?CLion,下面給大家分享JetBrains?CLion永久激活超詳細(xì)教程,感興趣的朋友一起看看吧2023-01-01
C++實(shí)現(xiàn)正整數(shù)的四則運(yùn)算表達(dá)式
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)正整數(shù)的四則運(yùn)算表達(dá)式,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-06-06
Qt實(shí)現(xiàn)制作簡(jiǎn)單的計(jì)算器
計(jì)算器是我們生活中很常見(jiàn)的東西,它可以由多種語(yǔ)言多種方式來(lái)實(shí)現(xiàn)。本文主要介紹的是利用Qt實(shí)現(xiàn)的簡(jiǎn)易計(jì)算器的制作,文中的示例代碼講解詳細(xì),需要的可以參考一下2022-12-12

