将UTF8编码的3字节中英文转成2字节中英文.
1. 首先要考虑将String转成一个bytes的数组, 每个汉字是3个bytes, 英文或者标点是1个byte.
2. 然后去判断一下每一个byte的前面几个bit, 看下面的表, 1个byte的字符, 就是英文跟标点, 它的第1个bit是0;
3. 重点是3个bytes的字符, 就是汉字, 或者说CJK, 它的第1个byte的前面4个bit, 是1110, 那么我们可以根据这个来判断.看起来好像下面的代码, 居然是java跟oc可以共用的 :)
if((bytes[i]>>4)==0x0E){
highPos=(bytes[i]<<4)|((bytes[i+1]&0x3f)>>2);
lowPos=((bytes[i+1]&0x03)<<6)|(bytes[i+2]&0x3f);
//NSLog(@"highPos: %x lowPos: %x",highPos,lowPos);
chnCodeArray[chnCodeIndex]=highPos;
chnCodeArray[chnCodeIndex+1]=lowPos;
chnCodeIndex=chnCodeIndex+2;
i=i+3;
}else if(bytes[i]>>7==0){
chnCodeArray[chnCodeIndex]=0;
chnCodeArray[chnCodeIndex+1]=bytes[i];
chnCodeIndex=chnCodeIndex+2;
i++;
}
chnCodeArray就是用来存放结果的byte的数组.
4. 根据协议包, 先发送字符串开始, 数据, 结束的数据包.
[self sendPackage:UTF8_CODE_TRANS_START_CMD withCmdNumber:number withLength:length];
[self sendUTF8Data:utf8Bytes];
[self sendPackage:UTF8_CODE_TRANS_END_CMD withCmdNumber:number withLength:length];
可以看得出, 分别发送cmdPackage跟dataPackage, 可以写得更好的, 希望在安卓的版本中有更好的设计方法.
5. 简单看一下, sendPackage这个方法:
for(int i=0;i<3;i++){
UInt64 startTime=[[NSDate date] timeIntervalSince1970]*1000;
if(packageType==UTF8_CODE_TRANS_START_CMD){
_packageID=[self sendUTF8TransStartCMDPackage:cmdNumber withLength:length];
}else if(packageType==UTF8_CODE_TRANS_END_CMD){
_packageID=[self sendUTF8TransEndCMDPackage:cmdNumber withLength:length];
}
if([self waitForACK:startTime]){
return;
}
}
重试3次, 如果有ACK就算发送成功.
-(bool)waitForACK:(UInt64) startTime{
[self readFFF1Value];
while(_ack==0){
UInt64 endTime=[[NSDate date] timeIntervalSince1970]*1000;
if((endTime-startTime)>1000){
NSLog(@"Over time");
return 0;
}
if(_ack==1){
//NSLog(@"ack==1");
return 1;
}
}
return 0;
}
sendUTF8TransStartCMDPackage方法的精髓是, 妈的, 自己看吧:
_packageSendingType.packageType=UTF8_CODE_TRANS_START_CMD;
_packageSendingType.warningCode=cmdNumber;
_packageSendingType.packageLength=length;
int packageID=arc4random()%255;
NSData *data=[_blePackageFactory createPackage:_packageSendingType withPackageID:packageID];
[self writeCharFFF1:data];
_ack=0;
return packageID;
这里会返回一个随机产生的packageID, 因为app校验ack的时候, 要用到这个packageID.
-(void) checkACK:(NSData*)charValue{
Byte byteBuffer[20];
//NSLog(@"checkACK thread is: %@",[NSThread currentThread]);
//NSLog(@"%d",(int)[notifObj length]);
[charValue getBytes:byteBuffer length:20];
if((byteBuffer[3]==_packageID)&&(byteBuffer[4]==TYPE_DATA_ACK)&&(byteBuffer[5]==ACK_YES)){
_ack=1;
NSLog(@"ACK check ok");
_transferErrorCount=0;
[[NSNotificationCenter defaultCenter] postNotificationName:@"BLECharValue" object:@"ACK!"];
}else{
_transferErrorCount++;
if(_transferErrorCount>5){
_transferErrorCount=0;
[[NSNotificationCenter defaultCenter] postNotificationName:@"BLECharValue" object:@"传输错误"];
}else{
//[self sendStartPackage];
[self resendPackage];
}
}
}
这个方法是在读取char值的回调中运行的, 所以用上了一个外部的全局变量, 叫_ack
6. 总结一下, 配合之前规定的通讯协议, 这个蓝牙通讯, 无非两种包, 一种是命令包, 即通讯开始, 通讯结束, 一种是数据的传输.
关键是中间有很多信息, 例如, 包ID, 校验码, 字符串编号, 开始位, 停止位, 诸如此类...
将UTF8编码的3字节中英文转成2字节中英文.的更多相关文章
- 【异常记录(六)】vs文件乱码:文件加载,使用Unicode(UTF-8)编码加载文件xxx时,有些字节已用Unicode替换字符替换。保存该文件将不会保留原始文件内容。
VS2013偶遇这种情况,页面汉字编码出现乱码. .... 按照网上查到的: 工具>选项>文本编辑器> 勾选了 然并卵,还是乱码... 其实炒鸡简单 用记事本打开另存为,选择 ...
- 如何将一个用utf-8编码的文本用java程序转换成ANSI编码的文本
jdk有一个关于UTF-8的bug所以加了一句 br.skip(1); bugID: http://bugs.java.com/view_bug.do?bug_id=4508058 public st ...
- Java不同编码方式,中英文字符所占字节数
测试代码 public class Test { public static void main(String[] args){ String[] charsetNames={ "UTF-8 ...
- 【Java】如何检测、替换4个字节的utf-8编码(此范围编码包含emoji表情)
> 参考的优秀文章 1.十分钟搞清字符集和字符编码 2.Java中byte与16进制字符串的互相转换 3.[异常处理]Incorrect string value: '\xF0\x90\x8D\ ...
- 在 Perl看来, 字符串只有两种形式. 一种是octets, 即8位序列, 也就是我们通常说的字节数组. 另一种utf8编码的字符串, perl管它叫string. 也就是说: Perl只熟悉两种编
在 Perl看来, 字符串只有两种形式. 一种是octets, 即8位序列, 也就是我们通常说的字节数组. 另一种utf8编码的字符串, perl管它叫string. 也就是说: Perl只熟悉两种编 ...
- python中文utf8编码后是占3个字符,unicode汉字为2字节
一个中文utf8编码后是占3个字符,所以求长度的函数可以这样写 def str_len(str): try: row_l=len(str) utf8_l=len(str.encode('utf-8') ...
- 刨根究底字符编码之十一——UTF-8编码方式与字节序标记
UTF-8编码方式与字节序标记 一.UTF-8编码方式 1. 接下来将分别介绍Unicode字符集的三种编码方式:UTF-8.UTF-16.UTF-32.这里先介绍应用最为广泛的UTF-8. 为满足基 ...
- 1个汉字在UTF-8编码占3个字节
http://blog.csdn.net/ns_code/article/details/14162087 http://www.ruanyifeng.com/blog/2007/10/ascii_u ...
- ASCII、Unicode和UTF-8编码的区别;中英文混合截取
ASCII编码是128个字符 中国把汉字编入GB2312,Shift_JIS/Euc-kr各国标准..... Unicode是为了解决各国乱码的,但浪费存储空间 UTF-8编码把一个Unicode字符 ...
随机推荐
- 使用APICloud平台一周时间开发出休闲娱乐内容类APP
这款app是我花一周左右时间做出来的,一款阅读笑话,段子,糗事,脑筋急转弯,神回复,语录,谜语等的休闲娱乐app,用户除了可以浏览他人发布的内容外,自己也可以发布相关内容,和其他人一同分享有趣的内容, ...
- 【原创】storyboard启动应用程序的大致流程
storyboard启动应用程序的大致流程 [原创] 转载请注明出处:http://i.cnblogs.com/EditPosts.aspx?postid=5395023 1. 用户点击APP图标—— ...
- MVC项目实践,在三层架构下实现SportsStore-06,实现购物车
SportsStore是<精通ASP.NET MVC3框架(第三版)>中演示的MVC项目,在该项目中涵盖了MVC的众多方面,包括:使用DI容器.URL优化.导航.分页.购物车.订单.产品管 ...
- yaffs文件系统
1. 概述yaffs文件系统专为Nandflash设计的日志文件系统,占用page中oob区域.目前有两个版本的yaffs文件系统.nandflash不可靠,存在坏块,存在数据错误,需要软件弥补纠正此 ...
- 第六篇 SQL Server代理深入作业步骤工作流
本篇文章是SQL Server代理系列的第六篇,详细内容请参考原文. 正如这一系列的前几篇所述,SQL Server代理作业是由一系列的作业步骤组成,每个步骤由一个独立的类型去执行.每个作业步骤在技术 ...
- iOS UILabel根据文字获取高度及UITableCell动态获取高度(以截取快递信息为例)
#import <UIKit/UIKit.h> @interface AppDelegate : UIResponder <UIApplicationDelegate> @pr ...
- Windows 8上使用SQL 2012 Configuration Manager
现在使用Windows 8 + Sql Server 2012,而今天想用SQL 2012 Configuraton Manager, 设置别名来访问不同的开发数据库.但发现没能找到SQL 2012 ...
- MS SQL SERVER 锁研究记录
首先创建一直数据表 ChenJi,有如下字段: ID, DanWeiID, Name, ChenJi CREATE TABLE [dbo].[ChenJi]( [ID] [int] NOT NUL ...
- String的replaceAll方法中的正则表达式用法
项目里面 需要对已手机号码进行 如下的显示 比如15088688388 要显示为150****8388的效果 实现这个简单的效果 方法有很多 我想试试用正则表达式去实现 查了点资料最终试出来以下方法可 ...
- 基于axis2框架的两种发布webservice的方法
这次在中韩的产品定义平台的开发,有幸接触到了通过自己写webservice给其他系统调用的项目. 具体开发背景:这个平台做了几个查询接口都是,都是用servlet方式处理请求,而这边系统之间是通过we ...