Meteor-Android-DDP通信 - 简书

mikel阅读(1942)

来源: Meteor-Android-DDP通信 – 简书

版本要求

配置

  • 将需要的包加入项目
    • 在根目录下的build.gradle里加如下两段代码
      • 申明Gradle仓库地址
        allprojects {
          repositories {
              maven { url "https://jitpack.io" }
          }
        }
      • 申明Gradle的依赖
        dependencies {
          compile 'com.github.delight-im:Android-DDP:v3.1.2'
        }
  • AndroidManifest.xml 中添加许可:
     <uses-permission android:name="android.permission.INTERNET" />

使用

  • 创建DDP client实例
    public class MyActivity extends Activity implements MeteorCallback {
    
        private Meteor mMeteor;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
    
            // 创建一个实例
            mMeteor = new Meteor(this, "ws://example.meteor.com/websocket");
    
            // 注册回调函数
            mMeteor.addCallback(this);
    
            // 建立连接
            mMeteor.connect();
        }
    
        public void onConnect(boolean signedInAutomatically) { }
    
        public void onDisconnect() { }
    
        public void onDataAdded(String collectionName, String documentID, String newValuesJson) {
            // 自行解析json数据 (不推荐)
            // 或者
            // 直接用数据库语句操作 (后面会有详细介绍) (推荐)
        }
    
        public void onDataChanged(String collectionName, String documentID, String updatedValuesJson, String removedValuesJson) {
            // 自行解析json数据 (不推荐)
            // 或者
            // 直接用数据库语句操作 (后面会有详细介绍) (推荐)
        }
    
        public void onDataRemoved(String collectionName, String documentID) {
            // 自行解析json数据 (不推荐)
            // 或者
            // 直接用数据库语句操作 (后面会有详细介绍) (推荐)
        }
    
        public void onException(Exception e) { }
    
        @Override
        public void onDestroy() {
            mMeteor.disconnect();
            mMeteor.removeCallback(this);
            // or
            // mMeteor.removeCallbacks();
    
            // ...
    
            super.onDestroy();
        }
    
    }
  • 单例模式
    • 在一开始就创建一个实例
      MeteorSingleton.createInstance(this, "ws://example.meteor.com/websocket")
      // 代替
      // new Meteor(this, "ws://example.meteor.com/websocket")
    • 实例可以直接用 (通过 Activity 实例)
      MeteorSingleton.getInstance()
      // 代替
      // mMeteor
    • 所有API方法都可以通过MeteorSingleton.getInstance() 调用。
  • 注册一个回调函数
    // MeteorCallback callback;
    mMeteor.addCallback(callback);
  • 取消一个回调函数
    mMeteor.removeCallbacks();
    // or
    // // MeteorCallback callback;
    // mMeteor.removeCallback(callback);
  • 向 collection 中添加数据
    Map<String, Object> values = new HashMap<String, Object>();
    values.put("_id", "my-id");
    values.put("some-key", "some-value");
    
    mMeteor.insert("my-collection", values);
    // or
    // mMeteor.insert("my-collection", values, new ResultListener() { });
  • 更新 collection 中的数据
    Map<String, Object> query = new HashMap<String, Object>();
    query.put("_id", "my-id");
    
    Map<String, Object> values = new HashMap<String, Object>();
    values.put("some-key", "some-value");
    
    mMeteor.update("my-collection", query, values);
    // or
    // mMeteor.update("my-collection", query, values, options);
    // or
    // mMeteor.update("my-collection", query, values, options, new ResultListener() { });
  • 删除 collection 中的数据
    mMeteor.remove("my-collection", "my-id");
    // or
    // mMeteor.remove("my-collection", "my-id", new ResultListener() { });
  • 从后端订阅数据
    String subscriptionId = mMeteor.subscribe("my-subscription");
    // or
    // String subscriptionId = mMeteor.subscribe("my-subscription", new Object[] { arg1, arg2 });
    // or
    // String subscriptionId = mMeteor.subscribe("my-subscription", new Object[] { arg1, arg2 }, new SubscribeListener() { });
  • 取消订阅
    mMeteor.unsubscribe(subscriptionId);
    // or
    // mMeteor.unsubscribe(subscriptionId, new UnsubscribeListener() { });
  • 调用后端自定义的方法
    mMeteor.call("myMethod");
    // or
    // mMeteor.call("myMethod", new Object[] { arg1, arg2 });
    // or
    // mMeteor.call("myMethod", new ResultListener() { });
    // or
    // mMeteor.call("myMethod", new Object[] { arg1, arg2 }, new ResultListener() { });
  • 与后端断开连接
    mMeteor.disconnect();
  • 创建新账户 (后端依赖包 accounts-password)
    mMeteor.registerAndLogin("john", "john.doe@example.com", "password", new ResultListener() { });
    // or
    // mMeteor.registerAndLogin("john", "john.doe@example.com", "password", profile, new ResultListener() { });
  • 通过 username 登录 (依赖后端包 accounts-password)
    mMeteor.loginWithUsername("john", "password", new ResultListener() { });
  • 通过邮箱登录 (依赖后端包 accounts-password)
    mMeteor.loginWithEmail("john.doe@example.com", "password", new ResultListener() { });
  • 检查是否已经登录 (依赖后端包 accounts-password)
    mMeteor.isLoggedIn();
  • 获取 user ID (前提是已登录) (依赖后端包 accounts-password)
    mMeteor.getUserId();
  • 登出 (依赖后端包 accounts-password)
    mMeteor.logout();
    // or
    // mMeteor.logout(new ResultListener() { });
  • 检查客户端是否连接
    mMeteor.isConnected();
  • 手动重连 (一般不需要)
    mMeteor.reconnect();

利用数据库语句处理数据

配置数据库

InMemoryDatabaseDatabase 实例的唯一子类,代码如下:

mMeteor = new Meteor(this, "ws://example.meteor.com/websocket", new InMemoryDatabase());

此时,所有数据都自动从后端接收,并解析完成存入数据库。不需要手动解析JSON!

接收数据时,注意一些状态, onDataAdded, onDataChanged or onDataRemoved

访问数据库

Database database = mMeteor.getDatabase();

这些和后续的方法,都支持链式调用。

通过集合名从数据库获取集合

// String collectionName = "myCollection";
Collection collection = mMeteor.getDatabase().getCollection(collectionName);

获取数据库中所有的集合名字

String[] collectionNames = mMeteor.getDatabase().getCollectionNames();

获取数据库中集合数量

int numCollections = mMeteor.getDatabase().count();

通过ID从集合中取数据

// String documentId = "wjQvNQ6sGjzLMDyiJ";
Document document = mMeteor.getDatabase().getCollection(collectionName).getDocument(documentId);

获取集合中所有数据的ID

String[] documentIds = mMeteor.getDatabase().getCollection(collectionName).getDocumentIds();

获取集合中数据数量

int numDocuments = mMeteor.getDatabase().getCollection(collectionName).count();

在集合中查找数据

以下所有方法都支持链式调用,并随意组合。

// String fieldName = "age";
// int fieldValue = 62;
Query query = mMeteor.getDatabase().getCollection(collectionName).whereEqual(fieldName, fieldValue);
// String fieldName = "active";
// int fieldValue = false;
Query query = mMeteor.getDatabase().getCollection(collectionName).whereNotEqual(fieldName, fieldValue);
// String fieldName = "accountBalance";
// float fieldValue = 100000.00f;
Query query = mMeteor.getDatabase().getCollection(collectionName).whereLessThan(fieldName, fieldValue);
// String fieldName = "numChildren";
// long fieldValue = 3L;
Query query = mMeteor.getDatabase().getCollection(collectionName).whereLessThanOrEqual(fieldName, fieldValue);
// String fieldName = "revenue";
// double fieldValue = 0.00;
Query query = mMeteor.getDatabase().getCollection(collectionName).whereGreaterThan(fieldName, fieldValue);
// String fieldName = "age";
// int fieldValue = 21;
Query query = mMeteor.getDatabase().getCollection(collectionName).whereGreaterThanOrEqual(fieldName, fieldValue);
// String fieldName = "address";
Query query = mMeteor.getDatabase().getCollection(collectionName).whereNull(fieldName);
// String fieldName = "modifiedAt";
Query query = mMeteor.getDatabase().getCollection(collectionName).whereNotNull(fieldName);
Document[] documents = mMeteor.getDatabase().getCollection(collectionName).find();
// int limit = 30;
Document[] documents = mMeteor.getDatabase().getCollection(collectionName).find(limit);
// int limit = 30;
// int offset = 5;
Document[] documents = mMeteor.getDatabase().getCollection(collectionName).find(limit, offset);
Document document = mMeteor.getDatabase().getCollection(collectionName).findOne();

