今天在对数据进行位操作和加操作混合运算的时候,发现得到的结果不对
int result = src << 4 + 10;
后来发现加法操作符的优先级要高于位操作,所以造成此问题。
附Java操作符的优先级表如下:
一共有三种方法可以实现StringBuilder的清空
下面通过代码验证三种方法的效率
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 |
public class StringBuilderClearTest { public static void main(String[] args){ long time = System.currentTimeMillis(); System.out.println("begin: " + time); StringBuilder sbMain1 = new StringBuilder(); StringBuilder sbMain2 = new StringBuilder(); StringBuilder sbMain3 = new StringBuilder(); StringBuilder sb1 = null; for(int i=0; i<1000000; i++){ sb1 = new StringBuilder("a1"); sb1.append("a2"); sb1.append("a3"); sbMain1.append(sb1); } long time1 = System.currentTimeMillis(); StringBuilder sb2 = new StringBuilder(); for(int i=0; i<1000000; i++){ sb2.delete(0, sb2.length()); sb2.append("b1"); sb2.append("b2"); sb2.append("b3"); sbMain2.append(sb2); } long time2 = System.currentTimeMillis(); StringBuilder sb3 = new StringBuilder(); for(int i=0; i< 1000000; i++){ sb3.setLength(0); sb3.append("c1"); sb3.append("c2"); sb3.append("c3"); sbMain3.append(sb3); } long time3 = System.currentTimeMillis(); System.out.println(sbMain1.length()); System.out.println(sbMain2.length()); System.out.println(sbMain3.length()); System.out.println("New耗时: " + (time1 - time)); System.out.println("delete耗时: " + (time2 - time1)); System.out.println("setLength: " + (time3 - time2)); } } |
测试了十几次,选了一次比较有代表性的结果:
1 2 3 |
New耗时 90 delete耗时: 73 setLength: 63 |
由结果可见setLength是最快的,delete不相上下,new是最慢的
这个错误一般是由于使用ListView时使用了列头(addHeaderView)或者列脚(addFooterView)。
不使用列头或者列脚时,listview.getAdapter()得到的adapter和你用setAdapter()设置的adapter一样,使用列头或者列脚后,setAdapter()传入的adapter就被重新赋值成了HeaderViewListAdapter ,这样在对getAdapter()进行强制转换时就会出现转换错误。
修复方法也很简单,把(MyAdapter)listview.getAdapter()改成如下就可以了:
1 |
(MyAdapter) ((HeaderViewListAdapter) listview.getAdapter()).getWrappedAdapter(); |
先将listview.getAdapter()强制转换成HeaderViewListAdapter,然后使用它的getWrappedAdapter()得到原本的adapter。
当电脑连接了多个android设置时,执行adb的相关命令的时候就会出现error: more than one device/emulator的错误,那么怎么才能正确的执行adb的命令呢?我们需要在adb和命令之间加上-s 设备序列号,比如adb shell应该这样写:adb -s 设备序列号 shell
首先要使用adb devices命令来显示当前已连接的设备,如下是我电脑运行的结果:
1 2 3 4 |
#adb devices List of devices attached a5449b40 device 011631d4e6fd50f4 device |
其中第一列是设备的序列号,是我们执行命令时要使用的.我们使用adb shell的时候应该使用如下命令:
1 |
adb -s 011631d4e6fd50f4 shell |
同样的道理,执行adb push或者adb pull的时候也应该在adb后面加上-s 设备序列号
一、先简单首先回顾一下Service的一些问题
1、为什么使用Service?
答:对于一些需要长时间才可执行完毕的操作放在Activity里不太合适,因为Activity关闭后导致操作不能再继续,但后台服务可以持续运行。
2、bindService和startService的区别?
答:1)两者创建的service的生命周期不同
2)startService启动的服务不能返回service的实例,bindService的可以
3、为什么不直接使用new XxxxService来创建服务?
答:在执行startService或者bindService时,android会把会启动的服务进行记录(未做深入研究),以确保拥有后台服务的进程具有较高的优先级,所以要使用startService或者bindService来创建服务。new XxxxService只是创建了一个服务类的实例,而不是一个服务。
4、为什么onServiceDisconnected一直未被调用到?
答:unbindService并不会导致onServieDisconnected被调用,只有在服务被系统销毁的时候才会被调用。
5、可以在一个应用中使用startService跨进程启动另外一个进程的服务吗?
答:可以,需要设置目标进程的包名,可以使用如下代码或者setPackage(未测试)
1 2 3 |
Intent intent = new Intent(); intent.setClassName("com.zht.car.testservice", "com.zht.car.testservice.MyTestService"); startService(intent); |
6、既然可以跨进程调用startService那为什么还要使用aidl?
答:startService只能启动另外进程的服务,最多还可以通过intent传入一些参数,但是不能从对方进程返回数据,所以如果跨进程通讯aidl还是很有必要的。
二、bindService的开发步骤简述
1、创建一个Service,比如本例中为MyTestService
2、在Service中定义一个Binder的子类MyServiceBinder,后面用来返回MyTestService的实例(见一、3为什么不能直接new MyTestService() )
3、定义并实例化一个MyServiceBinder对象,用来在Service的onBind函数中返回给调用者
4、在Service的onBind函数中返回MyServiceBinder实例对象,android sdk会通过各种调用将这个对象传到ServiceConnection的onServiceConnected
5、在Activity中定义一个MyTestService对象
6、在Activity中执行bindService
7、在Activity中实例化一个ServiceConnection对象
8、在ServiceConnection对象的onServiceConnected回调函数中,将参数中的IBinder对象强制转换为Service中定义的Binder子类(MyServiceBinder)对象
9、通过强制Binder子类对象的方法获取MyTestService实例,至此就得到这个Service的实例了
10、绑定状态的控制,通过一个boolean类型的变量来控制,以及其他的回调函数的设置
三、时序图
四、代码
1、Service的代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 |
package com.zht.car.testservice; import android.app.Service; import android.content.Intent; import android.os.Binder; import android.os.Handler; import android.os.IBinder; public class MyTestService extends Service { private MyServiceBinder mMyServiceBinder = new MyServiceBinder(); private OnWeightChangeListener mOnWeightChangeListener; private int mWeight = 20; public MyTestService() { } @Override public IBinder onBind(Intent intent) { new Handler().postDelayed(new Runnable() { @Override public void run() { mWeight = mWeight * 2; if(mOnWeightChangeListener != null){ mOnWeightChangeListener.onWeightChange(mWeight); } } }, 1000); return mMyServiceBinder; } class MyServiceBinder extends Binder { MyTestService getService(){ return MyTestService.this; } } public int getWeight() { return mWeight; } public void setOnWeightChangeListener(OnWeightChangeListener onWeightChangeListener) { mOnWeightChangeListener = onWeightChangeListener; } public interface OnWeightChangeListener { public void onWeightChange(int weight); } } |
2、Activity代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 |
package com.zht.car.testservice; import android.app.Service; import android.content.ComponentName; import android.content.Intent; import android.content.ServiceConnection; import android.os.IBinder; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.util.Log; import android.view.View; public class MainActivity extends AppCompatActivity { private MyTestService mMyTestService; private String LOG_TAG = "MainActivity"; private boolean mServiceBinded = false; private ServiceConnection mServiceConnection = new ServiceConnection() { @Override public void onServiceConnected(ComponentName name, IBinder service) { MyTestService.MyServiceBinder binder = (MyTestService.MyServiceBinder) service; mMyTestService = binder.getService(); mMyTestService.setOnWeightChangeListener(new MyTestService.OnWeightChangeListener() { @Override public void onWeightChange(int weight) { Log.d(LOG_TAG, "Weight changed: " + mMyTestService.getWeight()); } }); mServiceBinded = true; Log.d(LOG_TAG, "Service connected."); Log.d(LOG_TAG, "weight: " + mMyTestService.getWeight()); } @Override public void onServiceDisconnected(ComponentName name) { Log.d(LOG_TAG, "Service disconnected."); mServiceBinded = false; } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); } @Override protected void onDestroy() { if(mServiceBinded) { unbindService(mServiceConnection); } super.onDestroy(); } public void onBindServiceClick(View view) { if(mServiceBinded){ return; } Intent intent = new Intent(this, MyTestService.class); bindService(intent, mServiceConnection, Service.BIND_AUTO_CREATE); } public void onUnbindServiceClick(View view) { if(!mServiceBinded) { return; } unbindService(mServiceConnection); mServiceBinded = false; } } |
如果您的Android应用针对多个语言,那么您需要配置多个语言版本的strings.xml资源文件。
原来在Eclipse中我们都是在多个strings.xml文件中切换来翻译字符资源文件,但是在Android中提供了Translations Editor工具,让我们可以很方便的编辑多语言的资源文件。
打开Translations Editor的方法:
今天使用Android Studio给app打包的时候出现下面的提示错误,
Error:(107) Error: “hello_world” is not translated in “zh” (Chinese) [MissingTranslation]
这个错误是由于项目里有中英文两个语言的字符串资源,但是默认的英文的strings.xml里有一部分字符串值在中文的strings.xml里没有值造成的,要让Android Studio不提示这个问题,只要在resources节点里增加 xmlns:tools=”http://schemas.android.com/tools” tools:ignore=”MissingTranslation”就可以了,如下所示:
1 2 3 4 5 6 |
<resources xmlns:tools="http://schemas.android.com/tools" tools:ignore="MissingTranslation"> <string name="app_name">hello_world</string> <resources> |
LocalBroadcastManager是support v4包里提供的一个组件,它只负责进程内发送广播和接收消息,它的优点如下:
LocalBroadcastManager的实现原理:我们通常都是通过LocalBroadcastManager.getInstance(Context context)来获得它的实例的,通过查看LocalBroadcastManager的代码,笔者发现它的实现并不复杂,它其实是一个单实例对象,那么我们在整个进程内使用的其实是一个LocalBroadcastManager对象,这个单实例对象负责注册消息和分发消息。
了解了一下它的原理,下面我们就用一个小demo来使用一下它吧:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context="com.example.administrator.androidtest.LocalBroadCastMgrActivity"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="New Text" android:id="@+id/tvReceiver" android:layout_alignParentTop="true" android:layout_centerHorizontal="true" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Send" android:id="@+id/button24" android:layout_below="@+id/tvReceiver" android:layout_centerHorizontal="true" android:layout_marginTop="94dp" android:onClick="onBtnSendClick" /> </RelativeLayout> |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 |
package com.example.administrator.androidtest; import android.app.Activity; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.support.v4.content.LocalBroadcastManager; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.view.View; import android.widget.TextView; public class LocalBroadCastMgrActivity extends Activity { LocalBroadcastManager localBroadcastManager; MyMsgReceiver myMsgReceiver; TextView tvReceiver ; public static String BROADCAST_ACTION = "com.bcoder.test.localbroadcast"; public static String INTENTEXTRA_MYWEBSITE = "mywebsite"; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_local_broad_cast_mgr); tvReceiver = (TextView) findViewById(R.id.tvReceiver); } @Override protected void onStart() { super.onStart(); myMsgReceiver = new MyMsgReceiver(); IntentFilter intentFilter = new IntentFilter(BROADCAST_ACTION); localBroadcastManager = LocalBroadcastManager.getInstance(this); //注册广播消息接收器 localBroadcastManager.registerReceiver(myMsgReceiver, intentFilter); } @Override protected void onStop() { //删除已注册的广播 localBroadcastManager.unregisterReceiver(myMsgReceiver); super.onStop(); } public void onBtnSendClick(View v){ Intent intent = new Intent(BROADCAST_ACTION); intent.putExtra(INTENTEXTRA_MYWEBSITE, "http://svn1.bcoder.com"); //在其他Activity或者服务中使用时可以直接使用LocalBroadcastManager.getInstance,不用再定义变量 LocalBroadcastManager.getInstance(this).sendBroadcast(intent); } //定义自己的广播接收类 class MyMsgReceiver extends BroadcastReceiver{ @Override public void onReceive(Context context, Intent intent) { if(null != intent && null != intent.getStringExtra(INTENTEXTRA_MYWEBSITE)){ tvReceiver.setText(String.format("我的网站: %s", intent.getStringExtra(INTENTEXTRA_MYWEBSITE))); } } } } |
注意事项:
在Java中有 enum类型可以用于枚举,但是每个enum的类型都是做为一个类来处理,对于要求效率较高的程序不是一个很好的选择,android官方也不推荐在应用开发中使用此类型。
就是定义static final的整形值,这样虽然效率高,但是不方便查看代码和进行方法参数的描述,如下面的代码虽然加了注释,但是如果常量很多,还是无法轻松定位到想要查看的常量,有多个名称相似的常量时还容易混淆,甚至使用了错误的常量。
1 2 3 4 5 6 7 8 9 10 11 |
//声音警告方式 public static final int ALERT_TYPE_SOUND = 0; //振动警告方式 public static final int ALERT_TYPE_VERBERITE = 1; //文本便签 public static final int NOTETYPE_TEXT = 0; //语音便签 public static final int NOTETYPE_VOICE = 1; //手写便签 public static final int NOTETYPE_HANDWRITE = 2; |
如下是常量做为方法的参数时的描述
1 2 3 4 5 6 7 8 9 10 |
/** * * @param notetype 便签类型,可能是以下几种定义中的一种 * @see Const.NOTETYPE_TEXT * @see Const.NOTETYPE_VOICE * @see Const.NOTETYPE_HANDWRITE */ public void showNote(int notetype){ } |
将常量定义到一个类中,容易限制这些常量的范围,便于查看,而且在使用常量方法的方法描述里易于写注释,如下是常量的定义
1 2 3 4 5 |
public static final class NoteType{ public static final int TEXT = 0; public static final int VOICE = 1; public static final int HANDWRITE = 2; } |
而在方法描述中,只链接到Const.NoteType即可
1 2 3 4 5 6 7 |
/** * @param noteType 便签类型,类型的定义请查看Const.NoteType * @see Const.NoteType */ public void showNote(int noteType){ } |
一、Java部分
1.关于被私有访问控制符private修饰的成员变量,以下说法正确的是?
A) 可被三种类引用:该类自身、与它在同一个包中的其他类、在其他包中的该类的子类
B) 可以被两种类访问和引用:该类本身、该类的所有子类
C) 只能被该类自身所访问和修改 D) 只能被同一个包中的类访问
答案:C。
private定义的属性只能在类本身中使用,任何地方的子类都是不能访问的。
2. 以下声明合法的是
A) default String s; B) public final static native int w(); C) abstract double d;
D) abstract final double hyperbolicCosine();
答案: B
default是Java8引入的用于在接口中实现默认方法的,不能修饰变量;abstract同样不能修饰变量;abstract修饰说明方法必需要去实现,而final只能定义实现方法不能被修改,也就是必须在定义的时候已经实现,所以两者不能同时出现。
另外抽象方法必须定义在抽象类中,不能定义在普通类中。
3. 在调用方法时,若要使方法改变实参的值,可以()
A) 用基本数据类型作为参数 B) 用对象作为参数 C) A和B都对 D) A和B都不对
答案:B。
对于基本类型,参数传递的时候会将值复制,方法内的形参和实参只是值相同,但是两者没有关系;对于对象类型,传递的是对象的引用,修改形参将会导致实参被修改;有一个例外,String虽然是引用类型,但是在修改的时候会复制一个String对象,所以在方法内对String的修改不会影响方法外的实参。
4. Character流与Byte流的区别是
A) 每次读入的字节数不同 B) 前者带有缓冲,后者没有
C) 前者是字符读写,后者是字节读写 D) 二者没有区别,可以互换使用
答案:C。
Character是字符,而Byte是字节,所以选C
5. 线性表若采用链表存储结构,要求内存中可用存储单元地址?
A、必须连续 B、部分地址必须连续 C、一定不连续 D、连续不连续均可
答案:D
链式存储结构灵活性更高,不需要将对象存储在连续的空间内
6. 声明成员变量时,如果不使用任何访问控制符(public, protected, private),则 以下哪种类型的类不能对该成员进行直接访问
A)同一类 B)同一包中的子类 C)同一包中的非子类 D)不同包中的子类
答案:D。
访问级别 |
访问控制修饰符 |
同类 |
同包不同类(不含子类) |
同包子类 |
不同包不同类 (不含子类) |
不同包子类 |
公开 |
public |
√ |
√ |
√ |
√ |
√ |
受保护 |
protected |
√ |
√ |
√ |
— |
√(注意) |
默认 |
没有访问控制修饰符 |
√ |
√ |
√ |
— |
— |
私有 |
private |
√ |
— |
— |
— |
— |
参考:http://www.cnblogs.com/tjudzj/p/4443066.html
7. 一个线程在任何时刻都处于某种线程状态(thread state),例如运行状态、阻塞状态、就
绪状态等。一个线程可以由选项中的哪种线程状态直接到达运行状态?()
A.死亡状态 B.阻塞状态(对象lock池内) C.阻塞状态(对象wait池内) D.就绪状态
答案:D。
8. 在使用interface声明一个接口时,只可以使用( )修饰符修饰该接口?
A、private B、protected C、private protected D、public
答案:D。
接口本来就是要定义给对象用于交互的,当然必须是public的了。
9.下面程序中类ClassDemo中定义了一个静态变量sum,分析程序段的输出结果。()
1 2 3 4 5 6 |
class ClassDemo { public static int sum=1; public ClassDemo() { sum = sum + 5; } } |
1 2 3 4 5 6 7 |
public class ClassDemoTest{ public static void main(String args[]) { ClassDemo demo1=new ClassDemo(); ClassDemo demo2=new ClassDemo(); System.out.println(demo1.sum); } } |
A. 0 B. 6 C. 11 D. 2
答案:C。
static变量只初始化一次,是在链接阶段初始化的。创建两个ClassDemo实例后,执行两次+5,所以最后结果为11
10.下列哪些语句关于内存回收的说明是正确的?
A 程序员必须创建一个线程来释放内存; B 内存回收程序负责释放无用内存
C 内存回收程序允许程序员直接释放内存 D 内存回收程序可以在指定的时间释放内存对象
答案:B
内存回收机制不允许程序员手动释放内存,内存回收程序不定时的运行
二、Android部分
1.Intent传递数据时,下列的数据类型哪些可以被传递()(多选)
A、Serializable B、charsequence C、Parcelable D、Bundle
答案:ABCD