Android中定义了两种智能指针类型,一种是强指针sp(strong pointer),另外一种是弱指针(weak pointer)。其实称之为强引用和弱引用更合适一些。强指针与一般意义的智能指针概念相同,通过引用计数来记录有多少使用者在使用一个对象,如果所有使用者都放弃了对该对象的引用,则该对象将被自动销毁。
弱指针也指向一个对象,但是弱指针仅仅记录该对象的地址,不能通过弱指针来访问该对象,也就是说不能通过弱智真来调用对象的成员函数或访问对象的成员变量。要想访问弱指针所指向的对象,需首先通过wp类所提供的promote()方法将弱指针升级为强指针。弱指针所指向的对象是有可能在其它地方被销毁的,如果对象已经被销毁,wp的promote()方法将返回空指针,这样就能避免出现地址访问错的情况。
弱指针是怎么做到这一点的呢?其实说白了一点也不复杂,原因就在于每一个可以被智能指针引用的对象都同时被附加了另外一个weakref_impl类型的对象,这个对象中负责记录对象的强指针引用计数和弱指针引用计数。这个对象是智能指针的实现内部使用的,智能指针的使用者看不到这个对象。弱指针操作的就是这个对象,只有当强引用计数和弱引用计数都为0时,这个对象才会被销毁。
说了这么多原理,下面该看看智能指针该如何使用了。假设现在有一个类MyClass,如果要使用智能指针来引用这个类的对象,那么这个类需满足下列两个前提条件:
1:这个类是基类RefBase的子类或间接子类;
2:这个类必须定义虚构造函数,即它的构造函数需要这样定义:
virtual ~MyClass();
满足了上述条件的类就可以定义为Android智能指针了,定义方法和普通指针类似。比如普通指针是这样定义:
MyClass* p_obj;
智能指针是这样定义:
sp<MyClass> p_obj;
注意不要定义成sp<MyClass>* p_obj。这是初学者很容易犯的错误,这样其实相当于定义了一个指针的指针。尽管在语法上没有问题,但是最好不要这样定义。
定义了一个智能指针的变量,就可以象普通指针那样使用它,包括赋值、访问对象成员、作为函数的返回值、作为函数的参数等。比如:
- p_obj = new MyClass(); // 注意不要写成 p_obj = new sp<MyClass>
- sp<MyClass> p_obj2 = p_obj;
- p_obj->func();
- p_obj = create_obj();
- some_func(p_obj);
p_obj = NULL;
上面说的都是强指针,弱指针的定义方法和强指针类似,但是不能通过弱指针来访问对象的成员。下面是弱指针的示例:
- wp<MyClass> wp_obj = new MyClass();
- p_obj = wp_obj.promote(); // 升级为强指针。不过这里要用.而不是->,真是有负其指针之名啊
- wp_obj = NULL;
Android读取txt文件乱码问题
很多Android开发者在读取含有双字节字符的txt文件的时候,可能会中文乱码问题,解决办法如下:
- private String getTextString(String pathandname) throws IOException{
- String str = "";
-
- FileInputStream fis = new FileInputStream(pathandname);
- //InputStreamReader isr = new InputStreamReader(fis, "gbk");
- //BufferedReader br = new BufferedReader(isr);
- int size = fis.available();
- byte[] buffer = new byte[size];
- fis.read(buffer);
- fis.close();
- str = new String(buffer, "GBK");//支持双字节字符
- myApp.setCharNumofString(str.length());//存储总字符数
- return str;
- }
- public static boolean isRightfulTXT(File f) {
-
- String regexp = "[^\\x00-\\xff]";//双字节字符
- Pattern p = Pattern.compile(regexp);
-
- try {
- FileInputStream fis = new FileInputStream(f);
- //"GBK"编码方式支持双字节字符
- InputStreamReader isr = new InputStreamReader(fis, "GBK");
- BufferedReader br = new BufferedReader(isr);
- String line = "";
- while ((line = br.readLine()) != null){
- //循环读取文件每一行,检测是否含有双字节字符
- Matcher m = p.matcher(line);
- if (m.find()) {
- fis.close();
- isr.close();
- br.close();
- return false;
- }
- }
- fis.close();
- isr.close();
- br.close();
- }
- catch (FileNotFoundException e) {
- e.printStackTrace();
- }
- catch (UnsupportedEncodingException e) {
- e.printStackTrace();
- }
- catch (IOException e) {
- e.printStackTrace();
- }
-
- return true;
- }