可以组合到一起,如下:

Document document = mMeteor.getDatabase().getCollection("users").whereNotNull("lastLoginAt").whereGreaterThan("level", 3).findOne();

通过属性名查找属性

// String fieldName = "age";
Object field = mMeteor.getDatabase().getCollection(collectionName).getDocument(documentId).getField(fieldName);

获取所有属性名

String[] fieldNames = mMeteor.getDatabase().getCollection(collectionName).getDocument(documentId).getFieldNames();

获取属性数量

int numFields = mMeteor.getDatabase().getCollection(collectionName).getDocument(documentId).count();

Mongodb基础用法及查询操作[转载] - 周国选 - 博客园

mikel阅读(1147)

来源: Mongodb基础用法及查询操作[转载] – 周国选 – 博客园

插入多条测试数据
> for(i=1;i<=1000;i++){
… db.blog.insert({“title”:i,”content”:”mongodb测试文章。”,”name”:”刘”+i});
… }

db.blog.list.find().limit(10).forEach(function(data){print(“title:”+data.title);})   循环forEach 用法

 

db.blog.findOne();  取一条数据

db.blog.find();取多条数据
db.blog.remove(); 删除数据集
db.blog.drop();删除表

删除一个数据库:
1.use dbname
2.db.dropDatabase()
======================================查询操作=============================
db.blog.find() 相当于select * from blog
db.blog.find({“age”:27}) 相当于select * from blog where age=’27’
db.blog.find({“age”:27,”name”:”xing”}) 相当于select * from blog where age=’27’ and name=”xing”
db.blog.find({},{“name”:1}) select name from blog ,如果name:0则是不显示name字段
db.blog.find().limit(1) 相当于select * from blog limit 1

db.blog.find().sort({_id:-1}).limit(1)相当于select * from blog order by _id desc limit 1
db.blog.find().skip(10).limit(20) 相当于select * from blog limit 10,20
skip用的时候,一定要注意要是数量多的话skip就会变的很慢,所有的数据库都存在此问题,可以不用skip进行分页,用最后一条记录做为条件
db.blog.find({“age”:{“$gte”:18,”$lte”:30}})     select * from blog  where age>=27 and age<=50
$gt   >
$gte  >=
$lt   <
$lte  <=
$ne   !=
$in : in
$nin: not in
$all: all
$not: 反匹配

查询 creation_date > ‘2010-01-01’ and creation_date <= ‘2010-12-31’ 的数据
db.users.find({creation_date:{$gt:new Date(2010,0,1), $lte:new Date(2010,11,31)});

db.blog.find().sort({_id:-1})  相当于select * from blog  order by _id desc  按_id倒序取数据  1为正序,多个条件用,号分开如{name:1,age:-1}
db.blog.find({“_id”:{“$in”,[12,3,100]}})  相当于select * from blog where _id in (12,3,100)
db.blog.find({“_id”:{“$nin”,[12,3,100]}})  相当于select * from blog where _id not in (12,3,100)
db.blog.find({“$or”:[{“age”:16},{“name”:”xing”}]}) 相当于select * from blog where age = 16 or name = ‘xing’
db.blog.find({“id_num”:{“$mod”:[5,1]}}) 取的是id_num mod 5 = 1 的字段,如id_num=1,6,11,16
db.blog.find({“id_num”:{“$not”:{“$mod”:[5,1]}}}) 取的是id_num mod 5 != 1 的字段,如除了id_num=1,6,11,16等所有字段,多于正则一起用

$exists判断字段是否存在

db.blog.find({ a : { $exists : true }}); // 如果存在元素a,就返回
db.blog.find({ a : { $exists : false }}); // 如果不存在元素a,就返回
$type判断字段类型 
查询所有name字段是字符类型的
db.users.find({name: {$type: 2}});
查询所有age字段是整型的
db.users.find({age: {$type: 16}});
db.blog.find({“z”:null}) 返回没有z字段的所有记录
db.blog.find({“name”:/^joe/i}) 查找name=joe的所有记录,不区分大小写

db.blog.distinct(‘content’)  查指定的列,并去重
查询数组
db.blog.find({“fruit”:{“$all”:[“苹果”,”桃子”,”梨”]}})   fruit中必需有数组中的每一个才符合结果
db.blog.find({“fruit”:{“$size”:3}})  fruit数组长度为3的符合结果
db.blog.find({“$push”:{“fruit”:”桔子”}})相当于db.blog.find({“$push”:{“fruit”:”桔子”},”$inc”:{“size”:1}})
$slice 可以按偏移量返回记录,针对数组。如{“$slice”:10}返回前10条,{“$slice”:{[23,10]}}从24条取10条
如果对象有一个元素是数组,那么$elemMatch可以匹配内数组内的元素

db.people.find({“name.first”:”joe”,”name.last”:”schmoe”}) 子查询如:{“id”:34,”name”:{“first”:”joe”,”last”:”schmoe”}}

db.blog.find({“comments”:{“$elemMatch”:{“author”:”joe”,”score”:{“$gte”:5}}}}) 查joe发表的5分以上的评论,注意comments为二维数组
$where 在走投无路的时候可以用,但它的效率是很低的。

游标用法
cursor.hasNext()检查是否有后续结果存在,然后用cursor.next()将其获得。
>while(cursor.hasNext()){
var obj = cursor.next();
//do same
}

 

 

http://www.mongodb.org/display/DOCS/Advanced+Queries#AdvancedQueries-ConditionalOperators%3A%3C%2C%3C%3D%2C%3E%2C%3E%3D 手册

 

> use blog
> db.blog.insert({“title”:”华夏之星的博客”,”content”:”mongodb测试文章。”});
> db.blog.find();
{ “_id” : ObjectId(“4e29fd262ed6910732fa61df”), “title” : “华夏之星的博客”, “content” : “mongodb测试文章。” }
> db.blog.update({title:”华夏之星的博客”},{“author”:”星星”,”content”:”测试更新”});
> db.blog.find();
{ “_id” : ObjectId(“4e29fd262ed6910732fa61df”), “author” : “星星”, “content” : “测试更新” }

 

 

db.blog.insert不带括号则显示源码
db.blog.insert();插入
db.blog.update();更新

> db.blog.update({title:”华夏之星的博客”},{“author”:”星星”,”content”:”测试更新”});

update默认情况下只能对符合条件的第一个文档执行操作,要使所有的匹配的文档都得到更新,可以设置第四个参数为 true

> db.blog.update({title:”华夏之星的博客”},{“author”:”星星”,”content”:”测试更新”},false,true);

> db.runCommand({getLastError:1}) 可以查看更新了几条信息,n就是条数

 

 备份blog数据库到/soft目录
/usr/local/webserver/mongodb/bin/mongodump -d blog -o /soft

还原数据库

/usr/local/webserver/mongodb/bin/mongorestore -d blog -c blog   /soft/blog/blog.bson

导出数据库(备份出来的数据是二进制的,已经经过压缩。)
/usr/local/webserver/mongodb/bin/mongodump -h 127.0.0.1 -port 27017 -d demo -o /tmp/demo
导入数据
/usr/local/webserver/mongodb/bin/mongorestore -h 127.0.0.1  -port 27017 –directoryperdb /tmp/demo

5) $all
$all和$in类似,但是他需要匹配条件内所有的值:
如有一个对象:

{ a: [ 1, 2, 3 ] }

下面这个条件是可以匹配的:

db.things.find( { a: { $all: [ 2, 3 ] } } );

但是下面这个条件就不行了:

db.things.find( { a: { $all: [ 2, 3, 4 ] } } );

6) $size
$size是匹配数组内的元素数量的,如有一个对象:{a:[“foo”]},他只有一个元素:
下面的语句就可以匹配:

db.things.find( { a : { $size: 1 } } );

