readn和writen函數(shù), socket編程常用函數(shù)
Posted on 2009-01-11 21:10 Prayer 閱讀(2517) 評(píng)論(1) 編輯 收藏 引用 所屬分類: SOCKET寫函數(shù)write
#include <unistd.h>
ssize_t write(int fildes,const void *buf,size_t nbyte)
write函數(shù)將buf中的nbyte字節(jié)內(nèi)容寫入文件描述符fd引用的打開文件中,成功時(shí)返回寫的字節(jié)數(shù),失敗時(shí)返回-1。 并設(shè)置errno變量. 在網(wǎng)絡(luò)程序中,當(dāng)我們向套接字文件描述符寫時(shí)有兩種可能.
1)write的返回值大于0,表示寫了部分或者是全部的數(shù)據(jù).
2)返回的值小于0,此時(shí)出現(xiàn)了錯(cuò)誤.我們要根據(jù)錯(cuò)誤類型來處理.
如果錯(cuò)誤為EINTR表示在寫的時(shí)候出現(xiàn)了中斷錯(cuò)誤.
如果為EPIPE表示網(wǎng)絡(luò)連接出現(xiàn)了問題(對(duì)方已經(jīng)關(guān)閉了連接).
為了處理以上的情況,我們自己編寫一個(gè)寫函數(shù)來處理這幾種情況.
ssize_t writen (int fd, const void *buf, size_t num)
{
ssize_t res;
size_t n;
const char *ptr;
n = num;
ptr = buf;
while (n > 0) {
/* 開始寫*/
if ((res = write (fd, ptr, n)) <= 0) {
if (errno == EINTR)
res = 0;
else
return (-1);
}
ptr += res;/* 從剩下的地方繼續(xù)寫 */
n -= res;
}
return (num);
}
讀函數(shù)read
ssize_t read(int fildes,void *buf,size_t nbyte) read函數(shù)是負(fù)責(zé)從fd引用的打開文件中讀取nbyte字節(jié)到buf指向的緩沖區(qū),成功返回實(shí)際所讀的字節(jié)數(shù),如果返回的值是0 表示已經(jīng)讀到文件的結(jié)束了。小于0表示出現(xiàn)了錯(cuò)誤。如果錯(cuò)誤為EINTR說明讀是由中斷引起的, 如果是ECONNREST表示網(wǎng)絡(luò)連接出了問題。可能read讀取的字節(jié)數(shù)少于要求的nbyte字節(jié),有很多原因:1從規(guī)則文件中讀取,從終端讀取通常是每次讀取一行,從TCP套接字讀取根據(jù)如何接受包可以返回的任意字節(jié)數(shù)。 和上面一樣,我們也寫一個(gè)自己的讀函數(shù)。
ssize_t readn (int fd, void *buf, size_t num)
{
ssize_t res;
size_t n;
char *ptr;
n = num;
ptr = buf;
while (n > 0) {
if ((res = read (fd, ptr, n)) == -1) {
if (errno == EINTR)
res = 0;
else
return (-1);
}
else if (res == 0)
break;
ptr += res;
n -= res;
}
return (num - n);
}
#include <unistd.h>
ssize_t write(int fildes,const void *buf,size_t nbyte)
write函數(shù)將buf中的nbyte字節(jié)內(nèi)容寫入文件描述符fd引用的打開文件中,成功時(shí)返回寫的字節(jié)數(shù),失敗時(shí)返回-1。 并設(shè)置errno變量. 在網(wǎng)絡(luò)程序中,當(dāng)我們向套接字文件描述符寫時(shí)有兩種可能.
1)write的返回值大于0,表示寫了部分或者是全部的數(shù)據(jù).
2)返回的值小于0,此時(shí)出現(xiàn)了錯(cuò)誤.我們要根據(jù)錯(cuò)誤類型來處理.
如果錯(cuò)誤為EINTR表示在寫的時(shí)候出現(xiàn)了中斷錯(cuò)誤.
如果為EPIPE表示網(wǎng)絡(luò)連接出現(xiàn)了問題(對(duì)方已經(jīng)關(guān)閉了連接).
為了處理以上的情況,我們自己編寫一個(gè)寫函數(shù)來處理這幾種情況.
ssize_t writen (int fd, const void *buf, size_t num)
{
ssize_t res;
size_t n;
const char *ptr;
n = num;
ptr = buf;
while (n > 0) {
/* 開始寫*/
if ((res = write (fd, ptr, n)) <= 0) {
if (errno == EINTR)
res = 0;
else
return (-1);
}
ptr += res;/* 從剩下的地方繼續(xù)寫 */
n -= res;
}
return (num);
}
讀函數(shù)read
ssize_t read(int fildes,void *buf,size_t nbyte) read函數(shù)是負(fù)責(zé)從fd引用的打開文件中讀取nbyte字節(jié)到buf指向的緩沖區(qū),成功返回實(shí)際所讀的字節(jié)數(shù),如果返回的值是0 表示已經(jīng)讀到文件的結(jié)束了。小于0表示出現(xiàn)了錯(cuò)誤。如果錯(cuò)誤為EINTR說明讀是由中斷引起的, 如果是ECONNREST表示網(wǎng)絡(luò)連接出了問題。可能read讀取的字節(jié)數(shù)少于要求的nbyte字節(jié),有很多原因:1從規(guī)則文件中讀取,從終端讀取通常是每次讀取一行,從TCP套接字讀取根據(jù)如何接受包可以返回的任意字節(jié)數(shù)。 和上面一樣,我們也寫一個(gè)自己的讀函數(shù)。
ssize_t readn (int fd, void *buf, size_t num)
{
ssize_t res;
size_t n;
char *ptr;
n = num;
ptr = buf;
while (n > 0) {
if ((res = read (fd, ptr, n)) == -1) {
if (errno == EINTR)
res = 0;
else
return (-1);
}
else if (res == 0)
break;
ptr += res;
n -= res;
}
return (num - n);
}