首页 文章 精选 留言 我的

精选列表

搜索[网站开发],共10000篇文章
优秀的个人博客,低调大师

IOS开发-KVO

一、什么是kvo? key-value observing,观察者模式 观察者,观察对象属性的变化,当被观察者该属性发生变化时,观察者会接收到通知,可以在回调函数中做相应的处理 二、有什么作用? 变化处理操作可以在同一个函数中进行,先前本人都会在每次修改属性值的地方调用后续操作,比较繁琐,修改的地方也比较多,现在只要在同一个函数中操作就可以 用kvo只要做监控就行,更加方便易用,减少代码逻辑 三、使用场景: 当一个控件某个属性变化需要做别的相应操作时,比较适合用kvo,只要当该属性发生变化时,会发消息给观察者,在回调函数中做相应的操作 四、实际例子: 一)解释方法: typedef NS_OPTIONS(NSUInteger, NSKeyValueObservingOptions) { NSKeyValueObservingOptionNew = 0x01,//改变后的值 NSKeyValueObservingOptionOld = 0x02,//改变前的值 NSKeyValueObservingOptionInitial NS_ENUM_AVAILABLE(10_5, 2_0) = 0x04, //addobserving之后会马上调用observeValueForKeyPath,不会等到值改变 NSKeyValueObservingOptionPrior NS_ENUM_AVAILABLE(10_5, 2_0) = 0x08 //分2次调用。在值改变之前和值改变之后 }; NSKeyValueObservingOptionNew =0x01,//改变后的值NSKeyValueObservingOptionOld =0x02,//改变前的值这两个用到的比较多 NSObject(NSKeyValueObserving) //一旦被观察者属性发生改变,就会调用此方法后续操作在这个方法中进行 - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context; keyPath:是被观察对象的属性,字符串表示 object:被观察对象 change:属性改变的值,字典,通过objectForKey (key为 FOUNDATION_EXPORTNSString*constNSKeyValueChangeKindKey; FOUNDATION_EXPORTNSString*constNSKeyValueChangeNewKey; FOUNDATION_EXPORTNSString*constNSKeyValueChangeOldKey; FOUNDATION_EXPORTNSString*constNSKeyValueChangeIndexesKey; FOUNDATION_EXPORTNSString*constNSKeyValueChangeNotificationIsPriorKeyNS_AVAILABLE(10_5,2_0); 对应addobserving指定的NSKeyValueObservingOptions ) context:需要传输的数据(void *:任意指针类型),一般传(__bridgevoid*)self 或者 nil,用户也能传别的 for example: - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context { if (context == (__bridge void*)self) { if ([keyPath isEqualToString:kKeyPathForNavigationItemRightBarButtonItems]) { //取值 NSArray *rightBarButtonItems = [change objectForKey:NSKeyValueChangeNewKey]; //需要做操作 self.navigationItem.rightBarButtonItems = rightBarButtonItems; } } else { [super observeValueForKeyPath:keyPath ofObject:objectchange:changecontext:context]; } } -------------------------------------------- 二)接口方法 NSObject(NSKeyValueObserverRegistration) - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context; - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(void *)context NS_AVAILABLE(10_7, 5_0); - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath; NSArray(NSKeyValueObserverRegistration) - (void)addObserver:(NSObject *)observer toObjectsAtIndexes:(NSIndexSet *)indexes forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context; - (void)removeObserver:(NSObject *)observer fromObjectsAtIndexes:(NSIndexSet *)indexes forKeyPath:(NSString *)keyPath context:(void *)contextNS_AVAILABLE(10_7,5_0); - (void)removeObserver:(NSObject *)observer fromObjectsAtIndexes:(NSIndexSet *)indexes forKeyPath:(NSString *)keyPath; - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context; - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(void *)context NS_AVAILABLE(10_7, 5_0); - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath; NSOrderedSet(NSKeyValueObserverRegistration) - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context; - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(void *)context NS_AVAILABLE(10_7, 5_0); - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath; NSSet(NSKeyValueObserverRegistration) - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context; - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(void *)context NS_AVAILABLE(10_7, 5_0); - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath; NSObject(NSKeyValueObserverNotification) //这些方法都为了手动通知用到 - (void)willChangeValueForKey:(NSString *)key; - (void)didChangeValueForKey:(NSString *)key; - (void)willChange:(NSKeyValueChange)changeKind valuesAtIndexes:(NSIndexSet *)indexes forKey:(NSString *)key; - (void)didChange:(NSKeyValueChange)changeKind valuesAtIndexes:(NSIndexSet *)indexes forKey:(NSString *)key; - (void)willChangeValueForKey:(NSString *)key withSetMutation:(NSKeyValueSetMutationKind)mutationKind usingObjects:(NSSet *)objects; - (void)didChangeValueForKey:(NSString *)key withSetMutation:(NSKeyValueSetMutationKind)mutationKind usingObjects:(NSSet *)objects; NSObject(NSKeyValueObservingCustomization) + (NSSet *)keyPathsForValuesAffectingValueForKey:(NSString *)keyNS_AVAILABLE(10_5,2_0); + (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key; **addObserver和removeObserver要成对出现 ----------------------------------------------------------------------- **手动通知: 有两种通知观察者的方式,自动通知和手动通知。顾名思义,手动通知需要在值变化时调用 willChangeValueForKey:和didChangeValueForKey: 方法通知调用者。为求简便,我们一般使用自动通知。 要使用手动通知,需要在 automaticallyNotifiesObserversForKey方法中明确告诉cocoa,哪些键值要使用手动通知: forExample: [self willChangeValueForKey:@"frame"]; self.frame = CGRectMake(0,0,320,100); [self didChangeValueForKey:@"frame"]; 这时候就会调用 - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context; //重新实现NSObject类中的automaticallyNotifiesObserversForKey:方法,返回yes表示自动通知。 + (BOOL)automaticallyNotifiesObserversForKey:(NSString*)key { //当这两个值改变时,使用自动通知已注册过的观察者,观察者需要实现observeValueForKeyPath:ofObject:change:context:方法 if ([key isEqualToString:@"frame"]) { return NO; } return [super automaticallyNotifiesObserversForKey:key]; } 这时候frame就必须要手动通知 *手动通知一般不用,为了方便,都自动通知,所以这部分知道就可以了 ----------------------------------------------------------------------- 上面一些接口方法说明NSObject,NSArray,NSSet均实现了以上方法,因此我们不仅可以观察普通对象,还可以观察数组或结合类对象。 一般用的都是观察NSObject的某个属性 对NSArray进行观察是观察NSArray中每个model的属性 NSSet和NSArray差不多,只不过NSSet是无序集合 本文转自 咖啡机(K.F.J) 博客园博客,原文链接:http://www.cnblogs.com/strick/p/4036663.html ,如需转载请自行联系原作者

优秀的个人博客,低调大师

Hive UDF开发

HIVE允许用户使用UDF(user defined function)对数据进行处理。用户可以使用‘show functions’ 查看function list,可以使用'describe function function-name'查看函数说明。 [plain]view plaincopy hive>showfunctions; OK ! != ...... Timetaken:0.275seconds hive>descfunctionsubstr; OK substr(str,pos[,len])-returnsthesubstringofstrthatstartsatposandisoflengthlenorsubstr(bin,pos[,len])-returnsthesliceofbytearraythatstartsatposandisoflengthlen Timetaken:0.095seconds hive提供的build-in函数包括以下几类:1. 关系操作符:包括 = 、 <> 、 <= 、>=等2. 算数操作符:包括 + 、 - 、 *、/等3. 逻辑操作符:包括AND 、 && 、 OR 、 || 等4. 复杂类型构造函数:包括map、struct、create_union等5. 复杂类型操作符:包括A[n]、Map[key]、S.x6. 数学操作符:包括ln(double a)、sqrt(double a)等7. 集合操作符:包括size(Array<T>)、sort_array(Array<T>)等8. 类型转换函数:binary(string|binary)、cast(expr as <type>)9. 日期函数:包括from_unixtime(bigint unixtime[, string format])、unix_timestamp()等10.条件函数:包括if(boolean testCondition, T valueTrue, T valueFalseOrNull)等11. 字符串函数:包括acat(string|binary A, string|binary B...)等12. 其他:xpath、get_json_objectscii(string str)、con 编写Hive UDF有两种方式: 1. extends UDF ,重写evaluate方法 2. extends GenericUDF,重写initialize、getDisplayString、evaluate方法 编写UDF代码实例(更多例子参考https://svn.apache.org/repos/asf/hive/tags/release-0.8.1/ql/src/java/org/apache/hadoop/hive/ql/udf/):功能:大小转小写ToLowerCase.java: [plain]view plaincopy packagetest.udf; importorg.apache.hadoop.hive.ql.exec.UDF; importorg.apache.hadoop.io.Text; publicclassToLowerCaseextendsUDF{ publicTextevaluate(finalTexts){ if(s==null){returnnull;} returnnewText(s.toString().toLowerCase()); } } 功能:计算array中去重后元素个数 UDFArrayUniqElementNumber.java [java]view plaincopy packagetest.udf; importorg.apache.hadoop.hive.ql.exec.Description; importorg.apache.hadoop.hive.ql.exec.UDFArgumentException; importorg.apache.hadoop.hive.ql.exec.UDFArgumentTypeException; importorg.apache.hadoop.hive.ql.metadata.HiveException; importorg.apache.hadoop.hive.ql.udf.generic.GenericUDF; importorg.apache.hadoop.hive.serde2.objectinspector.ListObjectInspector; importorg.apache.hadoop.hive.serde2.objectinspector.ObjectInspector; importorg.apache.hadoop.hive.serde2.objectinspector.ObjectInspectorUtils; importorg.apache.hadoop.hive.serde2.objectinspector.ObjectInspector.Category; importorg.apache.hadoop.hive.serde2.objectinspector.primitive.PrimitiveObjectInspectorFactory; importorg.apache.hadoop.io.IntWritable; /** *UDF: *Getnubmerofobjectswithduplicateelementseliminated *@authorxiaomin.zhou */ @Description(name="array_uniq_element_number",value="_FUNC_(array)-Returnsnubmerofobjectswithduplicateelementseliminated.",extended="Example:\n" +">SELECT_FUNC_(array(1,2,2,3,3))FROMsrcLIMIT1;\n"+"3") publicclassUDFArrayUniqElementNumberextendsGenericUDF{ privatestaticfinalintARRAY_IDX=0; privatestaticfinalintARG_COUNT=1;//NumberofargumentstothisUDF privatestaticfinalStringFUNC_NAME="ARRAY_UNIQ_ELEMENT_NUMBER";//ExternalName privateListObjectInspectorarrayOI; privateObjectInspectorarrayElementOI; privatefinalIntWritableresult=newIntWritable(-1); publicObjectInspectorinitialize(ObjectInspector[]arguments) throwsUDFArgumentException{ //Checkiftwoargumentswerepassed if(arguments.length!=ARG_COUNT){ thrownewUDFArgumentException("Thefunction"+FUNC_NAME +"accepts"+ARG_COUNT+"arguments."); } //CheckifARRAY_IDXargumentisofcategoryLIST if(!arguments[ARRAY_IDX].getCategory().equals(Category.LIST)){ thrownewUDFArgumentTypeException(ARRAY_IDX,"\"" +org.apache.hadoop.hive.serde.Constants.LIST_TYPE_NAME +"\""+"expectedatfunctionARRAY_CONTAINS,but" +"\""+arguments[ARRAY_IDX].getTypeName()+"\"" +"isfound"); } arrayOI=(ListObjectInspector)arguments[ARRAY_IDX]; arrayElementOI=arrayOI.getListElementObjectInspector(); returnPrimitiveObjectInspectorFactory.writableIntObjectInspector; } publicIntWritableevaluate(DeferredObject[]arguments) throwsHiveException{ result.set(0); Objectarray=arguments[ARRAY_IDX].get(); intarrayLength=arrayOI.getListLength(array); if(arrayLength<=1){ result.set(arrayLength); returnresult; } //elementcompare;Algorithmcomplexity:O(N^2) intnum=1; inti,j; for(i=1;i<arrayLength;i++) { ObjectlistElement=arrayOI.getListElement(array,i); for(j=i-1;j>=0;j--) { if(listElement!=null){ Objecttmp=arrayOI.getListElement(array,j); if(ObjectInspectorUtils.compare(tmp,arrayElementOI,listElement, arrayElementOI)==0){ break; } } } if(-1==j) { num++; } } result.set(num); returnresult; } publicStringgetDisplayString(String[]children){ assert(children.length==ARG_COUNT); return"array_uniq_element_number("+children[ARRAY_IDX]+")"; } } 生成udf.jarhive有三种方法使用自定义的UDF函数 1. 临时添加UDF如下: [plain]view plaincopy hive>select*fromtest; OK Hello wORLD ZXM ljz Timetaken:13.76seconds hive>addjar/home/work/udf.jar; Added/home/work/udf.jartoclasspath Addedresource:/home/work/udf.jar hive>createtemporaryfunctionmytestas'test.udf.ToLowerCase'; OK Timetaken:0.103seconds hive>showfunctions; ...... mytest ...... hive>selectmytest(test.name)fromtest; ...... OK hello world zxm ljz Timetaken:38.218seconds 这种方式在会话结束后,函数自动销毁,因此每次打开新的会话,都需要重新add jar并且create temporary function2. 进入会话前自动创建使用hive -i参数在进入hive时自动初始化 [plain]view plaincopy $cathive_init addjar/home/work/udf.jar; createtemporaryfunctionmytestas'test.udf.ToLowerCase'; $hive-ihive_init Logginginitializedusingconfigurationinfile:/home/work/hive/hive-0.8.1/conf/hive-log4j.properties Hivehistoryfile=/tmp/work/hive_job_log_work_201209200147_1951517527.txt hive>showfunctions; ...... mytest ...... hive>selectmytest(test.name)fromtest; ...... OK hello world zxm ljz 方法2和方法1本质上是相同的,区别在于方法2在会话初始化时自动完成3. 自定义UDF注册为hive内置函数可参考:hive利器自定义UDF+重编译hive 和前两者相比,第三种方式直接将用户的自定义函数作为注册为内置函数,未来使用起来非常简单,但这种方式也非常危险,一旦出错,将是灾难性的,因此,建议如果不是特别通用,并且固化下来的函数,还是使用前两种方式比较靠谱。 本文转自 yntmdr 51CTO博客,原文链接:http://blog.51cto.com/yntmdr/1716940,如需转载请自行联系原作者

优秀的个人博客,低调大师

iOS开发——初识

pragram预处理指令 #pragram mark 代码逻辑分块 常用快捷键 查看文档说明 选中要查看的api,按住option+左键: 注释 cmd + / 框架 Masonry autolayouthttp://adad184.com/2014/09/28/use-masonry-to-quick-solve-autolayout/ ### 使用注意 + 案例 make.left.equalTo(self.contentView).offset(12); make.top.equalTo(self.contentView).offset(18); make.size.width.mas_equalTo(32); make.size.height.mas_equalTo(32); 每个属性设置都单独写;因为是链式语法,如果所有属性的配置都放在一起感觉后面的设置会覆盖前面的设置。 + 居中 > make.center + 垂直居中 > make.centerY + 水平居中 > meke.centerX 工具 cocoapods 程序依赖管理http://blog.devtang.com/2014/05/25/use-cocoapod-to-manage-ios-lib-dependency/ 控件 UIImageView http://blog.sina.com.cn/s/blog_9c2363ad0101e15l.html UITableView https://www.kancloud.cn/digest/ios-1/107419 设置UITableViewDelegate,试图加载委托 设置UITableViewDataSource,数据绑定 可以自定义Cell,需要继承UITableViewCell UITextView 类似于Android里面的EditText UILabel 文本 富文本 >https://www.jianshu.com/p/5d24d22f99c3 API UIFont 设置文本字体 UITableViewDelegate UITableViewDataSource 异常 编译失败,符号冲突 由于import了.m文件,导致;import .h文件后编译成功; unrecognized selector sent to instance 给nil对象发送了消息 自定义的UITableViewCell没有被UI TableView调用 需要调用UITableView的registerClass方法注册自定义的cell NSInternalInconsistencyException reason: 'couldn't find a common superview for <UITextView: 0x7fdb4d015200; 原因:是在设置view的约束前没有调用addSubView方法将view添加到parent view中。 解决办法:是在view初始化完成后调用parentview的addsubview方法; 本文转自wauoen51CTO博客,原文链接:http://blog.51cto.com/7183397/2061610 ,如需转载请自行联系原作者

优秀的个人博客,低调大师

IOS开发-KVC

1. 什么是kvc kvc--key-value coding,健值编码 可以通过key直接访问对象属性的value的方法 2.使用场景 kvc主要是为了让代码变的更简介明了 用的比较多的是在后台数据解析,还有访问一些没有setter,getter方法的属性 3.代码解析: 定义都在NSKeyValueCoding.h中 主要两个方法: //key为当前类的属性名 - (id)valueForKey:(NSString *)key; //当类没有名为key的属性时会调用抛出异常,可以复写valueForUndefinedKey:方法,作处理 - (void)setValue:(id)value forKey:(NSString *)key;//当类没有名为key的属性时会调用抛出异常,可以复写- (void)setValue:(id)value forUndefinedKey:(NSString *)key:方法,作处理 //keyPath 为访问属性的路径,比如:@"person.name",person是当前类的属性,name是person类的一个属性名 - (id)valueForKeyPath:(NSString *)keyPath; - (void)setValue:(NSValue *)value forKeyPath:(NSString *)keyPath; 异常处理 - (id)valueForUndefinedKey:(NSString *)key; - (void)setValue:(id)value forUndefinedKey:(NSString *)key 代码举例: @interface ViewController () @property (nonatomic, weak) NSString *viewString; @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; NSLog(@"----[%@]",[self valueForKey:@"viewString"]); NSLog(@"----[%@]",[self valueForKey:@"viewArray"]); [self setValue:@"testForViewString" forKey:@"viewString"]; [self setValue:@"1" forKey:@"viewArray"]; } - (void)setValue:(id)value forUndefinedKey:(NSString *)key { NSLog(@"【warning】!!!!!!---Undefined--key[%@]--!!!!!!",key); } - (id)valueForUndefinedKey:(NSString *)key { if ([key isEqualToString:@"viewArray"]) { return @"aaaa"; } else { return [super valueForUndefinedKey:key]; } } 实现原理 参考http://www.tuicool.com/articles/M7vQRj 感谢这篇文章,写的很详细 但有个疑问: 虽然说这几个方法一般不会去调用或者用到,但是我尝试了一下,系统并没有调用这些方法,而是直接走到了 - (id)valueForUndefinedKey:(NSString *)key 本文转自 咖啡机(K.F.J) 博客园博客,原文链接:http://www.cnblogs.com/strick/p/4046434.html,如需转载请自行联系原作者

资源下载

更多资源
Mario

Mario

马里奥是站在游戏界顶峰的超人气多面角色。马里奥靠吃蘑菇成长,特征是大鼻子、头戴帽子、身穿背带裤,还留着胡子。与他的双胞胎兄弟路易基一起,长年担任任天堂的招牌角色。

Nacos

Nacos

Nacos /nɑ:kəʊs/ 是 Dynamic Naming and Configuration Service 的首字母简称,一个易于构建 AI Agent 应用的动态服务发现、配置管理和AI智能体管理平台。Nacos 致力于帮助您发现、配置和管理微服务及AI智能体应用。Nacos 提供了一组简单易用的特性集,帮助您快速实现动态服务发现、服务配置、服务元数据、流量管理。Nacos 帮助您更敏捷和容易地构建、交付和管理微服务平台。

Rocky Linux

Rocky Linux

Rocky Linux(中文名:洛基)是由Gregory Kurtzer于2020年12月发起的企业级Linux发行版,作为CentOS稳定版停止维护后与RHEL(Red Hat Enterprise Linux)完全兼容的开源替代方案,由社区拥有并管理,支持x86_64、aarch64等架构。其通过重新编译RHEL源代码提供长期稳定性,采用模块化包装和SELinux安全架构,默认包含GNOME桌面环境及XFS文件系统,支持十年生命周期更新。

Sublime Text

Sublime Text

Sublime Text具有漂亮的用户界面和强大的功能,例如代码缩略图,Python的插件,代码段等。还可自定义键绑定,菜单和工具栏。Sublime Text 的主要功能包括:拼写检查,书签,完整的 Python API , Goto 功能,即时项目切换,多选择,多窗口等等。Sublime Text 是一个跨平台的编辑器,同时支持Windows、Linux、Mac OS X等操作系统。

用户登录
用户注册