官网上说不能用来匹配一个范围内的元素,如果想找$size<5之类的,他们建议创建一个字段来保存元素的数量。

 

8) $type

$type 基于 bson type来匹配一个元素的类型,像是按照类型ID来匹配,不过我没找到bson类型和id对照表。

db.things.find( { a : { $type : 2 } } ); // matches if a is a string
db.things.find( { a : { $type : 16 } } ); // matches if a is an int

9)正则表达式
mongo支持正则表达式,如:

db.customers.find( { name : /acme.*corp/i } ); // 后面的i的意思是区分大小写

10) 查询数据内的值
下面的查询是查询colors内red的记录,如果colors元素是一个数据,数据库将遍历这个数组的元素来查询。

db.things.find( { colors : “red” } );

11) $elemMatch
如果对象有一个元素是数组,那么$elemMatch可以匹配内数组内的元素:

> t.find( { x : { $elemMatch : { a : 1, b : { $gt : 1 } } } } )
{ “_id” : ObjectId(“4b5783300334000000000aa9”),
“x” : [ { “a” : 1, “b” : 3 }, 7, { “b” : 99 }, { “a” : 11 } ]
}

$elemMatch : { a : 1, b : { $gt : 1 } } 所有的条件都要匹配上才行。

注意,上面的语句和下面是不一样的。

> t.find( { “x.a” : 1, “x.b” : { $gt : 1 } } )

$elemMatch是匹配{ “a” : 1, “b” : 3 },而后面一句是匹配{ “b” : 99 }, { “a” : 11 }
12) 查询嵌入对象的值

db.postings.find( { “author.name” : “joe” } );

注意用法是author.name,用一个点就行了。更详细的可以看这个链接: dot notation

举个例子:

> db.blog.save({ title : “My First Post”, author: {name : “Jane”, id : 1}})

如果我们要查询 authors name 是Jane的, 我们可以这样:

> db.blog.findOne({“author.name” : “Jane”})

 

 

 

 
mongodb目前没有或(or)操作符,只能用变通的办法代替,可以参考下面的链接:

http://www.mongodb.org/display/DOCS/OR+operations+in+query+expressions

http://www.bumao.com/index.php/mongo_and_php

 

http://www.php.net/manual/en/mongocursor.count.php

ionic/cordova即时通讯解决方案(上) - 空灵羽少 - 博客园

mikel阅读(1304)

来源: ionic/cordova即时通讯解决方案(上) – 空灵羽少 – 博客园

webAPP即时通讯解决方案一开始总是找一些web端的第三方,其实做移动端还是比较推荐使用插件引入原生的第三方比较好。当然也试过用WebSocket协议来做,之前尝试过但是在PC上完美实现,当时在移动机上出了些问题,但是开发任务紧张后来放弃了。这里本章简单介绍使用融云的cordova来实现即时通讯。

相关文档http://www.rongcloud.cn/docs/cordova.html

 

1,在融云官网注册一个开发者账号,创建应用获取AppKey,AppSecret

2,在自己项目中导入插件,命令如下

3,让服务器端配置AppSecret为前端提供一个接口,通过UserId等参数返回Token
4,前端拿到Token之后连接融云SDK:

初始化sdk

RongCloudLibPlugin.init({

             appKey: “z3v46kbv8v30”

},
function(ret, err) {
if (ret) {
console.log(‘init:’ + JSON.stringify(ret));
}
if (err) {
console.log(‘init error:’ + JSON.stringify(err));
}
} );

获取账号状态

       RongCloudLibPlugin.setConnectionStatusListener(
function(ret, err) {
if (ret) {
console.log(‘setConnectionStatusListener:’ + JSON.stringify(ret));
if(ret.result.connectionStatus == ‘KICKED’){
alert(‘您的帐号已在其他端登录!’);
$rootScope.hideTabs = false;
$ionicHistory.clearCache();
$state.go(‘login’);
}
}
if (err) {
console.log(‘setConnectionStatusListener error:’ + JSON.stringify(err));
}
});

       连接融云
RongCloudLibPlugin.connect({

服务器接口获取的token
token: token

},
function(ret, err) {
if (ret) {
console.log(‘connect:’ + JSON.stringify(ret));
$rootScope.curUID = ret.result.userId;
$rootScope.$apply();
myUtil.setUserId(ret.result.userId);
$state.go(‘tab.friends’, {
userId: ret.result.userId
}, {
reload: true
});
}
if (err) {
console.log(‘init error:’ + JSON.stringify(err));
}
});
监听消息

RongCloudLibPlugin.setOnReceiveMessageListener(
function(ret, err) {
if (ret) {
console.log(‘setOnReceiveMessageListener:’ + JSON.stringify(ret));
var data=JSON.stringify(ret)
$rootScope.arrMsgs.push(JSON.stringify(ret.result.message));
$rootScope.$apply();
}
if (err) {
console.log(‘setOnReceiveMessageListener error:’ + JSON.stringify(err));
}
});

 

消息列表

RongCloudLibPlugin.getConversationList(
function(ret, err) {
if (ret) {
console.log(‘setOnReceiveMessageListener:’ + JSON.stringify(ret));
var data=JSON.stringify(ret)
$rootScope.arrMsgs.push(JSON.stringify(ret.result.message));
$rootScope.$apply();
}
if (err) {
console.log(‘setOnReceiveMessageListener error:’ + JSON.stringify(err));
}
});

 

聊天历史数据

RongCloudLibPlugin.getHistoryMessages({
对话类型(单聊)
conversationType:’PRIVATE’,

对话目标ID
targetId:’1234′,

count: 5,
oldestMessageId: oldestMessageId
},
function(ret, err) {
if (ret) {
console.log(“getHistoryMessages:” + JSON.stringify(ret));
var result = new Array(),tmp;
for (var i = ret.result.length – 1; i >= 0; i–) {
tmp = ret.result[i];
tmp = myUtil.resolveMsg(tmp);
result.push(tmp);
}
var hisArr = result.concat($scope.hisMsgs);
$scope.hisMsgs = hisArr;
}
if (err) {
alert(“getHistoryMessages error: ” + JSON.stringify(err));
}
});

获取对话

RongCloudLibPlugin.getConversation({

对话类型(单聊)
conversationType:’PRIVATE’,

对话目标ID
targetId:’1234′,
},
function(ret, err) {
if (ret) {
console.log(‘setOnReceiveMessageListener:’ + JSON.stringify(ret));
var data=JSON.stringify(ret)
$rootScope.arrMsgs.push(JSON.stringify(ret.result.message));
$rootScope.$apply();
}
if (err) {
console.log(‘setOnReceiveMessageListener error:’ + JSON.stringify(err));
}
});

发送消息

RongCloudLibPlugin.sendTextMessage({

conversationType: ‘PRIVATE’,

               targetId: ‘9527’,

               text: ‘Hello world.’, extra: ” },

               function (ret, err) {

                  if (ret.status == ‘prepare’) alert(JSON.stringify(ret.result.message) );

                 else if (ret.status == ‘success’) alert(ret.result.message.messageId );

                 else if (ret.status == ‘error’) alert(err.code );

               } );

 

5,聊天核心功能都已经列出来下面就是配上UI了下一节写界面部分包括小红点和未读数处理。

使用web方式开发cordova(ionic)应用即时聊天(im)功能(1) - Jax2000的学习旅程 - SegmentFault

mikel阅读(1042)

来源: 使用web方式开发cordova(ionic)应用即时聊天(im)功能(1) – Jax2000的学习旅程 – SegmentFault

最近在用ionic做一个app的时候,客户那边有即时聊天的需求,需要可以发文本、语音、图片和地理位置。为了快速开发,所以选择了采用第三方的库来开发,国内这方面做的比较好的有融云野狗leanCloud,最后选择了融云来做,理由有:

  1. 我的需求都在他们的免费服务里,我不需要交钱(〃^∇^)ぇ∧∧∧っ
  2. 看了下文档(web)足够简单,接口很清晰
  3. 他们的技术工作人员服务很到位,我在他们的工单库(也就是个提问平台)中提出的问题,他们会很快给我解答,而且回答的很到位,都解决或者指出了我的问题

