ftok函数 应用,原理及 shell 脚本实现
2009-02-05 20:47
579 查看
系统建立IPC通讯(如消息队列、共享内存时)必须指定一个ID值。通常情况下,该id值通过ftok函数得到。
ftok原型如下:
key_t ftok( char * fname, int id )
fname就时你指定的文件名,id是子序号。
在一般的UNIX实现中,是将文件的索引节点号取出,前面加上子序号得到key_t的返回值。
如指定文件的索引节点号为65538,换算成16进制为0x010002,而你指定的ID值为38,换算成16进制为0x26,则最后的key_t返回值为0x26010002。
查询文件索引节点号的方法是: ls -i
当删除重建文件后,索引节点号由操作系统根据当时文件系统的使用情况分配,因此与原来不同,所以得到的索引节点号也不同。
如果要确保key_t值不变,要目确保ftok的文件不被删除,要么不用ftok,指定一个固定的key_t值,比如:
#define IPCKEY 0x111
char path[256];
sprintf( path, "%s/etc/config.ini", (char*)getenv("HOME") );
msgid=ftok( path, IPCKEY );[/code]
同一段程序,用于保证两个不同用户下的两组相同程序获得互不干扰的IPC键值。
由于etc/config.ini(假定)为应用系统的关键配置文件,因此不存在被轻易删除的问题——即使被删,也会很快被发现并重建(此时应用系统也将被重起)。
ftok()的设计目的也在于此。
下面是一段具体的代码例子:
key_t keyShareMem ;
if((keyShareMem = ftok(AFC_SHARE_MEMORY_NAME.c_str(), 0)) == -1) {
cerr << "ERROR:"<< m_nThisThreadType<<"
CBaseMessageDeal() keyShareMem ftok: " << errno << ":"
<< strerror(errno)<<endl;
throw new AfcInitAfcResourceException("CBaseMessageDeal::CBaseMessageDeal ftok keyShareMem") ;
}
if ( (m_shmID = shmget(keyShareMem, 0, AFC_SHM_RW)) < 0) {
cerr << "ERROR:"<< m_nThisThreadType<<"
CBaseMessageDeal() shmget exist: "<<errno
<<":"<<strerror(errno)<<endl ;
throw new AfcInitAfcResourceException("CBaseMessageDeal::CBaseMessageDeal() shmget exist") ;
}
if((m_afcShareMemoryBegin=(char *)shmat(m_shmID, NULL, 0)) == (void *) -1){
cerr << "ERROR:"<< m_nThisThreadType<<"
CBaseMessageDeal() shmat: " << errno << ":" <<
strerror(errno) << endl ;
throw new AfcInitAfcResourceException("CBaseMessageDeal::CBaseMessageDeal shmat") ;
}
Trackback: http://tb.donews.net/TrackBack.aspx?PostId=893843 http://blog.csdn.net/hwz119/archive/2007/05/17/1613601.aspx
----------------------------------
let key=0
function ftok() {
pathname=$1;
proj_id=$2;
str_st_ino=`stat --format='%i' "${pathname}" 2>/dev/null`;
str_st_dev=`stat --format='%d' "${pathname}" 2>/dev/null`;
if [ "x${str_st_ino}" = "x" -o "x${str_st_dev}" = "x" ] ; then
return 1;
fi
let st_ino=${str_st_ino}
let st_dev=${str_st_dev}
# 注意这里的位操作运算符需要加转义符
let key1=${st_ino}/&16#FFFF
let key2=${st_dev}/&16#FF
let key2=${key2}/</<16
let key3=${proj_id}/&16#FF
let key3=${key3}/</<24
let key=${key1}/|${key2}
let key=${key}/|${key3}
}
function echohelp(){
echo "ftok generator"
echo "Usage:ftok pathname projid"
exit 5
}
if [ $# -ne 2 ] ; then
echohelp
fi
sPathName=$1
let nProjectID=$2
if [ "${sPathName:0:1}" != "/" ] ; then
sPathName=${PWD}/${sPathName}
fi
if ! test -f ${sPathName} ; then
echo "No File Found![${sPathName}]"
exit 4
fi
ftok "${sPathName}" "${nProjectID}"
echo ${key}
ftok原型如下:
key_t ftok( char * fname, int id )
fname就时你指定的文件名,id是子序号。
在一般的UNIX实现中,是将文件的索引节点号取出,前面加上子序号得到key_t的返回值。
如指定文件的索引节点号为65538,换算成16进制为0x010002,而你指定的ID值为38,换算成16进制为0x26,则最后的key_t返回值为0x26010002。
查询文件索引节点号的方法是: ls -i
当删除重建文件后,索引节点号由操作系统根据当时文件系统的使用情况分配,因此与原来不同,所以得到的索引节点号也不同。
如果要确保key_t值不变,要目确保ftok的文件不被删除,要么不用ftok,指定一个固定的key_t值,比如:
#define IPCKEY 0x111
char path[256];
sprintf( path, "%s/etc/config.ini", (char*)getenv("HOME") );
msgid=ftok( path, IPCKEY );[/code]
同一段程序,用于保证两个不同用户下的两组相同程序获得互不干扰的IPC键值。
由于etc/config.ini(假定)为应用系统的关键配置文件,因此不存在被轻易删除的问题——即使被删,也会很快被发现并重建(此时应用系统也将被重起)。
ftok()的设计目的也在于此。
下面是一段具体的代码例子:
key_t keyShareMem ;
if((keyShareMem = ftok(AFC_SHARE_MEMORY_NAME.c_str(), 0)) == -1) {
cerr << "ERROR:"<< m_nThisThreadType<<"
CBaseMessageDeal() keyShareMem ftok: " << errno << ":"
<< strerror(errno)<<endl;
throw new AfcInitAfcResourceException("CBaseMessageDeal::CBaseMessageDeal ftok keyShareMem") ;
}
if ( (m_shmID = shmget(keyShareMem, 0, AFC_SHM_RW)) < 0) {
cerr << "ERROR:"<< m_nThisThreadType<<"
CBaseMessageDeal() shmget exist: "<<errno
<<":"<<strerror(errno)<<endl ;
throw new AfcInitAfcResourceException("CBaseMessageDeal::CBaseMessageDeal() shmget exist") ;
}
if((m_afcShareMemoryBegin=(char *)shmat(m_shmID, NULL, 0)) == (void *) -1){
cerr << "ERROR:"<< m_nThisThreadType<<"
CBaseMessageDeal() shmat: " << errno << ":" <<
strerror(errno) << endl ;
throw new AfcInitAfcResourceException("CBaseMessageDeal::CBaseMessageDeal shmat") ;
}
Trackback: http://tb.donews.net/TrackBack.aspx?PostId=893843 http://blog.csdn.net/hwz119/archive/2007/05/17/1613601.aspx
----------------------------------
ftok函数的shell脚本实现
#!/bin/shlet key=0
function ftok() {
pathname=$1;
proj_id=$2;
str_st_ino=`stat --format='%i' "${pathname}" 2>/dev/null`;
str_st_dev=`stat --format='%d' "${pathname}" 2>/dev/null`;
if [ "x${str_st_ino}" = "x" -o "x${str_st_dev}" = "x" ] ; then
return 1;
fi
let st_ino=${str_st_ino}
let st_dev=${str_st_dev}
# 注意这里的位操作运算符需要加转义符
let key1=${st_ino}/&16#FFFF
let key2=${st_dev}/&16#FF
let key2=${key2}/</<16
let key3=${proj_id}/&16#FF
let key3=${key3}/</<24
let key=${key1}/|${key2}
let key=${key}/|${key3}
}
function echohelp(){
echo "ftok generator"
echo "Usage:ftok pathname projid"
exit 5
}
if [ $# -ne 2 ] ; then
echohelp
fi
sPathName=$1
let nProjectID=$2
if [ "${sPathName:0:1}" != "/" ] ; then
sPathName=${PWD}/${sPathName}
fi
if ! test -f ${sPathName} ; then
echo "No File Found![${sPathName}]"
exit 4
fi
ftok "${sPathName}" "${nProjectID}"
echo ${key}
相关文章推荐
- shell 典型应用之主控脚本实现
- Shell应用 之 场景主控脚本实现
- Shell脚本实现ftok函数
- Shell脚本实现应用自动安装部署
- Shell脚本实现ftok函数
- 使用shell脚本实现php应用的批量更新
- shell awk实现实时监控网卡流量脚本(常见应用二)
- shell awk实现实时监控网卡流量脚本(常见应用二)
- Shell脚本高级应用 --实现远程MySQL自动查询
- 使用shell脚本实现客户端应用自动化打包——mac
- Shell脚本高级应用 --实现远程MySQL自动查询[转
- shell awk实现实时监控网卡流量脚本(常见应用二)
- shell脚本实现RAID状态监控
- Shell脚本实现自动输入密码登录服务器
- shell练习:写一个脚本实现如下功能:输入一个数字,然后运行对应的一个命令。显示命令如下:*cmd
- Shell脚本中的trap应用
- 对生成对抗网络GANs原理、实现过程、应用场景的理解(附代码),另附:深度学习大神文章列表
- sphinxSE的实现原理和应用
- FrameBuffer 原理、实现与应用
- 9.2 Shell脚本应用1