14

发现一个非常好的网站,可以提供免费的ssh服务,而且有多种OS可供选择。
创始人具有真正的开放分享的精神,这样的人多一点,中国才能变得更好吧。
他们现在需要帮助,看到 & 有能力& 认同的人,尽自己的一份力吧。
可以购买服务,也可以直接捐款。
www.unix-center.net

#!/bin/sh
 
servers=("t1000.unix-center.net" "x4100.unix-center.net" "opensolaris.unix-center.net" "fedora.unix-center.net" "ubuntu.unix-center.net" "freebsd.unix-center.net" "aix.unix-center.net")
svrcnt=${#servers[@]}
 
function go()
{
	if [ $1 -ge $svrcnt ]; then
		echo "server not exist"
		exit
	fi
	echo "going ${servers[$1]}"
	ssh iver@${servers[$1]}
}
 
if [ $# -eq 1 ]; then
	go $1
else
	echo "Usage: $0 target\n"
	i=0
	while [ $i -lt $svrcnt ]
	do
		echo $i":" ${servers[$i]}
		((i++))
	done
fi
Tagged with:
18

当可能出现几个进程争用(读、写)同一个Critical Section的时候,加锁是常用的做法。
Linux加锁的方法,除了经典的IPC(Semophore)之外,记录锁(Record Locking)提供了更简单的方法。

其实记录锁的名字叫文件锁会比较贴切一点,因为其加锁和解锁都是通过对文件的操作完成的。
文件锁的粒度大可到整个文件,小可到一个字节,长度可变,但都可以说是对应一个Record(逻辑意义上)。

对锁的控制是通过调用fcntl实现的,基本的方式如下:

fcntl(fd, operation, flock);

fd是某个文件的句柄,该fd需要以与type相匹配的方式open
operation是操作类型

  • F_GETLK 读取锁信息
  • F_SETLK 设置锁,在锁已经被占用的情况下,马上返回错误,有点类似于pthread的trylock
  • F_SETLKW 设置锁,如果锁被其他进程占用,则阻塞

flock是个struct,用来传递锁的详细信息

  • short int l_type 锁的类型,可以是F_RDLCK、 F_WRLCK、F_UNLCK,分别对应读锁、写锁和解锁
  • short int l_whence 与l_start一起决定锁的起始位置,SEEK_SET、SEEK_CUR、SEEK_END分别对应文件的开始、当前位置和末尾,和fseek、lseek里的含义一致
  • off_t l_start 起始位置
  • off_t l_len 长度,0表示从l_start到文件的末尾。据说某些实现支持负数
  • pid_t l_pid 拥有锁的进程,operation为GETLK的时候会被设置

fd所对应的文件,本身不需要有数据。

由operation的取值,flock的定义可以看出,其实记录锁非常灵活。
它既可以实现排他锁(F_WRLCK),也可以实现共享锁(F_RDLCK);
同时也支持同步锁(F_SETLKW)和异步锁(F_SETLK)。

记录锁的另一个好处时,进程退出时,会自动释放掉自己所占用的锁。
这就避免了进程异常退出时资源无法回收的问题。

速度也是需要考虑的因素,根据测试,记录锁相对最慢;
但综合考虑易用性和灵活性,我认为这样的速度损耗是可以接受的。

flock(2)是另一种实现文件锁的方法,详情可以man 2 flock。

Tagged with:
十二 24

我肯定不是第一个不知道ls是怎么实现的人。
一般大家都知道怎么在linux下打开一个文件,然后读取里面的内容;
不过怎么样遍历一个目录,可能不少人还不太清楚用什么API。

下面是一个简单的ls demo程序,可以演示操作目录的模式。

#include <sys/types.h>
#include <dirent.h>
#include <stdio.h>
#include <sys/stat.h>
#include <unistd.h>
 
int ls(char *sPathName)
{
    DIR *dir = opendir(sPathName);
    if( dir == NULL )
    {
        printf("open %s fail.\n", sPathName);
        return -1;
    }
    chdir(sPathName);
    struct dirent *file;
    while( file = readdir( dir ) )
    {
#ifdef _DIRENT_HAVE_D_TYPE
        //如果是普通文件
        if( file-&gt;d_type &amp; DT_REG )
        {
            struct stat status;
            stat(file-&gt;d_name, &amp;status);
            printf("%20s last modified time: %d\n", file-&gt;d_name, int(status.st_mtime));
        }
        else
        {
            continue;
        }
#else
 
#endif
    }
    return 0;
}
 
int main(int c, char **v)
{
    return ls(v[1]);
}

ls的详细实现,可以在这里找到。

Tagged with:
preload preload preload

无觅相关文章插件,快速提升流量