因为我是用ionic开发也就是用h5跨平台的方式开发app,所以有两个选择,一个是直接用web的方式来做,一个是用cordova的插件的方式来做,这里我没有用cordova插件的方式来做的主要原因是,没法调试
web的方式的话,他们又提供了一种聊天插件,可以就几行代码就把聊天功能集成进来,不过我没有选择,因为那个界面不太适合移动端(人家本来就是为pc的用户编写的吗),而且没有集成语音和发送地理位置的功能。下面进入正题。

先提供几个资源:

大家通过开发者文档,肯定可以把从注册开发者账号到到初始化sdk到获取token的步骤全做了,我就不介绍了,不过中间有个坑需要注意一下。就是我用web sdk在cordova生成的Android上使用,发现连接不上,一直没响应,但是在pc上的开发过程是一点问题都没有的,最后还是他们的工作人员给了我解答,就是我们可能需要使用长连接的方式,因为我们生成的应用可能不支持webSocket,所以我们需要在引入sdk前设置:

<script type="text/javascript">

    window["SCHEMETYPE"] = "http";

    window.WEB_XHR_POLLING = true;

</script>
<script src="http://cdn.ronghub.com/RongIMLib-2.1.3.min.js"></script>

注意需要在引入融云的库之前执行
到此为止我们真正开始开发的前置工作就完成了。

转 push本地代码到github出错

mikel阅读(1013)

来源: 转 push本地代码到github出错

刚创建的github版本库,在push代码时出错:

$ git push -u origin master
To git@github.com:******/Demo.git
! [rejected] master -> master (non-fast-forward)
error: failed to push some refs to ‘git@github.com:******/Demo.git’
hint: Updates were rejected because the tip of your current branch is behind
hint: its remote counterpart. Merge the remote changes (e.g. ‘git pull’)
hint: before pushing again.
hint: See the ‘Note about fast-forwards’ in ‘git push –help’ for details.

网上搜索了下,是因为远程repository和我本地的repository冲突导致的,而我在创建版本库后,在github的版本库页面点击了创建README.md文件的按钮创建了说明文档,但是却没有pull到本地。这样就产生了版本冲突的问题。

有如下几种解决方法:

1.使用强制push的方法:

$ git push -u origin master -f

这样会使远程修改丢失,一般是不可取的,尤其是多人协作开发的时候。

2.push前先将远程repository修改pull下来

$ git pull origin master

$ git push -u origin master

3.若不想merge远程和本地修改,可以先创建新的分支:

$ git branch [name]

然后push

$ git push -u origin [name]

MongoDB学习笔记(查询) - Stephen_Liu - 博客园

mikel阅读(1025)

来源: MongoDB学习笔记(查询) – Stephen_Liu – 博客园

1.  基本查询:
构造查询数据。
> db.test.findOne()
{
“_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”),
“name” : “stephen”,
“age” : 35,
“genda” : “male”,
“email” : “stephen@hotmail.com
}

–多条件查询。下面的示例等同于SQL语句的where name = “stephen” and age = 35
> db.test.find({“name”:”stephen”,”age”:35})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “name” : “stephen”, “age” : 35, “genda” : “male”, “email” : “stephen@hotmail.com” }

–返回指定的文档键值对。下面的示例将只是返回name和age键值对。
> db.test.find({}, {“name”:1,”age”:1})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “name” : “stephen”, “age” : 35 }

 

–指定不返回的文档键值对。下面的示例将返回除name之外的所有键值对。
> db.test.find({}, {“name”:0})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “age” : 35, “genda” : “male”, “email” : “stephen@hotmail.com” }

2.  查询条件:
MongoDB提供了一组比较操作符:$lt/$lte/$gt/$gte/$ne,依次等价于</<=/>/>=/!=。
–下面的示例返回符合条件age >= 18 && age <= 40的文档。
> db.test.find({“age”:{“$gte”:18, “$lte”:40}})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “name” : “stephen”, “age” : 35,”genda” : “male”, “email” : “stephen@hotmail.com” }

–下面的示例返回条件符合name != “stephen1”
> db.test.find({“name”:{“$ne”:”stephen1″}})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “name” : “stephen”, “age” : 35,”genda” : “male”, “email” : “stephen@hotmail.com” }

–$in等同于SQL中的in,下面的示例等同于SQL中的in (“stephen”,”stephen1″)
> db.test.find({“name”:{“$in”:[“stephen”,”stephen1″]}})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “name” : “stephen”, “age” : 35,”genda” : “male”, “email” : “stephen@hotmail.com” }

–和SQL不同的是,MongoDB的in list中的数据可以是不同类型。这种情况可用于不同类型的别名场景。
> db.test.find({“name”:{“$in”:[“stephen”,123]}})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “name” : “stephen”, “age” : 35,”genda” : “male”, “email” : “stephen@hotmail.com” }

–$nin等同于SQL中的not in,同时也是$in的取反。如:
> db.test.find({“name”:{“$nin”:[“stephen2″,”stephen1”]}})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “name” : “stephen”, “age” : 35,”genda” : “male”, “email” : “stephen@hotmail.com” }

–$or等同于SQL中的or,$or所针对的条件被放到一个数组中,每个数组元素表示or的一个条件。
–下面的示例等同于name = “stephen1” or age = 35
> db.test.find({“$or”: [{“name”:”stephen1″}, {“age”:35}]})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “name” : “stephen”, “age” : 35,”genda” : “male”, “email” : “stephen@hotmail.com” }

–下面的示例演示了如何混合使用$or和$in。
> db.test.find({“$or”: [{“name”:{“$in”:[“stephen”,”stephen1″]}}, {“age”:36}]})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “name” : “stephen”, “age” : 35,”genda” : “male”, “email” : “stephen@hotmail.com” }

–$not表示取反,等同于SQL中的not。
> db.test.find({“name”: {“$not”: {“$in”:[“stephen2″,”stephen1”]}}})
{ “_id” : ObjectId(“4fd58ecbb9ac507e96276f1a”), “name” : “stephen”, “age” : 35,”genda” : “male”, “email” : “stephen@hotmail.com” }

 

3.  null数据类型的查询:
–在进行值为null数据的查询时,所有值为null,以及不包含指定键的文档均会被检索出来。
> db.test.find({“x”:null})
{ “_id” : ObjectId(“4fd59d30b9ac507e96276f1b”), “x” : null }
{ “_id” : ObjectId(“4fd59d49b9ac507e96276f1c”), “y” : 1 }

–需要将null作为数组中的一个元素进行相等性判断,即便这个数组中只有一个元素。
–再有就是通过$exists判断指定键是否存在。
> db.test.find({“x”: {“$in”: [null], “$exists”:true}})
{ “_id” : ObjectId(“4fd59d30b9ac507e96276f1b”), “x” : null }

4.  正则查询:
–MongoDB中使用了Perl规则的正则语法。如:
> db.test.find()
{ “_id” : ObjectId(“4fd59ed7b9ac507e96276f1d”), “name” : “stephen” }
{ “_id” : ObjectId(“4fd59edbb9ac507e96276f1e”), “name” : “stephen1” }
–i表示忽略大小写
> db.test.find({“name”:/stephen?/i})
{ “_id” : ObjectId(“4fd59ed7b9ac507e96276f1d”), “name” : “stephen” }
{ “_id” : ObjectId(“4fd59edbb9ac507e96276f1e”), “name” : “stephen1” }

5.  数组数据查询:
–基于数组的查找。
> db.test.find()
{ “_id” : ObjectId(“4fd5a177b9ac507e96276f1f”), “fruit” : [ “apple”, “banana”, “peach” ] }
{ “_id” : ObjectId(“4fd5a18cb9ac507e96276f20”), “fruit” : [ “apple”, “kumquat”,”orange” ] }
{ “_id” : ObjectId(“4fd5a1f0b9ac507e96276f21”), “fruit” : [ “cherry”, “banana”,”apple” ] }
–数组中所有包含banana的文档都会被检索出来。
> db.test.find({“fruit”:”banana”})
{ “_id” : ObjectId(“4fd5a177b9ac507e96276f1f”), “fruit” : [ “apple”, “banana”, “peach” ] }
{ “_id” : ObjectId(“4fd5a1f0b9ac507e96276f21”), “fruit” : [ “cherry”, “banana”,”apple” ] }
–检索数组中需要包含多个元素的情况,这里使用$all。下面的示例中,数组中必须同时包含apple和banana,但是他们的顺序无关紧要。
> db.test.find({“fruit”: {“$all”: [“banana”,”apple”]}})
{ “_id” : ObjectId(“4fd5a177b9ac507e96276f1f”), “fruit” : [ “apple”, “banana”, “peach” ] }
{ “_id” : ObjectId(“4fd5a1f0b9ac507e96276f21”), “fruit” : [ “cherry”, “banana”, “apple” ] }
–下面的示例表示精确匹配,即被检索出来的文档,fruit值中的数组数据必须和查询条件完全匹配,即不能多,也不能少,顺序也必须保持一致。
> db.test.find({“fruit”:[“apple”,”banana”,”peach”]})
{ “_id” : ObjectId(“4fd5a177b9ac507e96276f1f”), “fruit” : [ “apple”, “banana”, peach” ] }
–下面的示例将匹配数组中指定下标元素的值。数组的起始下标是0。
> db.test.find({“fruit.2″:”peach”})
{ “_id” : ObjectId(“4fd5a177b9ac507e96276f1f”), “fruit” : [ “apple”, “banana”, peach” ] }
–可以通过$size获取数组的长度,但是$size不能和比较操作符联合使用。
> db.test.find({“fruit”: {$size : 3}})
{ “_id” : ObjectId(“4fd5a177b9ac507e96276f1f”), “fruit” : [ “apple”, “banana”, “peach” ] }
{ “_id” : ObjectId(“4fd5a18cb9ac507e96276f20”), “fruit” : [ “apple”, “kumquat”,”orange” ] }
{ “_id” : ObjectId(“4fd5a1f0b9ac507e96276f21”), “fruit” : [ “cherry”, “banana”,”apple” ] }
–如果需要检索size > n的结果,不能直接使用$size,只能是添加一个额外的键表示数据中的元素数据,在操作数据中的元素时,需要同时更新size键的值。
–为后面的实验构造数据。
> db.test.update({}, {“$set”: {“size”:3}},false,true)
> db.test.find()
{ “_id” : ObjectId(“4fd5a18cb9ac507e96276f20”), “fruit” : [ “apple”, “kumquat”, “orange” ], “size” : 3 }
{ “_id” : ObjectId(“4fd5a1f0b9ac507e96276f21”), “fruit” : [ “cherry”, “banana”, “apple” ], “size” : 3 }
–每次添加一个新元素,都要原子性的自增size一次。
> test.update({},{“$push”: {“fruit”:”strawberry”},”$inc”:{“size”:1}},false,true)
> db.test.find()
{ “_id” : ObjectId(“4fd5a18cb9ac507e96276f20”), “fruit” : [ “apple”, “kumquat”, “orange”, “strawberry” ], “size” : 4 }
{ “_id” : ObjectId(“4fd5a1f0b9ac507e96276f21”), “fruit” : [ “cherry”, “banana”, “apple”, “strawberry” ], “size” : 4 }
–通过$slice返回数组中的部分数据。”$slice”:2表示数组中的前两个元素。
> db.test.find({},{“fruit”: {“$slice”:2}, “size”:0})
{ “_id” : ObjectId(“4fd5a18cb9ac507e96276f20”), “fruit” : [ “apple”, “kumquat” ]}
{ “_id” : ObjectId(“4fd5a1f0b9ac507e96276f21”), “fruit” : [ “cherry”, “banana” ]}
–通过$slice返回数组中的部分数据。”$slice”:-2表示数组中的后两个元素。
> db.test.find({},{“fruit”: {“$slice”:-2}, “size”:0})
{ “_id” : ObjectId(“4fd5a18cb9ac507e96276f20”), “fruit” : [ “orange”, “strawberry” ] }
{ “_id” : ObjectId(“4fd5a1f0b9ac507e96276f21”), “fruit” : [ “apple”, “strawberry” ] }
–$slice : [2,1],表示从第二个2元素开始取1个,如果获取数量大于2后面的元素数量,则取后面的全部数据。
> db.test.find({},{“fruit”: {“$slice”:[2,1]}, “size”:0})
{ “_id” : ObjectId(“4fd5a18cb9ac507e96276f20”), “fruit” : [ “orange” ] }
{ “_id” : ObjectId(“4fd5a1f0b9ac507e96276f21”), “fruit” : [ “apple” ] }

6.  内嵌文档查询:
–为后面的示例构造测试数据。
> db.test.find()
{ “_id” : ObjectId(“4fd5ada3b9ac507e96276f22”), “name” : { “first” : “Joe”, “last” : “He” }, “age” : 45 }
–当嵌入式文档为数组时,需要$elemMatch操作符来帮助定位某一个元素匹配的情况,否则嵌入式文件将进行全部的匹配。
–即检索时需要将所有元素都列出来作为查询条件方可。
> db.test.findOne()
{
“_id” : ObjectId(“4fd5af76b9ac507e96276f23”),
“comments” : [
{
“author” : “joe”,
“score” : 3
},
{
“author” : “mary”,
“score” : 6
}
]
}
> db.test.find({“comments”: {“$elemMatch”: {“author”:”joe”,”score”:{“$gte”:3}}}}
{ “_id” : ObjectId(“4fd5af76b9ac507e96276f23”), “comments” : [ { “author” : “joe”, “score” : 3 }, { “author” : “mary”, “score” : 6 } ] }

7.  游标:
数据库使用游标来返回find()的执行结果,客户端对游标可以进行有效的控制,如:限定结果集的数量、跳过部分结果、基于任意键的任意方向的排序等。
下面的例子将用于准备测试数据。
> db.testtable.remove()
> for (i = 0; i < 10; ++i) {
… db.testtable.insert({x:i})
… }
我们可以通过cursor提供的hasNext()方法判断是否还有未读取的数据,再通过next()方法读取结果集中的下一个文档。如:
> var c = db.testtable.find()
> while (c.hasNext()) {
… print(c.next().x)
… }
0
1
2
3
4
5
6
7
8
9
当调用find()的时候,shell并不立即查询数据库,而是等待真正开始要求获得结果的时候才发送查询,这样在执行之前可以给查询附加额外的选项。几乎所有的游标方法都返回本身,因此可以像下面这样将游标的方法链式组合起来。如:
> var c1 = db.testtable.find().sort({“x”:1}).limit(1).skip(4);
> var c2 = db.testtable.find().limit(1).sort({“x”:1}).skip(4);
> var c3 = db.testtable.find().skip(4).limit(1).sort({“x”:1});
此时,查询并未执行,所有这些函数都是在构造查询,当执行下面的语句时,查询将被真正执行,
> c.hasNext()
查询被发送到服务器,MongoDB服务器每次将返回一批数据,当本批被全部迭代后再从服务器读取下一批数据,直至查询结果需要的数据被全部迭代。

对于上面的示例,limit(1)表示输出结果仅为一个,如果小于1,则不输出,即limit(n)函数限定的是最多输出结果。skip(4)表示跳过查询结果中的前4个文档,如果结果小于4,则不会返回任何文档。sort({“x”:1})用于设定排序条件,即按照x键以升序(1)的方式排序,如果需要降序排序可以改为:sort({“x”:-1})。sort也可以支持多键排序,如:sort({username:1, age:-1})即先按照username进行升序排序,如果username的值相同,再以age键进行降序排序。这里需要指出的是,如果skip过多的文档,将会导致性能问题。

Git冲突:commit your changes or stash them before you can merge. - zzwdkxx的专栏 - 博客频道 - CSDN.NET

mikel阅读(955)

今天用git pull来更新代码,遇到了下面的问题: view plaincopyerror: Your local changes to the following files would be overwritten by merge:      xxx/xxx/xxx.java  Please, commit you

来源: Git冲突:commit your changes or stash them before you can merge. – zzwdkxx的专栏 – 博客频道 – CSDN.NET

今天用Git pull来更新代码,遇到了下面的问题:

  1. error: Your local changes to the following files would be overwritten by merge:
  2.     xxx/xxx/xxx.java
  3. Please, commit your changes or stash them before you can merge.
  4. Aborting

提示已经很友好了,从网友处得到的答案直接帮我解决问题。

1.stash

通常遇到这个问题,你可以直接commit你的修改;但我这次不想这样。

看看git stash是如何做的。

git stash
git pull
git stash pop

接下来diff一下此文件看看自动合并的情况,并作出相应修改。

git stash: 备份当前的工作区的内容,从最近的一次提交中读取相关内容,让工作区保证和上次提交的内容一致。同时,将当前的工作区内容保存到Git栈中。
git stash pop: 从Git栈中读取最近一次保存的内容,恢复工作区的相关内容。由于可能存在多个Stash的内容,所以用栈来管理,pop会从最近的一个stash中读取内容并恢复。
git stash list: 显示Git栈内的所有备份,可以利用这个列表来决定从那个地方恢复。
git stash clear: 清空Git栈。此时使用gitg等图形化工具会发现,原来stash的哪些节点都消失了。

2.放弃本地修改,直接覆盖之

git reset --hard
git pull

关于git reset参考这个:

http://www.cnblogs.com/craftor/archive/2012/11/04/2754140.html

Crontab的格式 - Cocowool - 博客园

mikel阅读(965)

来源: Crontab的格式 – Cocowool – 博客园

基本格式 :
*  *  *  *  *  command
分 时 日 月 周 命令

第1列表示分钟1~59 每分钟用*或者 */1表示
第2列表示小时1~23(0表示0点)
第3列表示日期1~31
第4列表示月份1~12
第5列标识号星期0~6(0表示星期天)
第6列要运行的命令

crontab文件的一些例子:

30 21 * * * /usr/local/etc/rc.d/lighttpd restart
上面的例子表示每晚的21:30重启apache。

45 4 1,10,22 * * /usr/local/etc/rc.d/lighttpd restart
上面的例子表示每月1、10、22日的4 : 45重启apache。

10 1 * * 6,0 /usr/local/etc/rc.d/lighttpd restart
上面的例子表示每周六、周日的1 : 10重启apache。

0,30 18-23 * * * /usr/local/etc/rc.d/lighttpd restart
上面的例子表示在每天18 : 00至23 : 00之间每隔30分钟重启apache。

0 23 * * 6 /usr/local/etc/rc.d/lighttpd restart
上面的例子表示每星期六的11 : 00 pm重启apache。

* */1 * * * /usr/local/etc/rc.d/lighttpd restart
每一小时重启apache

* 23-7/1 * * * /usr/local/etc/rc.d/lighttpd restart
晚上11点到早上7点之间,每隔一小时重启apache

0 11 4 * mon-wed /usr/local/etc/rc.d/lighttpd restart
每月的4号与每周一到周三的11点重启apache

0 4 1 jan * /usr/local/etc/rc.d/lighttpd restart
一月一号的4点重启apache

 

名称 : crontab

使用权限 : 所有使用者

使用方式 :

crontab file [-u user]-用指定的文件替代目前的crontab。

crontab-[-u user]-用标准输入替代目前的crontab.

crontab-1[user]-列出用户目前的crontab.

crontab-e[user]-编辑用户目前的crontab.

crontab-d[user]-删除用户目前的crontab.

crontab-c dir- 指定crontab的目录。

crontab文件的格式:M H D m d cmd.

 

M: 分钟(0-59)。

H:小时(0-23)。

D:天(1-31)。

m: 月(1-12)。

d: 一星期内的天(0~6,0为星期天)。

cmd要运行的程序,程序被送入sh执行,这个shell只有USER,HOME,SHELL这三个环境变量

说明 :

crontab 是用来让使用者在固定时间或固定间隔执行程序之用,换句话说,也就是类似使用者的时程表。-u user 是指设定指定

user 的时程表,这个前提是你必须要有其权限(比如说是 root)才能够指定他人的时程表。如果不使用 -u user 的话,就是表示设

定自己的时程表。

 

 

参数 :

crontab -e : 执行文字编辑器来设定时程表,内定的文字编辑器是 VI,如果你想用别的文字编辑器,则请先设定 VISUAL 环境变数

来指定使用那个文字编辑器(比如说 setenv VISUAL joe)

crontab -r : 删除目前的时程表

crontab -l : 列出目前的时程表

crontab file [-u user]-用指定的文件替代目前的crontab。

时程表的格式如下 :

f1 f2 f3 f4 f5 program

其中 f1 是表示分钟,f2 表示小时,f3 表示一个月份中的第几日,f4 表示月份,f5 表示一个星期中的第几天。program 表示要执

行的程序。

当 f1 为 * 时表示每分钟都要执行 program,f2 为 * 时表示每小时都要执行程序,其馀类推

当 f1 为 a-b 时表示从第 a 分钟到第 b 分钟这段时间内要执行,f2 为 a-b 时表示从第 a 到第 b 小时都要执行,其馀类推

当 f1 为 */n 时表示每 n 分钟个时间间隔执行一次,f2 为 */n 表示每 n 小时个时间间隔执行一次,其馀类推

当 f1 为 a, b, c,… 时表示第 a, b, c,… 分钟要执行,f2 为 a, b, c,… 时表示第 a, b, c…个小时要执行,其馀类推

使用者也可以将所有的设定先存放在档案 file 中,用 crontab file 的方式来设定时程表。

例子 :

#每天早上7点执行一次 /bin/ls :

0 7 * * * /bin/ls

在 12 月内, 每天的早上 6 点到 12 点中,每隔3个小时执行一次 /usr/bin/backup :

0 6-12/3 * 12 * /usr/bin/backup

周一到周五每天下午 5:00 寄一封信给 alex@domain.name :

0 17 * * 1-5 mail -s “hi” alex@domain.name < /tmp/maildata

每月每天的午夜 0 点 20 分, 2 点 20 分, 4 点 20 分….执行 echo “haha”

20 0-23/2 * * * echo “haha”

注意 :

当程序在你所指定的时间执行后,系统会寄一封信给你,显示该程序执行的内容,若是你不希望收到这样的信,请在每一行空一格之

后加上 > /dev/null 2>&1 即可

 

例子2 :

#每天早上6点10分

10 6 * * * date

#每两个小时

0 */2 * * * date

#晚上11点到早上8点之间每两个小时,早上8点

0 23-7/2,8 * * * date

#每个月的4号和每个礼拜的礼拜一到礼拜三的早上11点

0 11 4 * mon-wed date

#1月份日早上4点

0 4 1 jan * date

范例

$crontab -l 列出用户目前的crontab.

http://www.cnblogs.com/xingzc/p/5969777.html

前一天学习了 at 命令是针对仅运行一次的任务,循环运行的例行性计划任务,linux系统则是由 cron (crond) 这个系统服务来控制的。Linux 系统上面原本就有非常多的计划性工作,因此这个系统服务是默认启动的。另外, 由于使用者自己也可以设置计划任务,所以, Linux 系统也提供了使用者控制计划任务的命令 :crontab 命令。

一、crond简介

crond是linux下用来周期性的执行某种任务或等待处理某些事件的一个守护进程,与windows下的计划任务类似,当安装完成操作系统后,默认会安装此服务工具,并且会自动启动crond进程,crond进程每分钟会定期检查是否有要执行的任务,如果有要执行的任务,则自动执行该任务。

Linux下的任务调度分为两类,系统任务调度和用户任务调度。

系统任务调度:系统周期性所要执行的工作,比如写缓存数据到硬盘、日志清理等。在/etc目录下有一个crontab文件,这个就是系统任务调度的配置文件。

/etc/crontab文件包括下面几行:

[root@localhost ~]# cat /etc/crontab

SHELL=/bin/bash

PATH=/sbin:/bin:/usr/sbin:/usr/bin

MAILTO=””HOME=/

 

# run-parts

51 * * * * root run-parts /etc/cron.hourly

24 7 * * * root run-parts /etc/cron.daily

22 4 * * 0 root run-parts /etc/cron.weekly

42 4 1 * * root run-parts /etc/cron.monthly

[root@localhost ~]#

 

前四行是用来配置crond任务运行的环境变量,第一行SHELL变量指定了系统要使用哪个shell,这里是bash,第二行PATH变量指定了系统执行命令的路径,第三行MAILTO变量指定了crond的任务执行信息将通过电子邮件发送给root用户,如果MAILTO变量的值为空,则表示不发送任务执行信息给用户,第四行的HOME变量指定了在执行命令或者脚本时使用的主目录。第六至九行表示的含义将在下个小节详细讲述。这里不在多说。

用户任务调度:用户定期要执行的工作,比如用户数据备份、定时邮件提醒等。用户可以使用 crontab 工具来定制自己的计划任务。所有用户定义的crontab 文件都被保存在 /var/spool/cron目录中。其文件名与用户名一致。

 

使用者权限文件:

文件:

/etc/cron.deny

说明:

该文件中所列用户不允许使用crontab命令

文件:

/etc/cron.allow

说明:

该文件中所列用户允许使用crontab命令

文件:

/var/spool/cron/

说明:

所有用户crontab文件存放的目录,以用户名命名

 

crontab文件的含义:

用户所建立的crontab文件中,每一行都代表一项任务,每行的每个字段代表一项设置,它的格式共分为六个字段,前五段是时间设定段,第六段是要执行的命令段,格式如下:

minute   hour   day   month   week   command

其中:

minute: 表示分钟,可以是从0到59之间的任何整数。

hour:表示小时,可以是从0到23之间的任何整数。

day:表示日期,可以是从1到31之间的任何整数。

month:表示月份,可以是从1到12之间的任何整数。

week:表示星期几,可以是从0到7之间的任何整数,这里的0或7代表星期日。

command:要执行的命令,可以是系统命令,也可以是自己编写的脚本文件。

 

在以上各个字段中,还可以使用以下特殊字符:

星号(*):代表所有可能的值,例如month字段如果是星号,则表示在满足其它字段的制约条件后每月都执行该命令操作。

逗号(,):可以用逗号隔开的值指定一个列表范围,例如,“1,2,5,7,8,9”

中杠(-):可以用整数之间的中杠表示一个整数范围,例如“2-6”表示“2,3,4,5,6”

正斜线(/):可以用正斜线指定时间的间隔频率,例如“0-23/2”表示每两小时执行一次。同时正斜线可以和星号一起使用,例如*/10,如果用在minute字段,表示每十分钟执行一次。

 

二、crond服务

安装crontab:

yum install crontabs

服务操作说明:

/sbin/service crond start //启动服务

/sbin/service crond stop //关闭服务

/sbin/service crond restart //重启服务

/sbin/service crond reload //重新载入配置

查看crontab服务状态:

service crond status

手动启动crontab服务:

service crond start

查看crontab服务是否已设置为开机启动,执行命令:

ntsysv

加入开机自动启动:

chkconfig –level 35 crond on

 

三、crontab命令详解

1.命令格式:

crontab [-u user] file

crontab [-u user] [ -e | -l | -r ]

2.命令功能:

通过crontab 命令,我们可以在固定的间隔时间执行指定的系统指令或 shell script脚本。时间间隔的单位可以是分钟、小时、日、月、周及以上的任意组合。这个命令非常设合周期性的日志分析或数据备份等工作。

3.命令参数:

-u user:用来设定某个用户的crontab服务,例如,“-u ixdba”表示设定ixdba用户的crontab服务,此参数一般有root用户来运行。

file:file是命令文件的名字,表示将file做为crontab的任务列表文件并载入crontab。如果在命令行中没有指定这个文件,crontab命令将接受标准输入(键盘)上键入的命令,并将它们载入crontab。

-e:编辑某个用户的crontab文件内容。如果不指定用户,则表示编辑当前用户的crontab文件。

-l:显示某个用户的crontab文件内容,如果不指定用户,则表示显示当前用户的crontab文件内容。

-r:从/var/spool/cron目录中删除某个用户的crontab文件,如果不指定用户,则默认删除当前用户的crontab文件。

-i:在删除用户的crontab文件时给确认提示。

4.常用方法:

1). 创建一个新的crontab文件

 

在考虑向cron进程提交一个crontab文件之前,首先要做的一件事情就是设置环境变量EDITOR。cron进程根据它来确定使用哪个编辑器编辑crontab文件。9 9 %的UNIX和LINUX用户都使用vi,如果你也是这样,那么你就编辑$ HOME目录下的. profile文件,在其中加入这样一行:

EDITOR=vi; export EDITOR

然后保存并退出。不妨创建一个名为<user> cron的文件,其中<user>是用户名,例如, davecron。在该文件中加入如下的内容。

# (put your own initials here)echo the date to the console every

# 15minutes between 6pm and 6am

0,15,30,45 18-06 * * * /bin/echo ‘date’ > /dev/console

保存并退出。确信前面5个域用空格分隔。

在上面的例子中,系统将每隔1 5分钟向控制台输出一次当前时间。如果系统崩溃或挂起,从最后所显示的时间就可以一眼看出系统是什么时间停止工作的。在有些系统中,用tty1来表示控制台,可以根据实际情况对上面的例子进行相应的修改。为了提交你刚刚创建的crontab文件,可以把这个新创建的文件作为cron命令的参数:

$ crontab davecron

现在该文件已经提交给cron进程,它将每隔1 5分钟运行一次。

同时,新创建文件的一个副本已经被放在/var/spool/cron目录中,文件名就是用户名(即dave)。

2). 列出crontab文件

为了列出crontab文件,可以用:

$ crontab -l

0,15,30,45,18-06 * * * /bin/echo `date` > dev/tty1

你将会看到和上面类似的内容。可以使用这种方法在$ H O M E目录中对crontab文件做一备份:

$ crontab -l > $HOME/mycron

这样,一旦不小心误删了crontab文件,可以用上一节所讲述的方法迅速恢复。

3). 编辑crontab文件

如果希望添加、删除或编辑crontab文件中的条目,而E D I TO R环境变量又设置为v i,那么就可以用v i来编辑crontab文件,相应的命令为:

$ crontab -e

可以像使用v i编辑其他任何文件那样修改crontab文件并退出。如果修改了某些条目或添加了新的条目,那么在保存该文件时, c r o n会对其进行必要的完整性检查。如果其中的某个域出现了超出允许范围的值,它会提示你。

我们在编辑crontab文件时,没准会加入新的条目。例如,加入下面的一条:

# DT:delete core files,at 3.30am on 1,7,14,21,26,26 days of each month

30 3 1,7,14,21,26 * * /bin/find -name “core’ -exec rm {} \;

现在保存并退出。最好在crontab文件的每一个条目之上加入一条注释,这样就可以知道它的功能、运行时间,更为重要的是,知道这是哪位用户的作业。

现在让我们使用前面讲过的crontab -l命令列出它的全部信息:

$ crontab -l

# (crondave installed on Tue May 4 13:07:43 1999)

# DT:ech the date to the console every 30 minites

0,15,30,45 18-06 * * * /bin/echo `date` > /dev/tty1

# DT:delete core files,at 3.30am on 1,7,14,21,26,26 days of each month

30 3 1,7,14,21,26 * * /bin/find -name “core’ -exec rm {} \;

4). 删除crontab文件

要删除crontab文件,可以用:

$ crontab -r

5). 恢复丢失的crontab文件

如果不小心误删了crontab文件,假设你在自己的$ H O M E目录下还有一个备份,那么可以将其拷贝到/var/spool/cron/<username>,其中<username>是用户名。如果由于权限问题无法完成拷贝,可以用:

$ crontab <filename>

其中,<filename>是你在$ H O M E目录中副本的文件名。

我建议你在自己的$ H O M E目录中保存一个该文件的副本。我就有过类似的经历,有数次误删了crontab文件(因为r键紧挨在e键的右边)。这就是为什么有些系统文档建议不要直接编辑crontab文件,而是编辑该文件的一个副本,然后重新提交新的文件。

有些crontab的变体有些怪异,所以在使用crontab命令时要格外小心。如果遗漏了任何选项,crontab可能会打开一个空文件,或者看起来像是个空文件。这时敲delete键退出,不要按<Ctrl-D>,否则你将丢失crontab文件。

5.使用实例

实例1:每1分钟执行一次command

命令:

* * * * * command

 

实例2:每小时的第3和第15分钟执行

命令:

3,15 * * * * command

 

实例3:在上午8点到11点的第3和第15分钟执行

命令:

3,15 8-11 * * * command

 

实例4:每隔两天的上午8点到11点的第3和第15分钟执行

命令:

3,15 8-11 */2 * * command

 

实例5:每个星期一的上午8点到11点的第3和第15分钟执行

命令:

3,15 8-11 * * 1 command

 

实例6:每晚的21:30重启smb

命令:

30 21 * * * /etc/init.d/smb restart

 

实例7:每月1、10、22日的4 : 45重启smb

命令:

45 4 1,10,22 * * /etc/init.d/smb restart

 

实例8:每周六、周日的1 : 10重启smb

命令:

10 1 * * 6,0 /etc/init.d/smb restart

 

实例9:每天18 : 00至23 : 00之间每隔30分钟重启smb

命令:

0,30 18-23 * * * /etc/init.d/smb restart

 

实例10:每星期六的晚上11 : 00 pm重启smb

命令:

0 23 * * 6 /etc/init.d/smb restart

 

实例11:每一小时重启smb

命令:

* */1 * * * /etc/init.d/smb restart

 

实例12:晚上11点到早上7点之间,每隔一小时重启smb

命令:

* 23-7/1 * * * /etc/init.d/smb restart

 

实例13:每月的4号与每周一到周三的11点重启smb

命令:

0 11 4 * mon-wed /etc/init.d/smb restart

 

实例14:一月一号的4点重启smb

命令:

0 4 1 jan * /etc/init.d/smb restart

 

实例15:每小时执行/etc/cron.hourly目录内的脚本

命令:

01   *   *   *   *     root run-parts /etc/cron.hourly

说明:

run-parts这个参数了,如果去掉这个参数的话,后面就可以写要运行的某个脚本名,而不是目录名了

 

四、使用注意事项

1. 注意环境变量问题

有时我们创建了一个crontab,但是这个任务却无法自动执行,而手动执行这个任务却没有问题,这种情况一般是由于在crontab文件中没有配置环境变量引起的。

在crontab文件中定义多个调度任务时,需要特别注意的一个问题就是环境变量的设置,因为我们手动执行某个任务时,是在当前shell环境下进行的,程序当然能找到环境变量,而系统自动执行任务调度时,是不会加载任何环境变量的,因此,就需要在crontab文件中指定任务运行所需的所有环境变量,这样,系统执行任务调度时就没有问题了。

不要假定cron知道所需要的特殊环境,它其实并不知道。所以你要保证在shelll脚本中提供所有必要的路径和环境变量,除了一些自动设置的全局变量。所以注意如下3点:

1)脚本中涉及文件路径时写全局路径;

2)脚本执行要用到java或其他环境变量时,通过source命令引入环境变量,如:

cat start_cbp.sh

#!/bin/sh

source /etc/profile

export RUN_CONF=/home/d139/conf/platform/cbp/cbp_jboss.conf

/usr/local/jboss-4.0.5/bin/run.sh -c mev &

3)当手动执行脚本OK,但是crontab死活不执行时。这时必须大胆怀疑是环境变量惹的祸,并可以尝试在crontab中直接引入环境变量解决问题。如:

0 * * * * . /etc/profile;/bin/sh /var/www/java/audit_no_count/bin/restart_audit.sh

 

2. 注意清理系统用户的邮件日志

每条任务调度执行完毕,系统都会将任务输出信息通过电子邮件的形式发送给当前系统用户,这样日积月累,日志信息会非常大,可能会影响系统的正常运行,因此,将每条任务进行重定向处理非常重要。

例如,可以在crontab文件中设置如下形式,忽略日志输出:

0 */3 * * * /usr/local/apache2/apachectl restart >/dev/null 2>&1

“/dev/null 2>&1”表示先将标准输出重定向到/dev/null,然后将标准错误重定向到标准输出,由于标准输出已经重定向到了/dev/null,因此标准错误也会重定向到/dev/null,这样日志输出问题就解决了。

 

3. 系统级任务调度与用户级任务调度

系统级任务调度主要完成系统的一些维护操作,用户级任务调度主要完成用户自定义的一些任务,可以将用户级任务调度放到系统级任务调度来完成(不建议这么做),但是反过来却不行,root用户的任务调度操作可以通过“crontab –uroot –e”来设置,也可以将调度任务直接写入/etc/crontab文件,需要注意的是,如果要定义一个定时重启系统的任务,就必须将任务放到/etc/crontab文件,即使在root用户下创建一个定时重启系统的任务也是无效的。

4. 其他注意事项

新创建的cron job,不会马上执行,至少要过2分钟才执行。如果重启cron则马上执行。

当crontab突然失效时,可以尝试/etc/init.d/crond restart解决问题。或者查看日志看某个job有没有执行/报错tail -f /var/log/cron。

千万别乱运行crontab -r。它从Crontab目录(/var/spool/cron)中删除用户的Crontab文件。删除了该用户的所有crontab都没了。

在crontab中%是有特殊含义的,表示换行的意思。如果要用的话必须进行转义\%,如经常用的date ‘+%Y%m%d’在crontab里是不会执行的,应该换成date ‘+\%Y\%m\%d’。

crontab定时任务不执行的原因 - u011734144的专栏 - 博客频道 - CSDN.NET

mikel阅读(803)

我在网上找的时候发现网上主要说了这5个原因:1 crond服务未启动crontab不是Linux内核的功能,而是依赖一个crond服务,这个服务可以启动当然也可以停止。如果停止了就无法执行任何定时任务了,解决的方法是打开它:crond或service crond start如果提示crond命令不存在,可能被误删除了,CentOS下可以通过这个命

来源: crontab定时任务不执行的原因 – u011734144的专栏 – 博客频道 – CSDN.NET

我在网上找的时候发现网上主要说了这5个原因:

1 crond服务未启动

crontab不是Linux内核的功能,而是依赖一个crond服务,这个服务可以启动当然也可以停止。如果停止了就无法执行任何定时任务了,解决的方法是打开它:

crond

service crond start

如果提示crond命令不存在,可能被误删除了,CentOS下可以通过这个命令重新安装:

yum -y install crontabs

2 权限问题

比如:脚本没有x执行权限,解决方法:

增加执行权限,或者用bash abc.sh的方法执行

3 路径问题

有的命令在shell中执行正常,但是在crontab执行却总是失败。有可能是因为crontab使用的sh未正确识别路径,比如:以root身份登录shell后执行一个/root/test.sh,只要执行

./test.sh

就可以了。但是在crontab中,就会找不到这个脚本,比如写完整:

/root/test.sh

4 时差问题

因为服务器与客户端时差问题,所以crontab的时间以服务器时间为准。

5 变量问题

有时候命令中含有变量,但crontab执行时却没有,也会造成执行失败。

 

验证后,我的定时脚本test.sh不执行不是上述任何一种原因,其实我的脚本就一句话:

  1. #!/bin/bash
  2. echo 123 >> testFile

我希望通过这种方式来测试 我设置的定时脚本起作用了,于是我设置了该脚本每分钟执行一次,但是死活在脚本所在目录看不到这个文件,我手动执行

# sh test.sh

却能看到在脚本所在目录能看到这个文件

我怀疑是crontab根本没有执行,于是我在crontab中直接添加了

  1. */1 * * * * echo 123 >> /home/denglinjie/testFile

testFile文件生成了,说明crontab是执行了的,那看来是我脚本自身存在问题

最后发现,原来是testFile这里必须写完整的路径,我天真的以为testFile会生成在脚本所在的目录,所以改成了如下形式

  1. #!/bin/bash
  2. echo 123 >> /data/denglinjie/testFile

然后就可以了。

其实路径是个非常容易出问题的地方,假设在/home/denglinjie目录下有一个脚本文件test1.sh,然后在该目录下还有一个脚本文件test2.sh

在test1.sh中执行了test2.sh,而且用的是相对路径,即相对test1.sh所在的路径。

如果在crontab -e中编辑的时候,执行的方式是

sh /home/denglinjie/test1.sh,当执行到调用sh test2.sh的时候,系统会认为是从crontab文件所在的目录去找test2.sh,但是其实是找不到的,造成执行失败

最开始我想的方法是,我要将我写的待执行的脚本文件以及被调用的其他的脚本和crontab文件放到一个地方,这样就可以拉,但是失败了,可能是因为权限问题,我进不去/var/spool/cron目录。

所以另外一个解决方法就是在执行脚本之前先通过 cd   /home/denglinjie 命令进入到脚本所在目录